# Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/
require 'rails_helper'
RSpec.describe 'Ticket > Update > Simultaneously with two different user', type: :system do
let(:group) { Group.find_by(name: 'Users') }
let(:ticket) { create(:ticket, group: group) }
let(:agent) { User.find_by(login: 'agent1@example.com') }
# rubocop:disable RSpec/InstanceVariable
define :have_avatar do |expected|
chain(:changed, :text)
match do
elem = find_element
return false if elem.nil?
return true if !@icon && !@no_icon
return elem.has_no_css? '.icon' if @no_icon
elem.has_css? ".icon-#{@icon}"
end
def find_element
if expected.is_a? User
actual.find "#{base_selector}#{select_by_user}"
else
actual.find base_selector, text: expected
end
rescue
nil
end
match_when_negated do
if expected.is_a? User
return actual.has_no_css? "#{base_selector}#{select_by_user}"
end
actual.has_no_css? base_selector, text: expected
end
chain :changed! do
@changed = true
end
chain :with_icon do |icon|
@icon = icon
end
chain :with_no_icon! do
@no_icon = true
end
def select_by_user
"[data-id='#{expected.id}']"
end
def base_selector
changed_class = @changed ? 'changed' : 'not-changed'
".js-attributeBar .js-avatar .avatar--#{changed_class}"
end
end
# rubocop:enable RSpec/InstanceVariable
def check_taskbar_tab(ticket_id, title: nil, modified: false)
tab_data_key = "Ticket-#{ticket_id}"
if title
taskbar_tab = find(".tasks .task[data-key='#{tab_data_key}']")
expect(taskbar_tab).to have_css('.nav-tab-name', text: title)
end
if modified
expect(page).to have_css(".tasks .task[data-key='#{tab_data_key}'].is-modified")
else
expect(page).to have_no_css(".tasks .task[data-key='#{tab_data_key}'].is-modified")
end
end
context 'when two different users are simultaneously in one ticket' do
before do
visit "#ticket/zoom/#{ticket.id}"
using_session(:second_browser) do
login(
username: agent.login,
password: 'test',
)
visit "#ticket/zoom/#{ticket.id}"
end
end
it 'avatar from other user should be visible in ticket zoom' do
expect(page).to have_avatar('AT')
using_session(:second_browser) do
expect(page).to have_avatar('TA')
end
end
it 'check changes from the first user and added changes from the second user' do
within(:active_content) do
find('.js-textarea').send_keys('some note')
expect(page).to have_css('.js-reset')
end
expect(page).to have_avatar('AT')
using_session(:second_browser) do
expect(page).to have_avatar('TA').changed!
within(:active_content) do
find('.js-textarea').send_keys('some other note')
expect(page).to have_css('.js-reset')
end
end
expect(page).to have_avatar('AT').changed!
using_session(:second_browser) do
within(:active_content) do
click '.js-attributeBar .js-submit'
expect(page).to have_no_css('.js-reset')
expect(page).to have_css('.article-content', text: 'some other note')
end
expect(page).to have_avatar('TA').changed!
end
expect(page).to have_avatar('AT')
check_taskbar_tab(ticket.id, title: ticket.title, modified: true)
within(:active_content) do
expect(page).to have_css('.article-content', text: 'some other note')
click '.js-attributeBar .js-submit'
expect(page).to have_no_css('.js-reset')
expect(page).to have_css('.article-content', text: 'some note')
end
using_session(:second_browser) do
expect(page).to have_avatar('TA')
expect(page).to have_css('.article-content', text: 'some note')
check_taskbar_tab(ticket.id, title: ticket.title, modified: true)
end
# Reload browsers and check if state is correct.
refresh
using_session(:second_browser) do
refresh
expect(page).to have_avatar('TA')
expect(page).to have_no_css('.js-reset')
end
expect(page).to have_avatar('AT')
expect(page).to have_no_css('.js-reset')
end
it 'check refresh for unsaved changes and reset after refresh' do
using_session(:second_browser) do
within(:active_content) do
find('.js-textarea').send_keys('some other note')
expect(page).to have_css('.js-reset')
end
expect(page).to have_avatar('TA')
# We need to wait for the auto save feature.
wait.until do
Taskbar.find_by(key: "Ticket-#{ticket.id}", user_id: agent.id).state_changed?
end
refresh
end
expect(page).to have_avatar('AT').changed!
using_session(:second_browser) do
refresh
within(:active_content) do
click '.js-reset'
expect(page).to have_css('.js-textarea', text: '')
end
end
expect(page).to have_avatar('AT')
end
it 'change title with second user' do
find('.js-textarea').send_keys('some note')
using_session(:second_browser) do
find('.js-textarea').send_keys('some other note')
find('.ticketZoom-header .js-objectTitle').set('TTTsome level 2 subject 123äöü')
# Click in the body field, to trigger the title update.
find('.js-textarea').send_keys('trigger title')
expect(page).to have_css('.js-objectTitle', text: 'TTTsome level 2 subject 123äöü')
check_taskbar_tab(ticket.id, title: 'TTTsome level 2 subject 123äöü')
expect(page).to have_css('.js-textarea', text: 'some other note')
end
expect(page).to have_css('.js-objectTitle', text: 'TTTsome level 2 subject 123äöü')
expect(page).to have_css('.js-textarea', text: 'some note')
check_taskbar_tab(ticket.id, title: 'TTTsome level 2 subject 123äöü', modified: true)
# Refresh and check that modified flag is gone
refresh
check_taskbar_tab(ticket.id, title: 'TTTsome level 2 subject 123äöü', modified: false)
end
end
context 'when working on multiple platforms', authenticated_as: :user do
let(:ticket) { create(:ticket) }
let(:user) { create(:agent, groups: [ticket.group]) }
let(:another_user) { create(:agent, groups: [ticket.group]) }
let(:key) { "Ticket-#{ticket.id}" }
let(:path) { "ticket/zoom/#{ticket.id}" }
let(:taskbar_mobile) { create(:taskbar, user: user, app: :mobile, key: key) }
let(:taskbar_desktop) { create(:taskbar, user: user, app: :desktop, key: key) }
let(:another_taskbar_mobile) { create(:taskbar, user: another_user, app: :mobile, key: key) }
let(:another_taskbar_desktop) { create(:taskbar, user: another_user, app: :desktop, key: key) }
context 'when looking on a ticket' do
before do
taskbar_desktop
visit path
end
it 'does not show current user' do
expect(page).not_to have_avatar(user)
end
end
context 'when another user is looking on desktop' do
before do
another_taskbar_desktop
taskbar_desktop
visit path
end
it 'shows another user' do
expect(page).to have_avatar(another_user).with_no_icon!
end
end
context 'when another user is looking on mobile' do
before do
another_taskbar_mobile
taskbar_desktop
visit path
end
it 'shows another user' do
expect(page).to have_avatar(another_user).with_icon(:mobile)
end
end
context 'when another user is looking on mobile and desktop' do
before do
another_taskbar_mobile
another_taskbar_desktop
taskbar_desktop
visit path
end
it 'shows another user' do
expect(page).to have_avatar(another_user).with_no_icon!
end
end
context 'when another user is editing on desktop' do
before do
another_taskbar_desktop.update!(state: { a: 1 })
taskbar_desktop
visit path
end
it 'shows another user' do
expect(page).to have_avatar(another_user).with_icon(:pen).changed!
end
end
context 'when another user is editing on mobile' do
before do
another_taskbar_mobile.update!(state: { a: 1 })
taskbar_desktop
visit path
end
it 'shows another user' do
expect(page).to have_avatar(another_user).with_icon(:pen).changed!
end
end
context 'when same user is looking on mobile too' do
before do
taskbar_mobile
taskbar_desktop
visit path
end
it 'shows same user' do
expect(page).not_to have_avatar(user)
end
end
context 'when same user is editing' do
before do
taskbar_desktop.update!(state: { a: 1 })
visit path
end
it 'do not show same user' do
expect(page).not_to have_avatar(user)
end
end
context 'when same user is editing on mobile' do
before do
taskbar_mobile.update!(state: { a: 1 })
taskbar_desktop
visit path
end
it 'shows same user' do
expect(page).to have_avatar(user).with_icon(:'mobile-edit').changed!
end
end
end
end