debian-mirror-gitlab/spec/frontend/design_management/components/design_overlay_spec.js

425 lines
13 KiB
JavaScript
Raw Normal View History

2020-05-24 23:13:21 +05:30
import { mount } from '@vue/test-utils';
2022-04-04 11:22:00 +05:30
import { nextTick } from 'vue';
2020-05-24 23:13:21 +05:30
import DesignOverlay from '~/design_management/components/design_overlay.vue';
2021-03-11 19:13:27 +05:30
import { ACTIVE_DISCUSSION_SOURCE_TYPES } from '~/design_management/constants';
2020-05-24 23:13:21 +05:30
import updateActiveDiscussion from '~/design_management/graphql/mutations/update_active_discussion.mutation.graphql';
import notes from '../mock_data/notes';
const mutate = jest.fn(() => Promise.resolve());
describe('Design overlay component', () => {
let wrapper;
const mockDimensions = { width: 100, height: 100 };
const findAllNotes = () => wrapper.findAll('.js-image-badge');
const findCommentBadge = () => wrapper.find('.comment-indicator');
2021-03-08 18:12:59 +05:30
const findBadgeAtIndex = (noteIndex) => findAllNotes().at(noteIndex);
2020-11-24 15:15:51 +05:30
const findFirstBadge = () => findBadgeAtIndex(0);
const findSecondBadge = () => findBadgeAtIndex(1);
2020-05-24 23:13:21 +05:30
2022-04-04 11:22:00 +05:30
const clickAndDragBadge = async (elem, fromPoint, toPoint) => {
2020-05-24 23:13:21 +05:30
elem.trigger('mousedown', { clientX: fromPoint.x, clientY: fromPoint.y });
2022-04-04 11:22:00 +05:30
await nextTick();
elem.trigger('mousemove', { clientX: toPoint.x, clientY: toPoint.y });
await nextTick();
2020-05-24 23:13:21 +05:30
};
function createComponent(props = {}, data = {}) {
wrapper = mount(DesignOverlay, {
propsData: {
dimensions: mockDimensions,
position: {
top: '0',
left: '0',
},
2020-06-23 00:09:42 +05:30
resolvedDiscussionsExpanded: false,
2020-05-24 23:13:21 +05:30
...props,
},
data() {
return {
activeDiscussion: {
id: null,
source: null,
},
...data,
};
},
mocks: {
$apollo: {
mutate,
},
},
});
}
it('should have correct inline style', () => {
createComponent();
2020-11-24 15:15:51 +05:30
expect(wrapper.attributes().style).toBe('width: 100px; height: 100px; top: 0px; left: 0px;');
2020-05-24 23:13:21 +05:30
});
2022-04-04 11:22:00 +05:30
it('should emit `openCommentForm` when clicking on overlay', async () => {
2020-05-24 23:13:21 +05:30
createComponent();
const newCoordinates = {
x: 10,
y: 10,
};
wrapper
2020-11-24 15:15:51 +05:30
.find('[data-qa-selector="design_image_button"]')
2020-05-24 23:13:21 +05:30
.trigger('mouseup', { offsetX: newCoordinates.x, offsetY: newCoordinates.y });
2022-04-04 11:22:00 +05:30
await nextTick();
expect(wrapper.emitted('openCommentForm')).toEqual([
[{ x: newCoordinates.x, y: newCoordinates.y }],
]);
2020-05-24 23:13:21 +05:30
});
describe('with notes', () => {
2020-06-23 00:09:42 +05:30
it('should render only the first note', () => {
2020-05-24 23:13:21 +05:30
createComponent({
notes,
});
2020-06-23 00:09:42 +05:30
expect(findAllNotes()).toHaveLength(1);
2020-05-24 23:13:21 +05:30
});
2020-06-23 00:09:42 +05:30
describe('with resolved discussions toggle expanded', () => {
beforeEach(() => {
createComponent({
notes,
resolvedDiscussionsExpanded: true,
});
});
it('should render all notes', () => {
expect(findAllNotes()).toHaveLength(notes.length);
});
it('should have set the correct position for each note badge', () => {
expect(findFirstBadge().attributes().style).toBe('left: 10px; top: 15px;');
expect(findSecondBadge().attributes().style).toBe('left: 50px; top: 50px;');
});
it('should apply resolved class to the resolved note pin', () => {
expect(findSecondBadge().classes()).toContain('resolved');
});
2020-11-24 15:15:51 +05:30
describe('when no discussion is active', () => {
it('should not apply inactive class to any pins', () => {
expect(
2021-03-08 18:12:59 +05:30
findAllNotes(0).wrappers.every((designNote) => designNote.classes('gl-bg-blue-50')),
2020-11-24 15:15:51 +05:30
).toBe(false);
2020-06-23 00:09:42 +05:30
});
2020-11-24 15:15:51 +05:30
});
describe('when a discussion is active', () => {
it.each([notes[0].discussion.notes.nodes[1], notes[0].discussion.notes.nodes[0]])(
'should not apply inactive class to the pin for the active discussion',
2022-04-04 11:22:00 +05:30
async (note) => {
2022-03-02 08:16:31 +05:30
// setData usage is discouraged. See https://gitlab.com/groups/gitlab-org/-/epics/7330 for details
// eslint-disable-next-line no-restricted-syntax
2020-11-24 15:15:51 +05:30
wrapper.setData({
activeDiscussion: {
id: note.id,
source: 'discussion',
},
});
2020-05-24 23:13:21 +05:30
2022-04-04 11:22:00 +05:30
await nextTick();
expect(findBadgeAtIndex(0).classes()).not.toContain('inactive');
2020-11-24 15:15:51 +05:30
},
);
2022-04-04 11:22:00 +05:30
it('should apply inactive class to all pins besides the active one', async () => {
2022-03-02 08:16:31 +05:30
// setData usage is discouraged. See https://gitlab.com/groups/gitlab-org/-/epics/7330 for details
// eslint-disable-next-line no-restricted-syntax
2020-11-24 15:15:51 +05:30
wrapper.setData({
activeDiscussion: {
id: notes[0].id,
source: 'discussion',
},
});
2022-04-04 11:22:00 +05:30
await nextTick();
expect(findSecondBadge().classes()).toContain('inactive');
expect(findFirstBadge().classes()).not.toContain('inactive');
2020-06-23 00:09:42 +05:30
});
});
2020-05-24 23:13:21 +05:30
});
2022-04-04 11:22:00 +05:30
it('should recalculate badges positions on window resize', async () => {
2020-05-24 23:13:21 +05:30
createComponent({
notes,
dimensions: {
width: 400,
height: 400,
},
});
expect(findFirstBadge().attributes().style).toBe('left: 40px; top: 60px;');
wrapper.setProps({
dimensions: {
width: 200,
height: 200,
},
});
2022-04-04 11:22:00 +05:30
await nextTick();
expect(findFirstBadge().attributes().style).toBe('left: 20px; top: 30px;');
2020-05-24 23:13:21 +05:30
});
2022-04-04 11:22:00 +05:30
it('should call an update active discussion mutation when clicking a note without moving it', async () => {
2020-05-24 23:13:21 +05:30
const note = notes[0];
const { position } = note;
const mutationVariables = {
mutation: updateActiveDiscussion,
variables: {
id: note.id,
source: ACTIVE_DISCUSSION_SOURCE_TYPES.pin,
},
};
findFirstBadge().trigger('mousedown', { clientX: position.x, clientY: position.y });
2022-04-04 11:22:00 +05:30
await nextTick();
findFirstBadge().trigger('mouseup', { clientX: position.x, clientY: position.y });
expect(mutate).toHaveBeenCalledWith(mutationVariables);
2020-05-24 23:13:21 +05:30
});
});
describe('when moving notes', () => {
2022-04-04 11:22:00 +05:30
it('should update badge style when note is being moved', async () => {
2020-05-24 23:13:21 +05:30
createComponent({
notes,
});
const { position } = notes[0];
2022-04-04 11:22:00 +05:30
await clickAndDragBadge(findFirstBadge(), { x: position.x, y: position.y }, { x: 20, y: 20 });
expect(findFirstBadge().attributes().style).toBe('left: 20px; top: 20px;');
2020-05-24 23:13:21 +05:30
});
2022-04-04 11:22:00 +05:30
it('should emit `moveNote` event when note-moving action ends', async () => {
2020-05-24 23:13:21 +05:30
createComponent({ notes });
const note = notes[0];
const { position } = note;
const newCoordinates = { x: 20, y: 20 };
2022-03-02 08:16:31 +05:30
// setData usage is discouraged. See https://gitlab.com/groups/gitlab-org/-/epics/7330 for details
// eslint-disable-next-line no-restricted-syntax
2020-05-24 23:13:21 +05:30
wrapper.setData({
movingNoteNewPosition: {
...position,
...newCoordinates,
},
movingNoteStartPosition: {
noteId: notes[0].id,
discussionId: notes[0].discussion.id,
...position,
},
});
const badge = findFirstBadge();
2022-04-04 11:22:00 +05:30
await clickAndDragBadge(badge, { x: position.x, y: position.y }, newCoordinates);
badge.trigger('mouseup');
await nextTick();
expect(wrapper.emitted('moveNote')).toEqual([
[
{
noteId: notes[0].id,
discussionId: notes[0].discussion.id,
coordinates: newCoordinates,
},
],
]);
2020-05-24 23:13:21 +05:30
});
2021-01-29 00:20:46 +05:30
describe('without [repositionNote] permission', () => {
2020-06-23 00:09:42 +05:30
const mockNoteNotAuthorised = {
...notes[0],
userPermissions: {
2021-01-29 00:20:46 +05:30
repositionNote: false,
2020-06-23 00:09:42 +05:30
},
};
2020-05-24 23:13:21 +05:30
2020-06-23 00:09:42 +05:30
const mockNoteCoordinates = {
x: mockNoteNotAuthorised.position.x,
y: mockNoteNotAuthorised.position.y,
};
2022-04-04 11:22:00 +05:30
it('should be unable to move a note', async () => {
2020-06-23 00:09:42 +05:30
createComponent({
dimensions: mockDimensions,
notes: [mockNoteNotAuthorised],
});
const badge = findAllNotes().at(0);
2022-04-04 11:22:00 +05:30
await clickAndDragBadge(badge, { ...mockNoteCoordinates }, { x: 20, y: 20 });
// note position should not change after a click-and-drag attempt
expect(findFirstBadge().attributes().style).toContain(
`left: ${mockNoteCoordinates.x}px; top: ${mockNoteCoordinates.y}px;`,
);
2020-05-24 23:13:21 +05:30
});
});
});
describe('with a new form', () => {
it('should render a new comment badge', () => {
createComponent({
currentCommentForm: {
...notes[0].position,
},
});
expect(findCommentBadge().exists()).toBe(true);
expect(findCommentBadge().attributes().style).toBe('left: 10px; top: 15px;');
});
describe('when moving the comment badge', () => {
2022-04-04 11:22:00 +05:30
it('should update badge style to reflect new position', async () => {
2020-05-24 23:13:21 +05:30
const { position } = notes[0];
createComponent({
currentCommentForm: {
...position,
},
});
2022-04-04 11:22:00 +05:30
await clickAndDragBadge(
2020-05-24 23:13:21 +05:30
findCommentBadge(),
{ x: position.x, y: position.y },
{ x: 20, y: 20 },
2022-04-04 11:22:00 +05:30
);
expect(findCommentBadge().attributes().style).toBe('left: 20px; top: 20px;');
2020-05-24 23:13:21 +05:30
});
2022-04-04 11:22:00 +05:30
it('should update badge style when note-moving action ends', async () => {
2020-05-24 23:13:21 +05:30
const { position } = notes[0];
createComponent({
currentCommentForm: {
...position,
},
});
const commentBadge = findCommentBadge();
const toPoint = { x: 20, y: 20 };
2022-04-04 11:22:00 +05:30
await clickAndDragBadge(commentBadge, { x: position.x, y: position.y }, toPoint);
commentBadge.trigger('mouseup');
// simulates the currentCommentForm being updated in index.vue component, and
// propagated back down to this prop
wrapper.setProps({
currentCommentForm: { height: position.height, width: position.width, ...toPoint },
});
await nextTick();
expect(commentBadge.attributes().style).toBe('left: 20px; top: 20px;');
2020-05-24 23:13:21 +05:30
});
it.each`
element | getElementFunc | event
2020-11-24 15:15:51 +05:30
${'overlay'} | ${() => wrapper} | ${'mouseleave'}
2020-05-24 23:13:21 +05:30
${'comment badge'} | ${findCommentBadge} | ${'mouseup'}
`(
'should emit `openCommentForm` event when $event fired on $element element',
2022-04-04 11:22:00 +05:30
async ({ getElementFunc, event }) => {
2020-05-24 23:13:21 +05:30
createComponent({
notes,
currentCommentForm: {
...notes[0].position,
},
});
const newCoordinates = { x: 20, y: 20 };
2022-03-02 08:16:31 +05:30
// setData usage is discouraged. See https://gitlab.com/groups/gitlab-org/-/epics/7330 for details
// eslint-disable-next-line no-restricted-syntax
2020-05-24 23:13:21 +05:30
wrapper.setData({
movingNoteStartPosition: {
...notes[0].position,
},
movingNoteNewPosition: {
...notes[0].position,
...newCoordinates,
},
});
getElementFunc().trigger(event);
2022-04-04 11:22:00 +05:30
await nextTick();
expect(wrapper.emitted('openCommentForm')).toEqual([[newCoordinates]]);
2020-05-24 23:13:21 +05:30
},
);
});
});
describe('getMovingNotePositionDelta', () => {
it('should calculate delta correctly from state', () => {
createComponent();
2022-03-02 08:16:31 +05:30
// setData usage is discouraged. See https://gitlab.com/groups/gitlab-org/-/epics/7330 for details
// eslint-disable-next-line no-restricted-syntax
2020-05-24 23:13:21 +05:30
wrapper.setData({
movingNoteStartPosition: {
clientX: 10,
clientY: 20,
},
});
const mockMouseEvent = {
clientX: 30,
clientY: 10,
};
expect(wrapper.vm.getMovingNotePositionDelta(mockMouseEvent)).toEqual({
deltaX: 20,
deltaY: -10,
});
});
});
describe('isPositionInOverlay', () => {
createComponent({ dimensions: mockDimensions });
it.each`
test | coordinates | expectedResult
${'within overlay bounds'} | ${{ x: 50, y: 50 }} | ${true}
${'outside overlay bounds'} | ${{ x: 101, y: 101 }} | ${false}
`('returns [$expectedResult] when position is $test', ({ coordinates, expectedResult }) => {
const position = { ...mockDimensions, ...coordinates };
expect(wrapper.vm.isPositionInOverlay(position)).toBe(expectedResult);
});
});
describe('getNoteRelativePosition', () => {
it('calculates position correctly', () => {
createComponent({ dimensions: mockDimensions });
const position = { x: 50, y: 50, width: 200, height: 200 };
expect(wrapper.vm.getNoteRelativePosition(position)).toEqual({ left: 25, top: 25 });
});
});
describe('canMoveNote', () => {
it.each`
2021-01-29 00:20:46 +05:30
repositionNotePermission | canMoveNoteResult
${true} | ${true}
${false} | ${false}
${undefined} | ${false}
2020-05-24 23:13:21 +05:30
`(
2021-01-29 00:20:46 +05:30
'returns [$canMoveNoteResult] when [repositionNote permission] is [$repositionNotePermission]',
({ repositionNotePermission, canMoveNoteResult }) => {
2020-05-24 23:13:21 +05:30
createComponent();
const note = {
userPermissions: {
2021-01-29 00:20:46 +05:30
repositionNote: repositionNotePermission,
2020-05-24 23:13:21 +05:30
},
};
expect(wrapper.vm.canMoveNote(note)).toBe(canMoveNoteResult);
},
);
});
});