Commit 778092be authored by Mark Fletcher's avatar Mark Fletcher

Set active insights dashboard tab from hash fragment

- Allows linking to an individual dashboard page

- Hash fragment is used when tab has not been selected from dropdown
- Without hash fragment and selected tab the default (first tab) is used
parent d8f2eb01
...@@ -38,7 +38,14 @@ export default { ...@@ -38,7 +38,14 @@ export default {
} }
if (!activeTab) { if (!activeTab) {
this.setActiveTab(Object.keys(configData)[0]); if (this.validSpecifiedTab()) {
this.setActiveTab(this.specifiedTab);
} else {
const defaultTab = Object.keys(configData)[0];
this.setActiveTab(defaultTab);
this.$router.replace(defaultTab);
}
} }
return Object.keys(configData).map(key => ({ return Object.keys(configData).map(key => ({
...@@ -50,6 +57,9 @@ export default { ...@@ -50,6 +57,9 @@ export default {
configPresent() { configPresent() {
return !this.configLoading && this.configData != null; return !this.configLoading && this.configData != null;
}, },
specifiedTab() {
return this.$route.params.tabId;
},
}, },
mounted() { mounted() {
this.fetchConfigData(this.endpoint); this.fetchConfigData(this.endpoint);
...@@ -57,7 +67,15 @@ export default { ...@@ -57,7 +67,15 @@ export default {
methods: { methods: {
...mapActions('insights', ['fetchConfigData', 'setActiveTab']), ...mapActions('insights', ['fetchConfigData', 'setActiveTab']),
onChangePage(page) { onChangePage(page) {
this.setActiveTab(page); if (this.validTab(page) && this.activeTab !== page) {
this.$router.push(page);
}
},
validSpecifiedTab() {
return this.specifiedTab && this.validTab(this.specifiedTab);
},
validTab(tab) {
return Object.prototype.hasOwnProperty.call(this.configData, tab);
}, },
}, },
}; };
......
import Vue from 'vue'; import Vue from 'vue';
import Insights from './components/insights.vue'; import Insights from './components/insights.vue';
import createRouter from './insights_router';
import store from './stores'; import store from './stores';
export default () => { export default () => {
const el = document.querySelector('#js-insights-pane'); const el = document.querySelector('#js-insights-pane');
const { endpoint, queryEndpoint } = el.dataset;
const router = createRouter(endpoint);
if (!el) return null; if (!el) return null;
return new Vue({ return new Vue({
el, el,
store, store,
router,
components: { components: {
Insights, Insights,
}, },
render(createElement) { render(createElement) {
return createElement('insights', { return createElement('insights', {
props: { props: {
endpoint: el.dataset.endpoint, endpoint,
queryEndpoint: el.dataset.queryEndpoint, queryEndpoint,
}, },
}); });
}, },
......
import Vue from 'vue';
import VueRouter from 'vue-router';
import { joinPaths } from '~/lib/utils/url_utility';
import store from 'ee/insights/stores';
Vue.use(VueRouter);
export default function createRouter(base) {
const router = new VueRouter({
mode: 'hash',
base: joinPaths(gon.relative_url_root || '', base),
routes: [{ path: '/:tabId' }],
});
router.beforeEach((to, from, next) => {
const page = to.path.substr(1);
store.dispatch('insights/setActiveTab', page);
next();
});
return router;
}
...@@ -45,10 +45,16 @@ export const fetchChartData = ({ dispatch }, { endpoint, chart }) => ...@@ -45,10 +45,16 @@ export const fetchChartData = ({ dispatch }, { endpoint, chart }) =>
export const setActiveTab = ({ commit, state }, key) => { export const setActiveTab = ({ commit, state }, key) => {
const { configData } = state; const { configData } = state;
if (configData) {
const page = configData[key]; const page = configData[key];
if (page) {
commit(types.SET_ACTIVE_TAB, key); commit(types.SET_ACTIVE_TAB, key);
commit(types.SET_ACTIVE_PAGE, page); commit(types.SET_ACTIVE_PAGE, page);
} else {
createFlash(__('The specified tab is invalid, please select another'));
}
}
}; };
export const initChartData = ({ commit }, store) => commit(types.INIT_CHART_DATA, store); export const initChartData = ({ commit }, store) => commit(types.INIT_CHART_DATA, store);
......
---
title: Set active insights dashboard tab from hash fragment
merge_request: 16904
author:
type: added
...@@ -45,7 +45,7 @@ constraints(::Constraints::GroupUrlConstrainer.new) do ...@@ -45,7 +45,7 @@ constraints(::Constraints::GroupUrlConstrainer.new) do
resource :issues_analytics, only: [:show] resource :issues_analytics, only: [:show]
resource :insights, only: [:show] do resource :insights, only: [:show], trailing_slash: true do
collection do collection do
post :query post :query
end end
......
...@@ -77,7 +77,7 @@ constraints(::Constraints::ProjectUrlConstrainer.new) do ...@@ -77,7 +77,7 @@ constraints(::Constraints::ProjectUrlConstrainer.new) do
end end
end end
resource :insights, only: [:show] do resource :insights, only: [:show], trailing_slash: true do
collection do collection do
post :query post :query
end end
......
...@@ -10,6 +10,7 @@ describe Groups::InsightsController do ...@@ -10,6 +10,7 @@ describe Groups::InsightsController do
set(:user) { create(:user) } set(:user) { create(:user) }
let(:query_params) { { type: 'bar', query: { issuable_type: 'issue', collection_labels: ['bug'] }, projects: projects_params } } let(:query_params) { { type: 'bar', query: { issuable_type: 'issue', collection_labels: ['bug'] }, projects: projects_params } }
let(:projects_params) { { only: [project.id, project.full_path] } } let(:projects_params) { { only: [project.id, project.full_path] } }
let(:params) { { trailing_slash: true } }
before do before do
stub_licensed_features(insights: true) stub_licensed_features(insights: true)
...@@ -37,19 +38,19 @@ describe Groups::InsightsController do ...@@ -37,19 +38,19 @@ describe Groups::InsightsController do
context 'when insights configuration project cannot be read by current user' do context 'when insights configuration project cannot be read by current user' do
context 'when visiting the parent group' do context 'when visiting the parent group' do
describe 'GET #show.html' do describe 'GET #show.html' do
subject { get :show, params: { group_id: parent_group.to_param } } subject { get :show, params: params.merge(group_id: parent_group.to_param) }
it_behaves_like '404 status' it_behaves_like '404 status'
end end
describe 'GET #show.json' do describe 'GET #show.json' do
subject { get :show, params: { group_id: parent_group.to_param }, format: :json } subject { get :show, params: params.merge(group_id: parent_group.to_param), format: :json }
it_behaves_like '404 status' it_behaves_like '404 status'
end end
describe 'POST #query' do describe 'POST #query' do
subject { post :query, params: query_params.merge(group_id: parent_group.to_param) } subject { post :query, params: params.merge(query_params.merge(group_id: parent_group.to_param)) }
it_behaves_like '404 status' it_behaves_like '404 status'
end end
...@@ -57,7 +58,7 @@ describe Groups::InsightsController do ...@@ -57,7 +58,7 @@ describe Groups::InsightsController do
context 'when visiting a nested group' do context 'when visiting a nested group' do
describe 'GET #show.html' do describe 'GET #show.html' do
subject { get :show, params: { group_id: nested_group.to_param } } subject { get :show, params: params.merge(group_id: nested_group.to_param) }
# The following expectation should be changed to # The following expectation should be changed to
# it_behaves_like '404 status' # it_behaves_like '404 status'
...@@ -66,7 +67,7 @@ describe Groups::InsightsController do ...@@ -66,7 +67,7 @@ describe Groups::InsightsController do
end end
describe 'GET #show.json' do describe 'GET #show.json' do
subject { get :show, params: { group_id: nested_group.to_param }, format: :json } subject { get :show, params: params.merge(group_id: nested_group.to_param), format: :json }
# The following expectation should be changed to # The following expectation should be changed to
# it_behaves_like '404 status' # it_behaves_like '404 status'
...@@ -83,7 +84,7 @@ describe Groups::InsightsController do ...@@ -83,7 +84,7 @@ describe Groups::InsightsController do
end end
describe 'POST #query.json' do describe 'POST #query.json' do
subject { post :query, params: query_params.merge(group_id: nested_group.to_param), format: :json } subject { post :query, params: params.merge(query_params.merge(group_id: nested_group.to_param)), format: :json }
# The following expectation should be changed to # The following expectation should be changed to
# it_behaves_like '404 status' # it_behaves_like '404 status'
...@@ -100,19 +101,19 @@ describe Groups::InsightsController do ...@@ -100,19 +101,19 @@ describe Groups::InsightsController do
context 'when the configuration is attached to the current group' do context 'when the configuration is attached to the current group' do
describe 'GET #show.html' do describe 'GET #show.html' do
subject { get :show, params: { group_id: parent_group.to_param } } subject { get :show, params: params.merge(group_id: parent_group.to_param) }
it_behaves_like '200 status' it_behaves_like '200 status'
end end
describe 'GET #show.sjon' do describe 'GET #show.sjon' do
subject { get :show, params: { group_id: parent_group.to_param }, format: :json } subject { get :show, params: params.merge(group_id: parent_group.to_param), format: :json }
it_behaves_like '200 status' it_behaves_like '200 status'
end end
describe 'POST #query.json' do describe 'POST #query.json' do
subject { post :query, params: query_params.merge(group_id: parent_group.to_param), format: :json } subject { post :query, params: params.merge(query_params.merge(group_id: parent_group.to_param)), format: :json }
it_behaves_like '200 status' it_behaves_like '200 status'
end end
...@@ -120,19 +121,19 @@ describe Groups::InsightsController do ...@@ -120,19 +121,19 @@ describe Groups::InsightsController do
context 'when the configuration is attached to a nested group' do context 'when the configuration is attached to a nested group' do
describe 'GET #show.html' do describe 'GET #show.html' do
subject { get :show, params: { group_id: nested_group.to_param } } subject { get :show, params: params.merge(group_id: nested_group.to_param) }
it_behaves_like '200 status' it_behaves_like '200 status'
end end
describe 'GET #show.json' do describe 'GET #show.json' do
subject { get :show, params: { group_id: nested_group.to_param }, format: :json } subject { get :show, params: params.merge(group_id: nested_group.to_param), format: :json }
it_behaves_like '200 status' it_behaves_like '200 status'
end end
describe 'POST #query.json' do describe 'POST #query.json' do
subject { post :query, params: query_params.merge(group_id: nested_group.to_param), format: :json } subject { post :query, params: params.merge(query_params.merge(group_id: nested_group.to_param)), format: :json }
it_behaves_like '200 status' it_behaves_like '200 status'
end end
......
...@@ -6,5 +6,6 @@ describe 'Group Insights' do ...@@ -6,5 +6,6 @@ describe 'Group Insights' do
it_behaves_like 'Insights page' do it_behaves_like 'Insights page' do
set(:entity) { create(:group) } set(:entity) { create(:group) }
let(:route) { url_for([entity, :insights]) } let(:route) { url_for([entity, :insights]) }
let(:path) { group_insights_path(entity) }
end end
end end
...@@ -6,5 +6,6 @@ describe 'Project Insights' do ...@@ -6,5 +6,6 @@ describe 'Project Insights' do
it_behaves_like 'Insights page' do it_behaves_like 'Insights page' do
set(:entity) { create(:project) } set(:entity) { create(:project) }
let(:route) { url_for([entity.namespace, entity, :insights]) } let(:route) { url_for([entity.namespace, entity, :insights]) }
let(:path) { project_insights_path(entity) }
end end
end end
import Vue from 'vue'; import Vue from 'vue';
import Insights from 'ee/insights/components/insights.vue'; import Insights from 'ee/insights/components/insights.vue';
import { createStore } from 'ee/insights/stores'; import { createStore } from 'ee/insights/stores';
import { mountComponentWithStore } from 'spec/helpers/vue_mount_component_helper'; import createRouter from 'ee/insights/insights_router';
import { pageInfo } from '../mock_data';
describe('Insights component', () => { describe('Insights component', () => {
let vm; let vm;
let store; let store;
let mountComponent; let mountComponent;
const Component = Vue.extend(Insights); const Component = Vue.extend(Insights);
const router = createRouter('');
beforeEach(() => { beforeEach(() => {
store = createStore(); store = createStore();
spyOn(store, 'dispatch').and.stub(); spyOn(store, 'dispatch').and.stub();
mountComponent = data => { mountComponent = data => {
const el = null;
const props = data || { const props = data || {
endpoint: gl.TEST_HOST, endpoint: gl.TEST_HOST,
queryEndpoint: `${gl.TEST_HOST}/query`, queryEndpoint: `${gl.TEST_HOST}/query`,
}; };
return mountComponentWithStore(Component, { store, props });
return new Component({
store,
router,
propsData: props || {},
}).$mount(el);
}; };
vm = mountComponent(); vm = mountComponent();
...@@ -100,4 +109,41 @@ describe('Insights component', () => { ...@@ -100,4 +109,41 @@ describe('Insights component', () => {
}); });
}); });
}); });
describe('hash fragment present', () => {
const defaultKey = 'issues';
const selectedKey = 'mrs';
const configData = {};
configData[defaultKey] = {};
configData[selectedKey] = {};
beforeEach(() => {
vm.$store.state.insights.configLoading = false;
vm.$store.state.insights.configData = configData;
vm.$store.state.insights.activePage = pageInfo;
});
afterEach(() => {
window.location.hash = '';
});
it('selects the first tab if invalid', done => {
window.location.hash = '#/invalid';
vm.$nextTick(() => {
expect(store.dispatch).toHaveBeenCalledWith('insights/setActiveTab', defaultKey);
});
done();
});
it('selects the specified tab if valid', done => {
window.location.hash = `#/${selectedKey}`;
vm.$nextTick(() => {
expect(store.dispatch).toHaveBeenCalledWith('insights/setActiveTab', selectedKey);
});
done();
});
});
}); });
import createRouter from 'ee/insights/insights_router';
import store from 'ee/insights/stores';
describe('insights router', () => {
let router;
beforeEach(() => {
router = createRouter('base');
});
it(`sets the activeTab when route changed`, () => {
const route = 'route';
spyOn(store, 'dispatch').and.stub();
router.push(`/${route}`);
expect(store.dispatch).toHaveBeenCalledWith('insights/setActiveTab', route);
});
});
...@@ -220,9 +220,13 @@ describe('Insights store actions', () => { ...@@ -220,9 +220,13 @@ describe('Insights store actions', () => {
}); });
describe('setActiveTab', () => { describe('setActiveTab', () => {
it('commits SET_ACTIVE_TAB and SET_ACTIVE_PAGE', done => { let state;
const state = { configData };
beforeEach(() => {
state = { configData };
});
it('commits SET_ACTIVE_TAB and SET_ACTIVE_PAGE', done => {
testAction( testAction(
actions.setActiveTab, actions.setActiveTab,
key, key,
...@@ -232,6 +236,16 @@ describe('Insights store actions', () => { ...@@ -232,6 +236,16 @@ describe('Insights store actions', () => {
done, done,
); );
}); });
it('does not mutate with no configData', done => {
state = { configData: null };
testAction(actions.setActiveTab, key, state, [], [], done);
});
it('does not mutate with no matching tab', done => {
testAction(actions.setActiveTab, 'invalidTab', state, [], [], done);
});
}); });
describe('initChartData', () => { describe('initChartData', () => {
......
...@@ -24,6 +24,26 @@ RSpec.shared_examples 'Insights page' do ...@@ -24,6 +24,26 @@ RSpec.shared_examples 'Insights page' do
expect(page).to have_content('Insights') expect(page).to have_content('Insights')
end end
context 'hash fragment navigation', :js do
let(:config) { entity.insights_config }
let(:non_default_tab_id) { config.keys.last }
let(:non_default_tab_title) { config[non_default_tab_id][:title] }
let(:hash_fragment) { "#/#{non_default_tab_id}" }
let(:route) { path + hash_fragment }
before do
visit route
wait_for_requests
end
it 'loads the correct page' do
page.within(".insights-container") do
expect(page).to have_content(non_default_tab_title)
end
end
end
context 'when the feature flag is disabled globally' do context 'when the feature flag is disabled globally' do
before do before do
stub_feature_flags(insights: false) stub_feature_flags(insights: false)
......
...@@ -15930,6 +15930,9 @@ msgstr "" ...@@ -15930,6 +15930,9 @@ msgstr ""
msgid "The snippet is visible to any logged in user." msgid "The snippet is visible to any logged in user."
msgstr "" msgstr ""
msgid "The specified tab is invalid, please select another"
msgstr ""
msgid "The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time." msgid "The staging stage shows the time between merging the MR and deploying code to the production environment. The data will be automatically added once you deploy to production for the first time."
msgstr "" msgstr ""
......
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