Commit 2ed17c6e authored by Eulyeon Ko's avatar Eulyeon Ko Committed by Peter Hegman

Display iteration dates in issues filtered search

Auto-generated iterations (for a cadence) do not
hardcode start/due dates in their titles and
the frontend should utilize start/date attributes
and display them separately.
parent 1b3a4e6e
...@@ -329,13 +329,13 @@ export default { ...@@ -329,13 +329,13 @@ export default {
class="board-title-main-text gl-text-truncate" class="board-title-main-text gl-text-truncate"
> >
{{ listTitle }} {{ listTitle }}
<div <span
v-if="iterationCadencesAvailable" v-if="iterationCadencesAvailable"
class="gl-display-inline-block" class="gl-display-inline-block gl-text-gray-400"
data-testid="board-list-iteration-period" data-testid="board-list-iteration-period"
> >
<time class="gl-text-gray-400">{{ listIterationPeriod }}</time> {{ listIterationPeriod }}</span
</div> >
</span> </span>
<span <span
v-if="listType === 'assignee'" v-if="listType === 'assignee'"
......
fragment Iteration on Iteration { fragment Iteration on Iteration {
id id
title title
startDate
dueDate
iterationCadence { iterationCadence {
id id
title title
......
...@@ -4,6 +4,8 @@ import createFlash from '~/flash'; ...@@ -4,6 +4,8 @@ import createFlash from '~/flash';
import { getIdFromGraphQLId } from '~/graphql_shared/utils'; import { getIdFromGraphQLId } from '~/graphql_shared/utils';
import { __ } from '~/locale'; import { __ } from '~/locale';
import BaseToken from '~/vue_shared/components/filtered_search_bar/tokens/base_token.vue'; import BaseToken from '~/vue_shared/components/filtered_search_bar/tokens/base_token.vue';
import { formatDate } from '~/lib/utils/datetime_utility';
import glFeatureFlagMixin from '~/vue_shared/mixins/gl_feature_flags_mixin';
import { DEFAULT_ITERATIONS } from '../constants'; import { DEFAULT_ITERATIONS } from '../constants';
export default { export default {
...@@ -13,6 +15,7 @@ export default { ...@@ -13,6 +15,7 @@ export default {
GlDropdownSectionHeader, GlDropdownSectionHeader,
GlFilteredSearchSuggestion, GlFilteredSearchSuggestion,
}, },
mixins: [glFeatureFlagMixin()],
props: { props: {
active: { active: {
type: Boolean, type: Boolean,
...@@ -49,7 +52,11 @@ export default { ...@@ -49,7 +52,11 @@ export default {
return; return;
} }
const { title } = iteration.iterationCadence; const { title } = iteration.iterationCadence;
const cadenceIteration = { id: iteration.id, title: iteration.title }; const cadenceIteration = {
id: iteration.id,
title: iteration.title,
period: this.getIterationPeriod(iteration),
};
const cadence = cadences.find((cad) => cad.title === title); const cadence = cadences.find((cad) => cad.title === title);
if (cadence) { if (cadence) {
cadence.iterations.push(cadenceIteration); cadence.iterations.push(cadenceIteration);
...@@ -76,6 +83,16 @@ export default { ...@@ -76,6 +83,16 @@ export default {
getValue(iteration) { getValue(iteration) {
return String(getIdFromGraphQLId(iteration.id)); return String(getIdFromGraphQLId(iteration.id));
}, },
/**
* TODO: https://gitlab.com/gitlab-org/gitlab/-/issues/344619
* This method also exists as a utility function in ee/../iterations/utils.js
* Remove the duplication when iteration token is moved to EE.
*/
getIterationPeriod({ startDate, dueDate }) {
const start = formatDate(startDate, 'mmm d, yyyy', true);
const due = formatDate(dueDate, 'mmm d, yyyy', true);
return `${start} - ${due}`;
},
}, },
}; };
</script> </script>
...@@ -111,6 +128,9 @@ export default { ...@@ -111,6 +128,9 @@ export default {
:value="getValue(iteration)" :value="getValue(iteration)"
> >
{{ iteration.title }} {{ iteration.title }}
<div v-if="glFeatures.iterationCadences" class="gl-text-gray-400">
{{ iteration.period }}
</div>
</gl-filtered-search-suggestion> </gl-filtered-search-suggestion>
</template> </template>
</template> </template>
......
<template> <template>
<div> <div class="gl-text-gray-400">
<time class="gl-text-gray-400"> <slot></slot>
<slot></slot>
</time>
</div> </div>
</template> </template>
...@@ -123,18 +123,21 @@ RSpec.describe 'Filter issues by iteration', :js do ...@@ -123,18 +123,21 @@ RSpec.describe 'Filter issues by iteration', :js do
click_link '= is' click_link '= is'
end end
it 'shows cadence titles and iteration titles', :aggregate_failures do it 'shows cadence titles, and iteration titles and dates', :aggregate_failures do
within '.gl-filtered-search-suggestion-list' do within '.gl-filtered-search-suggestion-list' do
# cadence 1 grouping # cadence 1 grouping
expect(page).to have_css('li:nth-child(5)', text: cadence_1.title) expect(page).to have_css('li:nth-child(6)', text: "#{iteration_1.title} #{iteration_period(iteration_1)}")
expect(page).to have_css('li:nth-child(6)', text: iteration_1.title) expect(page).to have_css('li:nth-child(7)', text: "#{iteration_3.title} #{iteration_period(iteration_3)}")
expect(page).to have_css('li:nth-child(7)', text: iteration_3.title)
# cadence 2 grouping # cadence 2 grouping
expect(page).to have_css('li:nth-child(9)', text: cadence_2.title) expect(page).to have_css('li:nth-child(9)', text: cadence_2.title)
expect(page).to have_css('li:nth-child(10)', text: iteration_2.title) expect(page).to have_css('li:nth-child(10)', text: "#{iteration_2.title} #{iteration_period(iteration_2)}")
end end
end end
end end
def iteration_period(iteration)
"#{iteration.start_date.to_s(:medium)} - #{iteration.due_date.to_s(:medium)}"
end
end end
context 'project issues list' do context 'project issues list' do
......
...@@ -111,6 +111,18 @@ export const mockIterationToken = { ...@@ -111,6 +111,18 @@ export const mockIterationToken = {
fetchIterations: () => Promise.resolve(), fetchIterations: () => Promise.resolve(),
}; };
export const mockIterations = [
{
id: 1,
title: 'Iteration 1',
startDate: '2021-11-05',
dueDate: '2021-11-10',
iterationCadence: {
title: 'Cadence 1',
},
},
];
export const mockLabelToken = { export const mockLabelToken = {
type: 'label_name', type: 'label_name',
icon: 'labels', icon: 'labels',
......
import { GlFilteredSearchToken, GlFilteredSearchTokenSegment } from '@gitlab/ui'; import {
GlFilteredSearchToken,
GlFilteredSearchTokenSegment,
GlFilteredSearchSuggestion,
} from '@gitlab/ui';
import { mount } from '@vue/test-utils'; import { mount } from '@vue/test-utils';
import waitForPromises from 'helpers/wait_for_promises'; import waitForPromises from 'helpers/wait_for_promises';
import createFlash from '~/flash'; import createFlash from '~/flash';
import IterationToken from '~/vue_shared/components/filtered_search_bar/tokens/iteration_token.vue'; import IterationToken from '~/vue_shared/components/filtered_search_bar/tokens/iteration_token.vue';
import { mockIterationToken } from '../mock_data'; import { mockIterationToken, mockIterations } from '../mock_data';
jest.mock('~/flash'); jest.mock('~/flash');
...@@ -11,10 +15,16 @@ describe('IterationToken', () => { ...@@ -11,10 +15,16 @@ describe('IterationToken', () => {
const id = 123; const id = 123;
let wrapper; let wrapper;
const createComponent = ({ config = mockIterationToken, value = { data: '' } } = {}) => const createComponent = ({
config = mockIterationToken,
value = { data: '' },
active = false,
stubs = {},
provide = {},
} = {}) =>
mount(IterationToken, { mount(IterationToken, {
propsData: { propsData: {
active: false, active,
config, config,
value, value,
}, },
...@@ -22,13 +32,39 @@ describe('IterationToken', () => { ...@@ -22,13 +32,39 @@ describe('IterationToken', () => {
portalName: 'fake target', portalName: 'fake target',
alignSuggestions: function fakeAlignSuggestions() {}, alignSuggestions: function fakeAlignSuggestions() {},
suggestionsListClass: () => 'custom-class', suggestionsListClass: () => 'custom-class',
...provide,
}, },
stubs,
}); });
afterEach(() => { afterEach(() => {
wrapper.destroy(); wrapper.destroy();
}); });
describe('when iteration cadence feature is available', () => {
beforeEach(async () => {
wrapper = createComponent({
active: true,
config: { ...mockIterationToken, initialIterations: mockIterations },
value: { data: 'i' },
stubs: { Portal: true },
provide: {
glFeatures: {
iterationCadences: true,
},
},
});
await wrapper.setData({ loading: false });
});
it('renders iteration start date and due date', () => {
const suggestions = wrapper.findAll(GlFilteredSearchSuggestion);
expect(suggestions.at(3).text()).toContain('Nov 5, 2021 - Nov 10, 2021');
});
});
it('renders iteration value', async () => { it('renders iteration value', async () => {
wrapper = createComponent({ value: { data: id } }); wrapper = createComponent({ value: { data: id } });
......
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