diff --git a/app/assets/stylesheets/admin.scss b/app/assets/stylesheets/admin.scss index 337ce4b0e..8cf67d4d6 100644 --- a/app/assets/stylesheets/admin.scss +++ b/app/assets/stylesheets/admin.scss @@ -203,7 +203,7 @@ body.admin { margin-left: rem-calc(12); } -.proposal-description { +.moderation-description { max-height: rem-calc(65); overflow: hidden; max-width: rem-calc(750); diff --git a/app/controllers/moderation/bulk_controller.rb b/app/controllers/moderation/bulk_controller.rb deleted file mode 100644 index aa28b3213..000000000 --- a/app/controllers/moderation/bulk_controller.rb +++ /dev/null @@ -1,18 +0,0 @@ -class Moderation::BulkController < Moderation::BaseController - - def index - @debates = Debate.sort_for_moderation.page(params[:page]).per(100).includes(:author) - end - - def hide - debates = Debate.where(id: params[:debate_ids]) - if params[:commit] == t('moderation.bulk.index.hide_debates') - debates.each(&:hide) - elsif params[:commit] == t('moderation.bulk.index.block_authors') - debates.includes(:author).map(&:author).uniq.each(&:block) - end - - redirect_to action: :index - end - -end diff --git a/app/controllers/moderation/comments_controller.rb b/app/controllers/moderation/comments_controller.rb index 0ed71a8eb..07c0201ee 100644 --- a/app/controllers/moderation/comments_controller.rb +++ b/app/controllers/moderation/comments_controller.rb @@ -1,33 +1,43 @@ class Moderation::CommentsController < Moderation::BaseController has_filters %w{pending_flag_review all with_ignored_flag}, only: :index + has_orders %w{flags created_at}, only: :index - before_action :load_comments, only: :index + before_action :load_comments, only: [:index, :moderate] load_and_authorize_resource def index @comments = @comments.send(@current_filter) - @comments = @comments.page(params[:page]) + .send("sort_by_#{@current_order}") + .page(params[:page]) + .per(50) end def hide @comment.hide end - def hide_in_moderation_screen - @comment.hide - redirect_to request.query_parameters.merge(action: :index) - end + def moderate + @comments = @comments.where(id: params[:comment_ids]) + + if params[:hide_comments].present? + @comments.accessible_by(current_ability, :hide).each(&:hide) + + elsif params[:ignore_flags].present? + @comments.accessible_by(current_ability, :ignore_flag).each(&:ignore_flag) + + elsif params[:block_authors].present? + author_ids = @comments.pluck(:user_id).uniq + User.where(id: author_ids).accessible_by(current_ability, :block).each(&:block) + end - def ignore_flag - @comment.ignore_flag redirect_to request.query_parameters.merge(action: :index) end private def load_comments - @comments = Comment.accessible_by(current_ability, :hide).flagged.sort_for_moderation.includes(:commentable) + @comments = Comment.accessible_by(current_ability, :moderate) end end diff --git a/app/controllers/moderation/debates_controller.rb b/app/controllers/moderation/debates_controller.rb index c54851cb9..1fe3eb4b4 100644 --- a/app/controllers/moderation/debates_controller.rb +++ b/app/controllers/moderation/debates_controller.rb @@ -1,32 +1,43 @@ class Moderation::DebatesController < Moderation::BaseController has_filters %w{pending_flag_review all with_ignored_flag}, only: :index + has_orders %w{flags created_at}, only: :index - before_action :load_debates, only: :index + before_action :load_debates, only: [:index, :moderate] load_and_authorize_resource def index - @debates = @debates.send(@current_filter).page(params[:page]) + @debates = @debates.send(@current_filter) + .send("sort_by_#{@current_order}") + .page(params[:page]) + .per(50) end def hide @debate.hide end - def hide_in_moderation_screen - @debate.hide - redirect_to request.query_parameters.merge(action: :index) - end + def moderate + @debates = @debates.where(id: params[:debate_ids]) + + if params[:hide_debates].present? + @debates.accessible_by(current_ability, :hide).each(&:hide) + + elsif params[:ignore_flags].present? + @debates.accessible_by(current_ability, :ignore_flag).each(&:ignore_flag) + + elsif params[:block_authors].present? + author_ids = @debates.pluck(:author_id).uniq + User.where(id: author_ids).accessible_by(current_ability, :block).each(&:block) + end - def ignore_flag - @debate.ignore_flag redirect_to request.query_parameters.merge(action: :index) end private def load_debates - @debates = Debate.accessible_by(current_ability, :hide).flagged.sort_for_moderation + @debates = Debate.accessible_by(current_ability, :moderate) end end diff --git a/app/models/ability.rb b/app/models/ability.rb index 23c6f9f01..2d4c98536 100644 --- a/app/models/ability.rb +++ b/app/models/ability.rb @@ -59,12 +59,18 @@ class Ability can :ignore_flag, Comment, ignored_flag_at: nil, hidden_at: nil cannot :ignore_flag, Comment, user_id: user.id + can :moderate, Comment + cannot :moderate, Comment, user_id: user.id + can :hide, Debate, hidden_at: nil cannot :hide, Debate, author_id: user.id can :ignore_flag, Debate, ignored_flag_at: nil, hidden_at: nil cannot :ignore_flag, Debate, author_id: user.id + can :moderate, Debate + cannot :moderate, Debate, author_id: user.id + can :hide, Proposal, hidden_at: nil cannot :hide, Proposal, author_id: user.id diff --git a/app/models/comment.rb b/app/models/comment.rb index 237eb329e..6a3125285 100644 --- a/app/models/comment.rb +++ b/app/models/comment.rb @@ -18,9 +18,10 @@ class Comment < ActiveRecord::Base belongs_to :user, -> { with_hidden } scope :recent, -> { order(id: :desc) } - scope :sort_for_moderation, -> { order(flags_count: :desc, updated_at: :desc) } scope :for_render, -> { with_hidden.includes(user: :organization) } scope :with_visible_author, -> { joins(:user).where("users.hidden_at IS NULL") } + scope :sort_by_flags, -> { order(flags_count: :desc, updated_at: :desc) } + scope :sort_by_created_at, -> { order(created_at: :desc) } after_create :call_after_commented diff --git a/app/models/debate.rb b/app/models/debate.rb index 4785ab04f..d6483f0ad 100644 --- a/app/models/debate.rb +++ b/app/models/debate.rb @@ -25,13 +25,13 @@ class Debate < ActiveRecord::Base before_save :calculate_hot_score, :calculate_confidence_score - scope :sort_for_moderation, -> { order(flags_count: :desc, updated_at: :desc) } scope :for_render, -> { includes(:tags) } scope :sort_by_hot_score , -> { order(hot_score: :desc) } scope :sort_by_confidence_score , -> { order(confidence_score: :desc) } scope :sort_by_created_at, -> { order(created_at: :desc) } scope :sort_by_most_commented, -> { order(comments_count: :desc) } scope :sort_by_random, -> { order("RANDOM()") } + scope :sort_by_flags, -> { order(flags_count: :desc, updated_at: :desc) } # Ahoy setup visitable # Ahoy will automatically assign visit_id on create diff --git a/app/views/moderation/_menu.html.erb b/app/views/moderation/_menu.html.erb index 04bd02f4d..577b66342 100644 --- a/app/views/moderation/_menu.html.erb +++ b/app/views/moderation/_menu.html.erb @@ -31,12 +31,5 @@ <%= t("moderation.menu.users") %> <% end %> - -
  • > - <%= link_to moderation_bulk_path do %> - - <%= t("moderation.menu.bulk") %> - <% end %> -
  • diff --git a/app/views/moderation/bulk/index.html.erb b/app/views/moderation/bulk/index.html.erb deleted file mode 100644 index 5451d4b1c..000000000 --- a/app/views/moderation/bulk/index.html.erb +++ /dev/null @@ -1,46 +0,0 @@ -

    <%= t("moderation.bulk.index.title") %>

    - -

    <%= page_entries_info @debates %>

    - -<%= form_tag moderation_bulk_hide_path, method: :put do %> -

    - <%= t('moderation.bulk.index.check') %>: - <%= link_to t('moderation.bulk.index.check_all'), '#', data: {check_all: "debate_ids[]"} %> - | - <%= link_to t('moderation.bulk.index.check_none'), '#', data: {check_none: "debate_ids[]"} %> -

    - - - - - - - - <% @debates.each do |debate| %> - - - - - - <% end %> -
    - <%= t("moderation.bulk.index.headers.debate") %> - <%= t("moderation.bulk.index.headers.flags") %> - <%= t("moderation.bulk.index.headers.moderate") %> -
    - <%= link_to debate.title, debate, target: "_blank" %> -  •  - <%= debate.author.username %> -  •  - <%= l debate.updated_at.to_date %> -
    - <%= debate.description %> -
    <%= debate.flags_count %> - <%= check_box_tag "debate_ids[]", debate.id, nil, id: "#{dom_id(debate)}_check" %> -
    - - <%= submit_tag t('moderation.bulk.index.hide_debates'), class: "button radius", data: {confirm: t('moderation.bulk.index.confirm')} %> - <%= submit_tag t('moderation.bulk.index.block_authors'), class: "button radius", data: {confirm: t('moderation.bulk.index.confirm')} %> -<% end %> - -<%= paginate @debates %> diff --git a/app/views/moderation/comments/index.html.erb b/app/views/moderation/comments/index.html.erb index d18dca8e8..aeddc941c 100644 --- a/app/views/moderation/comments/index.html.erb +++ b/app/views/moderation/comments/index.html.erb @@ -2,44 +2,76 @@ <%= render 'shared/filter_subnav', i18n_namespace: "moderation.comments.index" %> -

    <%= page_entries_info @comments %>

    +
    +

    <%= page_entries_info @comments %>

    +
    +
    + <%= t("moderation.comments.index.order") %> + <%= render 'shared/order_selector', i18n_namespace: "moderation.comments.index" %> +
    +
    +
    - - - - - - - - <% @comments.each do |comment| %> - - - - - - <% if can? :ignore_flag, comment %> - - <% end %> - <% if comment.ignored_flag? %> - - <% end %> +<%= form_tag moderate_moderation_comments_path(request.query_parameters), method: :put do %> +

    + <%= t('shared.check') %>: + <%= link_to t('shared.check_all'), '#', data: {check_all: "comment_ids[]"} %> + | + <%= link_to t('shared.check_none'), '#', data: {check_none: "comment_ids[]"} %> +

    + +
    - <%= t("moderation.comments.index.headers.commentable") %> |  - <%= t("moderation.comments.index.headers.commentable_type") %> |  - <%= t("moderation.comments.index.headers.updated_at") %> - <%= t("moderation.comments.index.headers.comment") %><%= t("moderation.comments.index.headers.flags") %><%= t("moderation.debates.index.headers.actions") %>
    - <%= link_to comment.commentable.title, comment.commentable %> -
    - <%= comment.commentable_type.constantize.model_name.human %> - <%= l comment.updated_at.to_date %> -
    <%= text_with_links comment.body %><%= comment.flags_count %> - <%= link_to t("moderation.comments.index.hide"), hide_in_moderation_screen_moderation_comment_path(comment, request.query_parameters), method: :put, class: "delete" %> - - <%= link_to t("moderation.comments.index.ignore_flag"), ignore_flag_moderation_comment_path(comment, request.query_parameters), method: :put, class: "button radius tiny warning" %> - - <%= t("moderation.comments.index.ignored_flag") %> -
    + + + - <% end %> -
    + <%= t("moderation.comments.index.headers.comment") %> + + <%= t("moderation.comments.index.headers.moderate") %> +
    -<%= paginate @comments %> + <% @comments.each do |comment| %> + + + <%= comment.commentable_type.constantize.model_name.human %> - + <%= link_to comment.commentable.title, comment.commentable %> +
    + <%= l comment.updated_at.to_date %> +  •  + <%= comment.flags_count %> +  •  + <%= comment.author.username %> +
    +

    + <%= comment.body %> +

    + + + <%= check_box_tag "comment_ids[]", comment.id, nil, id: "#{dom_id(comment)}_check" %> + + + <% end %> + + + <%= submit_tag t('moderation.comments.index.block_authors'), + name: "block_authors", + class: "button radius alert", + data: {confirm: t('moderation.comments.index.confirm')} + %> + +
    + <%= submit_tag t('moderation.comments.index.hide_comments'), + name: "hide_comments", + class: "button radius alert", + data: {confirm: t('moderation.comments.index.confirm')} + %> + <%= submit_tag t('moderation.comments.index.ignore_flags'), + name: "ignore_flags", + class: "button radius success", + data: {confirm: t('moderation.comments.index.confirm')} + %> +
    + + <%= paginate @comments %> + +<% end %> diff --git a/app/views/moderation/debates/index.html.erb b/app/views/moderation/debates/index.html.erb index 3fc3f402b..60a5cf0dd 100644 --- a/app/views/moderation/debates/index.html.erb +++ b/app/views/moderation/debates/index.html.erb @@ -2,43 +2,74 @@ <%= render 'shared/filter_subnav', i18n_namespace: "moderation.debates.index" %> -

    <%= page_entries_info @debates %>

    +
    +

    <%= page_entries_info @debates %>

    +
    +
    + <%= t("moderation.debates.index.order") %> + <%= render 'shared/order_selector', i18n_namespace: "moderation.debates.index" %> +
    +
    +
    - - - - - - - <% @debates.each do |debate| %> - - - - - <% if can? :ignore_flag, debate %> - - <% end %> - <% if debate.ignored_flag? %> - - <% end %> +<%= form_tag moderate_moderation_debates_path(request.query_parameters), method: :put do %> +

    + <%= t('shared.check') %>: + <%= link_to t('shared.check_all'), '#', data: {check_all: "debate_ids[]"} %> + | + <%= link_to t('shared.check_none'), '#', data: {check_none: "debate_ids[]"} %> +

    + +
    - <%= t("moderation.debates.index.headers.title") %> |  - <%= t("moderation.debates.index.headers.updated_at") %> |  - <%= t("moderation.debates.index.headers.description") %> - <%= t("moderation.debates.index.headers.flags") %><%= t("moderation.debates.index.headers.actions") %>
    - <%= link_to debate.title, debate, target: "_blank" %> -
    - <%= l debate.updated_at.to_date %> -
    - <%= debate.description %> -
    <%= debate.flags_count %> - <%= link_to t("moderation.debates.index.hide"), hide_in_moderation_screen_moderation_debate_path(debate, request.query_parameters), method: :put, class: "delete" %> - - <%= link_to t("moderation.debates.index.ignore_flag"), ignore_flag_moderation_debate_path(debate, request.query_parameters), method: :put, class: "button radius tiny warning" %> - - <%= t("moderation.debates.index.ignored_flag") %> -
    + + + - <% end %> -
    + <%= t("moderation.debates.index.headers.debate") %> + + <%= t("moderation.debates.index.headers.moderate") %> +
    + <% @debates.each do |debate| %> + + + <%= link_to debate.title, debate, target: "_blank" %> +
    + <%= l debate.updated_at.to_date %> +  •  + <%= debate.flags_count %> +  •  + <%= debate.author.username %> +
    +
    + <%= debate.description %> +
    + + + <%= check_box_tag "debate_ids[]", debate.id, nil, id: "#{dom_id(debate)}_check" %> + + + <% end %> + -<%= paginate @debates %> + <%= submit_tag t('moderation.debates.index.block_authors'), + name: "block_authors", + class: "button radius alert", + data: {confirm: t('moderation.debates.index.confirm')} + %> + +
    + <%= submit_tag t('moderation.debates.index.hide_debates'), + name: "hide_debates", + class: "button radius alert", + data: {confirm: t('moderation.debates.index.confirm')} + %> + <%= submit_tag t('moderation.debates.index.ignore_flags'), + name: "ignore_flags", + class: "button radius success", + data: {confirm: t('moderation.debates.index.confirm')} + %> +
    + + <%= paginate @debates %> + +<% end %> diff --git a/app/views/moderation/proposals/index.html.erb b/app/views/moderation/proposals/index.html.erb index 59b9f4b7a..38957eb7d 100644 --- a/app/views/moderation/proposals/index.html.erb +++ b/app/views/moderation/proposals/index.html.erb @@ -40,7 +40,7 @@  •  <%= proposal.author.username %>
    -
    +
    <%= proposal.description %>
    diff --git a/config/i18n-tasks.yml b/config/i18n-tasks.yml index 496b202f7..6173538ef 100644 --- a/config/i18n-tasks.yml +++ b/config/i18n-tasks.yml @@ -110,9 +110,12 @@ ignore_unused: - 'admin.users.index.filter*' - 'admin.comments.index.hidden_*' - 'moderation.comments.index.filter*' + - 'moderation.comments.index.order*' - 'moderation.debates.index.filter*' - 'moderation.proposals.index.filter*' - 'moderation.proposals.index.order*' + - 'moderation.debates.index.filter*' + - 'moderation.debates.index.order*' - 'debates.index.orders.*' - 'debates.index.search_form.*' - 'proposals.index.orders.*' diff --git a/config/locales/moderation.en.yml b/config/locales/moderation.en.yml index 7a0d100d6..c49b3bc4a 100644 --- a/config/locales/moderation.en.yml +++ b/config/locales/moderation.en.yml @@ -5,44 +5,47 @@ en: flagged_debates: Debates flagged_comments: Comments users: Ban users - bulk: Bulk moderation dashboard: index: title: Moderation comments: index: - title: Comments flagged as inappropriate + hide_comments: Hide comments + block_authors: Block authors + ignore_flags: Ignore flags + title: Comments headers: - flags: Flags - updated_at: Date - commentable_type: Type - commentable: Root comment: Comment - hide: Hide - ignore_flag: Ignore - ignored_flag: Ignored + moderate: Moderate filter: Filter filters: all: All pending_flag_review: Pending with_ignored_flag: Ignored + order: Order + orders: + created_at: Newest + flags: Most flagged + confirm: Are you sure? debates: index: - title: Debates flagged as inappropriate + hide_debates: Hide debates + block_authors: Block authors + ignore_flags: Ignore flags + title: Debates headers: - flags: Flags - updated_at: Date - title: Title - description: Description - actions: Actions - hide: Hide - ignore_flag: Ignore - ignored_flag: Ignored + debate: Debate + moderate: Moderate filter: Filter filters: all: All pending_flag_review: Pending with_ignored_flag: Ignored + order: Order + orders: + created_at: Newest + flags: Most flagged + confirm: Are you sure? proposals: index: hide_proposals: Hide proposals @@ -62,19 +65,6 @@ en: created_at: Newest flags: Most flagged confirm: Are you sure? - bulk: - index: - title: Bulk moderation - headers: - debate: Debate - flags: Denuncias - moderate: Moderar - hide_debates: Hide debates - block_authors: Block authors - confirm: Are you sure? - check: Select - check_all: All - check_none: None users: notice_hide: User banned. index: diff --git a/config/locales/moderation.es.yml b/config/locales/moderation.es.yml index 42dcb6da3..bee1798df 100644 --- a/config/locales/moderation.es.yml +++ b/config/locales/moderation.es.yml @@ -5,44 +5,47 @@ es: flagged_debates: Debates flagged_comments: Comentarios users: Bloquear usuarios - bulk: Moderar en bloque dashboard: index: title: Moderación comments: index: - title: Comentarios denunciados como inapropiados + hide_comments: Ocultar comentarios + block_authors: Bloquear autores + ignore_flags: Marcar como revisados + title: Comentarios headers: - flags: Denuncias - updated_at: Fecha - commentable_type: Tipo - commentable: Raíz comment: Comentario - hide: Ocultar - ignore_flag: Ignorar - ignored_flag: Ignorado - filter: Filtrar - filters: - all: Todos - pending_flag_review: Pendientes - with_ignored_flag: Ignorados - debates: - index: - title: Debates denunciados como inapropiados - headers: - flags: Denuncias - updated_at: Fecha - title: Título - description: Descripción - actions: Acciones - hide: Ocultar - ignore_flag: Ignorar - ignored_flag: Ignorado + moderate: Moderar filter: Filtrar filters: all: Todos pending_flag_review: Pendientes with_ignored_flag: Marcados como revisados + order: Orden + orders: + created_at: Más nuevos + flags: Más denunciados + confirm: ¿Estás seguro? + debates: + index: + hide_debates: Ocultar debates + block_authors: Bloquear autores + ignore_flags: Marcar como revisados + title: Debates + headers: + debate: Debate + moderate: Moderar + filter: Filtrar + filters: + all: Todos + pending_flag_review: Pendientes + with_ignored_flag: Marcados como revisados + order: Orden + orders: + created_at: Más nuevos + flags: Más denunciados + confirm: ¿Estás seguro? proposals: index: hide_proposals: Ocultar Propuestas @@ -62,19 +65,6 @@ es: created_at: Más recientes flags: Más denunciadas confirm: ¿Estás seguro? - bulk: - index: - title: Moderar en bloque - headers: - debate: Debate - flags: Denuncias - moderate: Moderar - hide_debates: Ocultar debates - block_authors: Bloquear usuarios - confirm: ¿Estás seguro? - check: Seleccionar - check_all: Todos - check_none: Ninguno users: notice_hide: Usuario bloqueado. Se han ocultado todos sus debates y comentarios. index: diff --git a/config/routes.rb b/config/routes.rb index 2ad0d2199..ae52181bb 100644 --- a/config/routes.rb +++ b/config/routes.rb @@ -127,8 +127,9 @@ Rails.application.routes.draw do resources :debates, only: :index do member do put :hide - put :hide_in_moderation_screen - put :ignore_flag + end + collection do + put :moderate end end @@ -144,13 +145,11 @@ Rails.application.routes.draw do resources :comments, only: :index do member do put :hide - put :hide_in_moderation_screen - put :ignore_flag + end + collection do + put :moderate end end - - get '/bulk', to: "bulk#index" - put '/bulk/hide', to: "bulk#hide" end resource :stats, only: [:show] diff --git a/spec/features/debates_spec.rb b/spec/features/debates_spec.rb index 08ff1d8c1..32260992b 100644 --- a/spec/features/debates_spec.rb +++ b/spec/features/debates_spec.rb @@ -56,15 +56,15 @@ feature 'Debates' do login_as(author) visit new_debate_path - fill_in 'debate_title', with: 'End evictions' + fill_in 'debate_title', with: 'A title for a debate' fill_in 'debate_description', with: 'This is very important because...' fill_in 'debate_captcha', with: correct_captcha_text check 'debate_terms_of_service' click_button 'Start a debate' + expect(page).to have_content 'A title for a debate' expect(page).to have_content 'Debate was successfully created.' - expect(page).to have_content 'End evictions' expect(page).to have_content 'This is very important because...' expect(page).to have_content author.name expect(page).to have_content I18n.l(Debate.last.created_at.to_date) diff --git a/spec/features/moderation/bulk_spec.rb b/spec/features/moderation/bulk_spec.rb deleted file mode 100644 index fd3a408b3..000000000 --- a/spec/features/moderation/bulk_spec.rb +++ /dev/null @@ -1,52 +0,0 @@ -require 'rails_helper' - -feature 'Moderate in bulk' do - background do - moderator = create(:moderator) - login_as(moderator.user) - end - - feature "When a debate has been selected for moderation" do - background do - @debate = create(:debate) - visit moderation_bulk_path - - within("#debate_#{@debate.id}") do - check "debate_#{@debate.id}_check" - end - - expect(page).to_not have_css("debate_#{@debate.id}") - end - - scenario 'Hide the debate' do - click_on "Hide debates" - expect(page).to_not have_css("debate_#{@debate.id}") - expect(@debate.reload).to be_hidden - expect(@debate.author).to_not be_hidden - end - - scenario 'Block the author' do - click_on "Block authors" - expect(page).to_not have_css("debate_#{@debate.id}") - expect(@debate.reload).to be_hidden - expect(@debate.author).to be_hidden - end - end - - scenario "select all/none", :js do - create_list(:debate, 2) - - visit moderation_bulk_path - - click_on 'All' - all('input[type=checkbox]').each do |checkbox| - expect(checkbox).to be_checked - end - - click_on 'None' - all('input[type=checkbox]').each do |checkbox| - expect(checkbox).to_not be_checked - end - end - -end diff --git a/spec/features/moderation/comments_spec.rb b/spec/features/moderation/comments_spec.rb index 96a91a9b1..431e7a31d 100644 --- a/spec/features/moderation/comments_spec.rb +++ b/spec/features/moderation/comments_spec.rb @@ -1,105 +1,121 @@ require 'rails_helper' -feature 'Moderate Comments' do +feature 'Moderate comments' do - feature 'Hiding Comments' do - - scenario 'Hide without children hides the comment completely', :js do - citizen = create(:user) - moderator = create(:moderator) - - debate = create(:debate) - comment = create(:comment, commentable: debate, body: 'SPAM') - - login_as(moderator.user) - visit debate_path(debate) - - within("#comment_#{comment.id}") do - click_link 'Hide' - expect(page).to have_css('.comment .faded') - end - - login_as(citizen) - visit debate_path(debate) - - expect(page).to have_css('.comment', count: 1) - expect(page).to_not have_content('This comment has been deleted') - expect(page).to_not have_content('SPAM') - end - - scenario 'Children visible', :js do - citizen = create(:user) - moderator = create(:moderator) - - debate = create(:debate) - comment = create(:comment, commentable: debate, body: 'SPAM') - create(:comment, commentable: debate, body: 'Acceptable reply', parent_id: comment.id) - - login_as(moderator.user) - visit debate_path(debate) - - within("#comment_#{comment.id}") do - first(:link, "Hide").click - expect(page).to have_css('.comment .faded') - end - - login_as(citizen) - visit debate_path(debate) - - expect(page).to have_css('.comment', count: 2) - expect(page).to have_content('This comment has been deleted') - expect(page).to_not have_content('SPAM') - - expect(page).to have_content('Acceptable reply') - end - end - - scenario 'Moderator actions in the comment' do + scenario 'Hide', :js do citizen = create(:user) moderator = create(:moderator) - debate = create(:debate) - comment = create(:comment, commentable: debate) + comment = create(:comment) login_as(moderator.user) - visit debate_path(debate) + visit debate_path(comment.commentable) - within "#comment_#{comment.id}" do - expect(page).to have_link("Hide") - expect(page).to have_link("Ban author") + within("#comment_#{comment.id}") do + click_link 'Hide' + expect(page).to have_css('.comment .faded') end login_as(citizen) - visit debate_path(debate) + visit debate_path(comment.commentable) - within "#comment_#{comment.id}" do - expect(page).to_not have_link("Hide") - expect(page).to_not have_link("Ban author") - end + expect(page).to have_css('.comment', count: 1) + expect(page).to_not have_content('This comment has been deleted') + expect(page).to_not have_content('SPAM') end - scenario 'Moderator actions do not appear in own comments' do + scenario 'Can not hide own comment' do moderator = create(:moderator) - - debate = create(:debate) - comment = create(:comment, commentable: debate, user: moderator.user) + comment = create(:comment, user: moderator.user) login_as(moderator.user) - visit debate_path(debate) + visit debate_path(comment.commentable) - within "#comment_#{comment.id}" do - expect(page).to_not have_link("Hide") - expect(page).to_not have_link("Ban author") + within("#comment_#{comment.id}") do + expect(page).to_not have_link('Hide') + expect(page).to_not have_link('Block author') end end - feature '/moderation/ menu' do + feature '/moderation/ screen' do background do moderator = create(:moderator) login_as(moderator.user) end + feature 'moderate in bulk' do + feature "When a comment has been selected for moderation" do + background do + @comment = create(:comment) + visit moderation_comments_path + within('.sub-nav') do + click_link "All" + end + + within("#comment_#{@comment.id}") do + check "comment_#{@comment.id}_check" + end + + expect(page).to_not have_css("comment_#{@comment.id}") + end + + scenario 'Hide the comment' do + click_on "Hide comments" + expect(page).to_not have_css("comment_#{@comment.id}") + expect(@comment.reload).to be_hidden + expect(@comment.user).to_not be_hidden + end + + scenario 'Block the user' do + click_on "Block authors" + expect(page).to_not have_css("comment_#{@comment.id}") + expect(@comment.reload).to be_hidden + expect(@comment.user).to be_hidden + end + + scenario 'Ignore the comment' do + click_on "Ignore flags" + expect(page).to_not have_css("comment_#{@comment.id}") + expect(@comment.reload).to be_ignored_flag + expect(@comment.reload).to_not be_hidden + expect(@comment.user).to_not be_hidden + end + end + + scenario "select all/none", :js do + create_list(:comment, 2) + + visit moderation_comments_path + + within('.js-check') { click_on 'All' } + + all('input[type=checkbox]').each do |checkbox| + expect(checkbox).to be_checked + end + + within('.js-check') { click_on 'None' } + + all('input[type=checkbox]').each do |checkbox| + expect(checkbox).to_not be_checked + end + end + + scenario "remembering page, filter and order" do + create_list(:comment, 52) + + visit moderation_comments_path(filter: 'all', page: '2', order: 'created_at') + + click_on "Ignore flags" + + expect(page).to have_selector('.js-order-selector[data-order="created_at"]') + + expect(current_url).to include('filter=all') + expect(current_url).to include('page=2') + expect(current_url).to include('order=created_at') + end + end + scenario "Current filter is properly highlighted" do visit moderation_comments_path expect(page).to_not have_link('Pending') @@ -107,98 +123,74 @@ feature 'Moderate Comments' do expect(page).to have_link('Ignored') visit moderation_comments_path(filter: 'all') - expect(page).to_not have_link('All') - expect(page).to have_link('Pending') - expect(page).to have_link('Ignored') + within('.sub-nav') do + expect(page).to_not have_link('All') + expect(page).to have_link('Pending') + expect(page).to have_link('Ignored') + end visit moderation_comments_path(filter: 'pending_flag_review') - expect(page).to have_link('All') - expect(page).to_not have_link('Pending') - expect(page).to have_link('Ignored') + within('.sub-nav') do + expect(page).to have_link('All') + expect(page).to_not have_link('Pending') + expect(page).to have_link('Ignored') + end visit moderation_comments_path(filter: 'with_ignored_flag') - expect(page).to have_link('All') - expect(page).to have_link('Pending') - expect(page).to_not have_link('Ignored') + within('.sub-nav') do + expect(page).to have_link('All') + expect(page).to have_link('Pending') + expect(page).to_not have_link('Ignored') + end end scenario "Filtering comments" do + create(:comment, body: "Regular comment") create(:comment, :flagged, body: "Pending comment") - create(:comment, :flagged, :hidden, body: "Hidden comment") + create(:comment, :hidden, body: "Hidden comment") create(:comment, :flagged, :with_ignored_flag, body: "Ignored comment") visit moderation_comments_path(filter: 'all') + expect(page).to have_content('Regular comment') expect(page).to have_content('Pending comment') expect(page).to_not have_content('Hidden comment') expect(page).to have_content('Ignored comment') visit moderation_comments_path(filter: 'pending_flag_review') + expect(page).to_not have_content('Regular comment') expect(page).to have_content('Pending comment') expect(page).to_not have_content('Hidden comment') expect(page).to_not have_content('Ignored comment') visit moderation_comments_path(filter: 'with_ignored_flag') + expect(page).to_not have_content('Regular comment') expect(page).to_not have_content('Pending comment') expect(page).to_not have_content('Hidden comment') expect(page).to have_content('Ignored comment') end - scenario "Reviewing links remember the pagination setting and the filter" do - per_page = Kaminari.config.default_per_page - (per_page + 2).times { create(:comment, :flagged) } + scenario "sorting comments" do + create(:comment, body: "Flagged comment", created_at: Time.now - 1.day, flags_count: 5) + create(:comment, body: "Flagged newer comment", created_at: Time.now - 12.hours, flags_count: 3) + create(:comment, body: "Newer comment", created_at: Time.now) - visit moderation_comments_path(filter: 'pending_flag_review', page: 2) + visit moderation_comments_path(order: 'created_at') - click_link('Ignore', match: :first, exact: true) + expect("Flagged newer comment").to appear_before("Flagged comment") - expect(current_url).to include('filter=pending_flag_review') - expect(current_url).to include('page=2') - end + visit moderation_comments_path(order: 'flags') - feature 'A flagged comment exists' do + expect("Flagged comment").to appear_before("Flagged newer comment") - background do - debate = create(:debate, title: 'Democracy') - @comment = create(:comment, :flagged, commentable: debate, body: 'spammy spam') - visit moderation_comments_path - end + visit moderation_comments_path(filter: 'all', order: 'created_at') - scenario 'It is displayed with the correct attributes' do - within("#comment_#{@comment.id}") do - expect(page).to have_link('Democracy') - expect(page).to have_content('spammy spam') - expect(page).to have_content('1') - expect(page).to have_link('Hide') - expect(page).to have_link('Ignore') - end - end + expect("Newer comment").to appear_before("Flagged newer comment") + expect("Flagged newer comment").to appear_before("Flagged comment") - scenario 'Hiding the comment' do - within("#comment_#{@comment.id}") do - click_link('Hide') - end + visit moderation_comments_path(filter: 'all', order: 'flags') - expect(current_path).to eq(moderation_comments_path) - expect(page).to_not have_selector("#comment_#{@comment.id}") - - expect(@comment.reload).to be_hidden - end - - scenario 'Marking the comment as ignored' do - within("#comment_#{@comment.id}") do - click_link('Ignore') - end - - expect(current_path).to eq(moderation_comments_path) - - click_link('Ignored') - - within("#comment_#{@comment.id}") do - expect(page).to have_content('Ignored') - end - - expect(@comment.reload).to be_ignored_flag - end + expect("Flagged comment").to appear_before("Flagged newer comment") + expect("Flagged newer comment").to appear_before("Newer comment") end end end diff --git a/spec/features/moderation/debates_spec.rb b/spec/features/moderation/debates_spec.rb index 1df1776c7..303950e45 100644 --- a/spec/features/moderation/debates_spec.rb +++ b/spec/features/moderation/debates_spec.rb @@ -36,13 +36,85 @@ feature 'Moderate debates' do end end - feature '/moderation/ menu' do + feature '/moderation/ screen' do background do moderator = create(:moderator) login_as(moderator.user) end + feature 'moderate in bulk' do + feature "When a debate has been selected for moderation" do + background do + @debate = create(:debate) + visit moderation_debates_path + within('.sub-nav') do + click_link "All" + end + + within("#debate_#{@debate.id}") do + check "debate_#{@debate.id}_check" + end + + expect(page).to_not have_css("debate_#{@debate.id}") + end + + scenario 'Hide the debate' do + click_on "Hide debates" + expect(page).to_not have_css("debate_#{@debate.id}") + expect(@debate.reload).to be_hidden + expect(@debate.author).to_not be_hidden + end + + scenario 'Block the author' do + click_on "Block authors" + expect(page).to_not have_css("debate_#{@debate.id}") + expect(@debate.reload).to be_hidden + expect(@debate.author).to be_hidden + end + + scenario 'Ignore the debate' do + click_on "Ignore flags" + expect(page).to_not have_css("debate_#{@debate.id}") + expect(@debate.reload).to be_ignored_flag + expect(@debate.reload).to_not be_hidden + expect(@debate.author).to_not be_hidden + end + end + + scenario "select all/none", :js do + create_list(:debate, 2) + + visit moderation_debates_path + + within('.js-check') { click_on 'All' } + + all('input[type=checkbox]').each do |checkbox| + expect(checkbox).to be_checked + end + + within('.js-check') { click_on 'None' } + + all('input[type=checkbox]').each do |checkbox| + expect(checkbox).to_not be_checked + end + end + + scenario "remembering page, filter and order" do + create_list(:debate, 52) + + visit moderation_debates_path(filter: 'all', page: '2', order: 'created_at') + + click_on "Ignore flags" + + expect(page).to have_selector('.js-order-selector[data-order="created_at"]') + + expect(current_url).to include('filter=all') + expect(current_url).to include('page=2') + expect(current_url).to include('order=created_at') + end + end + scenario "Current filter is properly highlighted" do visit moderation_debates_path expect(page).to_not have_link('Pending') @@ -50,98 +122,74 @@ feature 'Moderate debates' do expect(page).to have_link('Ignored') visit moderation_debates_path(filter: 'all') - expect(page).to_not have_link('All') - expect(page).to have_link('Pending') - expect(page).to have_link('Ignored') + within('.sub-nav') do + expect(page).to_not have_link('All') + expect(page).to have_link('Pending') + expect(page).to have_link('Ignored') + end visit moderation_debates_path(filter: 'pending_flag_review') - expect(page).to have_link('All') - expect(page).to_not have_link('Pending') - expect(page).to have_link('Ignored') + within('.sub-nav') do + expect(page).to have_link('All') + expect(page).to_not have_link('Pending') + expect(page).to have_link('Ignored') + end visit moderation_debates_path(filter: 'with_ignored_flag') - expect(page).to have_link('All') - expect(page).to have_link('Pending') - expect(page).to_not have_link('Ignored') + within('.sub-nav') do + expect(page).to have_link('All') + expect(page).to have_link('Pending') + expect(page).to_not have_link('Ignored') + end end scenario "Filtering debates" do + create(:debate, title: "Regular debate") create(:debate, :flagged, title: "Pending debate") - create(:debate, :flagged, :hidden, title: "Hidden debate") + create(:debate, :hidden, title: "Hidden debate") create(:debate, :flagged, :with_ignored_flag, title: "Ignored debate") visit moderation_debates_path(filter: 'all') + expect(page).to have_content('Regular debate') expect(page).to have_content('Pending debate') expect(page).to_not have_content('Hidden debate') expect(page).to have_content('Ignored debate') visit moderation_debates_path(filter: 'pending_flag_review') + expect(page).to_not have_content('Regular debate') expect(page).to have_content('Pending debate') expect(page).to_not have_content('Hidden debate') expect(page).to_not have_content('Ignored debate') visit moderation_debates_path(filter: 'with_ignored_flag') + expect(page).to_not have_content('Regular debate') expect(page).to_not have_content('Pending debate') expect(page).to_not have_content('Hidden debate') expect(page).to have_content('Ignored debate') end - scenario "Reviewing links remember the pagination setting and the filter" do - per_page = Kaminari.config.default_per_page - (per_page + 2).times { create(:debate, :flagged) } + scenario "sorting debates" do + create(:debate, title: "Flagged debate", created_at: Time.now - 1.day, flags_count: 5) + create(:debate, title: "Flagged newer debate", created_at: Time.now - 12.hours, flags_count: 3) + create(:debate, title: "Newer debate", created_at: Time.now) - visit moderation_debates_path(filter: 'pending_flag_review', page: 2) + visit moderation_debates_path(order: 'created_at') - click_link('Ignore', match: :first, exact: true) + expect("Flagged newer debate").to appear_before("Flagged debate") - expect(current_url).to include('filter=pending_flag_review') - expect(current_url).to include('page=2') - end + visit moderation_debates_path(order: 'flags') - feature 'A flagged debate exists' do + expect("Flagged debate").to appear_before("Flagged newer debate") - background do - @debate = create(:debate, :flagged, title: 'spammy spam', description: 'buy buy buy') - visit moderation_debates_path - end + visit moderation_debates_path(filter: 'all', order: 'created_at') - scenario 'It is displayed with the correct attributes' do - within("#debate_#{@debate.id}") do - expect(page).to have_link('spammy spam') - expect(page).to have_content('buy buy buy') - expect(page).to have_content('1') - expect(page).to have_link('Hide') - expect(page).to have_link('Ignore') - end - end + expect("Newer debate").to appear_before("Flagged newer debate") + expect("Flagged newer debate").to appear_before("Flagged debate") - scenario 'Hiding the debate' do - within("#debate_#{@debate.id}") do - click_link('Hide') - end + visit moderation_debates_path(filter: 'all', order: 'flags') - expect(current_path).to eq(moderation_debates_path) - expect(page).to_not have_selector("#debate_#{@debate.id}") - - expect(@debate.reload).to be_hidden - end - - scenario 'Marking the debate as ignored' do - within("#debate_#{@debate.id}") do - click_link('Ignore') - end - - expect(current_path).to eq(moderation_debates_path) - - click_link('All') - - within("#debate_#{@debate.id}") do - expect(page).to have_content('Ignored') - end - - expect(@debate.reload).to be_ignored_flag - end + expect("Flagged debate").to appear_before("Flagged newer debate") + expect("Flagged newer debate").to appear_before("Newer debate") end end - end diff --git a/spec/models/ability_spec.rb b/spec/models/ability_spec.rb index a2e6212b5..badb9fda6 100644 --- a/spec/models/ability_spec.rb +++ b/spec/models/ability_spec.rb @@ -173,6 +173,9 @@ describe Ability do it { should_not be_able_to(:hide, hidden_comment) } it { should_not be_able_to(:hide, own_comment) } + it { should be_able_to(:moderate, comment) } + it { should_not be_able_to(:moderate, own_comment) } + it { should be_able_to(:hide, debate) } it { should be_able_to(:hide_in_moderation_screen, debate) } it { should_not be_able_to(:hide, hidden_debate) } @@ -193,6 +196,9 @@ describe Ability do it { should_not be_able_to(:ignore_flag, ignored_debate) } it { should_not be_able_to(:ignore_flag, own_debate) } + it { should be_able_to(:moderate, debate) } + it { should_not be_able_to(:moderate, own_debate) } + it { should be_able_to(:ignore_flag, proposal) } it { should_not be_able_to(:ignore_flag, hidden_proposal) } it { should_not be_able_to(:ignore_flag, ignored_proposal) }