debian-mirror-gitlab/spec/frontend/milestones/components/milestone_combobox_spec.js

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

527 lines
16 KiB
JavaScript
Raw Normal View History

2021-03-11 19:13:27 +05:30
import { GlLoadingIcon, GlSearchBoxByType, GlDropdownItem } from '@gitlab/ui';
2021-03-08 18:12:59 +05:30
import { mount } from '@vue/test-utils';
2021-01-29 00:20:46 +05:30
import axios from 'axios';
import MockAdapter from 'axios-mock-adapter';
2021-03-11 19:13:27 +05:30
import Vue, { nextTick } from 'vue';
import Vuex from 'vuex';
2023-04-23 21:23:45 +05:30
import { HTTP_STATUS_INTERNAL_SERVER_ERROR, HTTP_STATUS_OK } from '~/lib/utils/http_status';
2021-01-29 00:20:46 +05:30
import { ENTER_KEY } from '~/lib/utils/keys';
import MilestoneCombobox from '~/milestones/components/milestone_combobox.vue';
import createStore from '~/milestones/stores/';
2022-01-26 12:08:38 +05:30
import { projectMilestones, groupMilestones } from '../mock_data';
2021-01-29 00:20:46 +05:30
const extraLinks = [
{ text: 'Create new', url: 'http://127.0.0.1:3000/h5bp/html5-boilerplate/-/milestones/new' },
{ text: 'Manage milestones', url: '/h5bp/html5-boilerplate/-/milestones' },
];
2021-03-08 18:12:59 +05:30
Vue.use(Vuex);
2021-01-29 00:20:46 +05:30
describe('Milestone combobox component', () => {
const projectId = '8';
const groupId = '24';
const groupMilestonesAvailable = true;
const X_TOTAL_HEADER = 'x-total';
let wrapper;
let projectMilestonesApiCallSpy;
let groupMilestonesApiCallSpy;
let searchApiCallSpy;
const createComponent = (props = {}, attrs = {}) => {
2021-03-08 18:12:59 +05:30
const propsData = {
projectId,
groupId,
groupMilestonesAvailable,
extraLinks,
value: [],
...props,
};
2021-01-29 00:20:46 +05:30
wrapper = mount(MilestoneCombobox, {
2021-03-08 18:12:59 +05:30
propsData,
2021-01-29 00:20:46 +05:30
attrs,
listeners: {
// simulate a parent component v-model binding
2021-03-08 18:12:59 +05:30
input: (selectedMilestone) => {
// ugly hack because setProps plays bad with immediate watchers
// see https://github.com/vuejs/vue-test-utils/issues/1140 and
// https://github.com/vuejs/vue-test-utils/pull/1752
propsData.value = selectedMilestone;
2021-01-29 00:20:46 +05:30
wrapper.setProps({ value: selectedMilestone });
},
},
stubs: {
GlSearchBoxByType: true,
},
store: createStore(),
});
};
beforeEach(() => {
const mock = new MockAdapter(axios);
gon.api_version = 'v4';
projectMilestonesApiCallSpy = jest
.fn()
2023-04-23 21:23:45 +05:30
.mockReturnValue([HTTP_STATUS_OK, projectMilestones, { [X_TOTAL_HEADER]: '6' }]);
2021-01-29 00:20:46 +05:30
groupMilestonesApiCallSpy = jest
.fn()
2023-04-23 21:23:45 +05:30
.mockReturnValue([HTTP_STATUS_OK, groupMilestones, { [X_TOTAL_HEADER]: '6' }]);
2021-01-29 00:20:46 +05:30
searchApiCallSpy = jest
.fn()
2023-04-23 21:23:45 +05:30
.mockReturnValue([HTTP_STATUS_OK, projectMilestones, { [X_TOTAL_HEADER]: '6' }]);
2021-01-29 00:20:46 +05:30
mock
.onGet(`/api/v4/projects/${projectId}/milestones`)
2021-03-08 18:12:59 +05:30
.reply((config) => projectMilestonesApiCallSpy(config));
2021-01-29 00:20:46 +05:30
mock
.onGet(`/api/v4/groups/${groupId}/milestones`)
2021-03-08 18:12:59 +05:30
.reply((config) => groupMilestonesApiCallSpy(config));
2021-01-29 00:20:46 +05:30
2021-03-08 18:12:59 +05:30
mock.onGet(`/api/v4/projects/${projectId}/search`).reply((config) => searchApiCallSpy(config));
2021-01-29 00:20:46 +05:30
});
//
// Finders
//
const findButtonContent = () => wrapper.find('[data-testid="milestone-combobox-button-content"]');
const findNoResults = () => wrapper.find('[data-testid="milestone-combobox-no-results"]');
2022-10-11 01:57:18 +05:30
const findLoadingIcon = () => wrapper.findComponent(GlLoadingIcon);
2021-01-29 00:20:46 +05:30
2022-10-11 01:57:18 +05:30
const findSearchBox = () => wrapper.findComponent(GlSearchBoxByType);
2021-01-29 00:20:46 +05:30
const findProjectMilestonesSection = () =>
wrapper.find('[data-testid="project-milestones-section"]');
const findProjectMilestonesDropdownItems = () =>
2022-10-11 01:57:18 +05:30
findProjectMilestonesSection().findAllComponents(GlDropdownItem);
2021-01-29 00:20:46 +05:30
const findFirstProjectMilestonesDropdownItem = () => findProjectMilestonesDropdownItems().at(0);
const findGroupMilestonesSection = () => wrapper.find('[data-testid="group-milestones-section"]');
const findGroupMilestonesDropdownItems = () =>
2022-10-11 01:57:18 +05:30
findGroupMilestonesSection().findAllComponents(GlDropdownItem);
2021-01-29 00:20:46 +05:30
const findFirstGroupMilestonesDropdownItem = () => findGroupMilestonesDropdownItems().at(0);
//
// Expecters
//
const projectMilestoneSectionContainsErrorMessage = () => {
const projectMilestoneSection = findProjectMilestonesSection();
return projectMilestoneSection
.text()
2021-03-08 18:12:59 +05:30
.includes('An error occurred while searching for milestones');
2021-01-29 00:20:46 +05:30
};
const groupMilestoneSectionContainsErrorMessage = () => {
const groupMilestoneSection = findGroupMilestonesSection();
return groupMilestoneSection
.text()
2021-03-08 18:12:59 +05:30
.includes('An error occurred while searching for milestones');
2021-01-29 00:20:46 +05:30
};
//
// Convenience methods
//
2021-03-08 18:12:59 +05:30
const updateQuery = (newQuery) => {
2021-01-29 00:20:46 +05:30
findSearchBox().vm.$emit('input', newQuery);
};
const selectFirstProjectMilestone = () => {
findFirstProjectMilestonesDropdownItem().vm.$emit('click');
};
const selectFirstGroupMilestone = () => {
findFirstGroupMilestonesDropdownItem().vm.$emit('click');
};
2021-03-08 18:12:59 +05:30
const waitForRequests = async ({ andClearMocks } = { andClearMocks: false }) => {
await axios.waitForAll();
if (andClearMocks) {
projectMilestonesApiCallSpy.mockClear();
groupMilestonesApiCallSpy.mockClear();
}
};
2021-01-29 00:20:46 +05:30
describe('initialization behavior', () => {
it('initializes the dropdown with milestones when mounted', () => {
2022-08-13 15:12:31 +05:30
createComponent();
2021-01-29 00:20:46 +05:30
return waitForRequests().then(() => {
expect(projectMilestonesApiCallSpy).toHaveBeenCalledTimes(1);
expect(groupMilestonesApiCallSpy).toHaveBeenCalledTimes(1);
});
});
it('shows a spinner while network requests are in progress', () => {
2022-08-13 15:12:31 +05:30
createComponent();
2021-01-29 00:20:46 +05:30
expect(findLoadingIcon().exists()).toBe(true);
return waitForRequests().then(() => {
expect(findLoadingIcon().exists()).toBe(false);
});
});
it('shows additional links', () => {
2022-08-13 15:12:31 +05:30
createComponent();
2021-01-29 00:20:46 +05:30
const links = wrapper.findAll('[data-testid="milestone-combobox-extra-links"]');
links.wrappers.forEach((item, idx) => {
expect(item.text()).toBe(extraLinks[idx].text);
expect(item.attributes('href')).toBe(extraLinks[idx].url);
});
});
});
describe('post-initialization behavior', () => {
describe('when the parent component provides an `id` binding', () => {
const id = '8';
beforeEach(() => {
createComponent({}, { id });
return waitForRequests();
});
it('adds the provided ID to the GlDropdown instance', () => {
expect(wrapper.attributes().id).toBe(id);
});
});
describe('when milestones are pre-selected', () => {
beforeEach(() => {
createComponent({ value: projectMilestones });
return waitForRequests();
});
it('renders the pre-selected milestones', () => {
expect(findButtonContent().text()).toBe('v0.1 + 5 more');
});
});
describe('when the search query is updated', () => {
beforeEach(() => {
createComponent();
return waitForRequests({ andClearMocks: true });
});
it('requeries the search when the search query is updated', () => {
updateQuery('v1.2.3');
return waitForRequests().then(() => {
expect(searchApiCallSpy).toHaveBeenCalledTimes(1);
});
});
});
describe('when the Enter is pressed', () => {
beforeEach(() => {
createComponent();
return waitForRequests({ andClearMocks: true });
});
it('requeries the search when Enter is pressed', () => {
findSearchBox().vm.$emit('keydown', new KeyboardEvent({ key: ENTER_KEY }));
return waitForRequests().then(() => {
expect(searchApiCallSpy).toHaveBeenCalledTimes(1);
});
});
});
describe('when no results are found', () => {
beforeEach(() => {
projectMilestonesApiCallSpy = jest
.fn()
2023-04-23 21:23:45 +05:30
.mockReturnValue([HTTP_STATUS_OK, [], { [X_TOTAL_HEADER]: '0' }]);
2021-01-29 00:20:46 +05:30
2023-04-23 21:23:45 +05:30
groupMilestonesApiCallSpy = jest
.fn()
.mockReturnValue([HTTP_STATUS_OK, [], { [X_TOTAL_HEADER]: '0' }]);
2021-01-29 00:20:46 +05:30
createComponent();
return waitForRequests();
});
describe('when the search query is empty', () => {
it('renders a "no results" message', () => {
2021-03-08 18:12:59 +05:30
expect(findNoResults().text()).toBe('No matching results');
2021-01-29 00:20:46 +05:30
});
});
});
describe('project milestones', () => {
describe('when the project milestones search returns results', () => {
beforeEach(() => {
createComponent();
return waitForRequests();
});
it('renders the project milestones section in the dropdown', () => {
expect(findProjectMilestonesSection().exists()).toBe(true);
});
it('renders the "Project milestones" heading with a total number indicator', () => {
expect(
findProjectMilestonesSection()
.find('[data-testid="milestone-results-section-header"]')
.text(),
).toBe('Project milestones 6');
});
it("does not render an error message in the project milestone section's body", () => {
expect(projectMilestoneSectionContainsErrorMessage()).toBe(false);
});
it('renders each project milestones as a selectable item', () => {
const dropdownItems = findProjectMilestonesDropdownItems();
projectMilestones.forEach((milestone, i) => {
expect(dropdownItems.at(i).text()).toBe(milestone.title);
});
});
});
describe('when the project milestones search returns no results', () => {
beforeEach(() => {
projectMilestonesApiCallSpy = jest
.fn()
2023-04-23 21:23:45 +05:30
.mockReturnValue([HTTP_STATUS_OK, [], { [X_TOTAL_HEADER]: '0' }]);
2021-01-29 00:20:46 +05:30
createComponent();
return waitForRequests();
});
it('does not render the project milestones section in the dropdown', () => {
expect(findProjectMilestonesSection().exists()).toBe(false);
});
});
describe('when the project milestones search returns an error', () => {
beforeEach(() => {
2023-04-23 21:23:45 +05:30
projectMilestonesApiCallSpy = jest
.fn()
.mockReturnValue([HTTP_STATUS_INTERNAL_SERVER_ERROR]);
searchApiCallSpy = jest.fn().mockReturnValue([HTTP_STATUS_INTERNAL_SERVER_ERROR]);
2021-01-29 00:20:46 +05:30
createComponent({ value: [] });
return waitForRequests();
});
it('renders the project milestones section in the dropdown', () => {
expect(findProjectMilestonesSection().exists()).toBe(true);
});
it("renders an error message in the project milestones section's body", () => {
expect(projectMilestoneSectionContainsErrorMessage()).toBe(true);
});
});
describe('selection', () => {
beforeEach(() => {
createComponent();
return waitForRequests();
});
it('renders a checkmark by the selected item', async () => {
selectFirstProjectMilestone();
2021-03-08 18:12:59 +05:30
await nextTick();
2021-01-29 00:20:46 +05:30
expect(
2022-06-21 17:19:12 +05:30
findFirstProjectMilestonesDropdownItem()
.find('svg')
2023-03-04 22:38:38 +05:30
.classes('gl-dropdown-item-check-icon'),
2021-03-08 18:12:59 +05:30
).toBe(true);
2021-01-29 00:20:46 +05:30
selectFirstProjectMilestone();
2021-03-08 18:12:59 +05:30
await nextTick();
2021-01-29 00:20:46 +05:30
expect(
2022-06-21 17:19:12 +05:30
findFirstProjectMilestonesDropdownItem().find('svg').classes('gl-visibility-hidden'),
).toBe(true);
2021-01-29 00:20:46 +05:30
});
describe('when a project milestones is selected', () => {
beforeEach(() => {
createComponent();
projectMilestonesApiCallSpy = jest
.fn()
2023-04-23 21:23:45 +05:30
.mockReturnValue([HTTP_STATUS_OK, [{ title: 'v1.0' }], { [X_TOTAL_HEADER]: '1' }]);
2021-01-29 00:20:46 +05:30
return waitForRequests();
});
it("displays the project milestones name in the dropdown's button", async () => {
selectFirstProjectMilestone();
2021-03-08 18:12:59 +05:30
await nextTick();
2021-01-29 00:20:46 +05:30
2021-03-08 18:12:59 +05:30
expect(findButtonContent().text()).toBe('v1.0');
2021-01-29 00:20:46 +05:30
selectFirstProjectMilestone();
2021-03-08 18:12:59 +05:30
await nextTick();
2021-01-29 00:20:46 +05:30
2021-03-08 18:12:59 +05:30
expect(findButtonContent().text()).toBe('No milestone');
2021-01-29 00:20:46 +05:30
});
2021-03-08 18:12:59 +05:30
it('updates the v-model binding with the project milestone title', async () => {
2021-01-29 00:20:46 +05:30
selectFirstProjectMilestone();
2021-03-08 18:12:59 +05:30
await nextTick();
2021-01-29 00:20:46 +05:30
2021-03-08 18:12:59 +05:30
expect(wrapper.emitted().input[0][0]).toStrictEqual(['v1.0']);
2021-01-29 00:20:46 +05:30
});
});
});
});
describe('group milestones', () => {
describe('when the group milestones search returns results', () => {
beforeEach(() => {
createComponent();
return waitForRequests();
});
it('renders the group milestones section in the dropdown', () => {
expect(findGroupMilestonesSection().exists()).toBe(true);
});
it('renders the "Group milestones" heading with a total number indicator', () => {
expect(
findGroupMilestonesSection()
.find('[data-testid="milestone-results-section-header"]')
.text(),
).toBe('Group milestones 6');
});
it("does not render an error message in the group milestone section's body", () => {
expect(groupMilestoneSectionContainsErrorMessage()).toBe(false);
});
it('renders each group milestones as a selectable item', () => {
const dropdownItems = findGroupMilestonesDropdownItems();
groupMilestones.forEach((milestone, i) => {
expect(dropdownItems.at(i).text()).toBe(milestone.title);
});
});
});
describe('when the group milestones search returns no results', () => {
beforeEach(() => {
groupMilestonesApiCallSpy = jest
.fn()
2023-04-23 21:23:45 +05:30
.mockReturnValue([HTTP_STATUS_OK, [], { [X_TOTAL_HEADER]: '0' }]);
2021-01-29 00:20:46 +05:30
createComponent();
return waitForRequests();
});
it('does not render the group milestones section in the dropdown', () => {
expect(findGroupMilestonesSection().exists()).toBe(false);
});
});
describe('when the group milestones search returns an error', () => {
beforeEach(() => {
2023-04-23 21:23:45 +05:30
groupMilestonesApiCallSpy = jest
.fn()
.mockReturnValue([HTTP_STATUS_INTERNAL_SERVER_ERROR]);
searchApiCallSpy = jest.fn().mockReturnValue([HTTP_STATUS_INTERNAL_SERVER_ERROR]);
2021-01-29 00:20:46 +05:30
createComponent({ value: [] });
return waitForRequests();
});
it('renders the group milestones section in the dropdown', () => {
expect(findGroupMilestonesSection().exists()).toBe(true);
});
it("renders an error message in the group milestones section's body", () => {
expect(groupMilestoneSectionContainsErrorMessage()).toBe(true);
});
});
describe('selection', () => {
beforeEach(() => {
createComponent();
return waitForRequests();
});
it('renders a checkmark by the selected item', async () => {
selectFirstGroupMilestone();
2021-03-08 18:12:59 +05:30
await nextTick();
2021-01-29 00:20:46 +05:30
2022-06-21 17:19:12 +05:30
expect(
findFirstGroupMilestonesDropdownItem()
.find('svg')
2023-03-04 22:38:38 +05:30
.classes('gl-dropdown-item-check-icon'),
2022-06-21 17:19:12 +05:30
).toBe(true);
2021-01-29 00:20:46 +05:30
selectFirstGroupMilestone();
2021-03-08 18:12:59 +05:30
await nextTick();
2021-01-29 00:20:46 +05:30
2022-06-21 17:19:12 +05:30
expect(
findFirstGroupMilestonesDropdownItem().find('svg').classes('gl-visibility-hidden'),
).toBe(true);
2021-01-29 00:20:46 +05:30
});
describe('when a group milestones is selected', () => {
beforeEach(() => {
createComponent();
groupMilestonesApiCallSpy = jest
.fn()
2023-04-23 21:23:45 +05:30
.mockReturnValue([
HTTP_STATUS_OK,
[{ title: 'group-v1.0' }],
{ [X_TOTAL_HEADER]: '1' },
]);
2021-01-29 00:20:46 +05:30
return waitForRequests();
});
it("displays the group milestones name in the dropdown's button", async () => {
selectFirstGroupMilestone();
2021-03-08 18:12:59 +05:30
await nextTick();
2021-01-29 00:20:46 +05:30
2021-03-08 18:12:59 +05:30
expect(findButtonContent().text()).toBe('group-v1.0');
2021-01-29 00:20:46 +05:30
selectFirstGroupMilestone();
2021-03-08 18:12:59 +05:30
await nextTick();
2021-01-29 00:20:46 +05:30
2021-03-08 18:12:59 +05:30
expect(findButtonContent().text()).toBe('No milestone');
2021-01-29 00:20:46 +05:30
});
2021-03-08 18:12:59 +05:30
it('updates the v-model binding with the group milestone title', async () => {
2021-01-29 00:20:46 +05:30
selectFirstGroupMilestone();
2021-03-08 18:12:59 +05:30
await nextTick();
2021-01-29 00:20:46 +05:30
2021-03-08 18:12:59 +05:30
expect(wrapper.emitted().input[0][0]).toStrictEqual(['group-v1.0']);
2021-01-29 00:20:46 +05:30
});
});
});
});
});
});