Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Create dashboard drilldown - select dashboard via filtering combo box and navigate to dashboarrd #60087

Closed
Closed
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
50 commits
Select commit Hold shift + click to select a range
813a896
partial progress on async loading / searching of dashboard titles
mattkime Mar 13, 2020
65ff148
feat: 🎸 make combobox full width
streamich Mar 13, 2020
d46fcf0
filtering combobox polish
mattkime Mar 16, 2020
6e11bc2
Merge branch 'drilldowns_load_dashboard_list' of github.com:mattkime/…
mattkime Mar 16, 2020
699d853
storybook fix
mattkime Mar 16, 2020
d3ac7ad
implement navigating to dashboard, seems like a type problem
mattkime Mar 17, 2020
a5ee274
Merge branch 'drilldowns' into drilldowns_load_dashboard_list
mattkime Mar 17, 2020
4369399
try navToApp
mattkime Mar 17, 2020
94b08ae
Merge branch 'drilldowns' into drilldowns_load_dashboard_list
mattkime Mar 17, 2020
86f9768
Merge remote-tracking branch 'upstream/drilldowns' into drilldowns_lo…
mattkime Mar 17, 2020
a276b96
filter out current dashboard
mattkime Mar 17, 2020
a112732
Merge branch 'drilldowns' into drilldowns_load_dashboard_list
mattkime Mar 17, 2020
3b39dcd
rough draft linking to a dashboard
mattkime Mar 18, 2020
0bbf360
remove note
mattkime Mar 18, 2020
636d9ab
typefix
mattkime Mar 18, 2020
6ea19aa
fix navigation from dashboard to dashboard
Dosant Mar 18, 2020
0178ff5
Merge branch 'drilldowns' into drilldowns_load_dashboard_list
mattkime Mar 19, 2020
a431e86
partial progress getting filters from action data
mattkime Mar 20, 2020
4217b18
fix issue with getIndexPatterns undefined
Dosant Mar 20, 2020
177822f
fix filter / time passing into url
Dosant Mar 20, 2020
8bcbc1b
Merge branch 'drilldowns' into drilldowns_load_dashboard_list
mattkime Mar 21, 2020
f7b7add
typefix
mattkime Mar 21, 2020
346dc40
Merge branch 'drilldowns_load_dashboard_list' of github.com:mattkime/…
mattkime Mar 21, 2020
bd7a91c
dashboard to dashboard drilldown functional test and back button fix
Dosant Mar 21, 2020
5ea7818
documentation update
mattkime Mar 21, 2020
8297b55
Merge branch 'drilldowns' into drilldowns_load_dashboard_list
mattkime Mar 22, 2020
feccdfc
Merge branch 'drilldowns' into drilldowns_load_dashboard_list
mattkime Mar 22, 2020
29e885c
Merge branch 'drilldowns' of github.com:elastic/kibana into dev/drill…
Dosant Mar 23, 2020
100c4f0
chore clean-ups
Dosant Mar 23, 2020
340fd89
basic unit test for dashboard drilldown
Dosant Mar 23, 2020
406facd
remove test todos
Dosant Mar 23, 2020
d894574
Merge branch 'drilldowns' of github.com:elastic/kibana into dev/drill…
Dosant Mar 23, 2020
bf9e505
remove config
Dosant Mar 23, 2020
a24e29a
Merge branch 'drilldowns' into drilldowns_load_dashboard_list
mattkime Mar 23, 2020
d7633b3
improve back button with filter comparison tweak
Dosant Mar 23, 2020
59101ef
dashboard filters/date option off by default
Dosant Mar 23, 2020
9fdb358
Merge branch 'drilldowns_load_dashboard_list' of github.com:mattkime/…
mattkime Mar 23, 2020
bb308d6
revert change to config/kibana.yml
mattkime Mar 23, 2020
8d10133
Merge branch 'drilldowns' into drilldowns_load_dashboard_list
streamich Mar 23, 2020
31e28a3
remove unneeded comments
mattkime Mar 23, 2020
38ec7a0
use default time range as appropriate
mattkime Mar 24, 2020
c3708ca
Merge branch 'drilldowns' into drilldowns_load_dashboard_list
mattkime Mar 24, 2020
4a8d4a1
fix type, add filter icon, add text
mattkime Mar 24, 2020
0deb6b4
fix test
mattkime Mar 24, 2020
6a78486
change how time range is restored and improve back button for drilldowns
Dosant Mar 24, 2020
bc3fdd4
Merge branch 'drilldowns' into drilldowns_load_dashboard_list
mattkime Apr 3, 2020
8deeaa9
Merge branch 'drilldowns' of github.com:elastic/kibana into drilldown…
Dosant Apr 7, 2020
514561b
resolve conflicts
Dosant Apr 7, 2020
40e396c
fix async compile issue
mattkime Apr 8, 2020
a9b86a9
remove redundant test
Dosant Apr 8, 2020
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -9,5 +9,7 @@
```typescript
actions: {
createFiltersFromEvent: typeof createFiltersFromEvent;
valueClickActionGetFilters: typeof valueClickActionGetFilters;
selectRangeActionGetFilters: typeof selectRangeActionGetFilters;
};
```
Original file line number Diff line number Diff line change
Expand Up @@ -14,7 +14,7 @@ export interface DataPublicPluginStart

| Property | Type | Description |
| --- | --- | --- |
| [actions](./kibana-plugin-plugins-data-public.datapublicpluginstart.actions.md) | <code>{</code><br/><code> createFiltersFromEvent: typeof createFiltersFromEvent;</code><br/><code> }</code> | |
| [actions](./kibana-plugin-plugins-data-public.datapublicpluginstart.actions.md) | <code>{</code><br/><code> createFiltersFromEvent: typeof createFiltersFromEvent;</code><br/><code> valueClickActionGetFilters: typeof valueClickActionGetFilters;</code><br/><code> selectRangeActionGetFilters: typeof selectRangeActionGetFilters;</code><br/><code> }</code> | |
| [autocomplete](./kibana-plugin-plugins-data-public.datapublicpluginstart.autocomplete.md) | <code>AutocompleteStart</code> | |
| [fieldFormats](./kibana-plugin-plugins-data-public.datapublicpluginstart.fieldformats.md) | <code>FieldFormatsStart</code> | |
| [indexPatterns](./kibana-plugin-plugins-data-public.datapublicpluginstart.indexpatterns.md) | <code>IndexPatternsContract</code> | |
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -49,6 +49,7 @@ esFilters: {
generateFilters: typeof generateFilters;
onlyDisabledFiltersChanged: (newFilters?: import("../common").Filter[] | undefined, oldFilters?: import("../common").Filter[] | undefined) => boolean;
changeTimeFilter: typeof changeTimeFilter;
convertRangeFilterToTimeRangeString: typeof convertRangeFilterToTimeRangeString;
mapAndFlattenFilters: (filters: import("../common").Filter[]) => import("../common").Filter[];
extractTimeFilter: typeof extractTimeFilter;
}
Expand Down
2 changes: 1 addition & 1 deletion src/legacy/core_plugins/vis_type_vislib/public/plugin.ts
Original file line number Diff line number Diff line change
Expand Up @@ -111,6 +111,6 @@ export class VisTypeVislibPlugin implements Plugin<void, void> {

public start(core: CoreStart, { data }: VisTypeVislibPluginStartDependencies) {
setFormatService(data.fieldFormats);
setDataActions({ createFiltersFromEvent: data.actions.createFiltersFromEvent });
setDataActions(data.actions);
}
}
1 change: 1 addition & 0 deletions src/plugins/data/public/actions/apply_filter_action.ts
Original file line number Diff line number Diff line change
Expand Up @@ -42,6 +42,7 @@ export function createFilterAction(
return createAction<typeof ACTION_GLOBAL_APPLY_FILTER>({
type: ACTION_GLOBAL_APPLY_FILTER,
id: ACTION_GLOBAL_APPLY_FILTER,
getIconType: () => 'filter',
getDisplayName: () => {
return i18n.translate('data.filter.applyFilterActionTitle', {
defaultMessage: 'Apply filter to current view',
Expand Down
4 changes: 2 additions & 2 deletions src/plugins/data/public/actions/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -19,5 +19,5 @@

export { ACTION_GLOBAL_APPLY_FILTER, createFilterAction } from './apply_filter_action';
export { createFiltersFromEvent } from './filters/create_filters_from_event';
export { selectRangeAction } from './select_range_action';
export { valueClickAction } from './value_click_action';
export { selectRangeAction, selectRangeActionGetFilters } from './select_range_action';
export { valueClickAction, valueClickActionGetFilters } from './value_click_action';
60 changes: 34 additions & 26 deletions src/plugins/data/public/actions/select_range_action.ts
Original file line number Diff line number Diff line change
Expand Up @@ -41,44 +41,52 @@ async function isCompatible(context: SelectRangeActionContext) {
}
}

export const selectRangeActionGetFilters = async ({
timeFieldName,
data,
}: SelectRangeActionContext) => {
if (!(await isCompatible({ timeFieldName, data }))) {
throw new IncompatibleActionError();
}

const filter = await onBrushEvent(data);

if (!filter) {
return;
}

const selectedFilters = esFilters.mapAndFlattenFilters([filter]);

return esFilters.extractTimeFilter(timeFieldName || '', selectedFilters);
};

const selectRangeActionExecute = (
filterManager: FilterManager,
timeFilter: TimefilterContract
) => async ({ timeFieldName, data }: SelectRangeActionContext) => {
const { timeRangeFilter, restOfFilters } =
(await selectRangeActionGetFilters({ timeFieldName, data })) || {};

filterManager.addFilters(restOfFilters || []);
if (timeRangeFilter) {
esFilters.changeTimeFilter(timeFilter, timeRangeFilter);
}
};

export function selectRangeAction(
filterManager: FilterManager,
timeFilter: TimefilterContract
): ActionByType<typeof ACTION_SELECT_RANGE> {
return createAction<typeof ACTION_SELECT_RANGE>({
type: ACTION_SELECT_RANGE,
id: ACTION_SELECT_RANGE,
getIconType: () => 'filter',
getDisplayName: () => {
return i18n.translate('data.filter.applyFilterActionTitle', {
defaultMessage: 'Apply filter to current view',
});
},
isCompatible,
execute: async ({ timeFieldName, data }: SelectRangeActionContext) => {
if (!(await isCompatible({ timeFieldName, data }))) {
throw new IncompatibleActionError();
}

const filter = await onBrushEvent(data);

if (!filter) {
return;
}

const selectedFilters = esFilters.mapAndFlattenFilters([filter]);

if (timeFieldName) {
const { timeRangeFilter, restOfFilters } = esFilters.extractTimeFilter(
timeFieldName,
selectedFilters
);
filterManager.addFilters(restOfFilters);
if (timeRangeFilter) {
esFilters.changeTimeFilter(timeFilter, timeRangeFilter);
}
} else {
filterManager.addFilters(selectedFilters);
}
},
execute: selectRangeActionExecute(filterManager, timeFilter),
});
}
134 changes: 79 additions & 55 deletions src/plugins/data/public/actions/value_click_action.ts
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,7 @@
*/

import { i18n } from '@kbn/i18n';
import { RangeFilter } from 'src/plugins/data/public';
import { toMountPoint } from '../../../../plugins/kibana_react/public';
import {
ActionByType,
Expand Down Expand Up @@ -47,73 +48,96 @@ async function isCompatible(context: ValueClickActionContext) {
}
}

// this allows the user to select which filter to use
const filterSelectionFn = (filters: Filter[]): Promise<Filter[]> =>
new Promise(async resolve => {
const indexPatterns = await Promise.all(
filters.map(filter => {
return getIndexPatterns().get(filter.meta.index!);
})
);

const overlay = getOverlays().openModal(
toMountPoint(
applyFiltersPopover(
filters,
indexPatterns,
() => {
overlay.close();
resolve([]);
},
(filterSelection: Filter[]) => {
overlay.close();
resolve(filterSelection);
}
)
),
{
'data-test-subj': 'selectFilterOverlay',
}
);
});

// given a ValueClickActionContext, returns timeRangeFilter and Filters
export const valueClickActionGetFilters = async (
{ timeFieldName, data }: ValueClickActionContext,
filterSelection: (filters: Filter[]) => Promise<Filter[]> = async (filters: Filter[]) => filters
): Promise<{
timeRangeFilter?: RangeFilter;
restOfFilters: Filter[];
}> => {
if (!(await isCompatible({ timeFieldName, data }))) {
throw new IncompatibleActionError();
// note - the else statement is necessary due to odd compilation behavior
// code after the throw statement can't contain await statements
} else {
const filters: Filter[] =
(await createFiltersFromEvent(data.data || [data], data.negate)) || [];

let selectedFilters: Filter[] = esFilters.mapAndFlattenFilters(filters);

if (selectedFilters.length > 1) {
selectedFilters = await filterSelection(filters);
}

return esFilters.extractTimeFilter(timeFieldName || '', selectedFilters);
}
};

// gets and applies the filters
export const valueClickActionExecute = (
filterManager: FilterManager,
timeFilter: TimefilterContract,
filterSelection: (filters: Filter[]) => Promise<Filter[]> = async (filters: Filter[]) => filters
) => async ({ timeFieldName, data }: ValueClickActionContext) => {
const { timeRangeFilter, restOfFilters } = await valueClickActionGetFilters(
{
timeFieldName,
data,
},
filterSelection
);

filterManager.addFilters(restOfFilters);
if (timeRangeFilter) {
esFilters.changeTimeFilter(timeFilter, timeRangeFilter);
}
};

export function valueClickAction(
filterManager: FilterManager,
timeFilter: TimefilterContract
): ActionByType<typeof ACTION_VALUE_CLICK> {
return createAction<typeof ACTION_VALUE_CLICK>({
type: ACTION_VALUE_CLICK,
id: ACTION_VALUE_CLICK,
getIconType: () => 'filter',
getDisplayName: () => {
return i18n.translate('data.filter.applyFilterActionTitle', {
defaultMessage: 'Apply filter to current view',
});
},
isCompatible,
execute: async ({ timeFieldName, data }: ValueClickActionContext) => {
if (!(await isCompatible({ timeFieldName, data }))) {
throw new IncompatibleActionError();
}

const filters: Filter[] =
(await createFiltersFromEvent(data.data || [data], data.negate)) || [];

let selectedFilters: Filter[] = esFilters.mapAndFlattenFilters(filters);

if (selectedFilters.length > 1) {
const indexPatterns = await Promise.all(
filters.map(filter => {
return getIndexPatterns().get(filter.meta.index!);
})
);

const filterSelectionPromise: Promise<Filter[]> = new Promise(resolve => {
const overlay = getOverlays().openModal(
toMountPoint(
applyFiltersPopover(
filters,
indexPatterns,
() => {
overlay.close();
resolve([]);
},
(filterSelection: Filter[]) => {
overlay.close();
resolve(filterSelection);
}
)
),
{
'data-test-subj': 'selectFilterOverlay',
}
);
});

selectedFilters = await filterSelectionPromise;
}

if (timeFieldName) {
const { timeRangeFilter, restOfFilters } = esFilters.extractTimeFilter(
timeFieldName,
selectedFilters
);
filterManager.addFilters(restOfFilters);
if (timeRangeFilter) {
esFilters.changeTimeFilter(timeFilter, timeRangeFilter);
}
} else {
filterManager.addFilters(selectedFilters);
}
},
execute: valueClickActionExecute(filterManager, timeFilter, filterSelectionFn),
});
}
2 changes: 2 additions & 0 deletions src/plugins/data/public/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -59,6 +59,7 @@ import {
changeTimeFilter,
mapAndFlattenFilters,
extractTimeFilter,
convertRangeFilterToTimeRangeString,
} from './query';

// Filter helpers namespace:
Expand Down Expand Up @@ -96,6 +97,7 @@ export const esFilters = {
onlyDisabledFiltersChanged,

changeTimeFilter,
convertRangeFilterToTimeRangeString,
mapAndFlattenFilters,
extractTimeFilter,
};
Expand Down
2 changes: 2 additions & 0 deletions src/plugins/data/public/mocks.ts
Original file line number Diff line number Diff line change
Expand Up @@ -46,6 +46,8 @@ const createStartContract = (): Start => {
return {
actions: {
createFiltersFromEvent: jest.fn().mockResolvedValue(['yes']),
selectRangeActionGetFilters: jest.fn(),
valueClickActionGetFilters: jest.fn(),
},
autocomplete: autocompleteMock,
search: searchStartMock,
Expand Down
10 changes: 9 additions & 1 deletion src/plugins/data/public/plugin.ts
Original file line number Diff line number Diff line change
Expand Up @@ -55,7 +55,13 @@ import {
VALUE_CLICK_TRIGGER,
APPLY_FILTER_TRIGGER,
} from '../../ui_actions/public';
import { ACTION_GLOBAL_APPLY_FILTER, createFilterAction, createFiltersFromEvent } from './actions';
import {
ACTION_GLOBAL_APPLY_FILTER,
createFilterAction,
createFiltersFromEvent,
selectRangeActionGetFilters,
valueClickActionGetFilters,
} from './actions';
import { ApplyGlobalFilterActionContext } from './actions/apply_filter_action';
import {
selectRangeAction,
Expand Down Expand Up @@ -157,6 +163,8 @@ export class DataPublicPlugin implements Plugin<DataPublicPluginSetup, DataPubli
const dataServices = {
actions: {
createFiltersFromEvent,
selectRangeActionGetFilters,
valueClickActionGetFilters,
},
autocomplete: this.autocomplete.start(),
fieldFormats,
Expand Down
Loading