2020-04-08 14:13:33 +05:30
|
|
|
import Vuex from 'vuex';
|
|
|
|
import { shallowMount, createLocalVue } from '@vue/test-utils';
|
2020-10-24 23:57:45 +05:30
|
|
|
import { setHTMLFixture } from 'helpers/fixtures';
|
2020-04-08 14:13:33 +05:30
|
|
|
import * as utils from '~/lib/utils/common_utils';
|
|
|
|
import discussionNavigation from '~/notes/mixins/discussion_navigation';
|
|
|
|
import eventHub from '~/notes/event_hub';
|
2020-05-24 23:13:21 +05:30
|
|
|
import createEventHub from '~/helpers/event_hub_factory';
|
2020-04-08 14:13:33 +05:30
|
|
|
import notesModule from '~/notes/stores/modules';
|
|
|
|
|
|
|
|
const discussion = (id, index) => ({
|
|
|
|
id,
|
|
|
|
resolvable: index % 2 === 0,
|
|
|
|
active: true,
|
|
|
|
notes: [{}],
|
|
|
|
diff_discussion: true,
|
|
|
|
});
|
|
|
|
const createDiscussions = () => [...'abcde'].map(discussion);
|
|
|
|
const createComponent = () => ({
|
|
|
|
mixins: [discussionNavigation],
|
|
|
|
render() {
|
|
|
|
return this.$slots.default;
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('Discussion navigation mixin', () => {
|
|
|
|
const localVue = createLocalVue();
|
|
|
|
localVue.use(Vuex);
|
|
|
|
|
|
|
|
let wrapper;
|
|
|
|
let store;
|
|
|
|
let expandDiscussion;
|
|
|
|
|
|
|
|
beforeEach(() => {
|
|
|
|
setHTMLFixture(
|
|
|
|
[...'abcde']
|
|
|
|
.map(
|
|
|
|
id =>
|
|
|
|
`<ul class="notes" data-discussion-id="${id}"></ul>
|
|
|
|
<div class="discussion" data-discussion-id="${id}"></div>`,
|
|
|
|
)
|
|
|
|
.join(''),
|
|
|
|
);
|
|
|
|
|
2020-06-23 00:09:42 +05:30
|
|
|
jest.spyOn(utils, 'scrollToElementWithContext');
|
2020-04-08 14:13:33 +05:30
|
|
|
|
|
|
|
expandDiscussion = jest.fn();
|
|
|
|
const { actions, ...notesRest } = notesModule();
|
|
|
|
store = new Vuex.Store({
|
|
|
|
modules: {
|
|
|
|
notes: {
|
|
|
|
...notesRest,
|
|
|
|
actions: { ...actions, expandDiscussion },
|
|
|
|
},
|
|
|
|
},
|
|
|
|
});
|
|
|
|
store.state.notes.discussions = createDiscussions();
|
|
|
|
|
|
|
|
wrapper = shallowMount(createComponent(), { store, localVue });
|
|
|
|
});
|
|
|
|
|
|
|
|
afterEach(() => {
|
|
|
|
wrapper.vm.$destroy();
|
|
|
|
jest.clearAllMocks();
|
|
|
|
});
|
|
|
|
|
|
|
|
const findDiscussion = (selector, id) =>
|
|
|
|
document.querySelector(`${selector}[data-discussion-id="${id}"]`);
|
|
|
|
|
2020-10-24 23:57:45 +05:30
|
|
|
describe('jumpToFirstUnresolvedDiscussion method', () => {
|
|
|
|
let vm;
|
|
|
|
|
|
|
|
beforeEach(() => {
|
|
|
|
createComponent();
|
|
|
|
|
|
|
|
({ vm } = wrapper);
|
|
|
|
|
|
|
|
jest.spyOn(store, 'dispatch');
|
|
|
|
jest.spyOn(vm, 'jumpToNextDiscussion');
|
|
|
|
});
|
|
|
|
|
|
|
|
it('triggers the setCurrentDiscussionId action with null as the value', () => {
|
|
|
|
vm.jumpToFirstUnresolvedDiscussion();
|
|
|
|
|
|
|
|
expect(store.dispatch).toHaveBeenCalledWith('setCurrentDiscussionId', null);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('triggers the jumpToNextDiscussion action when the previous store action succeeds', () => {
|
|
|
|
store.dispatch.mockResolvedValue();
|
|
|
|
|
|
|
|
vm.jumpToFirstUnresolvedDiscussion();
|
|
|
|
|
|
|
|
return vm.$nextTick().then(() => {
|
|
|
|
expect(vm.jumpToNextDiscussion).toHaveBeenCalled();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2020-04-08 14:13:33 +05:30
|
|
|
describe('cycle through discussions', () => {
|
|
|
|
beforeEach(() => {
|
2020-05-24 23:13:21 +05:30
|
|
|
window.mrTabs = { eventHub: createEventHub(), tabShown: jest.fn() };
|
2020-04-08 14:13:33 +05:30
|
|
|
});
|
|
|
|
|
|
|
|
describe.each`
|
|
|
|
fn | args | currentId | expected
|
|
|
|
${'jumpToNextDiscussion'} | ${[]} | ${null} | ${'a'}
|
|
|
|
${'jumpToNextDiscussion'} | ${[]} | ${'a'} | ${'c'}
|
|
|
|
${'jumpToNextDiscussion'} | ${[]} | ${'e'} | ${'a'}
|
|
|
|
${'jumpToPreviousDiscussion'} | ${[]} | ${null} | ${'e'}
|
|
|
|
${'jumpToPreviousDiscussion'} | ${[]} | ${'e'} | ${'c'}
|
|
|
|
${'jumpToPreviousDiscussion'} | ${[]} | ${'c'} | ${'a'}
|
|
|
|
${'jumpToNextRelativeDiscussion'} | ${[null]} | ${null} | ${'a'}
|
|
|
|
${'jumpToNextRelativeDiscussion'} | ${['a']} | ${null} | ${'c'}
|
|
|
|
${'jumpToNextRelativeDiscussion'} | ${['e']} | ${'c'} | ${'a'}
|
|
|
|
`('$fn (args = $args, currentId = $currentId)', ({ fn, args, currentId, expected }) => {
|
|
|
|
beforeEach(() => {
|
|
|
|
store.state.notes.currentDiscussionId = currentId;
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('on `show` active tab', () => {
|
|
|
|
beforeEach(() => {
|
|
|
|
window.mrTabs.currentAction = 'show';
|
|
|
|
wrapper.vm[fn](...args);
|
2020-07-28 23:09:34 +05:30
|
|
|
|
|
|
|
return wrapper.vm.$nextTick();
|
2020-04-08 14:13:33 +05:30
|
|
|
});
|
|
|
|
|
|
|
|
it('sets current discussion', () => {
|
|
|
|
expect(store.state.notes.currentDiscussionId).toEqual(expected);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('expands discussion', () => {
|
|
|
|
expect(expandDiscussion).toHaveBeenCalled();
|
|
|
|
});
|
|
|
|
|
|
|
|
it('scrolls to element', () => {
|
2020-06-23 00:09:42 +05:30
|
|
|
expect(utils.scrollToElementWithContext).toHaveBeenCalledWith(
|
2020-04-08 14:13:33 +05:30
|
|
|
findDiscussion('div.discussion', expected),
|
|
|
|
);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('on `diffs` active tab', () => {
|
|
|
|
beforeEach(() => {
|
|
|
|
window.mrTabs.currentAction = 'diffs';
|
|
|
|
wrapper.vm[fn](...args);
|
2020-07-28 23:09:34 +05:30
|
|
|
|
|
|
|
return wrapper.vm.$nextTick();
|
2020-04-08 14:13:33 +05:30
|
|
|
});
|
|
|
|
|
|
|
|
it('sets current discussion', () => {
|
|
|
|
expect(store.state.notes.currentDiscussionId).toEqual(expected);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('expands discussion', () => {
|
|
|
|
expect(expandDiscussion).toHaveBeenCalled();
|
|
|
|
});
|
|
|
|
|
|
|
|
it('scrolls when scrollToDiscussion is emitted', () => {
|
2020-06-23 00:09:42 +05:30
|
|
|
expect(utils.scrollToElementWithContext).not.toHaveBeenCalled();
|
2020-04-08 14:13:33 +05:30
|
|
|
|
|
|
|
eventHub.$emit('scrollToDiscussion');
|
|
|
|
|
2020-06-23 00:09:42 +05:30
|
|
|
expect(utils.scrollToElementWithContext).toHaveBeenCalledWith(
|
|
|
|
findDiscussion('ul.notes', expected),
|
|
|
|
);
|
2020-04-08 14:13:33 +05:30
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('on `other` active tab', () => {
|
|
|
|
beforeEach(() => {
|
|
|
|
window.mrTabs.currentAction = 'other';
|
|
|
|
wrapper.vm[fn](...args);
|
2020-07-28 23:09:34 +05:30
|
|
|
|
|
|
|
return wrapper.vm.$nextTick();
|
2020-04-08 14:13:33 +05:30
|
|
|
});
|
|
|
|
|
|
|
|
it('sets current discussion', () => {
|
|
|
|
expect(store.state.notes.currentDiscussionId).toEqual(expected);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('does not expand discussion yet', () => {
|
|
|
|
expect(expandDiscussion).not.toHaveBeenCalled();
|
|
|
|
});
|
|
|
|
|
|
|
|
it('shows mrTabs', () => {
|
|
|
|
expect(window.mrTabs.tabShown).toHaveBeenCalledWith('show');
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('when tab is changed', () => {
|
|
|
|
beforeEach(() => {
|
|
|
|
window.mrTabs.eventHub.$emit('MergeRequestTabChange');
|
|
|
|
|
|
|
|
jest.runAllTimers();
|
|
|
|
});
|
|
|
|
|
|
|
|
it('expands discussion', () => {
|
|
|
|
expect(expandDiscussion).toHaveBeenCalledWith(
|
|
|
|
expect.anything(),
|
|
|
|
{
|
|
|
|
discussionId: expected,
|
|
|
|
},
|
|
|
|
undefined,
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('scrolls to discussion', () => {
|
2020-06-23 00:09:42 +05:30
|
|
|
expect(utils.scrollToElementWithContext).toHaveBeenCalledWith(
|
2020-04-08 14:13:33 +05:30
|
|
|
findDiscussion('div.discussion', expected),
|
|
|
|
);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|