require 'rails_helper' include ActionView::Helpers::DateHelper feature 'Commenting legislation questions' do let(:user) { create :user } let(:legislation_annotation) { create :legislation_annotation, author: user } scenario 'Index' do 3.times { create(:comment, commentable: legislation_annotation) } visit legislation_process_draft_version_annotation_path(legislation_annotation.draft_version.process, legislation_annotation.draft_version, legislation_annotation) expect(page).to have_css('.comment', count: 4) comment = Comment.first within first('.comment') do expect(page).to have_content comment.user.name expect(page).to have_content I18n.l(comment.created_at, format: :datetime) expect(page).to have_content comment.body end end scenario 'Show' do parent_comment = create(:comment, commentable: legislation_annotation) first_child = create(:comment, commentable: legislation_annotation, parent: parent_comment) second_child = create(:comment, commentable: legislation_annotation, parent: parent_comment) href = legislation_process_draft_version_annotation_path(legislation_annotation.draft_version.process, legislation_annotation.draft_version, legislation_annotation) visit comment_path(parent_comment) expect(page).to have_css(".comment", count: 3) expect(page).to have_content parent_comment.body expect(page).to have_content first_child.body expect(page).to have_content second_child.body expect(page).to have_link "Go back to #{legislation_annotation.title}", href: href expect(page).to have_selector("ul#comment_#{parent_comment.id}>li", count: 2) expect(page).to have_selector("ul#comment_#{first_child.id}>li", count: 1) expect(page).to have_selector("ul#comment_#{second_child.id}>li", count: 1) end scenario 'Collapsable comments', :js do parent_comment = legislation_annotation.comments.first child_comment = create(:comment, body: "First subcomment", commentable: legislation_annotation, parent: parent_comment) grandchild_comment = create(:comment, body: "Last subcomment", commentable: legislation_annotation, parent: child_comment) visit legislation_process_draft_version_annotation_path(legislation_annotation.draft_version.process, legislation_annotation.draft_version, legislation_annotation) expect(page).to have_css('.comment', count: 3) find("#comment_#{child_comment.id}_children_arrow").trigger('click') expect(page).to have_css('.comment', count: 2) expect(page).to_not have_content grandchild_comment.body find("#comment_#{child_comment.id}_children_arrow").trigger('click') expect(page).to have_css('.comment', count: 3) expect(page).to have_content grandchild_comment.body find("#comment_#{parent_comment.id}_children_arrow").trigger('click') expect(page).to have_css('.comment', count: 1) expect(page).to_not have_content child_comment.body expect(page).to_not have_content grandchild_comment.body end scenario 'Comment order' do c1 = create(:comment, :with_confidence_score, commentable: legislation_annotation, cached_votes_up: 100, cached_votes_total: 120, created_at: Time.current - 2) c2 = create(:comment, :with_confidence_score, commentable: legislation_annotation, cached_votes_up: 10, cached_votes_total: 12, created_at: Time.current - 1) c3 = create(:comment, :with_confidence_score, commentable: legislation_annotation, cached_votes_up: 1, cached_votes_total: 2, created_at: Time.current) visit legislation_process_draft_version_annotation_path(legislation_annotation.draft_version.process, legislation_annotation.draft_version, legislation_annotation, order: :most_voted) expect(c1.body).to appear_before(c2.body) expect(c2.body).to appear_before(c3.body) visit legislation_process_draft_version_annotation_path(legislation_annotation.draft_version.process, legislation_annotation.draft_version, legislation_annotation, order: :newest) expect(c3.body).to appear_before(c2.body) expect(c2.body).to appear_before(c1.body) visit legislation_process_draft_version_annotation_path(legislation_annotation.draft_version.process, legislation_annotation.draft_version, legislation_annotation, order: :oldest) expect(c1.body).to appear_before(c2.body) expect(c2.body).to appear_before(c3.body) end xscenario 'Creation date works differently in roots and in child comments, even when sorting by confidence_score' do old_root = create(:comment, commentable: legislation_annotation, created_at: Time.current - 10) new_root = create(:comment, commentable: legislation_annotation, created_at: Time.current) old_child = create(:comment, commentable: legislation_annotation, parent_id: new_root.id, created_at: Time.current - 10) new_child = create(:comment, commentable: legislation_annotation, parent_id: new_root.id, created_at: Time.current) visit legislation_process_draft_version_annotation_path(legislation_annotation.draft_version.process, legislation_annotation.draft_version, legislation_annotation, order: :most_voted) expect(new_root.body).to appear_before(old_root.body) expect(old_child.body).to appear_before(new_child.body) visit legislation_process_draft_version_annotation_path(legislation_annotation.draft_version.process, legislation_annotation.draft_version, legislation_annotation, order: :newest) expect(new_root.body).to appear_before(old_root.body) expect(new_child.body).to appear_before(old_child.body) visit legislation_process_draft_version_annotation_path(legislation_annotation.draft_version.process, legislation_annotation.draft_version, legislation_annotation, order: :oldest) expect(old_root.body).to appear_before(new_root.body) expect(old_child.body).to appear_before(new_child.body) end scenario 'Turns links into html links' do create :comment, commentable: legislation_annotation, body: 'Built with http://rubyonrails.org/' visit legislation_process_draft_version_annotation_path(legislation_annotation.draft_version.process, legislation_annotation.draft_version, legislation_annotation) within all('.comment').last do expect(page).to have_content 'Built with http://rubyonrails.org/' expect(page).to have_link('http://rubyonrails.org/', href: 'http://rubyonrails.org/') expect(find_link('http://rubyonrails.org/')[:rel]).to eq('nofollow') expect(find_link('http://rubyonrails.org/')[:target]).to eq('_blank') end end scenario 'Sanitizes comment body for security' do create :comment, commentable: legislation_annotation, body: " click me http://www.url.com" visit legislation_process_draft_version_annotation_path(legislation_annotation.draft_version.process, legislation_annotation.draft_version, legislation_annotation) within all('.comment').last do expect(page).to have_content "click me http://www.url.com" expect(page).to have_link('http://www.url.com', href: 'http://www.url.com') expect(page).not_to have_link('click me') end end scenario 'Paginated comments' do per_page = 10 (per_page + 2).times { create(:comment, commentable: legislation_annotation)} visit legislation_process_draft_version_annotation_path(legislation_annotation.draft_version.process, legislation_annotation.draft_version, legislation_annotation) expect(page).to have_css('.comment', count: per_page) within("ul.pagination") do expect(page).to have_content("1") expect(page).to have_content("2") expect(page).to_not have_content("3") click_link "Next", exact: false end expect(page).to have_css('.comment', count: 3) end feature 'Not logged user' do scenario 'can not see comments forms' do create(:comment, commentable: legislation_annotation) visit legislation_process_draft_version_annotation_path(legislation_annotation.draft_version.process, legislation_annotation.draft_version, legislation_annotation) expect(page).to have_content 'You must Sign in or Sign up to leave a comment' within('#comments') do expect(page).to_not have_content 'Write a comment' expect(page).to_not have_content 'Reply' end end end scenario 'Create', :js do login_as(user) visit legislation_process_draft_version_annotation_path(legislation_annotation.draft_version.process, legislation_annotation.draft_version, legislation_annotation) fill_in "comment-body-legislation_annotation_#{legislation_annotation.id}", with: 'Have you thought about...?' click_button 'Publish comment' within "#comments" do expect(page).to have_content 'Have you thought about...?' expect(page).to have_content '(2)' end end scenario 'Errors on create', :js do login_as(user) visit legislation_process_draft_version_annotation_path(legislation_annotation.draft_version.process, legislation_annotation.draft_version, legislation_annotation) click_button 'Publish comment' expect(page).to have_content "Can't be blank" end scenario 'Reply', :js do citizen = create(:user, username: 'Ana') manuela = create(:user, username: 'Manuela') legislation_annotation = create(:legislation_annotation, author: citizen) comment = legislation_annotation.comments.first login_as(manuela) visit legislation_process_draft_version_annotation_path(legislation_annotation.draft_version.process, legislation_annotation.draft_version, legislation_annotation) click_link "Reply" within "#js-comment-form-comment_#{comment.id}" do fill_in "comment-body-comment_#{comment.id}", with: 'It will be done next week.' click_button 'Publish reply' end within "#comment_#{comment.id}" do expect(page).to have_content 'It will be done next week.' end expect(page).to_not have_selector("#js-comment-form-comment_#{comment.id}", visible: true) end scenario 'Errors on reply', :js do comment = legislation_annotation.comments.first login_as(user) visit legislation_process_draft_version_annotation_path(legislation_annotation.draft_version.process, legislation_annotation.draft_version, legislation_annotation) click_link "Reply" within "#js-comment-form-comment_#{comment.id}" do click_button 'Publish reply' expect(page).to have_content "Can't be blank" end end scenario "N replies", :js do parent = create(:comment, commentable: legislation_annotation) 7.times do create(:comment, commentable: legislation_annotation, parent: parent) parent = parent.children.first end visit legislation_process_draft_version_annotation_path(legislation_annotation.draft_version.process, legislation_annotation.draft_version, legislation_annotation) expect(page).to have_css(".comment.comment.comment.comment.comment.comment.comment.comment") end scenario "Flagging as inappropriate", :js do comment = create(:comment, commentable: legislation_annotation) login_as(user) visit legislation_process_draft_version_annotation_path(legislation_annotation.draft_version.process, legislation_annotation.draft_version, legislation_annotation) within "#comment_#{comment.id}" do page.find("#flag-expand-comment-#{comment.id}").click page.find("#flag-comment-#{comment.id}").click expect(page).to have_css("#unflag-expand-comment-#{comment.id}") end expect(Flag.flagged?(user, comment)).to be end scenario "Undoing flagging as inappropriate", :js do comment = create(:comment, commentable: legislation_annotation) Flag.flag(user, comment) login_as(user) visit legislation_process_draft_version_annotation_path(legislation_annotation.draft_version.process, legislation_annotation.draft_version, legislation_annotation) within "#comment_#{comment.id}" do page.find("#unflag-expand-comment-#{comment.id}").click page.find("#unflag-comment-#{comment.id}").click expect(page).to have_css("#flag-expand-comment-#{comment.id}") end expect(Flag.flagged?(user, comment)).to_not be end scenario "Flagging turbolinks sanity check", :js do legislation_annotation = create(:legislation_annotation, text: "Should we change the world?") comment = create(:comment, commentable: legislation_annotation) login_as(user) visit legislation_process_draft_version_annotation_path(legislation_annotation.draft_version.process, legislation_annotation.draft_version, legislation_annotation) within "#comment_#{comment.id}" do page.find("#flag-expand-comment-#{comment.id}").click expect(page).to have_selector("#flag-comment-#{comment.id}") end end scenario "Erasing a comment's author" do legislation_annotation = create(:legislation_annotation) comment = create(:comment, commentable: legislation_annotation, body: 'this should be visible') comment.user.erase visit legislation_process_draft_version_annotation_path(legislation_annotation.draft_version.process, legislation_annotation.draft_version, legislation_annotation) within "#comment_#{comment.id}" do expect(page).to have_content('User deleted') expect(page).to have_content('this should be visible') end end scenario 'Submit button is disabled after clicking', :js do legislation_annotation = create(:legislation_annotation) login_as(user) visit legislation_process_draft_version_annotation_path(legislation_annotation.draft_version.process, legislation_annotation.draft_version, legislation_annotation) fill_in "comment-body-legislation_annotation_#{legislation_annotation.id}", with: 'Testing submit button!' click_button 'Publish comment' # The button's text should now be "..." # This should be checked before the Ajax request is finished expect(page).to_not have_button 'Publish comment' expect(page).to have_content('Testing submit button!') end feature "Moderators" do scenario "can create comment as a moderator", :js do moderator = create(:moderator) login_as(moderator.user) visit legislation_process_draft_version_annotation_path(legislation_annotation.draft_version.process, legislation_annotation.draft_version, legislation_annotation) fill_in "comment-body-legislation_annotation_#{legislation_annotation.id}", with: "I am moderating!" check "comment-as-moderator-legislation_annotation_#{legislation_annotation.id}" click_button "Publish comment" within "#comments" do expect(page).to have_content "I am moderating!" expect(page).to have_content "Moderator ##{moderator.id}" expect(page).to have_css "div.is-moderator" expect(page).to have_css "img.moderator-avatar" end end scenario "can create reply as a moderator", :js do citizen = create(:user, username: "Ana") manuela = create(:user, username: "Manuela") moderator = create(:moderator, user: manuela) legislation_annotation = create(:legislation_annotation, author: citizen) comment = legislation_annotation.comments.first login_as(manuela) visit legislation_process_draft_version_annotation_path(legislation_annotation.draft_version.process, legislation_annotation.draft_version, legislation_annotation) click_link "Reply" within "#js-comment-form-comment_#{comment.id}" do fill_in "comment-body-comment_#{comment.id}", with: "I am moderating!" check "comment-as-moderator-comment_#{comment.id}" click_button 'Publish reply' end within "#comment_#{comment.id}" do expect(page).to have_content "I am moderating!" expect(page).to have_content "Moderator ##{moderator.id}" expect(page).to have_css "div.is-moderator" expect(page).to have_css "img.moderator-avatar" end expect(page).to_not have_selector("#js-comment-form-comment_#{comment.id}", visible: true) end scenario "can not comment as an administrator" do moderator = create(:moderator) login_as(moderator.user) visit legislation_process_draft_version_annotation_path(legislation_annotation.draft_version.process, legislation_annotation.draft_version, legislation_annotation) expect(page).to_not have_content "Comment as administrator" end end feature "Administrators" do scenario "can create comment as an administrator", :js do admin = create(:administrator) login_as(admin.user) visit legislation_process_draft_version_annotation_path(legislation_annotation.draft_version.process, legislation_annotation.draft_version, legislation_annotation) fill_in "comment-body-legislation_annotation_#{legislation_annotation.id}", with: "I am your Admin!" check "comment-as-administrator-legislation_annotation_#{legislation_annotation.id}" click_button "Publish comment" within "#comments" do expect(page).to have_content "I am your Admin!" expect(page).to have_content "Administrator ##{admin.id}" expect(page).to have_css "div.is-admin" expect(page).to have_css "img.admin-avatar" end end scenario "can create reply as an administrator", :js do citizen = create(:user, username: "Ana") manuela = create(:user, username: "Manuela") admin = create(:administrator, user: manuela) legislation_annotation = create(:legislation_annotation, author: citizen) comment = legislation_annotation.comments.first login_as(manuela) visit legislation_process_draft_version_annotation_path(legislation_annotation.draft_version.process, legislation_annotation.draft_version, legislation_annotation) click_link "Reply" within "#js-comment-form-comment_#{comment.id}" do fill_in "comment-body-comment_#{comment.id}", with: "Top of the world!" check "comment-as-administrator-comment_#{comment.id}" click_button 'Publish reply' end within "#comment_#{comment.id}" do expect(page).to have_content "Top of the world!" expect(page).to have_content "Administrator ##{admin.id}" expect(page).to have_css "div.is-admin" expect(page).to have_css "img.admin-avatar" end expect(page).to_not have_selector("#js-comment-form-comment_#{comment.id}", visible: true) end scenario "can not comment as a moderator" do admin = create(:administrator) login_as(admin.user) visit legislation_process_draft_version_annotation_path(legislation_annotation.draft_version.process, legislation_annotation.draft_version, legislation_annotation) expect(page).to_not have_content "Comment as moderator" end end feature 'Voting comments' do background do @manuela = create(:user, verified_at: Time.current) @pablo = create(:user) @legislation_annotation = create(:legislation_annotation) @comment = create(:comment, commentable: @legislation_annotation) login_as(@manuela) end scenario 'Show' do create(:vote, voter: @manuela, votable: @comment, vote_flag: true) create(:vote, voter: @pablo, votable: @comment, vote_flag: false) visit legislation_process_draft_version_annotation_path(@legislation_annotation.draft_version.process, @legislation_annotation.draft_version, @legislation_annotation) within("#comment_#{@comment.id}_votes") do within(".in_favor") do expect(page).to have_content "1" end within(".against") do expect(page).to have_content "1" end expect(page).to have_content "2 votes" end end scenario 'Create', :js do visit legislation_process_draft_version_annotation_path(@legislation_annotation.draft_version.process, @legislation_annotation.draft_version, @legislation_annotation) within("#comment_#{@comment.id}_votes") do find(".in_favor a").click within(".in_favor") do expect(page).to have_content "1" end within(".against") do expect(page).to have_content "0" end expect(page).to have_content "1 vote" end end scenario 'Update', :js do visit legislation_process_draft_version_annotation_path(@legislation_annotation.draft_version.process, @legislation_annotation.draft_version, @legislation_annotation) within("#comment_#{@comment.id}_votes") do find('.in_favor a').click find('.against a').click within('.in_favor') do expect(page).to have_content "0" end within('.against') do expect(page).to have_content "1" end expect(page).to have_content "1 vote" end end xscenario 'Trying to vote multiple times', :js do visit legislation_process_draft_version_annotation_path(@legislation_annotation.draft_version.process, @legislation_annotation.draft_version, @legislation_annotation) within("#comment_#{@comment.id}_votes") do find('.in_favor a').click within('.in_favor') do expect(page).to have_content "1" end find('.in_favor a').click within('.in_favor') do expect(page).to_not have_content "2" expect(page).to have_content "1" end within('.against') do expect(page).to have_content "0" end expect(page).to have_content "1 vote" end end end feature "Merged comment threads", :js do let!(:draft_version) { create(:legislation_draft_version, :published) } let!(:annotation1) do create(:legislation_annotation, draft_version: draft_version, text: "my annotation", ranges: [{"start" => "/p[1]", "startOffset" => 1, "end" => "/p[1]", "endOffset" => 5}]) end let!(:annotation2) do create(:legislation_annotation, draft_version: draft_version, text: "my other annotation", ranges: [{"start" => "/p[1]", "startOffset" => 1, "end" => "/p[1]", "endOffset" => 10}]) end background do login_as user visit legislation_process_draft_version_path(draft_version.process, draft_version) expect(page).to have_css ".annotator-hl" first(:css, ".annotator-hl").click within(".comment-box") do expect(page).to have_content "my annotation" expect(page).to have_content "my other annotation" end end scenario 'View comments of annotations in an included range' do within("#annotation-link") do first(:css, "a").trigger('click') end expect(page).to have_css(".comment", count: 2) expect(page).to have_content("my annotation") expect(page).to have_content("my other annotation") end scenario "Reply on a single annotation thread and display it in the merged annotation thread" do within(".comment-box #annotation-#{annotation1.id}-comments") do first(:link, "0 replies").click end comment = annotation1.comments.first click_link "Reply" within "#js-comment-form-comment_#{comment.id}" do fill_in "comment-body-comment_#{comment.id}", with: 'replying in single annotation thread' click_button 'Publish reply' end within "#comment_#{comment.id}" do expect(page).to have_content 'replying in single annotation thread' end visit legislation_process_draft_version_path(draft_version.process, draft_version) expect(page).to have_css ".annotator-hl" first(:css, ".annotator-hl").click within(".comment-box") do expect(page).to have_content "my annotation" expect(page).to have_content "my other annotation" end within("#annotation-link") do first(:css, "a").trigger('click') end expect(page).to have_css(".comment", count: 3) expect(page).to have_content("my annotation") expect(page).to have_content("my other annotation") expect(page).to have_content("replying in single annotation thread") end scenario "Reply on a multiple annotation thread and display it in the single annotation thread" do within("#annotation-link") do first(:css, "a").trigger('click') end comment = annotation2.comments.first within("#comment_#{comment.id}") do click_link "Reply" end within "#js-comment-form-comment_#{comment.id}" do fill_in "comment-body-comment_#{comment.id}", with: 'replying in multiple annotation thread' click_button 'Publish reply' end within "#comment_#{comment.id}" do expect(page).to have_content 'replying in multiple annotation thread' end visit legislation_process_draft_version_path(draft_version.process, draft_version) expect(page).to have_css ".annotator-hl" first(:css, ".annotator-hl").click within(".comment-box #annotation-#{annotation2.id}-comments") do first(:link, "1 reply").click end expect(page).to have_css(".comment", count: 2) expect(page).to have_content("my other annotation") expect(page).to have_content("replying in multiple annotation thread") end end end