# Copyright (C) 2012-2022 Zammad Foundation, https://zammad-foundation.org/ require 'rails_helper' RSpec.describe 'Ticket > Update > Full Quote Header', current_user_id: -> { current_user.id }, type: :system, time_zone: 'Europe/London' do let(:group) { Group.find_by(name: 'Users') } let(:ticket) { create(:ticket, group: group) } let(:ticket_article) { create(:ticket_article, ticket: ticket, from: 'Example Name ') } let(:customer) { create(:customer) } let(:current_user) { customer } let(:selection) { '' } prepend_before do Setting.set 'ui_ticket_zoom_article_email_full_quote_header', full_quote_header_setting end before do visit "ticket/zoom/#{ticket_article.ticket.id}" end context 'when "ui_ticket_zoom_article_email_full_quote_header" is enabled' do let(:full_quote_header_setting) { true } it 'includes sender when forwarding' do within(:active_content) do click_forward within(:richtext) do expect(page).to contain_full_quote(ticket_article).formatted_for(:forward) end end end it 'includes sender when replying' do within(:active_content) do highlight_and_click_reply within(:richtext) do expect(page).to contain_full_quote(ticket_article).formatted_for(:reply) end end end it 'includes sender when article visibility toggled' do within(:active_content) do set_internal highlight_and_click_reply within(:richtext) do expect(page).to contain_full_quote(ticket_article).formatted_for(:reply) end end end context 'when customer is agent' do let(:customer) { create(:agent) } it 'includes sender without email when forwarding' do within(:active_content) do click_forward within(:richtext) do expect(page).to contain_full_quote(ticket_article).formatted_for(:forward).ensuring_privacy(true) end end end end # https://github.com/zammad/zammad/issues/3824 context 'when TO contains multiple senders and one of them is a known Zammad user' do let(:customer) { create(:customer) } let(:to_1) { "#{customer.fullname} <#{customer.email}>" } let(:to_2) { 'Example Two ' } let(:ticket_article) { create(:ticket_article, ticket: ticket, to: [to_1, to_2].join(', ')) } it 'includes all TO email address' do within(:active_content) do click_forward within(:richtext) do expect(page).to have_text(to_1).and(have_text(to_2)) end end end end context 'ticket is created by agent on behalf of customer' do let(:agent) { create(:agent) } let(:current_user) { agent } let(:ticket) { create(:ticket, group: group, title: 'Created by agent on behalf of a customer', customer: customer) } let(:ticket_article) { create(:ticket_article, ticket: ticket, from: 'Created by agent on behalf of a customer', origin_by_id: customer.id) } it 'includes sender without email when replying' do within(:active_content) do highlight_and_click_reply within(:richtext) do expect(page).to contain_full_quote(ticket_article).formatted_for(:reply) end end end end # https://github.com/zammad/zammad/issues/3855 context 'when ticket article has no recipient' do shared_examples 'when recipient is set to' do |recipient:, recipient_human:| context "when recipient is set to #{recipient_human}" do let(:ticket_article) { create(:ticket_article, :inbound_web, ticket: ticket, to: recipient) } it 'allows to forward without original recipient present' do within(:active_content) do click_forward within(:richtext) do expect(page).to contain_full_quote(ticket_article).formatted_for(:forward) end end end end end include_examples 'when recipient is set to', recipient: '', recipient_human: 'empty string' include_examples 'when recipient is set to', recipient: nil, recipient_human: 'nil' end end context 'when "ui_ticket_zoom_article_email_full_quote_header" is disabled' do let(:full_quote_header_setting) { false } it 'does not include sender when forwarding' do within(:active_content) do click_forward within(:richtext) do expect(page).not_to contain_full_quote(ticket_article).formatted_for(:forward) end end end it 'does not include sender when replying' do within(:active_content) do highlight_and_click_reply within(:richtext) do expect(page).not_to contain_full_quote(ticket_article).formatted_for(:reply) end end end end context 'when text is selected on page while replying' do let(:full_quote_header_setting) { false } let(:before_article_content_selector) { '.ticketZoom-header' } let(:after_article_content_selector) { '.ticket-article-item .humanTimeFromNow' } let(:article_content_selector) { '.ticket-article-item .richtext-content' } it 'does not quote article when bits other than the article are selected' do within(:active_content) do selection = highlight_and_get_selection(before_article_content_selector, '') click_reply within(:richtext) do expect(page).to have_no_text(selection) end end end it 'quotes article when bits inside the article are selected' do within(:active_content) do selection = highlight_and_get_selection(article_content_selector, '') click_reply within(:richtext) do expect(page).to have_text(selection) end end end it 'quotes only article when bits before the article are selected as well' do within(:active_content) do selection = highlight_and_get_selection(before_article_content_selector, article_content_selector) expected_text = find(article_content_selector).text click_reply within(:richtext) do expect(page).to have_no_text(selection) expect(page).to have_text(expected_text) end end end it 'quotes only article when bits after the article are selected as well' do within(:active_content) do selection = highlight_and_get_selection(article_content_selector, after_article_content_selector) expected_text = find(article_content_selector).text click_reply within(:richtext) do expect(page).to have_no_text(selection) expect(page).to have_text(expected_text) end end end it 'quotes only article when bits both before and after the article are selected as well' do within(:active_content) do selection = highlight_and_get_selection(before_article_content_selector, after_article_content_selector) expected_text = find(article_content_selector).text click_reply within(:richtext) do expect(page).to have_no_text(selection) expect(page).to have_text(expected_text) end end end context 'when full quote header setting is enabled' do let(:full_quote_header_setting) { true } it 'can breakout with enter from quote block' do within(:active_content) do highlight_and_click_reply within(:richtext) do blockquote_empty_line = first('blockquote br:nth-child(2)', visible: :all) page.driver.browser.action.move_to_location(blockquote_empty_line.native.location.x, blockquote_empty_line.native.location.y).click.perform end # Special handling for firefox, because the cursor is at the wrong location after the move to with click. if Capybara.current_driver == :zammad_firefox find(:richtext).send_keys(:down) end find(:richtext).send_keys(:enter) within(:richtext) do expect(page).to have_css('blockquote', count: 2) end end end end end def click_forward click '.js-ArticleAction[data-type=emailForward]' end def set_internal click '.js-ArticleAction[data-type=internal]' end def click_reply click '.js-ArticleAction[data-type=emailReply]' end def highlight_and_get_selection(start_selector, end_selector) find(start_selector) .execute_script(<<~JAVASCRIPT, end_selector) let [ end_selector ] = arguments let end_node = $(end_selector)[0] if(!end_node) { end_node = this.nextSibling } window.getSelection().removeAllRanges() var range = window.document.createRange() range.setStart(this, 0) range.setEnd(end_node, end_node.childNodes.length) window.getSelection().addRange(range) JAVASCRIPT find(start_selector).evaluate_script 'window.getSelection().toString().trim()' end def highlight_and_click_reply find('.ticket-article-item .richtext-content') .execute_script <<~JAVASCRIPT window.getSelection().removeAllRanges() var range = window.document.createRange() range.setStart(this, 0) range.setEnd(this.nextSibling, 0) window.getSelection().addRange(range) JAVASCRIPT click_reply end define :contain_full_quote do match do confirm_content && confirm_style end match_when_negated do confirm_no_content end # sets expected quote format # @param [Symbol] :forward or :reply, defaults to :reply if not set chain :formatted_for do |style| @style = style end def style @style || :reply # rubocop:disable RSpec/InstanceVariable end # sets expected privacy level # @param [Boolean] defaults to false if not set chain :ensuring_privacy do |flag| @ensuring_privacy = flag end def ensure_privacy? @ensuring_privacy || false # rubocop:disable RSpec/InstanceVariable end def confirm_content case style when :reply confirm_content_reply when :forward confirm_content_forward end end def confirm_content_reply citation.has_text?(name) && citation.has_no_text?(email) && citation.has_text?(timestamp_reply) end def confirm_content_forward if ensure_privacy? citation.has_text?(name) && citation.has_no_text?(email) && citation.has_text?(timestamp_forward) else citation.has_text?(name) && citation.has_text?(email) && citation.has_text?(timestamp_forward) end end def confirm_no_content citation.has_no_text?(name) && citation.has_no_text?(email) && citation.has_no_text?(timestamp_reply) && citation.has_no_text?(timestamp_forward) end def confirm_style case style when :forward citation.text.match?(%r{Subject(.+)\nDate(.+)}) when :reply citation.text.match?(%r{^On(.+)wrote:$}) end end def citation actual.first('blockquote[type=cite]') end def name (expected.origin_by || expected.created_by).fullname end def email expected.created_by.email end def timestamp_reply expected .created_at .in_time_zone('Europe/London') .strftime('%A, %B %1d, %Y, %1I:%M:%S %p') end def timestamp_forward expected .created_at .in_time_zone('Europe/London') .strftime('%m/%d/%Y %1I:%M %P') end end end