diff --git a/decidim-debates/app/commands/decidim/debates/admin/archive_debate.rb b/decidim-debates/app/commands/decidim/debates/admin/archive_debate.rb new file mode 100644 index 0000000000000..39a519c2f79ca --- /dev/null +++ b/decidim-debates/app/commands/decidim/debates/admin/archive_debate.rb @@ -0,0 +1,50 @@ +# frozen_string_literal: true + +module Decidim + module Debates + module Admin + # A command with all the business logic when an admin archives a debate. + class ArchiveDebate < Rectify::Command + # Public: Initializes the command. + # + # archive - Boolean, whether to archive (true) or unarchive (false) the debate. + # debate - The debate object to archive. + # user - The user performing the action. + def initialize(archive, debate, user) + @archive = archive + @debate = debate + @user = user + end + + # Executes the command. Broadcasts these events: + # + # - :ok when the debate is valid. + # - :invalid if the debate wasn't valid and we couldn't proceed. + # + # Returns nothing. + def call + archive_debate + broadcast(:ok) + rescue ActiveRecord::RecordInvalid + broadcast(:invalid) + end + + attr_reader :debate + + private + + def archive_debate + @debate = Decidim.traceability.perform_action!( + :close, + @debate, + @user + ) do + @debate.update!( + archived_at: @archive ? Time.zone.now : nil + ) + end + end + end + end + end +end diff --git a/decidim-debates/app/commands/decidim/debates/admin/close_debate.rb b/decidim-debates/app/commands/decidim/debates/admin/close_debate.rb index 8c60fbeaac36a..4c7940b516b4f 100644 --- a/decidim-debates/app/commands/decidim/debates/admin/close_debate.rb +++ b/decidim-debates/app/commands/decidim/debates/admin/close_debate.rb @@ -37,7 +37,8 @@ def close_debate ) do form.debate.update!( conclusions: form.conclusions, - closed_at: form.closed_at + closed_at: form.closed_at, + archived_at: (Time.zone.now if form.archive) ) end diff --git a/decidim-debates/app/controllers/decidim/debates/admin/debates_controller.rb b/decidim-debates/app/controllers/decidim/debates/admin/debates_controller.rb index ba2ae310cefaa..bfe20506c6b38 100644 --- a/decidim-debates/app/controllers/decidim/debates/admin/debates_controller.rb +++ b/decidim-debates/app/controllers/decidim/debates/admin/debates_controller.rb @@ -61,6 +61,24 @@ def update end end + def archive + enforce_permission_to :archive, :debate, debate: debate + + archive = params[:archive] == "true" + + ArchiveDebate.call(archive, debate, current_user) do + on(:ok) do + flash[:notice] = I18n.t("debates.#{archive ? "archive" : "unarchive"}.success", scope: "decidim.debates.admin") + redirect_to debates_path(archive ? {} : { filter: "archive" }) + end + + on(:invalid) do + flash.now[:alert] = I18n.t("debates.#{archive ? "archive" : "unarchive"}.invalid", scope: "decidim.debates.admin") + redirect_to debates_path(archive ? {} : { filter: "archive" }) + end + end + end + def destroy enforce_permission_to :delete, :debate, debate: debate @@ -74,11 +92,19 @@ def destroy private def debates - @debates ||= Debate.where(component: current_component) + @debates ||= archive? ? all_debates.archived : all_debates.not_archived end def debate - @debate ||= debates.find(params[:id]) + @debate ||= all_debates.find(params[:id]) + end + + def all_debates + Debate.where(component: current_component) + end + + def archive? + params[:filter] == "archive" end end end diff --git a/decidim-debates/app/forms/decidim/debates/admin/close_debate_form.rb b/decidim-debates/app/forms/decidim/debates/admin/close_debate_form.rb index cdc518decd74b..8ead2f9d065cc 100644 --- a/decidim-debates/app/forms/decidim/debates/admin/close_debate_form.rb +++ b/decidim-debates/app/forms/decidim/debates/admin/close_debate_form.rb @@ -15,6 +15,7 @@ class CloseDebateForm < Decidim::Form end attribute :debate, Debate + attribute :archive, Boolean validates :debate, presence: true validate :user_can_close_debate @@ -23,6 +24,10 @@ def closed_at debate&.closed_at || Time.current end + def map_model(model) + self.archive = model.archived_at.present? + end + private def user_can_close_debate diff --git a/decidim-debates/app/helpers/decidim/debates/admin/application_helper.rb b/decidim-debates/app/helpers/decidim/debates/admin/application_helper.rb index 2dbfb7caeab7b..30fb1c337eac0 100644 --- a/decidim-debates/app/helpers/decidim/debates/admin/application_helper.rb +++ b/decidim-debates/app/helpers/decidim/debates/admin/application_helper.rb @@ -7,6 +7,15 @@ module Admin # module ApplicationHelper include Decidim::Admin::ResourceScopeHelper + + def link_to_filtered_debates + unfiltered = params[:filter].blank? + link_to( + t("actions.#{unfiltered ? "archived" : "active"}", scope: "decidim.debates", name: t("models.debate.name", scope: "decidim.debates.admin")), + debates_path(unfiltered ? { filter: "archive" } : {}), + class: "button tiny button--title" + ) + end end end end diff --git a/decidim-debates/app/models/decidim/debates/debate.rb b/decidim-debates/app/models/decidim/debates/debate.rb index 8179485dac8d4..693a5711dc0f7 100644 --- a/decidim-debates/app/models/decidim/debates/debate.rb +++ b/decidim-debates/app/models/decidim/debates/debate.rb @@ -42,6 +42,8 @@ class Debate < Debates::ApplicationRecord scope :open, -> { where(closed_at: nil) } scope :closed, -> { where.not(closed_at: nil) } + scope :archived, -> { where.not(archived_at: nil) } + scope :not_archived, -> { where(archived_at: nil) } scope :authored_by, ->(author) { where(author: author) } scope :commented_by, lambda { |author| joins(:comments).where( @@ -86,6 +88,13 @@ def open? (ama? && open_ama?) || !ama? end + # Public: Checks if the debate is archived or not. + # + # Returns a boolean. + def archived? + archived_at.present? + end + # Public: Overrides the `commentable?` Commentable concern method. def commentable? component.settings.comments_enabled? @@ -157,6 +166,13 @@ def closeable_by?(user) authored_by?(user) end + # Checks whether the user can archive the debate. + # + # user - the user to check for authorship + def archivable_by?(user) + authored_by?(user) + end + # Public: Updates the comments counter cache. We have to do it these # way in order to properly calculate the counter with hidden # comments. diff --git a/decidim-debates/app/permissions/decidim/debates/admin/permissions.rb b/decidim-debates/app/permissions/decidim/debates/admin/permissions.rb index be721bbc436e8..958c23f0b15fa 100644 --- a/decidim-debates/app/permissions/decidim/debates/admin/permissions.rb +++ b/decidim-debates/app/permissions/decidim/debates/admin/permissions.rb @@ -12,6 +12,8 @@ def permissions case permission_action.action when :create, :read allow! + when :archive + toggle_allow(debate.present?) when :update toggle_allow(debate && !debate.closed? && debate.official?) when :delete, :close diff --git a/decidim-debates/app/views/decidim/debates/admin/debate_closes/edit.html.erb b/decidim-debates/app/views/decidim/debates/admin/debate_closes/edit.html.erb index 1bcc8c8e68c34..f6744df3607b7 100644 --- a/decidim-debates/app/views/decidim/debates/admin/debate_closes/edit.html.erb +++ b/decidim-debates/app/views/decidim/debates/admin/debate_closes/edit.html.erb @@ -9,6 +9,12 @@ <%= f.translated :editor, :conclusions, autofocus: true, rows: 15 %> +
<%= t(".archive_help") %>
+<%= t("models.debate.fields.title", scope: "decidim.debates") %> | -<%= t("models.debate.fields.start_time", scope: "decidim.debates") %> | -<%= t("models.debate.fields.end_time", scope: "decidim.debates") %> | - <%= th_resource_scope_label %> -<%= t("actions.title", scope: "decidim.debates") %> | -||||||||
---|---|---|---|---|---|---|---|---|---|---|---|
- <%= link_to present(debate).title, resource_locator(debate).path, target: :blank %> - |
- - <% if debate.start_time %> - <%= l debate.start_time, format: :long %> - <% end %> - | -- <% if debate.end_time %> - <%= l debate.end_time, format: :long %> - <% end %> - | - <%= td_resource_scope_for(debate.scope) %> -
- <% if allowed_to? :update, :debate, debate: debate %>
- <%= icon_link_to "pencil", edit_debate_path(debate), t("actions.edit", scope: "decidim.debates"), class: "action-icon--edit" %>
- <% end %>
+ <% if debates.any? %>
+
+
+ <% if allowed_to? :archive, :debate, debate: debate %>
+ <%= icon_link_to "folder", archive_debate_path(id: debate.id, archive: !debate.archived?), t("actions.#{ debate.archived? ? "unarchive" : "archive" }", scope: "decidim.debates"), class: "action-icon--archive #{"action-icon--highlighted" if debate.archived?}", method: :post %>
+ <% else %>
+
+ <% end %>
+
+ <% if allowed_to? :delete, :debate, debate: debate %>
+ <%= icon_link_to "circle-x", debate_path(debate), t("actions.destroy", scope: "decidim.debates"), method: :delete, class: "action-icon--remove", data: { confirm: t("actions.confirm_destroy", scope: "decidim.debates") } %>
+ <% else %>
+
+ <% end %>
+
|
+
", "
") do - Decidim::Faker::Localized.paragraph(3) + Decidim::Faker::Localized.paragraph(sentence_count: 3) end, instructions: Decidim::Faker::Localized.wrapped("", "
") do - Decidim::Faker::Localized.paragraph(3) + Decidim::Faker::Localized.paragraph(sentence_count: 3) end, start_time: 3.weeks.from_now, end_time: 3.weeks.from_now + 4.hours, @@ -109,22 +109,27 @@ Decidim::Comments::Seed.comments_for(debate) end - closed_debate = Decidim::Debates::Debate.last - closed_debate.conclusions = Decidim::Faker::Localized.wrapped("", "
") do - Decidim::Faker::Localized.paragraph(3) + Decidim::Debates::Debate.last(2).each do |debate| + debate.conclusions = Decidim::Faker::Localized.wrapped("", "
") do + Decidim::Faker::Localized.paragraph(sentence_count: 3) + end + debate.closed_at = Time.current + debate.save! end - closed_debate.closed_at = Time.current - closed_debate.save! + + archived_debate = Decidim::Debates::Debate.last + archived_debate.archived_at = Time.current + archived_debate.save! params = { component: component, category: participatory_space.categories.sample, - title: Decidim::Faker::Localized.sentence(2), + title: Decidim::Faker::Localized.sentence(word_count: 2), description: Decidim::Faker::Localized.wrapped("", "
") do - Decidim::Faker::Localized.paragraph(3) + Decidim::Faker::Localized.paragraph(sentence_count: 3) end, instructions: Decidim::Faker::Localized.wrapped("", "
") do - Decidim::Faker::Localized.paragraph(3) + Decidim::Faker::Localized.paragraph(sentence_count: 3) end, author: user } diff --git a/decidim-debates/spec/commands/decidim/debates/admin/archive_debate_spec.rb b/decidim-debates/spec/commands/decidim/debates/admin/archive_debate_spec.rb new file mode 100644 index 0000000000000..fc55ee80db4b4 --- /dev/null +++ b/decidim-debates/spec/commands/decidim/debates/admin/archive_debate_spec.rb @@ -0,0 +1,48 @@ +# frozen_string_literal: true + +require "spec_helper" + +describe Decidim::Debates::Admin::ArchiveDebate do + subject { described_class.new(archive, debate, user) } + + let(:organization) { create :organization, available_locales: [:en, :ca, :es], default_locale: :en } + let(:participatory_process) { create :participatory_process, organization: organization } + let(:current_component) { create :component, participatory_space: participatory_process, manifest_name: "debates" } + let(:category) { create :category, participatory_space: participatory_process } + let(:user) { create :user, :admin, :confirmed, organization: organization } + let(:debate) { create :debate, component: current_component, archived_at: archived_at } + let(:archive) { true } + let(:archived_at) { nil } + + context "when the debate object is not valid" do + before do + debate.title = nil + end + + it "is not valid" do + expect { subject.call }.to broadcast(:invalid) + end + end + + context "when everything is ok" do + context "when the archive parameter is true" do + let(:archived_at) { nil } + let(:archive) { true } + + it "archives the debate" do + subject.call + expect(debate.archived_at).to be_present + end + end + + context "when the archive parameter is false" do + let(:archived_at) { 1.day.ago } + let(:archive) { false } + + it "unarchives the debate" do + subject.call + expect(debate.archived_at).not_to be_present + end + end + end +end diff --git a/decidim-debates/spec/models/decidim/debates/debate_spec.rb b/decidim-debates/spec/models/decidim/debates/debate_spec.rb index 1e1ee727a443e..dee941e077b7b 100644 --- a/decidim-debates/spec/models/decidim/debates/debate_spec.rb +++ b/decidim-debates/spec/models/decidim/debates/debate_spec.rb @@ -125,4 +125,20 @@ it { is_expected.to be_falsey } end end + + describe "archived?" do + subject { debate.archived? } + + context "when the debate has an archived_at time" do + let(:debate) { build :debate, archived_at: 2.days.ago } + + it { is_expected.to be_truthy } + end + + context "when the debate does not have an archived_at time" do + let(:debate) { build :debate, archived_at: nil } + + it { is_expected.to be_falsey } + end + end end diff --git a/decidim-debates/spec/permissions/decidim/debates/admin/permissions_spec.rb b/decidim-debates/spec/permissions/decidim/debates/admin/permissions_spec.rb index e96cb68a8a5ae..781efc0b1f944 100644 --- a/decidim-debates/spec/permissions/decidim/debates/admin/permissions_spec.rb +++ b/decidim-debates/spec/permissions/decidim/debates/admin/permissions_spec.rb @@ -79,4 +79,34 @@ it { is_expected.to eq false } end end + + describe "debate archive" do + let(:action) do + { scope: :admin, action: :archive, subject: :debate } + end + + context "when the debate is closed" do + let(:debate) { create :debate, :closed, component: debates_component } + + it { is_expected.to eq true } + + context "and it is not official" do + let(:debate) { create :debate, :closed, author: user, component: debates_component } + + it { is_expected.to eq true } + end + end + + context "when debate is open" do + let(:debate) { create :debate, component: debates_component } + + it { is_expected.to eq true } + + context "and it is not official" do + let(:debate) { create :debate, author: user, component: debates_component } + + it { is_expected.to eq true } + end + end + end end diff --git a/decidim-debates/spec/shared/manage_debates_examples.rb b/decidim-debates/spec/shared/manage_debates_examples.rb index 2c0fe4f5b72dc..ae538fc12138f 100644 --- a/decidim-debates/spec/shared/manage_debates_examples.rb +++ b/decidim-debates/spec/shared/manage_debates_examples.rb @@ -63,7 +63,7 @@ it "creates a new debate" do within ".card-title" do - page.find(".button.button--title").click + click_link "New Debate" end within ".new_debate" do @@ -191,4 +191,46 @@ end end end + + describe "archiving a debate" do + let!(:debate) { create(:debate, :closed, component: current_component) } + + it "archives a debate" do + within find("tr", text: translated(debate.title)) do + page.find(".action-icon--archive").click + end + + within ".callout-wrapper" do + expect(page).to have_content("successfully") + end + + expect(page).to have_no_content(translated(debate.title)) + + click_link "Archived debates" + expect(page).to have_content(translated(debate.title)) + end + end + + describe "unarchiving a debate" do + let!(:debate) { create(:debate, :closed, component: current_component, archived_at: 1.day.ago) } + + before do + click_link "Archived debates" + end + + it "unarchives a debate" do + within find("tr", text: translated(debate.title)) do + page.find(".action-icon--archive").click + end + + within ".callout-wrapper" do + expect(page).to have_content("successfully") + end + + expect(page).to have_no_content(translated(debate.title)) + + click_link "Active debates" + expect(page).to have_content(translated(debate.title)) + end + end end