# frozen_string_literal: true require 'spec_helper' RSpec.describe 'User edit profile', feature_category: :users do include Spec::Support::Helpers::Features::NotesHelpers let_it_be(:user) { create(:user) } before do sign_in(user) visit(profile_path) end def submit_settings click_button 'Update profile settings' wait_for_requests if respond_to?(:wait_for_requests) end def update_user_email fill_in 'user_email', with: 'new-email@example.com' click_button 'Update profile settings' end def confirm_password(password) fill_in 'password-confirmation', with: password click_button 'Confirm password' wait_for_requests if respond_to?(:wait_for_requests) end def visit_user visit user_path(user) wait_for_requests end def toggle_busy_status find('[data-testid="user-availability-checkbox"]').set(true) end it 'changes user profile' do fill_in 'user_skype', with: 'testskype' fill_in 'user_linkedin', with: 'testlinkedin' fill_in 'user_twitter', with: 'testtwitter' fill_in 'user_website_url', with: 'http://testurl.com' fill_in 'user_location', with: 'Ukraine' fill_in 'user_bio', with: 'I <3 GitLab :tada:' fill_in 'user_job_title', with: 'Frontend Engineer' fill_in 'user_organization', with: 'GitLab' submit_settings expect(user.reload).to have_attributes( skype: 'testskype', linkedin: 'testlinkedin', twitter: 'testtwitter', website_url: 'http://testurl.com', bio: 'I <3 GitLab :tada:', job_title: 'Frontend Engineer', organization: 'GitLab' ) expect(find('#user_location').value).to eq 'Ukraine' expect(page).to have_content('Profile was successfully updated') end it 'does not set secondary emails without user input' do fill_in 'user_organization', with: 'GitLab' submit_settings user.reload expect(page).to have_field('user_commit_email', with: '') expect(page).to have_field('user_public_email', with: '') User::SECONDARY_EMAIL_ATTRIBUTES.each do |attribute| expect(user.read_attribute(attribute)).to be_blank end end it 'shows an error if the full name contains an emoji', :js do simulate_input('#user_name', 'Martin 😀') submit_settings page.within('.rspec-full-name') do expect(page).to have_css '.gl-field-error-outline' expect(find('.gl-field-error')).not_to have_selector('.hidden') expect(find('.gl-field-error')).to have_content('Using emojis in names seems fun, but please try to set a status message instead') end end it 'shows an error if the website url is not valid' do fill_in 'user_website_url', with: 'admin@gitlab.com' submit_settings expect(user.reload).to have_attributes( website_url: '' ) expect(page).to have_content('Website url is not a valid URL') end describe 'when I change my email', :js do before do user.send_reset_password_instructions end it 'will prompt to confirm my password' do expect(user.reset_password_token?).to be true update_user_email expect(page).to have_selector('[data-testid="password-prompt-modal"]') end context 'when prompted to confirm password' do before do update_user_email end it 'with the correct password successfully updates' do confirm_password(user.password) expect(page).to have_text("Profile was successfully updated") end it 'with the incorrect password fails to update' do confirm_password("Fake password") expect(page).to have_text("Invalid password") end end it 'clears the reset password token' do expect(user.reset_password_token?).to be true update_user_email confirm_password(user.password) user.reload expect(user.confirmation_token).not_to be_nil expect(user.reset_password_token?).to be false end end context 'user avatar' do before do attach_file(:user_avatar, Rails.root.join('spec', 'fixtures', 'banana_sample.gif')) submit_settings end it 'changes user avatar' do expect(page).to have_link('Remove avatar') user.reload expect(user.avatar).to be_instance_of AvatarUploader expect(user.avatar.url).to eq "/uploads/-/system/user/avatar/#{user.id}/banana_sample.gif" end it 'removes user avatar' do click_link 'Remove avatar' user.reload expect(user.avatar?).to eq false expect(page).not_to have_link('Remove avatar') expect(page).to have_link('gravatar.com') end end context 'user status', :js do def select_emoji(emoji_name, is_modal = false) toggle_button = find('.emoji-menu-toggle-button') toggle_button.click emoji_button = find("gl-emoji[data-name=\"#{emoji_name}\"]") emoji_button.click end context 'profile edit form' do it 'shows the user status form' do expect(page).to have_content('Current status') end it 'adds emoji to user status' do emoji = 'basketball' select_emoji(emoji) submit_settings visit_user within('.cover-status') do expect(page).to have_emoji(emoji) end end it 'adds message to user status' do message = 'I have something to say' fill_in s_("SetStatusModal|What's your status?"), with: message submit_settings visit_user within('.cover-status') do expect(page).to have_emoji('speech_balloon') expect(page).to have_content message end end it 'adds message and emoji to user status' do emoji = '8ball' message = 'Playing outside' select_emoji(emoji) fill_in s_("SetStatusModal|What's your status?"), with: message submit_settings visit_user within('.cover-status') do expect(page).to have_emoji(emoji) expect(page).to have_content message end end it 'clears the user status' do user_status = create(:user_status, user: user, message: 'Eating bread', emoji: 'stuffed_flatbread') visit_user within('.cover-status') do expect(page).to have_emoji(user_status.emoji) expect(page).to have_content user_status.message end visit(profile_path) click_button s_('SetStatusModal|Clear status') submit_settings visit_user expect(page).not_to have_selector '.cover-status' end it 'displays a default emoji if only message is entered' do message = 'a status without emoji' fill_in s_("SetStatusModal|What's your status?"), with: message within('.emoji-menu-toggle-button') do expect(page).to have_emoji('speech_balloon') end end it 'sets the users status to busy' do busy_status = find('[data-testid="user-availability-checkbox"]') expect(busy_status.checked?).to eq(false) toggle_busy_status submit_settings visit profile_path expect(busy_status.checked?).to eq(true) end context 'with user status set to busy' do let(:project) { create(:project, :public) } let(:issue) { create(:issue, project: project, author: user) } before do toggle_busy_status submit_settings project.add_developer(user) visit project_issue_path(project, issue) end it 'shows author as busy in the assignee dropdown' do page.within('.assignee') do click_button('Edit') wait_for_requests end page.within '.dropdown-menu-user' do expect(page).to have_content("#{user.name} (Busy)") end end it 'displays the assignee busy status' do click_button 'assign yourself' wait_for_requests visit project_issue_path(project, issue) wait_for_requests expect(page.find('.issuable-assignees')).to have_content("#{user.name} (Busy)") end end end context 'user menu' do let(:issue) { create(:issue, project: project) } let(:project) { create(:project) } def open_modal(button_text) find('.header-user-dropdown-toggle').click page.within ".header-user" do find('.js-set-status-modal-trigger.ready') click_button button_text end end def open_user_status_modal open_modal 'Set status' end def open_edit_status_modal open_modal 'Edit status' end def set_user_status_in_modal page.within "#set-user-status-modal" do click_button 'Set status' end wait_for_requests end before do visit root_path(user) end it 'shows the "Set status" menu item in the user menu' do find('.header-user-dropdown-toggle').click page.within ".header-user" do expect(page).to have_content('Set status') end end it 'shows the "Edit status" menu item in the user menu' do user_status = create(:user_status, user: user, message: 'Eating bread', emoji: 'stuffed_flatbread') visit root_path(user) find('.header-user-dropdown-toggle').click page.within ".header-user" do expect(page).to have_emoji(user_status.emoji) expect(page).to have_content user_status.message expect(page).to have_content('Edit status') end end it 'shows user status modal' do open_user_status_modal expect(page.find('#set-user-status-modal')).to be_visible expect(page).to have_content('Set a status') end it 'adds emoji to user status' do emoji = '8ball' open_user_status_modal select_emoji(emoji, true) set_user_status_in_modal visit_user within('.cover-status') do expect(page).to have_emoji(emoji) end end it 'sets the users status to busy' do open_user_status_modal busy_status = find('[data-testid="user-availability-checkbox"]') expect(busy_status.checked?).to eq(false) toggle_busy_status set_user_status_in_modal wait_for_requests visit root_path(user) open_edit_status_modal expect(busy_status.checked?).to eq(true) end it 'opens the emoji modal again after closing it' do open_user_status_modal select_emoji('8ball', true) find('.emoji-menu-toggle-button').click expect(page).to have_selector('.emoji-picker-emoji') end it 'does not update the awards panel emoji' do project.add_maintainer(user) visit(project_issue_path(project, issue)) emoji = '8ball' open_user_status_modal select_emoji(emoji, true) expect(page.all('.award-control .js-counter')).to all(have_content('0')) end it 'adds message to user status' do message = 'I have something to say' open_user_status_modal find_field(s_("SetStatusModal|What's your status?")).native.send_keys(message) set_user_status_in_modal visit_user within('.cover-status') do expect(page).to have_emoji('speech_balloon') expect(page).to have_content message end end it 'adds message and emoji to user status' do emoji = '8ball' message = 'Playing outside' open_user_status_modal select_emoji(emoji, true) find_field(s_("SetStatusModal|What's your status?")).native.send_keys(message) set_user_status_in_modal visit_user within('.cover-status') do expect(page).to have_emoji(emoji) expect(page).to have_content message end end it 'clears the user status with the "X" button' do user_status = create(:user_status, user: user, message: 'Eating bread', emoji: 'stuffed_flatbread') visit_user wait_for_requests within('.cover-status') do expect(page).to have_emoji(user_status.emoji) expect(page).to have_content user_status.message end open_edit_status_modal click_button s_('SetStatusModal|Clear status') set_user_status_in_modal visit_user wait_for_requests expect(page).not_to have_selector '.cover-status' end context 'Remove status button' do let(:user) { create(:user) } before do user.status = UserStatus.new(message: 'Eating bread', emoji: 'stuffed_flatbread') visit_user wait_for_requests open_edit_status_modal page.within "#set-user-status-modal" do click_button 'Remove status' end wait_for_requests end it 'clears the user status with the "Remove status" button' do visit_user expect(page).not_to have_selector '.cover-status' end it 'shows the "Set status" menu item in the user menu' do visit root_path(user) find('.header-user-dropdown-toggle').click page.within ".header-user" do expect(page).to have_content('Set status') end end end it 'displays a default emoji if only message is entered' do message = 'a status without emoji' open_user_status_modal find_field(s_("SetStatusModal|What's your status?")).native.send_keys(message) expect(page).to have_emoji('speech_balloon') end end context 'User time preferences', :js do let(:issue) { create(:issue, project: project) } let(:project) { create(:project) } before do stub_feature_flags(user_time_settings: true) end it 'shows the user time preferences form' do expect(page).to have_content('Time settings') end it 'allows the user to select a time zone from a dropdown list of options' do expect(page.find('.user-time-preferences .dropdown')).not_to have_css('.show') page.find('.user-time-preferences .dropdown').click expect(page.find('.user-time-preferences .dropdown')).to have_css('.show') page.find("button", text: "Arizona").click expect(page).to have_field(:user_timezone, with: 'America/Phoenix', type: :hidden) end it 'timezone defaults to empty' do expect(page).to have_field(:user_timezone, with: '', type: :hidden) end end end context 'work information', :js do context 'when job title and organziation are entered' do it "shows job title and organzation on user's profile" do fill_in 'user_job_title', with: 'Frontend Engineer' fill_in 'user_organization', with: 'GitLab - work info test' submit_settings visit_user expect(page).to have_content('Frontend Engineer at GitLab - work info test') end end context 'when only job title is entered' do it "shows only job title on user's profile" do fill_in 'user_job_title', with: 'Frontend Engineer - work info test' submit_settings visit_user expect(page).to have_content('Frontend Engineer - work info test') end end context 'when only organization is entered' do it "shows only organization on user's profile" do fill_in 'user_organization', with: 'GitLab - work info test' submit_settings visit_user expect(page).to have_content('GitLab - work info test') end end end end