Commit c6c8c8b5 authored by Kushal Pandya's avatar Kushal Pandya

Fix EpicToken rendering and filtering

- Fixes EpicToken rendering to prevent duplicate keys.
- Fixes filtering of epics to handle epics from child groups.
parent cb611101
...@@ -288,6 +288,7 @@ export default { ...@@ -288,6 +288,7 @@ export default {
icon: 'epic', icon: 'epic',
token: EpicToken, token: EpicToken,
unique: true, unique: true,
idProperty: 'id',
fetchEpics: this.fetchEpics, fetchEpics: this.fetchEpics,
}); });
} }
...@@ -320,13 +321,23 @@ export default { ...@@ -320,13 +321,23 @@ export default {
); );
}, },
urlParams() { urlParams() {
const filterParams = {
...this.urlFilterParams,
};
if (filterParams.epic_id) {
filterParams.epic_id = encodeURIComponent(filterParams.epic_id);
} else if (filterParams['not[epic_id]']) {
filterParams['not[epic_id]'] = encodeURIComponent(filterParams['not[epic_id]']);
}
return { return {
due_date: this.dueDateFilter, due_date: this.dueDateFilter,
page: this.page, page: this.page,
search: this.searchQuery, search: this.searchQuery,
state: this.state, state: this.state,
...urlSortParams[this.sortKey], ...urlSortParams[this.sortKey],
...this.urlFilterParams, ...filterParams,
}; };
}, },
}, },
...@@ -358,7 +369,7 @@ export default { ...@@ -358,7 +369,7 @@ export default {
fetchEmojis(search) { fetchEmojis(search) {
return this.fetchWithCache(this.autocompleteAwardEmojisPath, 'emojis', 'name', search); return this.fetchWithCache(this.autocompleteAwardEmojisPath, 'emojis', 'name', search);
}, },
async fetchEpics(search) { async fetchEpics({ search }) {
const epics = await this.fetchWithCache(this.groupEpicsPath, 'epics'); const epics = await this.fetchWithCache(this.groupEpicsPath, 'epics');
if (!search) { if (!search) {
return epics.slice(0, MAX_LIST_SIZE); return epics.slice(0, MAX_LIST_SIZE);
...@@ -387,6 +398,16 @@ export default { ...@@ -387,6 +398,16 @@ export default {
this.isLoading = true; this.isLoading = true;
const filterParams = {
...this.apiFilterParams,
};
if (filterParams.epic_id) {
filterParams.epic_id = filterParams.epic_id.split('::&').pop();
} else if (filterParams['not[epic_id]']) {
filterParams['not[epic_id]'] = filterParams['not[epic_id]'].split('::&').pop();
}
return axios return axios
.get(this.endpoint, { .get(this.endpoint, {
params: { params: {
...@@ -397,7 +418,7 @@ export default { ...@@ -397,7 +418,7 @@ export default {
state: this.state, state: this.state,
with_labels_details: true, with_labels_details: true,
...apiSortParams[this.sortKey], ...apiSortParams[this.sortKey],
...this.apiFilterParams, ...filterParams,
}, },
}) })
.then(({ data, headers }) => { .then(({ data, headers }) => {
......
...@@ -11,6 +11,7 @@ import { __ } from '~/locale'; ...@@ -11,6 +11,7 @@ import { __ } from '~/locale';
import { DEBOUNCE_DELAY, DEFAULT_NONE_ANY } from '../constants'; import { DEBOUNCE_DELAY, DEFAULT_NONE_ANY } from '../constants';
export default { export default {
separator: '::&',
components: { components: {
GlDropdownDivider, GlDropdownDivider,
GlFilteredSearchToken, GlFilteredSearchToken,
...@@ -34,17 +35,35 @@ export default { ...@@ -34,17 +35,35 @@ export default {
}; };
}, },
computed: { computed: {
idProperty() {
return this.config.idProperty || 'iid';
},
currentValue() { currentValue() {
return Number(this.value.data); const epicIid = Number(this.value.data);
if (epicIid) {
return epicIid;
}
return this.value.data;
}, },
defaultEpics() { defaultEpics() {
return this.config.defaultEpics || DEFAULT_NONE_ANY; return this.config.defaultEpics || DEFAULT_NONE_ANY;
}, },
idProperty() {
return this.config.idProperty || 'id';
},
activeEpic() { activeEpic() {
return this.epics.find((epic) => epic[this.idProperty] === this.currentValue); if (this.currentValue && this.epics.length) {
// Check if current value is an epic ID.
if (typeof this.currentValue === 'number') {
return this.epics.find((epic) => epic[this.idProperty] === this.currentValue);
}
// Current value is a string.
const [groupPath, idProperty] = this.currentValue?.split('::&');
return this.epics.find(
(epic) =>
epic.group_full_path === groupPath &&
epic[this.idProperty] === parseInt(idProperty, 10),
);
}
return null;
}, },
}, },
watch: { watch: {
...@@ -58,10 +77,10 @@ export default { ...@@ -58,10 +77,10 @@ export default {
}, },
}, },
methods: { methods: {
fetchEpicsBySearchTerm(searchTerm = '') { fetchEpicsBySearchTerm({ epicPath = '', search = '' }) {
this.loading = true; this.loading = true;
this.config this.config
.fetchEpics(searchTerm) .fetchEpics({ epicPath, search })
.then((response) => { .then((response) => {
this.epics = Array.isArray(response) ? response : response.data; this.epics = Array.isArray(response) ? response : response.data;
}) })
...@@ -71,11 +90,21 @@ export default { ...@@ -71,11 +90,21 @@ export default {
}); });
}, },
searchEpics: debounce(function debouncedSearch({ data }) { searchEpics: debounce(function debouncedSearch({ data }) {
this.fetchEpicsBySearchTerm(data); let epicPath = this.activeEpic?.web_url;
// When user visits the page with token value already included in filters
// We don't have any information about selected token except for its
// group path and iid joined by separator, so we need to manually
// compose epic path from it.
if (data.includes(this.$options.separator)) {
const [groupPath, epicIid] = data.split(this.$options.separator);
epicPath = `/groups/${groupPath}/-/epics/${epicIid}`;
}
this.fetchEpicsBySearchTerm({ epicPath, search: data });
}, DEBOUNCE_DELAY), }, DEBOUNCE_DELAY),
getEpicDisplayText(epic) { getEpicDisplayText(epic) {
return `${epic.title}::&${epic[this.idProperty]}`; return `${epic.title}${this.$options.separator}${epic.iid}`;
}, },
}, },
}; };
...@@ -104,8 +133,8 @@ export default { ...@@ -104,8 +133,8 @@ export default {
<template v-else> <template v-else>
<gl-filtered-search-suggestion <gl-filtered-search-suggestion
v-for="epic in epics" v-for="epic in epics"
:key="epic[idProperty]" :key="epic.id"
:value="String(epic[idProperty])" :value="`${epic.group_full_path}::&${epic[idProperty]}`"
> >
{{ epic.title }} {{ epic.title }}
</gl-filtered-search-suggestion> </gl-filtered-search-suggestion>
......
...@@ -207,7 +207,7 @@ export default { ...@@ -207,7 +207,7 @@ export default {
:current-tab="currentState" :current-tab="currentState"
:tab-counts="epicsCount" :tab-counts="epicsCount"
:search-input-placeholder="__('Search or filter results...')" :search-input-placeholder="__('Search or filter results...')"
:search-tokens="getFilteredSearchTokens()" :search-tokens="getFilteredSearchTokens({ supportsEpic: false })"
:sort-options="$options.EpicsSortOptions" :sort-options="$options.EpicsSortOptions"
:initial-filter-value="getFilteredSearchValue()" :initial-filter-value="getFilteredSearchValue()"
:initial-sort-by="sortedBy" :initial-sort-by="sortedBy"
......
...@@ -36,13 +36,13 @@ export default { ...@@ -36,13 +36,13 @@ export default {
milestone_title: milestoneTitle, milestone_title: milestoneTitle,
confidential, confidential,
my_reaction_emoji: myReactionEmoji, my_reaction_emoji: myReactionEmoji,
epic_iid: epicIid && Number(epicIid), epic_iid: epicIid,
search, search,
}; };
}, },
}, },
methods: { methods: {
getFilteredSearchTokens() { getFilteredSearchTokens({ supportsEpic = true } = {}) {
const tokens = [ const tokens = [
{ {
type: 'author_username', type: 'author_username',
...@@ -113,7 +113,10 @@ export default { ...@@ -113,7 +113,10 @@ export default {
{ icon: 'eye', value: false, title: __('No') }, { icon: 'eye', value: false, title: __('No') },
], ],
}, },
{ ];
if (supportsEpic) {
tokens.push({
type: 'epic_iid', type: 'epic_iid',
icon: 'epic', icon: 'epic',
title: __('Epic'), title: __('Epic'),
...@@ -121,16 +124,28 @@ export default { ...@@ -121,16 +124,28 @@ export default {
symbol: '&', symbol: '&',
token: EpicToken, token: EpicToken,
operators: OPERATOR_IS_ONLY, operators: OPERATOR_IS_ONLY,
idProperty: 'iid',
defaultEpics: [], defaultEpics: [],
fetchEpics: (search = '') => { fetchEpics: ({ epicPath = '', search = '' }) => {
const number = Number(search); const epicId = Number(search) || null;
return !search || Number.isNaN(number)
? axios.get(this.listEpicsPath, { params: { search } }) // No search criteria or path has been provided, fetch all epics.
: axios.get(joinPaths(this.listEpicsPath, search)).then(({ data }) => [data]); if (!epicPath && !search) {
return axios.get(this.listEpicsPath);
} else if (epicPath) {
// Just epicPath has been provided, fetch a specific epic.
return axios.get(epicPath).then(({ data }) => [data]);
} else if (!epicPath && epicId) {
// Exact epic ID provided, fetch the epic.
return axios
.get(joinPaths(this.listEpicsPath, String(epicId)))
.then(({ data }) => [data]);
}
// Search for an epic.
return axios.get(this.listEpicsPath, { params: { search } });
}, },
}, });
]; }
if (gon.current_user_id) { if (gon.current_user_id) {
// Appending to tokens only when logged-in // Appending to tokens only when logged-in
......
...@@ -79,7 +79,7 @@ export default () => { ...@@ -79,7 +79,7 @@ export default () => {
}), }),
...(rawFilterParams.epicIid && { ...(rawFilterParams.epicIid && {
epicIid: parseInt(rawFilterParams.epicIid, 10), epicIid: rawFilterParams.epicIid,
}), }),
}; };
const timeframe = getTimeframeForPreset( const timeframe = getTimeframeForPreset(
......
...@@ -47,7 +47,7 @@ const fetchGroupEpics = ( ...@@ -47,7 +47,7 @@ const fetchGroupEpics = (
}; };
if (filterParams?.epicIid) { if (filterParams?.epicIid) {
variables.iid = filterParams.epicIid; variables.iid = filterParams.epicIid.split('::&').pop();
} }
} }
......
...@@ -9,6 +9,9 @@ class EpicEntity < IssuableEntity ...@@ -9,6 +9,9 @@ class EpicEntity < IssuableEntity
expose :group_full_name do |epic| expose :group_full_name do |epic|
epic.group.full_name epic.group.full_name
end end
expose :group_full_path do |epic|
epic.group.full_path
end
expose :start_date expose :start_date
expose :start_date_is_fixed?, as: :start_date_is_fixed expose :start_date_is_fixed?, as: :start_date_is_fixed
......
...@@ -22,6 +22,7 @@ ...@@ -22,6 +22,7 @@
"labels": { "type": ["array", "null"] }, "labels": { "type": ["array", "null"] },
"group_name": { "type": "string" }, "group_name": { "type": "string" },
"group_full_name": { "type": "string" }, "group_full_name": { "type": "string" },
"group_full_path": { "type": "string" },
"current_user": { "current_user": {
"can_create_note": { "type": "boolean" } "can_create_note": { "type": "boolean" }
}, },
...@@ -49,6 +50,7 @@ ...@@ -49,6 +50,7 @@
"labels", "labels",
"group_name", "group_name",
"group_full_name", "group_full_name",
"group_full_path",
"current_user", "current_user",
"create_note_path", "create_note_path",
"preview_note_path" "preview_note_path"
......
...@@ -170,6 +170,7 @@ describe('EpicsListRoot', () => { ...@@ -170,6 +170,7 @@ describe('EpicsListRoot', () => {
const getIssuableList = () => wrapper.find(IssuableList); const getIssuableList = () => wrapper.find(IssuableList);
it('renders issuable-list component', async () => { it('renders issuable-list component', async () => {
jest.spyOn(wrapper.vm, 'getFilteredSearchTokens');
wrapper.setData({ wrapper.setData({
filterParams: { filterParams: {
search: 'foo', search: 'foo',
...@@ -192,6 +193,10 @@ describe('EpicsListRoot', () => { ...@@ -192,6 +193,10 @@ describe('EpicsListRoot', () => {
issuableSymbol: '&', issuableSymbol: '&',
recentSearchesStorageKey: 'epics', recentSearchesStorageKey: 'epics',
}); });
expect(wrapper.vm.getFilteredSearchTokens).toHaveBeenCalledWith({
supportsEpic: false,
});
}); });
it.each` it.each`
......
...@@ -218,7 +218,6 @@ describe('RoadmapFilters', () => { ...@@ -218,7 +218,6 @@ describe('RoadmapFilters', () => {
symbol: '&', symbol: '&',
token: EpicToken, token: EpicToken,
operators, operators,
idProperty: 'iid',
defaultEpics: [], defaultEpics: [],
fetchEpics: expect.any(Function), fetchEpics: expect.any(Function),
}, },
......
...@@ -16,6 +16,6 @@ RSpec.describe EpicEntity do ...@@ -16,6 +16,6 @@ RSpec.describe EpicEntity do
end end
it 'has epic specific attributes' do it 'has epic specific attributes' do
expect(subject).to include(:start_date, :end_date, :group_id, :group_name, :group_full_name, :web_url) expect(subject).to include(:start_date, :end_date, :group_id, :group_name, :group_full_name, :group_full_path, :web_url)
end end
end end
...@@ -21,8 +21,8 @@ export const locationSearch = [ ...@@ -21,8 +21,8 @@ export const locationSearch = [
'confidential=no', 'confidential=no',
'iteration_title=season:+%234', 'iteration_title=season:+%234',
'not[iteration_title]=season:+%2320', 'not[iteration_title]=season:+%2320',
'epic_id=12', 'epic_id=gitlab-org%3A%3A%2612',
'not[epic_id]=34', 'not[epic_id]=gitlab-org%3A%3A%2634',
'weight=1', 'weight=1',
'not[weight]=3', 'not[weight]=3',
].join('&'); ].join('&');
...@@ -53,8 +53,8 @@ export const filteredTokens = [ ...@@ -53,8 +53,8 @@ export const filteredTokens = [
{ type: 'confidential', value: { data: 'no', operator: OPERATOR_IS } }, { type: 'confidential', value: { data: 'no', operator: OPERATOR_IS } },
{ type: 'iteration', value: { data: 'season: #4', operator: OPERATOR_IS } }, { type: 'iteration', value: { data: 'season: #4', operator: OPERATOR_IS } },
{ type: 'iteration', value: { data: 'season: #20', operator: OPERATOR_IS_NOT } }, { type: 'iteration', value: { data: 'season: #20', operator: OPERATOR_IS_NOT } },
{ type: 'epic_id', value: { data: '12', operator: OPERATOR_IS } }, { type: 'epic_id', value: { data: 'gitlab-org::&12', operator: OPERATOR_IS } },
{ type: 'epic_id', value: { data: '34', operator: OPERATOR_IS_NOT } }, { type: 'epic_id', value: { data: 'gitlab-org::&34', operator: OPERATOR_IS_NOT } },
{ type: 'weight', value: { data: '1', operator: OPERATOR_IS } }, { type: 'weight', value: { data: '1', operator: OPERATOR_IS } },
{ type: 'weight', value: { data: '3', operator: OPERATOR_IS_NOT } }, { type: 'weight', value: { data: '3', operator: OPERATOR_IS_NOT } },
{ type: 'filtered-search-term', value: { data: 'find' } }, { type: 'filtered-search-term', value: { data: 'find' } },
...@@ -84,7 +84,7 @@ export const apiParams = { ...@@ -84,7 +84,7 @@ export const apiParams = {
iteration_title: 'season: #4', iteration_title: 'season: #4',
'not[iteration_title]': 'season: #20', 'not[iteration_title]': 'season: #20',
epic_id: '12', epic_id: '12',
'not[epic_id]': '34', 'not[epic_id]': 'gitlab-org::&34',
weight: '1', weight: '1',
'not[weight]': '3', 'not[weight]': '3',
}; };
...@@ -111,8 +111,8 @@ export const urlParams = { ...@@ -111,8 +111,8 @@ export const urlParams = {
confidential: 'no', confidential: 'no',
iteration_title: 'season: #4', iteration_title: 'season: #4',
'not[iteration_title]': 'season: #20', 'not[iteration_title]': 'season: #20',
epic_id: '12', epic_id: 'gitlab-org%3A%3A%2612',
'not[epic_id]': '34', 'not[epic_id]': 'gitlab-org::&34',
weight: '1', weight: '1',
'not[weight]': '3', 'not[weight]': '3',
}; };
......
...@@ -82,7 +82,10 @@ describe('getFilterTokens', () => { ...@@ -82,7 +82,10 @@ describe('getFilterTokens', () => {
describe('convertToParams', () => { describe('convertToParams', () => {
it('returns api params given filtered tokens', () => { it('returns api params given filtered tokens', () => {
expect(convertToParams(filteredTokens, API_PARAM)).toEqual(apiParams); expect(convertToParams(filteredTokens, API_PARAM)).toEqual({
...apiParams,
epic_id: 'gitlab-org::&12',
});
}); });
it('returns api params given filtered tokens with special values', () => { it('returns api params given filtered tokens with special values', () => {
...@@ -92,7 +95,10 @@ describe('convertToParams', () => { ...@@ -92,7 +95,10 @@ describe('convertToParams', () => {
}); });
it('returns url params given filtered tokens', () => { it('returns url params given filtered tokens', () => {
expect(convertToParams(filteredTokens, URL_PARAM)).toEqual(urlParams); expect(convertToParams(filteredTokens, URL_PARAM)).toEqual({
...urlParams,
epic_id: 'gitlab-org::&12',
});
}); });
it('returns url params given filtered tokens with special values', () => { it('returns url params given filtered tokens with special values', () => {
......
...@@ -65,8 +65,8 @@ export const mockMilestones = [ ...@@ -65,8 +65,8 @@ export const mockMilestones = [
]; ];
export const mockEpics = [ export const mockEpics = [
{ iid: 1, id: 1, title: 'Foo' }, { iid: 1, id: 1, title: 'Foo', group_full_path: 'gitlab-org' },
{ iid: 2, id: 2, title: 'Bar' }, { iid: 2, id: 2, title: 'Bar', group_full_path: 'gitlab-org/design' },
]; ];
export const mockEmoji1 = { export const mockEmoji1 = {
......
...@@ -67,18 +67,6 @@ describe('EpicToken', () => { ...@@ -67,18 +67,6 @@ describe('EpicToken', () => {
await wrapper.vm.$nextTick(); await wrapper.vm.$nextTick();
}); });
describe('activeEpic', () => {
it('returns object for currently present `value.data`', async () => {
wrapper.setProps({
value: { data: `${mockEpics[0].iid}` },
});
await wrapper.vm.$nextTick();
expect(wrapper.vm.activeEpic).toEqual(mockEpics[0]);
});
});
}); });
describe('methods', () => { describe('methods', () => {
...@@ -86,9 +74,12 @@ describe('EpicToken', () => { ...@@ -86,9 +74,12 @@ describe('EpicToken', () => {
it('calls `config.fetchEpics` with provided searchTerm param', () => { it('calls `config.fetchEpics` with provided searchTerm param', () => {
jest.spyOn(wrapper.vm.config, 'fetchEpics'); jest.spyOn(wrapper.vm.config, 'fetchEpics');
wrapper.vm.fetchEpicsBySearchTerm('foo'); wrapper.vm.fetchEpicsBySearchTerm({ search: 'foo' });
expect(wrapper.vm.config.fetchEpics).toHaveBeenCalledWith('foo'); expect(wrapper.vm.config.fetchEpics).toHaveBeenCalledWith({
epicPath: '',
search: 'foo',
});
}); });
it('sets response to `epics` when request is successful', async () => { it('sets response to `epics` when request is successful', async () => {
...@@ -96,7 +87,7 @@ describe('EpicToken', () => { ...@@ -96,7 +87,7 @@ describe('EpicToken', () => {
data: mockEpics, data: mockEpics,
}); });
wrapper.vm.fetchEpicsBySearchTerm(); wrapper.vm.fetchEpicsBySearchTerm({});
await waitForPromises(); await waitForPromises();
...@@ -106,7 +97,7 @@ describe('EpicToken', () => { ...@@ -106,7 +97,7 @@ describe('EpicToken', () => {
it('calls `createFlash` with flash error message when request fails', async () => { it('calls `createFlash` with flash error message when request fails', async () => {
jest.spyOn(wrapper.vm.config, 'fetchEpics').mockRejectedValue({}); jest.spyOn(wrapper.vm.config, 'fetchEpics').mockRejectedValue({});
wrapper.vm.fetchEpicsBySearchTerm('foo'); wrapper.vm.fetchEpicsBySearchTerm({ search: 'foo' });
await waitForPromises(); await waitForPromises();
...@@ -118,7 +109,7 @@ describe('EpicToken', () => { ...@@ -118,7 +109,7 @@ describe('EpicToken', () => {
it('sets `loading` to false when request completes', async () => { it('sets `loading` to false when request completes', async () => {
jest.spyOn(wrapper.vm.config, 'fetchEpics').mockRejectedValue({}); jest.spyOn(wrapper.vm.config, 'fetchEpics').mockRejectedValue({});
wrapper.vm.fetchEpicsBySearchTerm('foo'); wrapper.vm.fetchEpicsBySearchTerm({ search: 'foo' });
await waitForPromises(); await waitForPromises();
...@@ -128,9 +119,11 @@ describe('EpicToken', () => { ...@@ -128,9 +119,11 @@ describe('EpicToken', () => {
}); });
describe('template', () => { describe('template', () => {
const getTokenValueEl = () => wrapper.findAllComponents(GlFilteredSearchTokenSegment).at(2);
beforeEach(async () => { beforeEach(async () => {
wrapper = createComponent({ wrapper = createComponent({
value: { data: `${mockEpics[0].iid}` }, value: { data: `${mockEpics[0].group_full_path}::&${mockEpics[0].iid}` },
data: { epics: mockEpics }, data: { epics: mockEpics },
}); });
...@@ -147,5 +140,19 @@ describe('EpicToken', () => { ...@@ -147,5 +140,19 @@ describe('EpicToken', () => {
expect(tokenSegments).toHaveLength(3); expect(tokenSegments).toHaveLength(3);
expect(tokenSegments.at(2).text()).toBe(`${mockEpics[0].title}::&${mockEpics[0].iid}`); expect(tokenSegments.at(2).text()).toBe(`${mockEpics[0].title}::&${mockEpics[0].iid}`);
}); });
it.each`
value | valueType | tokenValueString
${`${mockEpics[0].group_full_path}::&${mockEpics[0].iid}`} | ${'string'} | ${`${mockEpics[0].title}::&${mockEpics[0].iid}`}
${`${mockEpics[1].group_full_path}::&${mockEpics[1].iid}`} | ${'number'} | ${`${mockEpics[1].title}::&${mockEpics[1].iid}`}
`('renders token item when selection is a $valueType', async ({ value, tokenValueString }) => {
wrapper.setProps({
value: { data: value },
});
await wrapper.vm.$nextTick();
expect(getTokenValueEl().text()).toBe(tokenValueString);
});
}); });
}); });
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