import { GlAccordionItem } from '@gitlab/ui'; import { shallowMount } from '@vue/test-utils'; import { nextTick } from 'vue'; import DesignDiscussion from '~/design_management/components/design_notes/design_discussion.vue'; import DesignNoteSignedOut from '~/design_management/components/design_notes/design_note_signed_out.vue'; import DesignSidebar from '~/design_management/components/design_sidebar.vue'; import DesignTodoButton from '~/design_management/components/design_todo_button.vue'; import updateActiveDiscussionMutation from '~/design_management/graphql/mutations/update_active_discussion.mutation.graphql'; import Participants from '~/sidebar/components/participants/participants.vue'; import design from '../mock_data/design'; const scrollIntoViewMock = jest.fn(); HTMLElement.prototype.scrollIntoView = scrollIntoViewMock; const updateActiveDiscussionMutationVariables = { mutation: updateActiveDiscussionMutation, variables: { id: design.discussions.nodes[0].notes.nodes[0].id, source: 'discussion', }, }; const $route = { params: { id: '1', }, }; const mutate = jest.fn().mockResolvedValue(); describe('Design management design sidebar component', () => { const originalGon = window.gon; let wrapper; const findDiscussions = () => wrapper.findAll(DesignDiscussion); const findFirstDiscussion = () => findDiscussions().at(0); const findUnresolvedDiscussions = () => wrapper.findAll('[data-testid="unresolved-discussion"]'); const findResolvedDiscussions = () => wrapper.findAll('[data-testid="resolved-discussion"]'); const findParticipants = () => wrapper.find(Participants); const findResolvedCommentsToggle = () => wrapper.find(GlAccordionItem); const findNewDiscussionDisclaimer = () => wrapper.find('[data-testid="new-discussion-disclaimer"]'); function createComponent(props = {}) { wrapper = shallowMount(DesignSidebar, { propsData: { design, resolvedDiscussionsExpanded: false, markdownPreviewPath: '', isLoading: false, ...props, }, mocks: { $route, $apollo: { mutate, }, }, provide: { registerPath: '/users/sign_up?redirect_to_referer=yes', signInPath: '/users/sign_in?redirect_to_referer=yes', }, }); } beforeEach(() => { window.gon = { current_user_id: 1 }; }); afterEach(() => { wrapper.destroy(); window.gon = originalGon; }); it('renders participants', () => { createComponent(); expect(findParticipants().exists()).toBe(true); }); it('passes the correct amount of participants to the Participants component', () => { createComponent(); expect(findParticipants().props('participants')).toHaveLength(1); }); it('renders To-Do button', () => { createComponent(); expect(wrapper.find(DesignTodoButton).exists()).toBe(true); }); describe('when has no discussions', () => { beforeEach(() => { createComponent({ design: { ...design, discussions: { nodes: [], }, }, }); }); it('does not render discussions', () => { expect(findDiscussions().exists()).toBe(false); }); it('renders a message about possibility to create a new discussion', () => { expect(findNewDiscussionDisclaimer().exists()).toBe(true); }); }); describe('when has discussions', () => { beforeEach(() => { createComponent(); }); it('renders correct amount of unresolved discussions', () => { expect(findUnresolvedDiscussions()).toHaveLength(1); }); it('renders correct amount of resolved discussions', () => { expect(findResolvedDiscussions()).toHaveLength(1); }); it('has resolved comments accordion item collapsed', () => { expect(findResolvedCommentsToggle().props('visible')).toBe(false); }); it('emits toggleResolveComments event on resolve comments button click', async () => { findResolvedCommentsToggle().vm.$emit('input', true); await nextTick(); expect(wrapper.emitted('toggleResolvedComments')).toHaveLength(1); }); it('opens the accordion item when resolvedDiscussionsExpanded prop changes to true', async () => { expect(findResolvedCommentsToggle().props('visible')).toBe(false); wrapper.setProps({ resolvedDiscussionsExpanded: true, }); await nextTick(); expect(findResolvedCommentsToggle().props('visible')).toBe(true); }); it('sends a mutation to set an active discussion when clicking on a discussion', () => { findFirstDiscussion().trigger('click'); expect(mutate).toHaveBeenCalledWith(updateActiveDiscussionMutationVariables); }); it('sends a mutation to reset an active discussion when clicking outside of discussion', () => { wrapper.trigger('click'); expect(mutate).toHaveBeenCalledWith({ ...updateActiveDiscussionMutationVariables, variables: { id: undefined, source: 'discussion' }, }); }); it('emits correct event on discussion create note error', () => { findFirstDiscussion().vm.$emit('create-note-error', 'payload'); expect(wrapper.emitted('onDesignDiscussionError')).toEqual([['payload']]); }); it('emits correct event on discussion update note error', () => { findFirstDiscussion().vm.$emit('update-note-error', 'payload'); expect(wrapper.emitted('updateNoteError')).toEqual([['payload']]); }); it('emits correct event on discussion resolve error', () => { findFirstDiscussion().vm.$emit('resolve-discussion-error', 'payload'); expect(wrapper.emitted('resolveDiscussionError')).toEqual([['payload']]); }); it('changes prop correctly on opening discussion form', async () => { findFirstDiscussion().vm.$emit('open-form', 'some-id'); await nextTick(); expect(findFirstDiscussion().props('discussionWithOpenForm')).toBe('some-id'); }); }); describe('when all discussions are resolved', () => { beforeEach(() => { createComponent({ design: { ...design, discussions: { nodes: [ { id: 'discussion-id', replyId: 'discussion-reply-id', resolved: true, notes: { nodes: [ { id: 'note-id', body: '123', author: { name: 'Administrator', username: 'root', webUrl: 'link-to-author', avatarUrl: 'link-to-avatar', }, }, ], }, }, ], }, }, }); }); it('renders a message about possibility to create a new discussion', () => { expect(findNewDiscussionDisclaimer().exists()).toBe(true); }); it('does not render unresolved discussions', () => { expect(findUnresolvedDiscussions()).toHaveLength(0); }); }); describe('when user is not logged in', () => { const findDesignNoteSignedOut = () => wrapper.findComponent(DesignNoteSignedOut); beforeEach(() => { window.gon = { current_user_id: null }; }); describe('design has no discussions', () => { beforeEach(() => { createComponent({ design: { ...design, discussions: { nodes: [], }, }, }); }); it('does not render a message about possibility to create a new discussion', () => { expect(findNewDiscussionDisclaimer().exists()).toBe(false); }); it('renders design-note-signed-out component', () => { expect(findDesignNoteSignedOut().exists()).toBe(true); }); }); describe('design has discussions', () => { beforeEach(() => { createComponent(); }); it('renders design-note-signed-out component', () => { expect(findDesignNoteSignedOut().exists()).toBe(true); }); }); }); });