From e112129f3d21579a2d7fac3f01077504a1d9fa9e Mon Sep 17 00:00:00 2001 From: Andrew Fontaine <afontaine@gitlab.com> Date: Tue, 26 Jan 2021 07:56:11 +0000 Subject: [PATCH] Add URL Navigation to Pipeline Analytics Charts If a query string of `?chart=XXX` is added to the URL, the appropriate tab is selected. Currently, the only correct values are `pipelines` and `deployments`. All other values are ignored, and `pipelines` is used as the default. If the application would not display tabs, nothing is affected. --- .../pipelines/charts/components/app.vue | 14 +++++- ...dd-url-navigation-pipeline-charts-tabs.yml | 5 ++ .../pipelines/charts/components/app_spec.js | 48 +++++++++++++++++++ 3 files changed, 66 insertions(+), 1 deletion(-) create mode 100644 ee/changelogs/unreleased/add-url-navigation-pipeline-charts-tabs.yml diff --git a/app/assets/javascripts/projects/pipelines/charts/components/app.vue b/app/assets/javascripts/projects/pipelines/charts/components/app.vue index 7bb62cf4a731d..61b899896bc75 100644 --- a/app/assets/javascripts/projects/pipelines/charts/components/app.vue +++ b/app/assets/javascripts/projects/pipelines/charts/components/app.vue @@ -4,6 +4,7 @@ import { s__ } from '~/locale'; import getPipelineCountByStatus from '../graphql/queries/get_pipeline_count_by_status.query.graphql'; import getProjectPipelineStatistics from '../graphql/queries/get_project_pipeline_statistics.query.graphql'; import PipelineCharts from './pipeline_charts.vue'; +import { mergeUrlParams, updateHistory, getParameterValues } from '~/lib/utils/url_utility'; import { DEFAULT, @@ -36,6 +37,8 @@ const defaultCountValues = { }, }; +const charts = ['pipelines', 'deployments']; + export default { components: { GlAlert, @@ -56,7 +59,11 @@ export default { }, }, data() { + const [chart] = getParameterValues('chart') || charts; + const tab = charts.indexOf(chart); return { + chart, + selectedTab: tab >= 0 ? tab : 0, showFailureAlert: false, failureType: null, analytics: { ...defaultAnalyticsValues }, @@ -172,6 +179,11 @@ export default { this.showFailureAlert = true; this.failureType = type; }, + onTabChange(index) { + this.selectedTab = index; + const path = mergeUrlParams({ chart: charts[index] }, window.location.pathname); + updateHistory({ url: path }); + }, }, errorTexts: { [LOAD_ANALYTICS_FAILURE]: s__( @@ -190,7 +202,7 @@ export default { <gl-alert v-if="showFailureAlert" :variant="failure.variant" @dismiss="hideAlert">{{ failure.text }}</gl-alert> - <gl-tabs v-if="shouldRenderDeploymentFrequencyCharts"> + <gl-tabs v-if="shouldRenderDeploymentFrequencyCharts" :value="selectedTab" @input="onTabChange"> <gl-tab :title="__('Pipelines')"> <pipeline-charts :counts="formattedCounts" diff --git a/ee/changelogs/unreleased/add-url-navigation-pipeline-charts-tabs.yml b/ee/changelogs/unreleased/add-url-navigation-pipeline-charts-tabs.yml new file mode 100644 index 0000000000000..26e4cc71a090a --- /dev/null +++ b/ee/changelogs/unreleased/add-url-navigation-pipeline-charts-tabs.yml @@ -0,0 +1,5 @@ +--- +title: Add URL Navigation to Pipeline Analytics Charts +merge_request: 52338 +author: +type: added diff --git a/spec/frontend/projects/pipelines/charts/components/app_spec.js b/spec/frontend/projects/pipelines/charts/components/app_spec.js index 44329944097eb..07eb8198edb21 100644 --- a/spec/frontend/projects/pipelines/charts/components/app_spec.js +++ b/spec/frontend/projects/pipelines/charts/components/app_spec.js @@ -3,12 +3,17 @@ import { createLocalVue, shallowMount } from '@vue/test-utils'; import VueApollo from 'vue-apollo'; import { GlTabs, GlTab } from '@gitlab/ui'; import createMockApollo from 'helpers/mock_apollo_helper'; +import setWindowLocation from 'helpers/set_window_location_helper'; +import { TEST_HOST } from 'helpers/test_constants'; +import { mergeUrlParams, updateHistory, getParameterValues } from '~/lib/utils/url_utility'; import Component from '~/projects/pipelines/charts/components/app.vue'; import PipelineCharts from '~/projects/pipelines/charts/components/pipeline_charts.vue'; import getPipelineCountByStatus from '~/projects/pipelines/charts/graphql/queries/get_pipeline_count_by_status.query.graphql'; import getProjectPipelineStatistics from '~/projects/pipelines/charts/graphql/queries/get_project_pipeline_statistics.query.graphql'; import { mockPipelineCount, mockPipelineStatistics } from '../mock_data'; +jest.mock('~/lib/utils/url_utility'); + const projectPath = 'gitlab-org/gitlab'; const localVue = createLocalVue(); localVue.use(VueApollo); @@ -115,6 +120,49 @@ describe('ProjectsPipelinesChartsApp', () => { expect(findGlTabAt(1).attributes('title')).toBe('Deployments'); expect(findDeploymentFrequencyCharts().exists()).toBe(true); }); + + it('sets the tab and url when a tab is clicked', async () => { + let chartsPath; + setWindowLocation(`${TEST_HOST}/gitlab-org/gitlab-test/-/pipelines/charts`); + + mergeUrlParams.mockImplementation(({ chart }, path) => { + expect(chart).toBe('deployments'); + expect(path).toBe(window.location.pathname); + chartsPath = `${path}?chart=${chart}`; + return chartsPath; + }); + + updateHistory.mockImplementation(({ url }) => { + expect(url).toBe(chartsPath); + }); + const tabs = findGlTabs(); + + expect(tabs.attributes('value')).toBe('0'); + + tabs.vm.$emit('input', 1); + + await wrapper.vm.$nextTick(); + + expect(tabs.attributes('value')).toBe('1'); + }); + }); + + describe('when provided with a query param', () => { + it.each` + chart | tab + ${'deployments'} | ${'1'} + ${'pipelines'} | ${'0'} + ${'fake'} | ${'0'} + ${''} | ${'0'} + `('shows the correct tab for URL parameter "$chart"', ({ chart, tab }) => { + setWindowLocation(`${TEST_HOST}/gitlab-org/gitlab-test/-/pipelines/charts?chart=${chart}`); + getParameterValues.mockImplementation((name) => { + expect(name).toBe('chart'); + return chart ? [chart] : []; + }); + createComponent({ provide: { shouldRenderDeploymentFrequencyCharts: true } }); + expect(findGlTabs().attributes('value')).toBe(tab); + }); }); describe('when shouldRenderDeploymentFrequencyCharts is false', () => { -- GitLab