Commit c88b5e3b authored by Vitaly Slobodin's avatar Vitaly Slobodin

Merge branch...

Merge branch '232569-update-the-milestone-dropdown-combobox-to-display-separated-sections-and-badge-counters' into 'master'

Display sections and badge counters in milestone dropdown combobox

See merge request gitlab-org/gitlab!43427
parents 07999dca 6d2bc034
...@@ -8,14 +8,17 @@ import { ...@@ -8,14 +8,17 @@ import {
GlSearchBoxByType, GlSearchBoxByType,
GlIcon, GlIcon,
} from '@gitlab/ui'; } from '@gitlab/ui';
import { intersection, debounce } from 'lodash'; import { debounce, isEqual } from 'lodash';
import { __, sprintf } from '~/locale'; import { mapActions, mapGetters, mapState } from 'vuex';
import Api from '~/api'; import { s__, __, sprintf } from '~/locale';
import { deprecatedCreateFlash as createFlash } from '~/flash'; import createStore from '../stores';
import MilestoneResultsSection from './milestone_results_section.vue';
const SEARCH_DEBOUNCE_MS = 250; const SEARCH_DEBOUNCE_MS = 250;
export default { export default {
name: 'MilestoneCombobox',
store: createStore(),
components: { components: {
GlDropdown, GlDropdown,
GlDropdownDivider, GlDropdownDivider,
...@@ -24,21 +27,18 @@ export default { ...@@ -24,21 +27,18 @@ export default {
GlLoadingIcon, GlLoadingIcon,
GlSearchBoxByType, GlSearchBoxByType,
GlIcon, GlIcon,
}, MilestoneResultsSection,
model: {
prop: 'preselectedMilestones',
event: 'change',
}, },
props: { props: {
value: {
type: Array,
required: false,
default: () => [],
},
projectId: { projectId: {
type: String, type: String,
required: true, required: true,
}, },
preselectedMilestones: {
type: Array,
default: () => [],
required: false,
},
extraLinks: { extraLinks: {
type: Array, type: Array,
default: () => [], default: () => [],
...@@ -48,47 +48,60 @@ export default { ...@@ -48,47 +48,60 @@ export default {
data() { data() {
return { return {
searchQuery: '', searchQuery: '',
projectMilestones: [],
searchResults: [],
selectedMilestones: [],
requestCount: 0,
}; };
}, },
translations: { translations: {
milestone: __('Milestone'), milestone: s__('MilestoneCombobox|Milestone'),
selectMilestone: __('Select milestone'), selectMilestone: s__('MilestoneCombobox|Select milestone'),
noMilestone: __('No milestone'), noMilestone: s__('MilestoneCombobox|No milestone'),
noResultsLabel: __('No matching results'), noResultsLabel: s__('MilestoneCombobox|No matching results'),
searchMilestones: __('Search Milestones'), searchMilestones: s__('MilestoneCombobox|Search Milestones'),
searhErrorMessage: s__('MilestoneCombobox|An error occurred while searching for milestones'),
projectMilestones: s__('MilestoneCombobox|Project milestones'),
}, },
computed: { computed: {
...mapState(['matches', 'selectedMilestones']),
...mapGetters(['isLoading']),
selectedMilestonesLabel() { selectedMilestonesLabel() {
if (this.milestoneTitles.length === 1) { const { selectedMilestones } = this;
return this.milestoneTitles[0]; const firstMilestoneName = selectedMilestones[0];
if (selectedMilestones.length === 0) {
return this.$options.translations.noMilestone;
} }
if (this.milestoneTitles.length > 1) { if (selectedMilestones.length === 1) {
const firstMilestoneName = this.milestoneTitles[0]; return firstMilestoneName;
const numberOfOtherMilestones = this.milestoneTitles.length - 1;
return sprintf(__('%{firstMilestoneName} + %{numberOfOtherMilestones} more'), {
firstMilestoneName,
numberOfOtherMilestones,
});
} }
return this.$options.translations.noMilestone; const numberOfOtherMilestones = selectedMilestones.length - 1;
}, return sprintf(__('%{firstMilestoneName} + %{numberOfOtherMilestones} more'), {
milestoneTitles() { firstMilestoneName,
return this.preselectedMilestones.map(milestone => milestone.title); numberOfOtherMilestones,
});
}, },
dropdownItems() { showProjectMilestoneSection() {
return this.searchResults.length ? this.searchResults : this.projectMilestones; return Boolean(
this.matches.projectMilestones.totalCount > 0 || this.matches.projectMilestones.error,
);
}, },
noResults() { showNoResults() {
return this.searchQuery.length > 2 && this.searchResults.length === 0; return !this.showProjectMilestoneSection;
}, },
isLoading() { },
return this.requestCount !== 0; watch: {
// Keep the Vuex store synchronized if the parent
// component updates the selected milestones through v-model
value: {
immediate: true,
handler() {
const milestoneTitles = this.value.map(milestone =>
milestone.title ? milestone.title : milestone,
);
if (!isEqual(milestoneTitles, this.selectedMilestones)) {
this.setSelectedMilestones(milestoneTitles);
}
},
}, },
}, },
created() { created() {
...@@ -97,100 +110,48 @@ export default { ...@@ -97,100 +110,48 @@ export default {
// lodash attaches to the function, which is // lodash attaches to the function, which is
// made inaccessible by Vue. More info: // made inaccessible by Vue. More info:
// https://stackoverflow.com/a/52988020/1063392 // https://stackoverflow.com/a/52988020/1063392
this.debouncedSearchMilestones = debounce(this.searchMilestones, SEARCH_DEBOUNCE_MS); this.debouncedSearch = debounce(function search() {
}, this.search(this.searchQuery);
mounted() { }, SEARCH_DEBOUNCE_MS);
this.setProjectId(this.projectId);
this.fetchMilestones(); this.fetchMilestones();
}, },
methods: { methods: {
...mapActions([
'setProjectId',
'setSelectedMilestones',
'clearSelectedMilestones',
'toggleMilestones',
'search',
'fetchMilestones',
]),
focusSearchBox() { focusSearchBox() {
this.$refs.searchBox.$el.querySelector('input').focus(); this.$refs.searchBox.$el.querySelector('input').focus();
}, },
fetchMilestones() {
this.requestCount += 1;
Api.projectMilestones(this.projectId)
.then(({ data }) => {
this.projectMilestones = this.getTitles(data);
this.selectedMilestones = intersection(this.projectMilestones, this.milestoneTitles);
})
.catch(() => {
createFlash(__('An error occurred while loading milestones'));
})
.finally(() => {
this.requestCount -= 1;
});
},
searchMilestones() {
this.requestCount += 1;
const options = {
search: this.searchQuery,
scope: 'milestones',
};
if (this.searchQuery.length < 3) {
this.requestCount -= 1;
this.searchResults = [];
return;
}
Api.projectSearch(this.projectId, options)
.then(({ data }) => {
const searchResults = this.getTitles(data);
this.searchResults = searchResults.length ? searchResults : [];
})
.catch(() => {
createFlash(__('An error occurred while searching for milestones'));
})
.finally(() => {
this.requestCount -= 1;
});
},
onSearchBoxInput() {
this.debouncedSearchMilestones();
},
onSearchBoxEnter() { onSearchBoxEnter() {
this.debouncedSearchMilestones.cancel(); this.debouncedSearch.cancel();
this.searchMilestones(); this.search(this.searchQuery);
}, },
toggleMilestoneSelection(clickedMilestone) { onSearchBoxInput() {
if (!clickedMilestone) return []; this.debouncedSearch();
let milestones = [...this.preselectedMilestones];
const hasMilestone = this.milestoneTitles.includes(clickedMilestone);
if (hasMilestone) {
milestones = milestones.filter(({ title }) => title !== clickedMilestone);
} else {
milestones.push({ title: clickedMilestone });
}
return milestones;
},
onMilestoneClicked(clickedMilestone) {
const milestones = this.toggleMilestoneSelection(clickedMilestone);
this.$emit('change', milestones);
this.selectedMilestones = intersection(
this.projectMilestones,
milestones.map(milestone => milestone.title),
);
}, },
isSelectedMilestone(milestoneTitle) { selectMilestone(milestone) {
return this.selectedMilestones.includes(milestoneTitle); this.toggleMilestones(milestone);
this.$emit('input', this.selectedMilestones);
}, },
getTitles(milestones) { selectNoMilestone() {
return milestones.filter(({ state }) => state === 'active').map(({ title }) => title); this.clearSelectedMilestones();
this.$emit('input', this.selectedMilestones);
}, },
}, },
}; };
</script> </script>
<template> <template>
<gl-dropdown v-bind="$attrs" class="project-milestone-combobox" @shown="focusSearchBox"> <gl-dropdown v-bind="$attrs" class="milestone-combobox" @shown="focusSearchBox">
<template slot="button-content"> <template slot="button-content">
<span ref="buttonText" class="flex-grow-1 ml-1 text-muted">{{ <span data-testid="milestone-combobox-button-content" class="gl-flex-grow-1 text-muted">{{
selectedMilestonesLabel selectedMilestonesLabel
}}</span> }}</span>
<gl-icon name="chevron-down" /> <gl-icon name="chevron-down" />
...@@ -205,13 +166,14 @@ export default { ...@@ -205,13 +166,14 @@ export default {
<gl-search-box-by-type <gl-search-box-by-type
ref="searchBox" ref="searchBox"
v-model.trim="searchQuery" v-model.trim="searchQuery"
class="gl-m-3"
:placeholder="this.$options.translations.searchMilestones" :placeholder="this.$options.translations.searchMilestones"
@input="onSearchBoxInput" @input="onSearchBoxInput"
@keydown.enter.prevent="onSearchBoxEnter" @keydown.enter.prevent="onSearchBoxEnter"
/> />
<gl-dropdown-item @click="onMilestoneClicked(null)"> <gl-dropdown-item @click="selectNoMilestone()">
<span :class="{ 'pl-4': true, 'selected-item': selectedMilestones.length === 0 }"> <span :class="{ 'gl-pl-6': true, 'selected-item': selectedMilestones.length === 0 }">
{{ $options.translations.noMilestone }} {{ $options.translations.noMilestone }}
</span> </span>
</gl-dropdown-item> </gl-dropdown-item>
...@@ -222,28 +184,33 @@ export default { ...@@ -222,28 +184,33 @@ export default {
<gl-loading-icon /> <gl-loading-icon />
<gl-dropdown-divider /> <gl-dropdown-divider />
</template> </template>
<template v-else-if="noResults"> <template v-else-if="showNoResults">
<div class="dropdown-item-space"> <div class="dropdown-item-space">
<span ref="noResults" class="pl-4">{{ $options.translations.noResultsLabel }}</span> <span data-testid="milestone-combobox-no-results" class="gl-pl-6">{{
$options.translations.noResultsLabel
}}</span>
</div> </div>
<gl-dropdown-divider /> <gl-dropdown-divider />
</template> </template>
<template v-else-if="dropdownItems.length"> <template v-else>
<gl-dropdown-item <milestone-results-section
v-for="item in dropdownItems" :section-title="$options.translations.projectMilestones"
:key="item" :total-count="matches.projectMilestones.totalCount"
role="milestone option" :items="matches.projectMilestones.list"
@click="onMilestoneClicked(item)" :selected-milestones="selectedMilestones"
> :error="matches.projectMilestones.error"
<span :class="{ 'pl-4': true, 'selected-item': isSelectedMilestone(item) }"> :error-message="$options.translations.searhErrorMessage"
{{ item }} data-testid="project-milestones-section"
</span> @selected="selectMilestone($event)"
</gl-dropdown-item> />
<gl-dropdown-divider />
</template> </template>
<gl-dropdown-item
<gl-dropdown-item v-for="(item, idx) in extraLinks" :key="idx" :href="item.url"> v-for="(item, idx) in extraLinks"
<span class="pl-4">{{ item.text }}</span> :key="idx"
:href="item.url"
data-testid="milestone-combobox-extra-links"
>
<span class="gl-pl-6">{{ item.text }}</span>
</gl-dropdown-item> </gl-dropdown-item>
</gl-dropdown> </gl-dropdown>
</template> </template>
<script>
import {
GlDropdownSectionHeader,
GlDropdownDivider,
GlDropdownItem,
GlBadge,
GlIcon,
} from '@gitlab/ui';
import { s__ } from '~/locale';
export default {
name: 'MilestoneResultsSection',
components: {
GlDropdownSectionHeader,
GlDropdownDivider,
GlDropdownItem,
GlBadge,
GlIcon,
},
props: {
sectionTitle: {
type: String,
required: true,
},
totalCount: {
type: Number,
required: true,
},
items: {
type: Array,
required: true,
},
selectedMilestones: {
type: Array,
required: true,
default: () => [],
},
error: {
type: Error,
required: false,
default: null,
},
errorMessage: {
type: String,
required: false,
default: '',
},
},
computed: {
totalCountText() {
return this.totalCount > 999 ? s__('TotalMilestonesIndicator|1000+') : `${this.totalCount}`;
},
},
methods: {
isSelectedMilestone(item) {
return this.selectedMilestones.includes(item);
},
},
};
</script>
<template>
<div>
<gl-dropdown-section-header>
<div
class="gl-display-flex gl-align-items-center gl-pl-6"
data-testid="milestone-results-section-header"
>
<span class="gl-mr-2 gl-mb-1">{{ sectionTitle }}</span>
<gl-badge variant="neutral">{{ totalCountText }}</gl-badge>
</div>
</gl-dropdown-section-header>
<template v-if="error">
<div class="gl-display-flex align-items-start gl-text-red-500 gl-ml-4 gl-mr-4 gl-mb-3">
<gl-icon name="error" class="gl-mr-2 gl-mt-2 gl-flex-shrink-0" />
<span>{{ errorMessage }}</span>
</div>
</template>
<template v-else>
<gl-dropdown-item
v-for="{ title } in items"
:key="title"
role="milestone option"
@click="$emit('selected', title)"
>
<span class="gl-pl-6" :class="{ 'selected-item': isSelectedMilestone(title) }">
{{ title }}
</span>
</gl-dropdown-item>
<gl-dropdown-divider />
</template>
</div>
</template>
...@@ -6,6 +6,8 @@ export const setProjectId = ({ commit }, projectId) => commit(types.SET_PROJECT_ ...@@ -6,6 +6,8 @@ export const setProjectId = ({ commit }, projectId) => commit(types.SET_PROJECT_
export const setSelectedMilestones = ({ commit }, selectedMilestones) => export const setSelectedMilestones = ({ commit }, selectedMilestones) =>
commit(types.SET_SELECTED_MILESTONES, selectedMilestones); commit(types.SET_SELECTED_MILESTONES, selectedMilestones);
export const clearSelectedMilestones = ({ commit }) => commit(types.CLEAR_SELECTED_MILESTONES);
export const toggleMilestones = ({ commit, state }, selectedMilestone) => { export const toggleMilestones = ({ commit, state }, selectedMilestone) => {
const removeMilestone = state.selectedMilestones.includes(selectedMilestone); const removeMilestone = state.selectedMilestones.includes(selectedMilestone);
...@@ -16,8 +18,8 @@ export const toggleMilestones = ({ commit, state }, selectedMilestone) => { ...@@ -16,8 +18,8 @@ export const toggleMilestones = ({ commit, state }, selectedMilestone) => {
} }
}; };
export const search = ({ dispatch, commit }, query) => { export const search = ({ dispatch, commit }, searchQuery) => {
commit(types.SET_QUERY, query); commit(types.SET_SEARCH_QUERY, searchQuery);
dispatch('searchMilestones'); dispatch('searchMilestones');
}; };
...@@ -41,7 +43,7 @@ export const searchMilestones = ({ commit, state }) => { ...@@ -41,7 +43,7 @@ export const searchMilestones = ({ commit, state }) => {
commit(types.REQUEST_START); commit(types.REQUEST_START);
const options = { const options = {
search: state.query, search: state.searchQuery,
scope: 'milestones', scope: 'milestones',
}; };
......
export const SET_PROJECT_ID = 'SET_PROJECT_ID'; export const SET_PROJECT_ID = 'SET_PROJECT_ID';
export const SET_SELECTED_MILESTONES = 'SET_SELECTED_MILESTONES'; export const SET_SELECTED_MILESTONES = 'SET_SELECTED_MILESTONES';
export const CLEAR_SELECTED_MILESTONES = 'CLEAR_SELECTED_MILESTONES';
export const ADD_SELECTED_MILESTONE = 'ADD_SELECTED_MILESTONE'; export const ADD_SELECTED_MILESTONE = 'ADD_SELECTED_MILESTONE';
export const REMOVE_SELECTED_MILESTONE = 'REMOVE_SELECTED_MILESTONE'; export const REMOVE_SELECTED_MILESTONE = 'REMOVE_SELECTED_MILESTONE';
export const SET_QUERY = 'SET_QUERY'; export const SET_SEARCH_QUERY = 'SET_SEARCH_QUERY';
export const REQUEST_START = 'REQUEST_START'; export const REQUEST_START = 'REQUEST_START';
export const REQUEST_FINISH = 'REQUEST_FINISH'; export const REQUEST_FINISH = 'REQUEST_FINISH';
......
...@@ -9,6 +9,9 @@ export default { ...@@ -9,6 +9,9 @@ export default {
[types.SET_SELECTED_MILESTONES](state, selectedMilestones) { [types.SET_SELECTED_MILESTONES](state, selectedMilestones) {
Vue.set(state, 'selectedMilestones', selectedMilestones); Vue.set(state, 'selectedMilestones', selectedMilestones);
}, },
[types.CLEAR_SELECTED_MILESTONES](state) {
Vue.set(state, 'selectedMilestones', []);
},
[types.ADD_SELECTED_MILESTONE](state, selectedMilestone) { [types.ADD_SELECTED_MILESTONE](state, selectedMilestone) {
state.selectedMilestones.push(selectedMilestone); state.selectedMilestones.push(selectedMilestone);
}, },
...@@ -18,8 +21,8 @@ export default { ...@@ -18,8 +21,8 @@ export default {
); );
Vue.set(state, 'selectedMilestones', filteredMilestones); Vue.set(state, 'selectedMilestones', filteredMilestones);
}, },
[types.SET_QUERY](state, query) { [types.SET_SEARCH_QUERY](state, searchQuery) {
state.query = query; state.searchQuery = searchQuery;
}, },
[types.REQUEST_START](state) { [types.REQUEST_START](state) {
state.requestCount += 1; state.requestCount += 1;
......
export default () => ({ export default () => ({
projectId: null, projectId: null,
groupId: null, groupId: null,
query: '', searchQuery: '',
matches: { matches: {
projectMilestones: { projectMilestones: {
list: [], list: [],
......
...@@ -6,7 +6,7 @@ import MarkdownField from '~/vue_shared/components/markdown/field.vue'; ...@@ -6,7 +6,7 @@ import MarkdownField from '~/vue_shared/components/markdown/field.vue';
import { BACK_URL_PARAM } from '~/releases/constants'; import { BACK_URL_PARAM } from '~/releases/constants';
import { getParameterByName } from '~/lib/utils/common_utils'; import { getParameterByName } from '~/lib/utils/common_utils';
import AssetLinksForm from './asset_links_form.vue'; import AssetLinksForm from './asset_links_form.vue';
import MilestoneCombobox from '~/milestones/project_milestone_combobox.vue'; import MilestoneCombobox from '~/milestones/components/milestone_combobox.vue';
import TagField from './tag_field.vue'; import TagField from './tag_field.vue';
export default { export default {
......
...@@ -15,15 +15,13 @@ import { ...@@ -15,15 +15,13 @@ import {
export const releaseToApiJson = (release, createFrom = null) => { export const releaseToApiJson = (release, createFrom = null) => {
const name = release.name?.trim().length > 0 ? release.name.trim() : null; const name = release.name?.trim().length > 0 ? release.name.trim() : null;
const milestones = release.milestones ? release.milestones.map(milestone => milestone.title) : [];
return convertObjectPropsToSnakeCase( return convertObjectPropsToSnakeCase(
{ {
name, name,
tagName: release.tagName, tagName: release.tagName,
ref: createFrom, ref: createFrom,
description: release.description, description: release.description,
milestones, milestones: release.milestones,
assets: release.assets, assets: release.assets,
}, },
{ deep: true }, { deep: true },
......
---
title: Update the milestone dropdown combobox to display separated sections
and badge counters
merge_request: 43427
author:
type: added
...@@ -2994,9 +2994,6 @@ msgstr "" ...@@ -2994,9 +2994,6 @@ msgstr ""
msgid "An error occurred while loading merge requests." msgid "An error occurred while loading merge requests."
msgstr "" msgstr ""
msgid "An error occurred while loading milestones"
msgstr ""
msgid "An error occurred while loading project creation UI" msgid "An error occurred while loading project creation UI"
msgstr "" msgstr ""
...@@ -3078,9 +3075,6 @@ msgstr "" ...@@ -3078,9 +3075,6 @@ msgstr ""
msgid "An error occurred while saving assignees" msgid "An error occurred while saving assignees"
msgstr "" msgstr ""
msgid "An error occurred while searching for milestones"
msgstr ""
msgid "An error occurred while subscribing to notifications." msgid "An error occurred while subscribing to notifications."
msgstr "" msgstr ""
...@@ -16992,6 +16986,27 @@ msgstr "" ...@@ -16992,6 +16986,27 @@ msgstr ""
msgid "Milestone lists show all issues from the selected milestone." msgid "Milestone lists show all issues from the selected milestone."
msgstr "" msgstr ""
msgid "MilestoneCombobox|An error occurred while searching for milestones"
msgstr ""
msgid "MilestoneCombobox|Milestone"
msgstr ""
msgid "MilestoneCombobox|No matching results"
msgstr ""
msgid "MilestoneCombobox|No milestone"
msgstr ""
msgid "MilestoneCombobox|Project milestones"
msgstr ""
msgid "MilestoneCombobox|Search Milestones"
msgstr ""
msgid "MilestoneCombobox|Select milestone"
msgstr ""
msgid "MilestoneSidebar|Closed:" msgid "MilestoneSidebar|Closed:"
msgstr "" msgstr ""
...@@ -23061,9 +23076,6 @@ msgstr "" ...@@ -23061,9 +23076,6 @@ msgstr ""
msgid "Search Jira issues" msgid "Search Jira issues"
msgstr "" msgstr ""
msgid "Search Milestones"
msgstr ""
msgid "Search an environment spec" msgid "Search an environment spec"
msgstr "" msgstr ""
...@@ -27873,6 +27885,9 @@ msgstr "" ...@@ -27873,6 +27885,9 @@ msgstr ""
msgid "Total: %{total}" msgid "Total: %{total}"
msgstr "" msgstr ""
msgid "TotalMilestonesIndicator|1000+"
msgstr ""
msgid "TotalRefCountIndicator|1000+" msgid "TotalRefCountIndicator|1000+"
msgstr "" msgstr ""
......
import Vuex from 'vuex';
import { mount, createLocalVue } from '@vue/test-utils';
import axios from 'axios';
import MockAdapter from 'axios-mock-adapter';
import { GlLoadingIcon, GlSearchBoxByType, GlDropdownItem } from '@gitlab/ui';
import { s__ } from '~/locale';
import { ENTER_KEY } from '~/lib/utils/keys';
import MilestoneCombobox from '~/milestones/components/milestone_combobox.vue';
import { milestones as projectMilestones } from './mock_data';
import createStore from '~/milestones/stores/';
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' },
];
const localVue = createLocalVue();
localVue.use(Vuex);
describe('Milestone combobox component', () => {
const projectId = '8';
const X_TOTAL_HEADER = 'x-total';
let wrapper;
let projectMilestonesApiCallSpy;
let searchApiCallSpy;
const createComponent = (props = {}, attrs = {}) => {
wrapper = mount(MilestoneCombobox, {
propsData: {
projectId,
extraLinks,
value: [],
...props,
},
attrs,
listeners: {
// simulate a parent component v-model binding
input: selectedMilestone => {
wrapper.setProps({ value: selectedMilestone });
},
},
stubs: {
GlSearchBoxByType: true,
},
localVue,
store: createStore(),
});
};
beforeEach(() => {
const mock = new MockAdapter(axios);
gon.api_version = 'v4';
projectMilestonesApiCallSpy = jest
.fn()
.mockReturnValue([200, projectMilestones, { [X_TOTAL_HEADER]: '6' }]);
searchApiCallSpy = jest
.fn()
.mockReturnValue([200, projectMilestones, { [X_TOTAL_HEADER]: '6' }]);
mock
.onGet(`/api/v4/projects/${projectId}/milestones`)
.reply(config => projectMilestonesApiCallSpy(config));
mock.onGet(`/api/v4/projects/${projectId}/search`).reply(config => searchApiCallSpy(config));
});
afterEach(() => {
wrapper.destroy();
wrapper = null;
});
//
// Finders
//
const findButtonContent = () => wrapper.find('[data-testid="milestone-combobox-button-content"]');
const findNoResults = () => wrapper.find('[data-testid="milestone-combobox-no-results"]');
const findLoadingIcon = () => wrapper.find(GlLoadingIcon);
const findSearchBox = () => wrapper.find(GlSearchBoxByType);
const findProjectMilestonesSection = () =>
wrapper.find('[data-testid="project-milestones-section"]');
const findProjectMilestonesDropdownItems = () =>
findProjectMilestonesSection().findAll(GlDropdownItem);
const findFirstProjectMilestonesDropdownItem = () => findProjectMilestonesDropdownItems().at(0);
//
// Expecters
//
const projectMilestoneSectionContainsErrorMessage = () => {
const projectMilestoneSection = findProjectMilestonesSection();
return projectMilestoneSection
.text()
.includes(s__('MilestoneCombobox|An error occurred while searching for milestones'));
};
//
// Convenience methods
//
const updateQuery = newQuery => {
findSearchBox().vm.$emit('input', newQuery);
};
const selectFirstProjectMilestone = () => {
findFirstProjectMilestonesDropdownItem().vm.$emit('click');
};
const waitForRequests = ({ andClearMocks } = { andClearMocks: false }) =>
axios.waitForAll().then(() => {
if (andClearMocks) {
projectMilestonesApiCallSpy.mockClear();
}
});
describe('initialization behavior', () => {
beforeEach(createComponent);
it('initializes the dropdown with project milestones when mounted', () => {
return waitForRequests().then(() => {
expect(projectMilestonesApiCallSpy).toHaveBeenCalledTimes(1);
});
});
it('shows a spinner while network requests are in progress', () => {
expect(findLoadingIcon().exists()).toBe(true);
return waitForRequests().then(() => {
expect(findLoadingIcon().exists()).toBe(false);
});
});
it('shows additional links', () => {
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 project 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()
.mockReturnValue([200, [], { [X_TOTAL_HEADER]: '0' }]);
createComponent();
return waitForRequests();
});
describe('when the search query is empty', () => {
it('renders a "no results" message', () => {
expect(findNoResults().text()).toBe(s__('MilestoneCombobox|No matching results'));
});
});
});
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()
.mockReturnValue([200, [], { [X_TOTAL_HEADER]: '0' }]);
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(() => {
projectMilestonesApiCallSpy = jest.fn().mockReturnValue([500]);
searchApiCallSpy = jest.fn().mockReturnValue([500]);
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();
await localVue.nextTick();
expect(
findFirstProjectMilestonesDropdownItem()
.find('span')
.classes('selected-item'),
).toBe(false);
selectFirstProjectMilestone();
return localVue.nextTick().then(() => {
expect(
findFirstProjectMilestonesDropdownItem()
.find('span')
.classes('selected-item'),
).toBe(true);
});
});
describe('when a project milestones is selected', () => {
beforeEach(() => {
createComponent();
projectMilestonesApiCallSpy = jest
.fn()
.mockReturnValue([200, [{ title: 'v1.0' }], { [X_TOTAL_HEADER]: '1' }]);
return waitForRequests();
});
it("displays the project milestones name in the dropdown's button", async () => {
selectFirstProjectMilestone();
await localVue.nextTick();
expect(findButtonContent().text()).toBe(s__('MilestoneCombobox|No milestone'));
selectFirstProjectMilestone();
await localVue.nextTick();
expect(findButtonContent().text()).toBe('v1.0');
});
it('updates the v-model binding with the project milestone title', () => {
expect(wrapper.vm.value).toEqual([]);
selectFirstProjectMilestone();
expect(wrapper.vm.value).toEqual(['v1.0']);
});
});
});
});
});
import axios from 'axios';
import MockAdapter from 'axios-mock-adapter';
import { shallowMount } from '@vue/test-utils';
import { GlDropdown, GlLoadingIcon, GlSearchBoxByType } from '@gitlab/ui';
import { ENTER_KEY } from '~/lib/utils/keys';
import MilestoneCombobox from '~/milestones/project_milestone_combobox.vue';
import { milestones as projectMilestones } from './mock_data';
const TEST_SEARCH_ENDPOINT = '/api/v4/projects/8/search';
const TEST_SEARCH = 'TEST_SEARCH';
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' },
];
const preselectedMilestones = [];
const projectId = '8';
describe('Milestone selector', () => {
let wrapper;
let mock;
const findNoResultsMessage = () => wrapper.find({ ref: 'noResults' });
const findSearchBox = () => wrapper.find(GlSearchBoxByType);
const factory = (options = {}) => {
wrapper = shallowMount(MilestoneCombobox, {
...options,
});
};
beforeEach(() => {
mock = new MockAdapter(axios);
gon.api_version = 'v4';
mock.onGet('/api/v4/projects/8/milestones').reply(200, projectMilestones);
factory({
propsData: {
projectId,
preselectedMilestones,
extraLinks,
},
});
});
afterEach(() => {
mock.restore();
wrapper.destroy();
wrapper = null;
});
it('renders the dropdown', () => {
expect(wrapper.find(GlDropdown)).toExist();
});
it('renders additional links', () => {
const links = wrapper.findAll('[href]');
links.wrappers.forEach((item, idx) => {
expect(item.text()).toBe(extraLinks[idx].text);
expect(item.attributes('href')).toBe(extraLinks[idx].url);
});
});
describe('before results', () => {
it('should show a loading icon', () => {
const request = mock.onGet(TEST_SEARCH_ENDPOINT, {
params: { search: TEST_SEARCH, scope: 'milestones' },
});
expect(wrapper.find(GlLoadingIcon).exists()).toBe(true);
return wrapper.vm.$nextTick().then(() => {
request.reply(200, []);
});
});
it('should not show any dropdown items', () => {
expect(wrapper.findAll('[role="milestone option"]')).toHaveLength(0);
});
it('should have "No milestone" as the button text', () => {
expect(wrapper.find({ ref: 'buttonText' }).text()).toBe('No milestone');
});
});
describe('with empty results', () => {
beforeEach(() => {
mock
.onGet(TEST_SEARCH_ENDPOINT, { params: { search: TEST_SEARCH, scope: 'milestones' } })
.reply(200, []);
findSearchBox().vm.$emit('input', TEST_SEARCH);
return axios.waitForAll();
});
it('should display that no matching items are found', () => {
expect(findNoResultsMessage().exists()).toBe(true);
});
});
describe('with results', () => {
let items;
beforeEach(() => {
mock
.onGet(TEST_SEARCH_ENDPOINT, { params: { search: 'v0.1', scope: 'milestones' } })
.reply(200, [
{
id: 41,
iid: 6,
project_id: 8,
title: 'v0.1',
description: '',
state: 'active',
created_at: '2020-04-04T01:30:40.051Z',
updated_at: '2020-04-04T01:30:40.051Z',
due_date: null,
start_date: null,
web_url: 'http://127.0.0.1:3000/h5bp/html5-boilerplate/-/milestones/6',
},
]);
findSearchBox().vm.$emit('input', 'v0.1');
return axios.waitForAll().then(() => {
items = wrapper.findAll('[role="milestone option"]');
});
});
it('should display one item per result', () => {
expect(items).toHaveLength(1);
});
it('should emit a change if an item is clicked', () => {
items.at(0).vm.$emit('click');
expect(wrapper.emitted().change.length).toBe(1);
expect(wrapper.emitted().change[0]).toEqual([[{ title: 'v0.1' }]]);
});
it('should not have a selecton icon on any item', () => {
items.wrappers.forEach(item => {
expect(item.find('.selected-item').exists()).toBe(false);
});
});
it('should have a selecton icon if an item is clicked', () => {
items.at(0).vm.$emit('click');
expect(wrapper.find('.selected-item').exists()).toBe(true);
});
it('should not display a message about no results', () => {
expect(findNoResultsMessage().exists()).toBe(false);
});
});
describe('when Enter is pressed', () => {
beforeEach(() => {
factory({
propsData: {
projectId,
preselectedMilestones,
extraLinks,
},
data() {
return {
searchQuery: 'TEST_SEARCH',
};
},
});
mock
.onGet(TEST_SEARCH_ENDPOINT, { params: { search: 'TEST_SEARCH', scope: 'milestones' } })
.reply(200, []);
});
it('should trigger a search', async () => {
mock.resetHistory();
findSearchBox().vm.$emit('keydown', new KeyboardEvent({ key: ENTER_KEY }));
await axios.waitForAll();
expect(mock.history.get.length).toBe(1);
expect(mock.history.get[0].url).toBe(TEST_SEARCH_ENDPOINT);
});
});
});
...@@ -41,6 +41,14 @@ describe('Milestone combobox Vuex store actions', () => { ...@@ -41,6 +41,14 @@ describe('Milestone combobox Vuex store actions', () => {
}); });
}); });
describe('clearSelectedMilestones', () => {
it(`commits ${types.CLEAR_SELECTED_MILESTONES} with the new selected milestones name`, () => {
testAction(actions.clearSelectedMilestones, null, state, [
{ type: types.CLEAR_SELECTED_MILESTONES },
]);
});
});
describe('toggleMilestones', () => { describe('toggleMilestones', () => {
const selectedMilestone = 'v1.2.3'; const selectedMilestone = 'v1.2.3';
it(`commits ${types.ADD_SELECTED_MILESTONE} with the new selected milestone name`, () => { it(`commits ${types.ADD_SELECTED_MILESTONE} with the new selected milestone name`, () => {
...@@ -58,13 +66,13 @@ describe('Milestone combobox Vuex store actions', () => { ...@@ -58,13 +66,13 @@ describe('Milestone combobox Vuex store actions', () => {
}); });
describe('search', () => { describe('search', () => {
it(`commits ${types.SET_QUERY} with the new search query`, () => { it(`commits ${types.SET_SEARCH_QUERY} with the new search query`, () => {
const query = 'v1.0'; const searchQuery = 'v1.0';
testAction( testAction(
actions.search, actions.search,
query, searchQuery,
state, state,
[{ type: types.SET_QUERY, payload: query }], [{ type: types.SET_SEARCH_QUERY, payload: searchQuery }],
[{ type: 'searchMilestones' }], [{ type: 'searchMilestones' }],
); );
}); });
......
...@@ -14,7 +14,7 @@ describe('Milestones combobox Vuex store mutations', () => { ...@@ -14,7 +14,7 @@ describe('Milestones combobox Vuex store mutations', () => {
expect(state).toEqual({ expect(state).toEqual({
projectId: null, projectId: null,
groupId: null, groupId: null,
query: '', searchQuery: '',
matches: { matches: {
projectMilestones: { projectMilestones: {
list: [], list: [],
...@@ -46,6 +46,20 @@ describe('Milestones combobox Vuex store mutations', () => { ...@@ -46,6 +46,20 @@ describe('Milestones combobox Vuex store mutations', () => {
}); });
}); });
describe(`${types.CLEAR_SELECTED_MILESTONES}`, () => {
it('clears the selected milestones', () => {
const selectedMilestones = ['v1.2.3'];
// Set state.selectedMilestones
mutations[types.SET_SELECTED_MILESTONES](state, selectedMilestones);
// Clear state.selectedMilestones
mutations[types.CLEAR_SELECTED_MILESTONES](state);
expect(state.selectedMilestones).toEqual([]);
});
});
describe(`${types.ADD_SELECTED_MILESTONESs}`, () => { describe(`${types.ADD_SELECTED_MILESTONESs}`, () => {
it('adds the selected milestones', () => { it('adds the selected milestones', () => {
const selectedMilestone = 'v1.2.3'; const selectedMilestone = 'v1.2.3';
...@@ -67,12 +81,12 @@ describe('Milestones combobox Vuex store mutations', () => { ...@@ -67,12 +81,12 @@ describe('Milestones combobox Vuex store mutations', () => {
}); });
}); });
describe(`${types.SET_QUERY}`, () => { describe(`${types.SET_SEARCH_QUERY}`, () => {
it('updates the search query', () => { it('updates the search query', () => {
const newQuery = 'hello'; const newQuery = 'hello';
mutations[types.SET_QUERY](state, newQuery); mutations[types.SET_SEARCH_QUERY](state, newQuery);
expect(state.query).toBe(newQuery); expect(state.searchQuery).toBe(newQuery);
}); });
}); });
......
...@@ -22,7 +22,7 @@ describe('releases/util.js', () => { ...@@ -22,7 +22,7 @@ describe('releases/util.js', () => {
tagName: 'tag-name', tagName: 'tag-name',
name: 'Release name', name: 'Release name',
description: 'Release description', description: 'Release description',
milestones: [{ id: 1, title: '13.2' }, { id: 2, title: '13.3' }], milestones: ['13.2', '13.3'],
assets: { assets: {
links: [{ url: 'https://gitlab.example.com/link', linkType: 'other' }], links: [{ url: 'https://gitlab.example.com/link', linkType: 'other' }],
}, },
...@@ -73,18 +73,6 @@ describe('releases/util.js', () => { ...@@ -73,18 +73,6 @@ describe('releases/util.js', () => {
expect(releaseToApiJson(release)).toMatchObject(expectedJson); expect(releaseToApiJson(release)).toMatchObject(expectedJson);
}); });
}); });
describe('when release.milestones is falsy', () => {
it('includes a "milestone" property in the returned result as an empty array', () => {
const release = {};
const expectedJson = {
milestones: [],
};
expect(releaseToApiJson(release)).toMatchObject(expectedJson);
});
});
}); });
describe('apiJsonToRelease', () => { describe('apiJsonToRelease', () => {
......
...@@ -66,7 +66,7 @@ module Spec ...@@ -66,7 +66,7 @@ module Spec
focused_element.send_keys(:enter) focused_element.send_keys(:enter)
# Wait for the dropdown to be rendered # Wait for the dropdown to be rendered
page.find('.project-milestone-combobox .dropdown-menu') page.find('.milestone-combobox .dropdown-menu')
# Clear any existing input # Clear any existing input
focused_element.attribute('value').length.times { focused_element.send_keys(:backspace) } focused_element.attribute('value').length.times { focused_element.send_keys(:backspace) }
...@@ -75,7 +75,7 @@ module Spec ...@@ -75,7 +75,7 @@ module Spec
focused_element.send_keys(milestone_title, :enter) focused_element.send_keys(milestone_title, :enter)
# Wait for the search to return # Wait for the search to return
page.find('.project-milestone-combobox .dropdown-item', text: milestone_title, match: :first) page.find('.milestone-combobox .dropdown-item', text: milestone_title, match: :first)
focused_element.send_keys(:arrow_down, :arrow_down, :enter) focused_element.send_keys(:arrow_down, :arrow_down, :enter)
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment