[Canvas] Top Menu (#59982)
* Redesigned workpad_header to top menu layout * Added comments * Fixed element spec * Removed element preview images * Removed tooltip from menus * Removed extraneous JSX * Fixed element fixtures * Moved component strings * top menu design tweaks * Added filter debug element * Fix file picker in asset manager * Sort components strings object keys * Removed ElementTypes component in favor of SavedElementsModal * Updated stories * Fixed custom elements functional tests * Removed unused tag strings * Fixed test fixtures * Updated element_menu stories * Updated view_menu stories * TS for SavedElementsModal * Updated types * Fixed TS errors * Fix i18n errors * Renamed stories * Fixed test file name * Fixed stories * Updated storyshots * Reverted storybook webpack config change * Fixed SavedElementsModal stories * Updated comments * Removed unnecessary ts-ignores * Moved workpad_shortcuts back to /components * Unskip custom elements functional test * Reverted workpad_loader changes * Added element_menu stories and mocks * Fixed element i18n strings * Updated storyshots * Updated storyshot Co-authored-by: Ryan Keairns <contactryank@gmail.com> Co-authored-by: Elastic Machine <elasticmachine@users.noreply.github.com>
|
@ -7,6 +7,7 @@
|
||||||
import path from 'path';
|
import path from 'path';
|
||||||
import moment from 'moment';
|
import moment from 'moment';
|
||||||
import 'moment-timezone';
|
import 'moment-timezone';
|
||||||
|
import ReactDOM from "react-dom";
|
||||||
|
|
||||||
import initStoryshots, { multiSnapshotWithOptions } from '@storybook/addon-storyshots';
|
import initStoryshots, { multiSnapshotWithOptions } from '@storybook/addon-storyshots';
|
||||||
import styleSheetSerializer from 'jest-styled-components/src/styleSheetSerializer';
|
import styleSheetSerializer from 'jest-styled-components/src/styleSheetSerializer';
|
||||||
|
@ -24,6 +25,9 @@ moment.tz.setDefault('UTC');
|
||||||
const testTime = new Date(Date.UTC(2019, 5, 1)); // June 1 2019
|
const testTime = new Date(Date.UTC(2019, 5, 1)); // June 1 2019
|
||||||
Date.now = jest.fn(() => testTime);
|
Date.now = jest.fn(() => testTime);
|
||||||
|
|
||||||
|
// Mock telemetry service
|
||||||
|
jest.mock('../public/lib/ui_metric', () => ({ trackCanvasUiMetric: () => { } }));
|
||||||
|
|
||||||
// Mock EUI generated ids to be consistently predictable for snapshots.
|
// Mock EUI generated ids to be consistently predictable for snapshots.
|
||||||
jest.mock(`@elastic/eui/lib/components/form/form_row/make_id`, () => () => `generated-id`);
|
jest.mock(`@elastic/eui/lib/components/form/form_row/make_id`, () => () => `generated-id`);
|
||||||
|
|
||||||
|
@ -32,7 +36,7 @@ jest.mock(`@elastic/eui/lib/components/form/form_row/make_id`, () => () => `gene
|
||||||
jest.mock('@elastic/eui/lib/components/code/code', () => {
|
jest.mock('@elastic/eui/lib/components/code/code', () => {
|
||||||
const React = require.requireActual('react');
|
const React = require.requireActual('react');
|
||||||
return {
|
return {
|
||||||
EuiCode: ({children, className}) => (
|
EuiCode: ({ children, className }) => (
|
||||||
<span>
|
<span>
|
||||||
<code>{children}</code>
|
<code>{children}</code>
|
||||||
</span>
|
</span>
|
||||||
|
@ -61,6 +65,12 @@ jest.mock('@elastic/eui/packages/react-datepicker', () => {
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
||||||
|
// Mock React Portal for components that use modals, tooltips, etc
|
||||||
|
ReactDOM.createPortal = jest.fn((element) => {
|
||||||
|
return element;
|
||||||
|
});
|
||||||
|
|
||||||
jest.mock('@elastic/eui/lib/services/accessibility/html_id_generator', () => {
|
jest.mock('@elastic/eui/lib/services/accessibility/html_id_generator', () => {
|
||||||
return {
|
return {
|
||||||
htmlIdGenerator: () => () => `generated-id`,
|
htmlIdGenerator: () => () => `generated-id`,
|
||||||
|
@ -71,7 +81,7 @@ jest.mock('plugins/interpreter/registries', () => ({}));
|
||||||
|
|
||||||
// Disabling this test due to https://github.com/elastic/eui/issues/2242
|
// Disabling this test due to https://github.com/elastic/eui/issues/2242
|
||||||
jest.mock(
|
jest.mock(
|
||||||
'../public/components/workpad_header/workpad_export/flyout/__examples__/share_website_flyout.stories',
|
'../public/components/workpad_header/share_menu/flyout/__examples__/share_website_flyout.stories',
|
||||||
() => {
|
() => {
|
||||||
return 'Disabled Panel';
|
return 'Disabled Panel';
|
||||||
}
|
}
|
||||||
|
|
|
@ -177,8 +177,10 @@ module.exports = async ({ config }) => {
|
||||||
}),
|
}),
|
||||||
|
|
||||||
// Mock out libs used by a few componets to avoid loading in kibana_legacy and platform
|
// Mock out libs used by a few componets to avoid loading in kibana_legacy and platform
|
||||||
new webpack.NormalModuleReplacementPlugin(/lib\/notify/, path.resolve(__dirname, '../tasks/mocks/uiNotify')),
|
new webpack.NormalModuleReplacementPlugin(/(lib)?\/notify/, path.resolve(__dirname, '../tasks/mocks/uiNotify')),
|
||||||
new webpack.NormalModuleReplacementPlugin(/lib\/download_workpad/, path.resolve(__dirname, '../tasks/mocks/downloadWorkpad')),
|
new webpack.NormalModuleReplacementPlugin(/lib\/download_workpad/, path.resolve(__dirname, '../tasks/mocks/downloadWorkpad')),
|
||||||
|
new webpack.NormalModuleReplacementPlugin(/(lib)?\/custom_element_service/, path.resolve(__dirname, '../tasks/mocks/customElementService')),
|
||||||
|
new webpack.NormalModuleReplacementPlugin(/(lib)?\/ui_metric/, path.resolve(__dirname, '../tasks/mocks/uiMetric')),
|
||||||
);
|
);
|
||||||
|
|
||||||
// Tell Webpack about relevant extensions
|
// Tell Webpack about relevant extensions
|
||||||
|
|
Before Width: | Height: | Size: 149 KiB |
|
@ -5,14 +5,13 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import { ElementFactory } from '../../../types';
|
import { ElementFactory } from '../../../types';
|
||||||
import header from './header.png';
|
|
||||||
|
|
||||||
export const areaChart: ElementFactory = () => ({
|
export const areaChart: ElementFactory = () => ({
|
||||||
name: 'areaChart',
|
name: 'areaChart',
|
||||||
displayName: 'Area chart',
|
displayName: 'Area',
|
||||||
help: 'A line chart with a filled body',
|
help: 'A line chart with a filled body',
|
||||||
tags: ['chart'],
|
type: 'chart',
|
||||||
image: header,
|
icon: 'visArea',
|
||||||
expression: `filters
|
expression: `filters
|
||||||
| demodata
|
| demodata
|
||||||
| pointseries x="time" y="mean(price)"
|
| pointseries x="time" y="mean(price)"
|
||||||
|
|
Before Width: | Height: | Size: 67 KiB |
|
@ -5,16 +5,15 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import { ElementFactory } from '../../../types';
|
import { ElementFactory } from '../../../types';
|
||||||
import header from './header.png';
|
|
||||||
|
|
||||||
export const bubbleChart: ElementFactory = () => ({
|
export const bubbleChart: ElementFactory = () => ({
|
||||||
name: 'bubbleChart',
|
name: 'bubbleChart',
|
||||||
displayName: 'Bubble chart',
|
displayName: 'Bubble',
|
||||||
tags: ['chart'],
|
type: 'chart',
|
||||||
help: 'A customizable bubble chart',
|
help: 'A customizable bubble chart',
|
||||||
width: 700,
|
width: 700,
|
||||||
height: 300,
|
height: 300,
|
||||||
image: header,
|
icon: 'heatmap',
|
||||||
expression: `filters
|
expression: `filters
|
||||||
| demodata
|
| demodata
|
||||||
| pointseries x="project" y="sum(price)" color="state" size="size(username)"
|
| pointseries x="project" y="sum(price)" color="state" size="size(username)"
|
||||||
|
|
Before Width: | Height: | Size: 15 KiB |
|
@ -5,14 +5,12 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import { ElementFactory } from '../../../types';
|
import { ElementFactory } from '../../../types';
|
||||||
import header from './header.png';
|
|
||||||
|
|
||||||
export const debug: ElementFactory = () => ({
|
export const debug: ElementFactory = () => ({
|
||||||
name: 'debug',
|
name: 'debug',
|
||||||
displayName: 'Debug',
|
displayName: 'Debug data',
|
||||||
tags: ['text'],
|
|
||||||
help: 'Just dumps the configuration of the element',
|
help: 'Just dumps the configuration of the element',
|
||||||
image: header,
|
icon: 'bug',
|
||||||
expression: `demodata
|
expression: `demodata
|
||||||
| render as=debug`,
|
| render as=debug`,
|
||||||
});
|
});
|
||||||
|
|
Before Width: | Height: | Size: 32 KiB |
|
@ -1,21 +0,0 @@
|
||||||
/*
|
|
||||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
|
||||||
* or more contributor license agreements. Licensed under the Elastic License;
|
|
||||||
* you may not use this file except in compliance with the Elastic License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
import { ElementFactory } from '../../../types';
|
|
||||||
import header from './header.png';
|
|
||||||
|
|
||||||
export const donut: ElementFactory = () => ({
|
|
||||||
name: 'donut',
|
|
||||||
displayName: 'Donut chart',
|
|
||||||
tags: ['chart', 'proportion'],
|
|
||||||
help: 'A customizable donut chart',
|
|
||||||
image: header,
|
|
||||||
expression: `filters
|
|
||||||
| demodata
|
|
||||||
| pointseries color="project" size="max(price)"
|
|
||||||
| pie hole=50 labels=false legend="ne"
|
|
||||||
| render`,
|
|
||||||
});
|
|
Before Width: | Height: | Size: 16 KiB |
|
@ -5,14 +5,13 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import { ElementFactory } from '../../../types';
|
import { ElementFactory } from '../../../types';
|
||||||
import header from './header.png';
|
|
||||||
|
|
||||||
export const dropdownFilter: ElementFactory = () => ({
|
export const dropdownFilter: ElementFactory = () => ({
|
||||||
name: 'dropdown_filter',
|
name: 'dropdownFilter',
|
||||||
displayName: 'Dropdown filter',
|
displayName: 'Dropdown select',
|
||||||
tags: ['filter'],
|
type: 'filter',
|
||||||
help: 'A dropdown from which you can select values for an "exactly" filter',
|
help: 'A dropdown from which you can select values for an "exactly" filter',
|
||||||
image: header,
|
icon: 'filter',
|
||||||
height: 50,
|
height: 50,
|
||||||
expression: `demodata
|
expression: `demodata
|
||||||
| dropdownControl valueColumn=project filterColumn=project | render`,
|
| dropdownControl valueColumn=project filterColumn=project | render`,
|
||||||
|
|
|
@ -0,0 +1,16 @@
|
||||||
|
/*
|
||||||
|
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||||
|
* or more contributor license agreements. Licensed under the Elastic License;
|
||||||
|
* you may not use this file except in compliance with the Elastic License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { ElementFactory } from '../../../types';
|
||||||
|
|
||||||
|
export const filterDebug: ElementFactory = () => ({
|
||||||
|
name: 'filterDebug',
|
||||||
|
displayName: 'Debug filter',
|
||||||
|
help: 'Shows the underlying global filters in a workpad',
|
||||||
|
icon: 'bug',
|
||||||
|
expression: `filters
|
||||||
|
| render as=debug`,
|
||||||
|
});
|
Before Width: | Height: | Size: 19 KiB |
|
@ -5,14 +5,13 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import { ElementFactory } from '../../../types';
|
import { ElementFactory } from '../../../types';
|
||||||
import header from './header.png';
|
|
||||||
|
|
||||||
export const horizontalBarChart: ElementFactory = () => ({
|
export const horizontalBarChart: ElementFactory = () => ({
|
||||||
name: 'horizontalBarChart',
|
name: 'horizontalBarChart',
|
||||||
displayName: 'Horizontal bar chart',
|
displayName: 'Bar horizontal',
|
||||||
tags: ['chart'],
|
type: 'chart',
|
||||||
help: 'A customizable horizontal bar chart',
|
help: 'A customizable horizontal bar chart',
|
||||||
image: header,
|
icon: 'visBarHorizontal',
|
||||||
expression: `filters
|
expression: `filters
|
||||||
| demodata
|
| demodata
|
||||||
| pointseries x="size(cost)" y="project" color="project"
|
| pointseries x="size(cost)" y="project" color="project"
|
||||||
|
|
Before Width: | Height: | Size: 12 KiB |
|
@ -6,16 +6,14 @@
|
||||||
|
|
||||||
import { openSans } from '../../../common/lib/fonts';
|
import { openSans } from '../../../common/lib/fonts';
|
||||||
import { ElementFactory } from '../../../types';
|
import { ElementFactory } from '../../../types';
|
||||||
import header from './header.png';
|
|
||||||
|
|
||||||
export const horizontalProgressBar: ElementFactory = () => ({
|
export const horizontalProgressBar: ElementFactory = () => ({
|
||||||
name: 'horizontalProgressBar',
|
name: 'horizontalProgressBar',
|
||||||
displayName: 'Horizontal progress bar',
|
displayName: 'Horizontal bar',
|
||||||
tags: ['chart', 'proportion'],
|
type: 'progress',
|
||||||
help: 'Displays progress as a portion of a horizontal bar',
|
help: 'Displays progress as a portion of a horizontal bar',
|
||||||
width: 400,
|
width: 400,
|
||||||
height: 30,
|
height: 30,
|
||||||
image: header,
|
|
||||||
expression: `filters
|
expression: `filters
|
||||||
| demodata
|
| demodata
|
||||||
| math "mean(percent_uptime)"
|
| math "mean(percent_uptime)"
|
||||||
|
|
Before Width: | Height: | Size: 19 KiB |
|
@ -6,16 +6,14 @@
|
||||||
|
|
||||||
import { openSans } from '../../../common/lib/fonts';
|
import { openSans } from '../../../common/lib/fonts';
|
||||||
import { ElementFactory } from '../../../types';
|
import { ElementFactory } from '../../../types';
|
||||||
import header from './header.png';
|
|
||||||
|
|
||||||
export const horizontalProgressPill: ElementFactory = () => ({
|
export const horizontalProgressPill: ElementFactory = () => ({
|
||||||
name: 'horizontalProgressPill',
|
name: 'horizontalProgressPill',
|
||||||
displayName: 'Horizontal progress pill',
|
displayName: 'Horizontal pill',
|
||||||
tags: ['chart', 'proportion'],
|
type: 'progress',
|
||||||
help: 'Displays progress as a portion of a horizontal pill',
|
help: 'Displays progress as a portion of a horizontal pill',
|
||||||
width: 400,
|
width: 400,
|
||||||
height: 30,
|
height: 30,
|
||||||
image: header,
|
|
||||||
expression: `filters
|
expression: `filters
|
||||||
| demodata
|
| demodata
|
||||||
| math "mean(percent_uptime)"
|
| math "mean(percent_uptime)"
|
||||||
|
|
Before Width: | Height: | Size: 91 KiB |
|
@ -5,14 +5,13 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import { ElementFactory } from '../../../types';
|
import { ElementFactory } from '../../../types';
|
||||||
import header from './header.png';
|
|
||||||
|
|
||||||
export const image: ElementFactory = () => ({
|
export const image: ElementFactory = () => ({
|
||||||
name: 'image',
|
name: 'image',
|
||||||
displayName: 'Image',
|
displayName: 'Image',
|
||||||
tags: ['graphic'],
|
type: 'image',
|
||||||
help: 'A static image',
|
help: 'A static image',
|
||||||
image: header,
|
icon: 'image',
|
||||||
expression: `image dataurl=null mode="contain"
|
expression: `image dataurl=null mode="contain"
|
||||||
| render`,
|
| render`,
|
||||||
});
|
});
|
||||||
|
|
|
@ -8,8 +8,8 @@ import { applyElementStrings } from '../../i18n/elements';
|
||||||
import { areaChart } from './area_chart';
|
import { areaChart } from './area_chart';
|
||||||
import { bubbleChart } from './bubble_chart';
|
import { bubbleChart } from './bubble_chart';
|
||||||
import { debug } from './debug';
|
import { debug } from './debug';
|
||||||
import { donut } from './donut';
|
|
||||||
import { dropdownFilter } from './dropdown_filter';
|
import { dropdownFilter } from './dropdown_filter';
|
||||||
|
import { filterDebug } from './filter_debug';
|
||||||
import { horizontalBarChart } from './horizontal_bar_chart';
|
import { horizontalBarChart } from './horizontal_bar_chart';
|
||||||
import { horizontalProgressBar } from './horizontal_progress_bar';
|
import { horizontalProgressBar } from './horizontal_progress_bar';
|
||||||
import { horizontalProgressPill } from './horizontal_progress_pill';
|
import { horizontalProgressPill } from './horizontal_progress_pill';
|
||||||
|
@ -26,7 +26,6 @@ import { repeatImage } from './repeat_image';
|
||||||
import { revealImage } from './reveal_image';
|
import { revealImage } from './reveal_image';
|
||||||
import { shape } from './shape';
|
import { shape } from './shape';
|
||||||
import { table } from './table';
|
import { table } from './table';
|
||||||
import { tiltedPie } from './tilted_pie';
|
|
||||||
import { timeFilter } from './time_filter';
|
import { timeFilter } from './time_filter';
|
||||||
import { verticalBarChart } from './vert_bar_chart';
|
import { verticalBarChart } from './vert_bar_chart';
|
||||||
import { verticalProgressBar } from './vertical_progress_bar';
|
import { verticalProgressBar } from './vertical_progress_bar';
|
||||||
|
@ -39,8 +38,8 @@ const elementSpecs = [
|
||||||
areaChart,
|
areaChart,
|
||||||
bubbleChart,
|
bubbleChart,
|
||||||
debug,
|
debug,
|
||||||
donut,
|
|
||||||
dropdownFilter,
|
dropdownFilter,
|
||||||
|
filterDebug,
|
||||||
image,
|
image,
|
||||||
horizontalBarChart,
|
horizontalBarChart,
|
||||||
horizontalProgressBar,
|
horizontalProgressBar,
|
||||||
|
@ -56,7 +55,6 @@ const elementSpecs = [
|
||||||
revealImage,
|
revealImage,
|
||||||
shape,
|
shape,
|
||||||
table,
|
table,
|
||||||
tiltedPie,
|
|
||||||
timeFilter,
|
timeFilter,
|
||||||
verticalBarChart,
|
verticalBarChart,
|
||||||
verticalProgressBar,
|
verticalProgressBar,
|
||||||
|
|
Before Width: | Height: | Size: 58 KiB |
|
@ -5,14 +5,13 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import { ElementFactory } from '../../../types';
|
import { ElementFactory } from '../../../types';
|
||||||
import header from './header.png';
|
|
||||||
|
|
||||||
export const lineChart: ElementFactory = () => ({
|
export const lineChart: ElementFactory = () => ({
|
||||||
name: 'lineChart',
|
name: 'lineChart',
|
||||||
displayName: 'Line chart',
|
displayName: 'Line',
|
||||||
tags: ['chart'],
|
type: 'chart',
|
||||||
help: 'A customizable line chart',
|
help: 'A customizable line chart',
|
||||||
image: header,
|
icon: 'visLine',
|
||||||
expression: `filters
|
expression: `filters
|
||||||
| demodata
|
| demodata
|
||||||
| pointseries x="time" y="mean(price)"
|
| pointseries x="time" y="mean(price)"
|
||||||
|
|
Before Width: | Height: | Size: 33 KiB |
|
@ -4,15 +4,13 @@
|
||||||
* you may not use this file except in compliance with the Elastic License.
|
* you may not use this file except in compliance with the Elastic License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import header from './header.png';
|
|
||||||
|
|
||||||
import { ElementFactory } from '../../../types';
|
import { ElementFactory } from '../../../types';
|
||||||
export const markdown: ElementFactory = () => ({
|
export const markdown: ElementFactory = () => ({
|
||||||
name: 'markdown',
|
name: 'markdown',
|
||||||
displayName: 'Markdown',
|
displayName: 'Text',
|
||||||
tags: ['text'],
|
type: 'text',
|
||||||
help: 'Markup from Markdown',
|
help: 'Add text using Markdown',
|
||||||
image: header,
|
icon: 'visText',
|
||||||
expression: `filters
|
expression: `filters
|
||||||
| demodata
|
| demodata
|
||||||
| markdown "### Welcome to the Markdown element
|
| markdown "### Welcome to the Markdown element
|
||||||
|
|
Before Width: | Height: | Size: 12 KiB |
|
@ -5,8 +5,6 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import { openSans } from '../../../common/lib/fonts';
|
import { openSans } from '../../../common/lib/fonts';
|
||||||
import header from './header.png';
|
|
||||||
|
|
||||||
import { ElementFactory } from '../../../types';
|
import { ElementFactory } from '../../../types';
|
||||||
import { SetupInitializer } from '../../plugin';
|
import { SetupInitializer } from '../../plugin';
|
||||||
|
|
||||||
|
@ -14,11 +12,11 @@ export const metricElementInitializer: SetupInitializer<ElementFactory> = (core,
|
||||||
return () => ({
|
return () => ({
|
||||||
name: 'metric',
|
name: 'metric',
|
||||||
displayName: 'Metric',
|
displayName: 'Metric',
|
||||||
tags: ['text'],
|
type: 'chart',
|
||||||
help: 'A number with a label',
|
help: 'A number with a label',
|
||||||
width: 200,
|
width: 200,
|
||||||
height: 100,
|
height: 100,
|
||||||
image: header,
|
icon: 'visMetric',
|
||||||
expression: `filters
|
expression: `filters
|
||||||
| demodata
|
| demodata
|
||||||
| math "unique(country)"
|
| math "unique(country)"
|
||||||
|
|
Before Width: | Height: | Size: 12 KiB |
|
@ -4,17 +4,15 @@
|
||||||
* you may not use this file except in compliance with the Elastic License.
|
* you may not use this file except in compliance with the Elastic License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import header from './header.png';
|
|
||||||
|
|
||||||
import { ElementFactory } from '../../../types';
|
import { ElementFactory } from '../../../types';
|
||||||
export const pie: ElementFactory = () => ({
|
export const pie: ElementFactory = () => ({
|
||||||
name: 'pie',
|
name: 'pie',
|
||||||
displayName: 'Pie chart',
|
displayName: 'Pie',
|
||||||
tags: ['chart', 'proportion'],
|
type: 'chart',
|
||||||
width: 300,
|
width: 300,
|
||||||
height: 300,
|
height: 300,
|
||||||
help: 'A simple pie chart',
|
help: 'A simple pie chart',
|
||||||
image: header,
|
icon: 'visPie',
|
||||||
expression: `filters
|
expression: `filters
|
||||||
| demodata
|
| demodata
|
||||||
| pointseries color="state" size="max(price)"
|
| pointseries color="state" size="max(price)"
|
||||||
|
|
Before Width: | Height: | Size: 37 KiB |
|
@ -5,14 +5,12 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import { ElementFactory } from '../../../types';
|
import { ElementFactory } from '../../../types';
|
||||||
import header from './header.png';
|
|
||||||
|
|
||||||
export const plot: ElementFactory = () => ({
|
export const plot: ElementFactory = () => ({
|
||||||
name: 'plot',
|
name: 'plot',
|
||||||
displayName: 'Coordinate plot',
|
displayName: 'Coordinate plot',
|
||||||
tags: ['chart'],
|
type: 'chart',
|
||||||
help: 'Mixed line, bar or dot charts',
|
help: 'Mixed line, bar or dot charts',
|
||||||
image: header,
|
|
||||||
expression: `filters
|
expression: `filters
|
||||||
| demodata
|
| demodata
|
||||||
| pointseries x="time" y="sum(price)" color="state"
|
| pointseries x="time" y="sum(price)" color="state"
|
||||||
|
|
Before Width: | Height: | Size: 22 KiB |
|
@ -6,16 +6,15 @@
|
||||||
|
|
||||||
import { openSans } from '../../../common/lib/fonts';
|
import { openSans } from '../../../common/lib/fonts';
|
||||||
import { ElementFactory } from '../../../types';
|
import { ElementFactory } from '../../../types';
|
||||||
import header from './header.png';
|
|
||||||
|
|
||||||
export const progressGauge: ElementFactory = () => ({
|
export const progressGauge: ElementFactory = () => ({
|
||||||
name: 'progressGauge',
|
name: 'progressGauge',
|
||||||
displayName: 'Progress gauge',
|
displayName: 'Gauge',
|
||||||
tags: ['chart', 'proportion'],
|
type: 'progress',
|
||||||
help: 'Displays progress as a portion of a gauge',
|
help: 'Displays progress as a portion of a gauge',
|
||||||
width: 200,
|
width: 200,
|
||||||
height: 200,
|
height: 200,
|
||||||
image: header,
|
icon: 'visGoal',
|
||||||
expression: `filters
|
expression: `filters
|
||||||
| demodata
|
| demodata
|
||||||
| math "mean(percent_uptime)"
|
| math "mean(percent_uptime)"
|
||||||
|
|
Before Width: | Height: | Size: 17 KiB |
|
@ -6,16 +6,14 @@
|
||||||
|
|
||||||
import { openSans } from '../../../common/lib/fonts';
|
import { openSans } from '../../../common/lib/fonts';
|
||||||
import { ElementFactory } from '../../../types';
|
import { ElementFactory } from '../../../types';
|
||||||
import header from './header.png';
|
|
||||||
|
|
||||||
export const progressSemicircle: ElementFactory = () => ({
|
export const progressSemicircle: ElementFactory = () => ({
|
||||||
name: 'progressSemicircle',
|
name: 'progressSemicircle',
|
||||||
displayName: 'Progress semicircle',
|
displayName: 'Semicircle',
|
||||||
tags: ['chart', 'proportion'],
|
type: 'progress',
|
||||||
help: 'Displays progress as a portion of a semicircle',
|
help: 'Displays progress as a portion of a semicircle',
|
||||||
width: 200,
|
width: 200,
|
||||||
height: 100,
|
height: 100,
|
||||||
image: header,
|
|
||||||
expression: `filters
|
expression: `filters
|
||||||
| demodata
|
| demodata
|
||||||
| math "mean(percent_uptime)"
|
| math "mean(percent_uptime)"
|
||||||
|
|
Before Width: | Height: | Size: 26 KiB |
|
@ -6,16 +6,14 @@
|
||||||
|
|
||||||
import { openSans } from '../../../common/lib/fonts';
|
import { openSans } from '../../../common/lib/fonts';
|
||||||
import { ElementFactory } from '../../../types';
|
import { ElementFactory } from '../../../types';
|
||||||
import header from './header.png';
|
|
||||||
|
|
||||||
export const progressWheel: ElementFactory = () => ({
|
export const progressWheel: ElementFactory = () => ({
|
||||||
name: 'progressWheel',
|
name: 'progressWheel',
|
||||||
displayName: 'Progress wheel',
|
displayName: 'Wheel',
|
||||||
tags: ['chart', 'proportion'],
|
type: 'progress',
|
||||||
help: 'Displays progress as a portion of a wheel',
|
help: 'Displays progress as a portion of a wheel',
|
||||||
width: 200,
|
width: 200,
|
||||||
height: 200,
|
height: 200,
|
||||||
image: header,
|
|
||||||
expression: `filters
|
expression: `filters
|
||||||
| demodata
|
| demodata
|
||||||
| math "mean(percent_uptime)"
|
| math "mean(percent_uptime)"
|
||||||
|
|
Before Width: | Height: | Size: 82 KiB |
|
@ -5,14 +5,12 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import { ElementFactory } from '../../../types';
|
import { ElementFactory } from '../../../types';
|
||||||
import header from './header.png';
|
|
||||||
|
|
||||||
export const repeatImage: ElementFactory = () => ({
|
export const repeatImage: ElementFactory = () => ({
|
||||||
name: 'repeatImage',
|
name: 'repeatImage',
|
||||||
displayName: 'Image repeat',
|
displayName: 'Image repeat',
|
||||||
tags: ['graphic', 'proportion'],
|
type: 'image',
|
||||||
help: 'Repeats an image N times',
|
help: 'Repeats an image N times',
|
||||||
image: header,
|
|
||||||
expression: `filters
|
expression: `filters
|
||||||
| demodata
|
| demodata
|
||||||
| math "mean(cost)"
|
| math "mean(cost)"
|
||||||
|
|
Before Width: | Height: | Size: 8.8 KiB |
|
@ -5,14 +5,12 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import { ElementFactory } from '../../../types';
|
import { ElementFactory } from '../../../types';
|
||||||
import header from './header.png';
|
|
||||||
|
|
||||||
export const revealImage: ElementFactory = () => ({
|
export const revealImage: ElementFactory = () => ({
|
||||||
name: 'revealImage',
|
name: 'revealImage',
|
||||||
displayName: 'Image reveal',
|
displayName: 'Image reveal',
|
||||||
tags: ['graphic', 'proportion'],
|
type: 'image',
|
||||||
help: 'Reveals a percentage of an image',
|
help: 'Reveals a percentage of an image',
|
||||||
image: header,
|
|
||||||
expression: `filters
|
expression: `filters
|
||||||
| demodata
|
| demodata
|
||||||
| math "mean(percent_uptime)"
|
| math "mean(percent_uptime)"
|
||||||
|
|
Before Width: | Height: | Size: 10 KiB |
|
@ -5,16 +5,15 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import { ElementFactory } from '../../../types';
|
import { ElementFactory } from '../../../types';
|
||||||
import header from './header.png';
|
|
||||||
|
|
||||||
export const shape: ElementFactory = () => ({
|
export const shape: ElementFactory = () => ({
|
||||||
name: 'shape',
|
name: 'shape',
|
||||||
displayName: 'Shape',
|
displayName: 'Shape',
|
||||||
tags: ['graphic'],
|
type: 'shape',
|
||||||
help: 'A customizable shape',
|
help: 'A customizable shape',
|
||||||
width: 200,
|
width: 200,
|
||||||
height: 200,
|
height: 200,
|
||||||
image: header,
|
icon: 'node',
|
||||||
expression:
|
expression:
|
||||||
'shape "square" fill="#4cbce4" border="rgba(255,255,255,0)" borderWidth=0 maintainAspect=false | render',
|
'shape "square" fill="#4cbce4" border="rgba(255,255,255,0)" borderWidth=0 maintainAspect=false | render',
|
||||||
});
|
});
|
||||||
|
|
Before Width: | Height: | Size: 16 KiB |
|
@ -5,14 +5,13 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import { ElementFactory } from '../../../types';
|
import { ElementFactory } from '../../../types';
|
||||||
import header from './header.png';
|
|
||||||
|
|
||||||
export const table: ElementFactory = () => ({
|
export const table: ElementFactory = () => ({
|
||||||
name: 'table',
|
name: 'table',
|
||||||
displayName: 'Data table',
|
displayName: 'Data table',
|
||||||
tags: ['text'],
|
type: 'chart',
|
||||||
help: 'A scrollable grid for displaying data in a tabular format',
|
help: 'A scrollable grid for displaying data in a tabular format',
|
||||||
image: header,
|
icon: 'visTable',
|
||||||
expression: `filters
|
expression: `filters
|
||||||
| demodata
|
| demodata
|
||||||
| table
|
| table
|
||||||
|
|
Before Width: | Height: | Size: 32 KiB |
|
@ -1,23 +0,0 @@
|
||||||
/*
|
|
||||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
|
||||||
* or more contributor license agreements. Licensed under the Elastic License;
|
|
||||||
* you may not use this file except in compliance with the Elastic License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
import { ElementFactory } from '../../../types';
|
|
||||||
import header from './header.png';
|
|
||||||
|
|
||||||
export const tiltedPie: ElementFactory = () => ({
|
|
||||||
name: 'tiltedPie',
|
|
||||||
displayName: 'Tilted pie chart',
|
|
||||||
tags: ['chart', 'proportion'],
|
|
||||||
width: 500,
|
|
||||||
height: 250,
|
|
||||||
help: 'A customizable tilted pie chart',
|
|
||||||
image: header,
|
|
||||||
expression: `filters
|
|
||||||
| demodata
|
|
||||||
| pointseries color="project" size="max(price)"
|
|
||||||
| pie tilt=0.5
|
|
||||||
| render`,
|
|
||||||
});
|
|
Before Width: | Height: | Size: 18 KiB |
|
@ -5,14 +5,13 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import { ElementFactory } from '../../../types';
|
import { ElementFactory } from '../../../types';
|
||||||
import header from './header.png';
|
|
||||||
|
|
||||||
export const timeFilter: ElementFactory = () => ({
|
export const timeFilter: ElementFactory = () => ({
|
||||||
name: 'time_filter',
|
name: 'timeFilter',
|
||||||
displayName: 'Time filter',
|
displayName: 'Time filter',
|
||||||
tags: ['filter'],
|
type: 'filter',
|
||||||
help: 'Set a time window',
|
help: 'Set a time window',
|
||||||
image: header,
|
icon: 'calendar',
|
||||||
height: 50,
|
height: 50,
|
||||||
expression: `timefilterControl compact=true column=@timestamp
|
expression: `timefilterControl compact=true column=@timestamp
|
||||||
| render`,
|
| render`,
|
||||||
|
|
Before Width: | Height: | Size: 17 KiB |
|
@ -5,14 +5,13 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import { ElementFactory } from '../../../types';
|
import { ElementFactory } from '../../../types';
|
||||||
import header from './header.png';
|
|
||||||
|
|
||||||
export const verticalBarChart: ElementFactory = () => ({
|
export const verticalBarChart: ElementFactory = () => ({
|
||||||
name: 'verticalBarChart',
|
name: 'verticalBarChart',
|
||||||
displayName: 'Vertical bar chart',
|
displayName: 'Vertical bar chart',
|
||||||
tags: ['chart'],
|
type: 'chart',
|
||||||
help: 'A customizable vertical bar chart',
|
help: 'A customizable vertical bar chart',
|
||||||
image: header,
|
icon: 'visBarVertical',
|
||||||
expression: `filters
|
expression: `filters
|
||||||
| demodata
|
| demodata
|
||||||
| pointseries x="project" y="size(cost)" color="project"
|
| pointseries x="project" y="size(cost)" color="project"
|
||||||
|
|
Before Width: | Height: | Size: 15 KiB |
|
@ -6,16 +6,14 @@
|
||||||
|
|
||||||
import { openSans } from '../../../common/lib/fonts';
|
import { openSans } from '../../../common/lib/fonts';
|
||||||
import { ElementFactory } from '../../../types';
|
import { ElementFactory } from '../../../types';
|
||||||
import header from './header.png';
|
|
||||||
|
|
||||||
export const verticalProgressBar: ElementFactory = () => ({
|
export const verticalProgressBar: ElementFactory = () => ({
|
||||||
name: 'verticalProgressBar',
|
name: 'verticalProgressBar',
|
||||||
displayName: 'Vertical progress bar',
|
displayName: 'Vertical progress bar',
|
||||||
tags: ['chart', 'proportion'],
|
type: 'progress',
|
||||||
help: 'Displays progress as a portion of a vertical bar',
|
help: 'Displays progress as a portion of a vertical bar',
|
||||||
width: 80,
|
width: 80,
|
||||||
height: 400,
|
height: 400,
|
||||||
image: header,
|
|
||||||
expression: `filters
|
expression: `filters
|
||||||
| demodata
|
| demodata
|
||||||
| math "mean(percent_uptime)"
|
| math "mean(percent_uptime)"
|
||||||
|
|
Before Width: | Height: | Size: 20 KiB |
|
@ -6,16 +6,14 @@
|
||||||
|
|
||||||
import { openSans } from '../../../common/lib/fonts';
|
import { openSans } from '../../../common/lib/fonts';
|
||||||
import { ElementFactory } from '../../../types';
|
import { ElementFactory } from '../../../types';
|
||||||
import header from './header.png';
|
|
||||||
|
|
||||||
export const verticalProgressPill: ElementFactory = () => ({
|
export const verticalProgressPill: ElementFactory = () => ({
|
||||||
name: 'verticalProgressPill',
|
name: 'verticalProgressPill',
|
||||||
displayName: 'Vertical progress pill',
|
displayName: 'Vertical progress pill',
|
||||||
tags: ['chart', 'proportion'],
|
type: 'progress',
|
||||||
help: 'Displays progress as a portion of a vertical pill',
|
help: 'Displays progress as a portion of a vertical pill',
|
||||||
width: 80,
|
width: 80,
|
||||||
height: 400,
|
height: 400,
|
||||||
image: header,
|
|
||||||
expression: `filters
|
expression: `filters
|
||||||
| demodata
|
| demodata
|
||||||
| math "mean(percent_uptime)"
|
| math "mean(percent_uptime)"
|
||||||
|
|
|
@ -1,15 +0,0 @@
|
||||||
/*
|
|
||||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
|
||||||
* or more contributor license agreements. Licensed under the Elastic License;
|
|
||||||
* you may not use this file except in compliance with the Elastic License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
import { euiPaletteColorBlind } from '@elastic/eui';
|
|
||||||
import { TagFactory } from '../../../public/lib/tag';
|
|
||||||
import { TagStrings as strings } from '../../../i18n';
|
|
||||||
const euiVisPalette = euiPaletteColorBlind();
|
|
||||||
|
|
||||||
export const chart: TagFactory = () => ({
|
|
||||||
name: strings.chart(),
|
|
||||||
color: euiVisPalette[4],
|
|
||||||
});
|
|
|
@ -1,16 +0,0 @@
|
||||||
/*
|
|
||||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
|
||||||
* or more contributor license agreements. Licensed under the Elastic License;
|
|
||||||
* you may not use this file except in compliance with the Elastic License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
import { euiPaletteColorBlind } from '@elastic/eui';
|
|
||||||
import { TagFactory } from '../../../public/lib/tag';
|
|
||||||
import { TagStrings as strings } from '../../../i18n';
|
|
||||||
|
|
||||||
const euiVisPalette = euiPaletteColorBlind();
|
|
||||||
|
|
||||||
export const filter: TagFactory = () => ({
|
|
||||||
name: strings.filter(),
|
|
||||||
color: euiVisPalette[1],
|
|
||||||
});
|
|
|
@ -1,15 +0,0 @@
|
||||||
/*
|
|
||||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
|
||||||
* or more contributor license agreements. Licensed under the Elastic License;
|
|
||||||
* you may not use this file except in compliance with the Elastic License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
import { euiPaletteColorBlind } from '@elastic/eui';
|
|
||||||
import { TagFactory } from '../../../public/lib/tag';
|
|
||||||
import { TagStrings as strings } from '../../../i18n';
|
|
||||||
const euiVisPalette = euiPaletteColorBlind();
|
|
||||||
|
|
||||||
export const graphic: TagFactory = () => ({
|
|
||||||
name: strings.graphic(),
|
|
||||||
color: euiVisPalette[5],
|
|
||||||
});
|
|
|
@ -4,13 +4,8 @@
|
||||||
* you may not use this file except in compliance with the Elastic License.
|
* you may not use this file except in compliance with the Elastic License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import { chart } from './chart';
|
|
||||||
import { filter } from './filter';
|
|
||||||
import { graphic } from './graphic';
|
|
||||||
import { presentation } from './presentation';
|
import { presentation } from './presentation';
|
||||||
import { proportion } from './proportion';
|
|
||||||
import { report } from './report';
|
import { report } from './report';
|
||||||
import { text } from './text';
|
|
||||||
|
|
||||||
// Registry expects a function that returns a spec object
|
// Registry expects a function that returns a spec object
|
||||||
export const tagSpecs = [chart, filter, graphic, presentation, proportion, report, text];
|
export const tagSpecs = [presentation, report];
|
||||||
|
|
|
@ -1,15 +0,0 @@
|
||||||
/*
|
|
||||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
|
||||||
* or more contributor license agreements. Licensed under the Elastic License;
|
|
||||||
* you may not use this file except in compliance with the Elastic License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
import { euiPaletteColorBlind } from '@elastic/eui';
|
|
||||||
import { TagFactory } from '../../../public/lib/tag';
|
|
||||||
import { TagStrings as strings } from '../../../i18n';
|
|
||||||
const euiVisPalette = euiPaletteColorBlind();
|
|
||||||
|
|
||||||
export const proportion: TagFactory = () => ({
|
|
||||||
name: strings.proportion(),
|
|
||||||
color: euiVisPalette[3],
|
|
||||||
});
|
|
|
@ -1,13 +0,0 @@
|
||||||
/*
|
|
||||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
|
||||||
* or more contributor license agreements. Licensed under the Elastic License;
|
|
||||||
* you may not use this file except in compliance with the Elastic License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
import { TagFactory } from '../../../public/lib/tag';
|
|
||||||
import { TagStrings as strings } from '../../../i18n';
|
|
||||||
|
|
||||||
export const text: TagFactory = () => ({
|
|
||||||
name: strings.text(),
|
|
||||||
color: '#D3DAE6',
|
|
||||||
});
|
|
|
@ -40,3 +40,4 @@ export const API_ROUTE_SHAREABLE_ZIP = '/public/canvas/zip';
|
||||||
export const API_ROUTE_SHAREABLE_RUNTIME = '/public/canvas/runtime';
|
export const API_ROUTE_SHAREABLE_RUNTIME = '/public/canvas/runtime';
|
||||||
export const API_ROUTE_SHAREABLE_RUNTIME_DOWNLOAD = `/public/canvas/${SHAREABLE_RUNTIME_NAME}.js`;
|
export const API_ROUTE_SHAREABLE_RUNTIME_DOWNLOAD = `/public/canvas/${SHAREABLE_RUNTIME_NAME}.js`;
|
||||||
export const CANVAS_EMBEDDABLE_CLASSNAME = `canvasEmbeddable`;
|
export const CANVAS_EMBEDDABLE_CLASSNAME = `canvasEmbeddable`;
|
||||||
|
export const CONTEXT_MENU_TOP_BORDER_CLASSNAME = 'canvasContextMenu--topBorder';
|
||||||
|
|
|
@ -15,7 +15,7 @@ export const ComponentStrings = {
|
||||||
}),
|
}),
|
||||||
getTitleText: () =>
|
getTitleText: () =>
|
||||||
i18n.translate('xpack.canvas.embedObject.titleText', {
|
i18n.translate('xpack.canvas.embedObject.titleText', {
|
||||||
defaultMessage: 'Embed Object',
|
defaultMessage: 'Add from Visualize library',
|
||||||
}),
|
}),
|
||||||
},
|
},
|
||||||
AdvancedFilter: {
|
AdvancedFilter: {
|
||||||
|
@ -305,21 +305,21 @@ export const ComponentStrings = {
|
||||||
}),
|
}),
|
||||||
},
|
},
|
||||||
ElementControls: {
|
ElementControls: {
|
||||||
getEditTooltip: () =>
|
getDeleteAriaLabel: () =>
|
||||||
i18n.translate('xpack.canvas.elementControls.editToolTip', {
|
i18n.translate('xpack.canvas.elementControls.deleteAriaLabel', {
|
||||||
defaultMessage: 'Edit',
|
defaultMessage: 'Delete element',
|
||||||
}),
|
|
||||||
getEditAriaLabel: () =>
|
|
||||||
i18n.translate('xpack.canvas.elementControls.editAriaLabel', {
|
|
||||||
defaultMessage: 'Edit element',
|
|
||||||
}),
|
}),
|
||||||
getDeleteTooltip: () =>
|
getDeleteTooltip: () =>
|
||||||
i18n.translate('xpack.canvas.elementControls.deleteToolTip', {
|
i18n.translate('xpack.canvas.elementControls.deleteToolTip', {
|
||||||
defaultMessage: 'Delete',
|
defaultMessage: 'Delete',
|
||||||
}),
|
}),
|
||||||
getDeleteAriaLabel: () =>
|
getEditAriaLabel: () =>
|
||||||
i18n.translate('xpack.canvas.elementControls.deleteAriaLabel', {
|
i18n.translate('xpack.canvas.elementControls.editAriaLabel', {
|
||||||
defaultMessage: 'Delete element',
|
defaultMessage: 'Edit element',
|
||||||
|
}),
|
||||||
|
getEditTooltip: () =>
|
||||||
|
i18n.translate('xpack.canvas.elementControls.editToolTip', {
|
||||||
|
defaultMessage: 'Edit',
|
||||||
}),
|
}),
|
||||||
},
|
},
|
||||||
ElementSettings: {
|
ElementSettings: {
|
||||||
|
@ -336,53 +336,6 @@ export const ComponentStrings = {
|
||||||
description: 'This tab contains the settings for how data is displayed in a Canvas element',
|
description: 'This tab contains the settings for how data is displayed in a Canvas element',
|
||||||
}),
|
}),
|
||||||
},
|
},
|
||||||
ElementTypes: {
|
|
||||||
getEditElementTitle: () =>
|
|
||||||
i18n.translate('xpack.canvas.elementTypes.editElementTitle', {
|
|
||||||
defaultMessage: 'Edit element',
|
|
||||||
}),
|
|
||||||
getDeleteElementTitle: (elementName: string) =>
|
|
||||||
i18n.translate('xpack.canvas.elementTypes.deleteElementTitle', {
|
|
||||||
defaultMessage: `Delete element '{elementName}'?`,
|
|
||||||
values: {
|
|
||||||
elementName,
|
|
||||||
},
|
|
||||||
}),
|
|
||||||
getDeleteElementDescription: () =>
|
|
||||||
i18n.translate('xpack.canvas.elementTypes.deleteElementDescription', {
|
|
||||||
defaultMessage: 'Are you sure you want to delete this element?',
|
|
||||||
}),
|
|
||||||
getCancelButtonLabel: () =>
|
|
||||||
i18n.translate('xpack.canvas.elementTypes.cancelButtonLabel', {
|
|
||||||
defaultMessage: 'Cancel',
|
|
||||||
}),
|
|
||||||
getDeleteButtonLabel: () =>
|
|
||||||
i18n.translate('xpack.canvas.elementTypes.deleteButtonLabel', {
|
|
||||||
defaultMessage: 'Delete',
|
|
||||||
}),
|
|
||||||
getAddNewElementTitle: () =>
|
|
||||||
i18n.translate('xpack.canvas.elementTypes.addNewElementTitle', {
|
|
||||||
defaultMessage: 'Add new elements',
|
|
||||||
}),
|
|
||||||
getAddNewElementDescription: () =>
|
|
||||||
i18n.translate('xpack.canvas.elementTypes.addNewElementDescription', {
|
|
||||||
defaultMessage: 'Group and save workpad elements to create new elements',
|
|
||||||
}),
|
|
||||||
getFindElementPlaceholder: () =>
|
|
||||||
i18n.translate('xpack.canvas.elementTypes.findElementPlaceholder', {
|
|
||||||
defaultMessage: 'Find element',
|
|
||||||
}),
|
|
||||||
getElementsTitle: () =>
|
|
||||||
i18n.translate('xpack.canvas.elementTypes.elementsTitle', {
|
|
||||||
defaultMessage: 'Elements',
|
|
||||||
description: 'Title for the "Elements" tab when adding a new element',
|
|
||||||
}),
|
|
||||||
getMyElementsTitle: () =>
|
|
||||||
i18n.translate('xpack.canvas.elementTypes.myElementsTitle', {
|
|
||||||
defaultMessage: 'My elements',
|
|
||||||
description: 'Title for the "My elements" tab when adding a new element',
|
|
||||||
}),
|
|
||||||
},
|
|
||||||
Error: {
|
Error: {
|
||||||
getDescription: () =>
|
getDescription: () =>
|
||||||
i18n.translate('xpack.canvas.errorComponent.description', {
|
i18n.translate('xpack.canvas.errorComponent.description', {
|
||||||
|
@ -633,6 +586,61 @@ export const ComponentStrings = {
|
||||||
defaultMessage: 'Delete',
|
defaultMessage: 'Delete',
|
||||||
}),
|
}),
|
||||||
},
|
},
|
||||||
|
SavedElementsModal: {
|
||||||
|
getAddNewElementDescription: () =>
|
||||||
|
i18n.translate('xpack.canvas.savedElementsModal.addNewElementDescription', {
|
||||||
|
defaultMessage: 'Group and save workpad elements to create new elements',
|
||||||
|
}),
|
||||||
|
getAddNewElementTitle: () =>
|
||||||
|
i18n.translate('xpack.canvas.savedElementsModal.addNewElementTitle', {
|
||||||
|
defaultMessage: 'Add new elements',
|
||||||
|
}),
|
||||||
|
getCancelButtonLabel: () =>
|
||||||
|
i18n.translate('xpack.canvas.savedElementsModal.cancelButtonLabel', {
|
||||||
|
defaultMessage: 'Cancel',
|
||||||
|
}),
|
||||||
|
getDeleteButtonLabel: () =>
|
||||||
|
i18n.translate('xpack.canvas.savedElementsModal.deleteButtonLabel', {
|
||||||
|
defaultMessage: 'Delete',
|
||||||
|
}),
|
||||||
|
getDeleteElementDescription: () =>
|
||||||
|
i18n.translate('xpack.canvas.savedElementsModal.deleteElementDescription', {
|
||||||
|
defaultMessage: 'Are you sure you want to delete this element?',
|
||||||
|
}),
|
||||||
|
getDeleteElementTitle: (elementName: string) =>
|
||||||
|
i18n.translate('xpack.canvas.savedElementsModal.deleteElementTitle', {
|
||||||
|
defaultMessage: `Delete element '{elementName}'?`,
|
||||||
|
values: {
|
||||||
|
elementName,
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
getEditElementTitle: () =>
|
||||||
|
i18n.translate('xpack.canvas.savedElementsModal.editElementTitle', {
|
||||||
|
defaultMessage: 'Edit element',
|
||||||
|
}),
|
||||||
|
getElementsTitle: () =>
|
||||||
|
i18n.translate('xpack.canvas.savedElementsModal.elementsTitle', {
|
||||||
|
defaultMessage: 'Elements',
|
||||||
|
description: 'Title for the "Elements" tab when adding a new element',
|
||||||
|
}),
|
||||||
|
getFindElementPlaceholder: () =>
|
||||||
|
i18n.translate('xpack.canvas.savedElementsModal.findElementPlaceholder', {
|
||||||
|
defaultMessage: 'Find element',
|
||||||
|
}),
|
||||||
|
getModalTitle: () =>
|
||||||
|
i18n.translate('xpack.canvas.savedElementsModal.modalTitle', {
|
||||||
|
defaultMessage: 'My elements',
|
||||||
|
}),
|
||||||
|
getMyElementsTitle: () =>
|
||||||
|
i18n.translate('xpack.canvas.savedElementsModal.myElementsTitle', {
|
||||||
|
defaultMessage: 'My elements',
|
||||||
|
description: 'Title for the "My elements" tab when adding a new element',
|
||||||
|
}),
|
||||||
|
getSavedElementsModalCloseButtonLabel: () =>
|
||||||
|
i18n.translate('xpack.canvas.workpadHeader.addElementModalCloseButtonLabel', {
|
||||||
|
defaultMessage: 'Close',
|
||||||
|
}),
|
||||||
|
},
|
||||||
ShareWebsiteFlyout: {
|
ShareWebsiteFlyout: {
|
||||||
getRuntimeStepTitle: () =>
|
getRuntimeStepTitle: () =>
|
||||||
i18n.translate('xpack.canvas.shareWebsiteFlyout.snippetsStep.downloadRuntimeTitle', {
|
i18n.translate('xpack.canvas.shareWebsiteFlyout.snippetsStep.downloadRuntimeTitle', {
|
||||||
|
@ -652,7 +660,7 @@ export const ComponentStrings = {
|
||||||
defaultMessage: 'Share on a website',
|
defaultMessage: 'Share on a website',
|
||||||
}),
|
}),
|
||||||
getUnsupportedRendererWarning: () =>
|
getUnsupportedRendererWarning: () =>
|
||||||
i18n.translate('xpack.canvas.workpadHeaderWorkpadExport.unsupportedRendererWarning', {
|
i18n.translate('xpack.canvas.workpadHeaderShareMenu.unsupportedRendererWarning', {
|
||||||
defaultMessage:
|
defaultMessage:
|
||||||
'This workpad contains render functions that are not supported by the {CANVAS} Shareable Workpad Runtime. These elements will not be rendered:',
|
'This workpad contains render functions that are not supported by the {CANVAS} Shareable Workpad Runtime. These elements will not be rendered:',
|
||||||
values: {
|
values: {
|
||||||
|
@ -900,6 +908,10 @@ export const ComponentStrings = {
|
||||||
i18n.translate('xpack.canvas.textStylePicker.alignRightOption', {
|
i18n.translate('xpack.canvas.textStylePicker.alignRightOption', {
|
||||||
defaultMessage: 'Align right',
|
defaultMessage: 'Align right',
|
||||||
}),
|
}),
|
||||||
|
getFontColorLabel: () =>
|
||||||
|
i18n.translate('xpack.canvas.textStylePicker.fontColorLabel', {
|
||||||
|
defaultMessage: 'Font Color',
|
||||||
|
}),
|
||||||
getStyleBoldOption: () =>
|
getStyleBoldOption: () =>
|
||||||
i18n.translate('xpack.canvas.textStylePicker.styleBoldOption', {
|
i18n.translate('xpack.canvas.textStylePicker.styleBoldOption', {
|
||||||
defaultMessage: 'Bold',
|
defaultMessage: 'Bold',
|
||||||
|
@ -912,10 +924,6 @@ export const ComponentStrings = {
|
||||||
i18n.translate('xpack.canvas.textStylePicker.styleUnderlineOption', {
|
i18n.translate('xpack.canvas.textStylePicker.styleUnderlineOption', {
|
||||||
defaultMessage: 'Underline',
|
defaultMessage: 'Underline',
|
||||||
}),
|
}),
|
||||||
getFontColorLabel: () =>
|
|
||||||
i18n.translate('xpack.canvas.textStylePicker.fontColorLabel', {
|
|
||||||
defaultMessage: 'Font Color',
|
|
||||||
}),
|
|
||||||
},
|
},
|
||||||
TimePicker: {
|
TimePicker: {
|
||||||
getApplyButtonLabel: () =>
|
getApplyButtonLabel: () =>
|
||||||
|
@ -962,6 +970,10 @@ export const ComponentStrings = {
|
||||||
description:
|
description:
|
||||||
'"stylesheet" refers to the collection of CSS style rules entered by the user.',
|
'"stylesheet" refers to the collection of CSS style rules entered by the user.',
|
||||||
}),
|
}),
|
||||||
|
getBackgroundColorLabel: () =>
|
||||||
|
i18n.translate('xpack.canvas.workpadConfig.backgroundColorLabel', {
|
||||||
|
defaultMessage: 'Background color',
|
||||||
|
}),
|
||||||
getFlipDimensionAriaLabel: () =>
|
getFlipDimensionAriaLabel: () =>
|
||||||
i18n.translate('xpack.canvas.workpadConfig.swapDimensionsAriaLabel', {
|
i18n.translate('xpack.canvas.workpadConfig.swapDimensionsAriaLabel', {
|
||||||
defaultMessage: `Swap the page's width and height`,
|
defaultMessage: `Swap the page's width and height`,
|
||||||
|
@ -1013,10 +1025,6 @@ export const ComponentStrings = {
|
||||||
defaultMessage: 'US Letter',
|
defaultMessage: 'US Letter',
|
||||||
description: 'This is referring to the dimensions of U.S. standard letter paper.',
|
description: 'This is referring to the dimensions of U.S. standard letter paper.',
|
||||||
}),
|
}),
|
||||||
getBackgroundColorLabel: () =>
|
|
||||||
i18n.translate('xpack.canvas.workpadConfig.backgroundColorLabel', {
|
|
||||||
defaultMessage: 'Background color',
|
|
||||||
}),
|
|
||||||
},
|
},
|
||||||
WorkpadCreate: {
|
WorkpadCreate: {
|
||||||
getWorkpadCreateButtonLabel: () =>
|
getWorkpadCreateButtonLabel: () =>
|
||||||
|
@ -1029,14 +1037,6 @@ export const ComponentStrings = {
|
||||||
i18n.translate('xpack.canvas.workpadHeader.addElementButtonLabel', {
|
i18n.translate('xpack.canvas.workpadHeader.addElementButtonLabel', {
|
||||||
defaultMessage: 'Add element',
|
defaultMessage: 'Add element',
|
||||||
}),
|
}),
|
||||||
getAddElementModalCloseButtonLabel: () =>
|
|
||||||
i18n.translate('xpack.canvas.workpadHeader.addElementModalCloseButtonLabel', {
|
|
||||||
defaultMessage: 'Close',
|
|
||||||
}),
|
|
||||||
getEmbedObjectButtonLabel: () =>
|
|
||||||
i18n.translate('xpack.canvas.workpadHeader.embedObjectButtonLabel', {
|
|
||||||
defaultMessage: 'Embed object',
|
|
||||||
}),
|
|
||||||
getFullScreenButtonAriaLabel: () =>
|
getFullScreenButtonAriaLabel: () =>
|
||||||
i18n.translate('xpack.canvas.workpadHeader.fullscreenButtonAriaLabel', {
|
i18n.translate('xpack.canvas.workpadHeader.fullscreenButtonAriaLabel', {
|
||||||
defaultMessage: 'View fullscreen',
|
defaultMessage: 'View fullscreen',
|
||||||
|
@ -1080,9 +1080,9 @@ export const ComponentStrings = {
|
||||||
}),
|
}),
|
||||||
},
|
},
|
||||||
WorkpadHeaderControlSettings: {
|
WorkpadHeaderControlSettings: {
|
||||||
getTooltip: () =>
|
getButtonLabel: () =>
|
||||||
i18n.translate('xpack.canvas.workpadHeaderControlSettings.settingsTooltip', {
|
i18n.translate('xpack.canvas.workpadHeaderControlSettings.buttonLabel', {
|
||||||
defaultMessage: 'Control settings',
|
defaultMessage: 'Options',
|
||||||
}),
|
}),
|
||||||
},
|
},
|
||||||
WorkpadHeaderCustomInterval: {
|
WorkpadHeaderCustomInterval: {
|
||||||
|
@ -1105,6 +1105,56 @@ export const ComponentStrings = {
|
||||||
defaultMessage: 'Set a custom interval',
|
defaultMessage: 'Set a custom interval',
|
||||||
}),
|
}),
|
||||||
},
|
},
|
||||||
|
WorkpadHeaderElementMenu: {
|
||||||
|
getAssetsMenuItemLabel: () =>
|
||||||
|
i18n.translate('xpack.canvas.workpadHeaderElementMenu.manageAssetsMenuItemLabel', {
|
||||||
|
defaultMessage: 'Manage assets',
|
||||||
|
}),
|
||||||
|
getChartMenuItemLabel: () =>
|
||||||
|
i18n.translate('xpack.canvas.workpadHeaderElementMenu.chartMenuItemLabel', {
|
||||||
|
defaultMessage: 'Chart',
|
||||||
|
}),
|
||||||
|
getElementMenuButtonLabel: () =>
|
||||||
|
i18n.translate('xpack.canvas.workpadHeaderElementMenu.elementMenuButtonLabel', {
|
||||||
|
defaultMessage: 'Add element',
|
||||||
|
}),
|
||||||
|
getElementMenuLabel: () =>
|
||||||
|
i18n.translate('xpack.canvas.workpadHeaderElementMenu.elementMenuLabel', {
|
||||||
|
defaultMessage: 'Add an element',
|
||||||
|
}),
|
||||||
|
getEmbedObjectMenuItemLabel: () =>
|
||||||
|
i18n.translate('xpack.canvas.workpadHeaderElementMenu.embedObjectMenuItemLabel', {
|
||||||
|
defaultMessage: 'Add from Visualize library',
|
||||||
|
}),
|
||||||
|
getFilterMenuItemLabel: () =>
|
||||||
|
i18n.translate('xpack.canvas.workpadHeaderElementMenu.filterMenuItemLabel', {
|
||||||
|
defaultMessage: 'Filter',
|
||||||
|
}),
|
||||||
|
getImageMenuItemLabel: () =>
|
||||||
|
i18n.translate('xpack.canvas.workpadHeaderElementMenu.imageMenuItemLabel', {
|
||||||
|
defaultMessage: 'Image',
|
||||||
|
}),
|
||||||
|
getMyElementsMenuItemLabel: () =>
|
||||||
|
i18n.translate('xpack.canvas.workpadHeaderElementMenu.myElementsMenuItemLabel', {
|
||||||
|
defaultMessage: 'My elements',
|
||||||
|
}),
|
||||||
|
getOtherMenuItemLabel: () =>
|
||||||
|
i18n.translate('xpack.canvas.workpadHeaderElementMenu.otherMenuItemLabel', {
|
||||||
|
defaultMessage: 'Other',
|
||||||
|
}),
|
||||||
|
getProgressMenuItemLabel: () =>
|
||||||
|
i18n.translate('xpack.canvas.workpadHeaderElementMenu.progressMenuItemLabel', {
|
||||||
|
defaultMessage: 'Progress',
|
||||||
|
}),
|
||||||
|
getShapeMenuItemLabel: () =>
|
||||||
|
i18n.translate('xpack.canvas.workpadHeaderElementMenu.shapeMenuItemLabel', {
|
||||||
|
defaultMessage: 'Shape',
|
||||||
|
}),
|
||||||
|
getTextMenuItemLabel: () =>
|
||||||
|
i18n.translate('xpack.canvas.workpadHeaderElementMenu.textMenuItemLabel', {
|
||||||
|
defaultMessage: 'Text',
|
||||||
|
}),
|
||||||
|
},
|
||||||
WorkpadHeaderKioskControls: {
|
WorkpadHeaderKioskControls: {
|
||||||
getCycleFormLabel: () =>
|
getCycleFormLabel: () =>
|
||||||
i18n.translate('xpack.canvas.workpadHeaderKioskControl.cycleFormLabel', {
|
i18n.translate('xpack.canvas.workpadHeaderKioskControl.cycleFormLabel', {
|
||||||
|
@ -1129,9 +1179,9 @@ export const ComponentStrings = {
|
||||||
defaultMessage: 'Refresh data',
|
defaultMessage: 'Refresh data',
|
||||||
}),
|
}),
|
||||||
},
|
},
|
||||||
WorkpadHeaderWorkpadExport: {
|
WorkpadHeaderShareMenu: {
|
||||||
getCopyPDFMessage: () =>
|
getCopyPDFMessage: () =>
|
||||||
i18n.translate('xpack.canvas.workpadHeaderWorkpadExport.copyPDFMessage', {
|
i18n.translate('xpack.canvas.workpadHeaderShareMenu.copyPDFMessage', {
|
||||||
defaultMessage: 'The {PDF} generation {URL} was copied to your clipboard.',
|
defaultMessage: 'The {PDF} generation {URL} was copied to your clipboard.',
|
||||||
values: {
|
values: {
|
||||||
PDF,
|
PDF,
|
||||||
|
@ -1139,15 +1189,15 @@ export const ComponentStrings = {
|
||||||
},
|
},
|
||||||
}),
|
}),
|
||||||
getCopyReportingConfigMessage: () =>
|
getCopyReportingConfigMessage: () =>
|
||||||
i18n.translate('xpack.canvas.workpadHeaderWorkpadExport.copyReportingConfigMessage', {
|
i18n.translate('xpack.canvas.workpadHeaderShareMenu.copyReportingConfigMessage', {
|
||||||
defaultMessage: 'Copied reporting configuration to clipboard',
|
defaultMessage: 'Copied reporting configuration to clipboard',
|
||||||
}),
|
}),
|
||||||
getCopyShareConfigMessage: () =>
|
getCopyShareConfigMessage: () =>
|
||||||
i18n.translate('xpack.canvas.workpadHeaderWorkpadExport.copyShareConfigMessage', {
|
i18n.translate('xpack.canvas.workpadHeaderShareMenu.copyShareConfigMessage', {
|
||||||
defaultMessage: 'Copied share markup to clipboard',
|
defaultMessage: 'Copied share markup to clipboard',
|
||||||
}),
|
}),
|
||||||
getExportPDFErrorTitle: (workpadName: string) =>
|
getExportPDFErrorTitle: (workpadName: string) =>
|
||||||
i18n.translate('xpack.canvas.workpadHeaderWorkpadExport.exportPDFErrorMessage', {
|
i18n.translate('xpack.canvas.workpadHeaderShareMenu.exportPDFErrorMessage', {
|
||||||
defaultMessage: "Failed to create {PDF} for '{workpadName}'",
|
defaultMessage: "Failed to create {PDF} for '{workpadName}'",
|
||||||
values: {
|
values: {
|
||||||
PDF,
|
PDF,
|
||||||
|
@ -1155,14 +1205,14 @@ export const ComponentStrings = {
|
||||||
},
|
},
|
||||||
}),
|
}),
|
||||||
getExportPDFMessage: () =>
|
getExportPDFMessage: () =>
|
||||||
i18n.translate('xpack.canvas.workpadHeaderWorkpadExport.exportPDFMessage', {
|
i18n.translate('xpack.canvas.workpadHeaderShareMenu.exportPDFMessage', {
|
||||||
defaultMessage: 'Exporting {PDF}. You can track the progress in Management.',
|
defaultMessage: 'Exporting {PDF}. You can track the progress in Management.',
|
||||||
values: {
|
values: {
|
||||||
PDF,
|
PDF,
|
||||||
},
|
},
|
||||||
}),
|
}),
|
||||||
getExportPDFTitle: (workpadName: string) =>
|
getExportPDFTitle: (workpadName: string) =>
|
||||||
i18n.translate('xpack.canvas.workpadHeaderWorkpadExport.exportPDFTitle', {
|
i18n.translate('xpack.canvas.workpadHeaderShareMenu.exportPDFTitle', {
|
||||||
defaultMessage: "{PDF} export of workpad '{workpadName}'",
|
defaultMessage: "{PDF} export of workpad '{workpadName}'",
|
||||||
values: {
|
values: {
|
||||||
PDF,
|
PDF,
|
||||||
|
@ -1170,7 +1220,7 @@ export const ComponentStrings = {
|
||||||
},
|
},
|
||||||
}),
|
}),
|
||||||
getPDFPanelCopyAriaLabel: () =>
|
getPDFPanelCopyAriaLabel: () =>
|
||||||
i18n.translate('xpack.canvas.workpadHeaderWorkpadExport.pdfPanelCopyAriaLabel', {
|
i18n.translate('xpack.canvas.workpadHeaderShareMenu.pdfPanelCopyAriaLabel', {
|
||||||
defaultMessage:
|
defaultMessage:
|
||||||
'Alternatively, you can generate a {PDF} from a script or with Watcher by using this {URL}. Press Enter to copy the {URL} to clipboard.',
|
'Alternatively, you can generate a {PDF} from a script or with Watcher by using this {URL}. Press Enter to copy the {URL} to clipboard.',
|
||||||
values: {
|
values: {
|
||||||
|
@ -1179,7 +1229,7 @@ export const ComponentStrings = {
|
||||||
},
|
},
|
||||||
}),
|
}),
|
||||||
getPDFPanelCopyButtonLabel: () =>
|
getPDFPanelCopyButtonLabel: () =>
|
||||||
i18n.translate('xpack.canvas.workpadHeaderWorkpadExport.pdfPanelCopyButtonLabel', {
|
i18n.translate('xpack.canvas.workpadHeaderShareMenu.pdfPanelCopyButtonLabel', {
|
||||||
defaultMessage: 'Copy {POST} {URL}',
|
defaultMessage: 'Copy {POST} {URL}',
|
||||||
values: {
|
values: {
|
||||||
POST,
|
POST,
|
||||||
|
@ -1187,7 +1237,7 @@ export const ComponentStrings = {
|
||||||
},
|
},
|
||||||
}),
|
}),
|
||||||
getPDFPanelCopyDescription: () =>
|
getPDFPanelCopyDescription: () =>
|
||||||
i18n.translate('xpack.canvas.workpadHeaderWorkpadExport.pdfPanelCopyDescription', {
|
i18n.translate('xpack.canvas.workpadHeaderShareMenu.pdfPanelCopyDescription', {
|
||||||
defaultMessage:
|
defaultMessage:
|
||||||
'Alternatively, copy this {POST} {URL} to call generation from outside {KIBANA} or from Watcher.',
|
'Alternatively, copy this {POST} {URL} to call generation from outside {KIBANA} or from Watcher.',
|
||||||
values: {
|
values: {
|
||||||
|
@ -1197,14 +1247,14 @@ export const ComponentStrings = {
|
||||||
},
|
},
|
||||||
}),
|
}),
|
||||||
getPDFPanelGenerateButtonLabel: () =>
|
getPDFPanelGenerateButtonLabel: () =>
|
||||||
i18n.translate('xpack.canvas.workpadHeaderWorkpadExport.pdfPanelGenerateButtonLabel', {
|
i18n.translate('xpack.canvas.workpadHeaderShareMenu.pdfPanelGenerateButtonLabel', {
|
||||||
defaultMessage: 'Generate {PDF}',
|
defaultMessage: 'Generate {PDF}',
|
||||||
values: {
|
values: {
|
||||||
PDF,
|
PDF,
|
||||||
},
|
},
|
||||||
}),
|
}),
|
||||||
getPDFPanelGenerateDescription: () =>
|
getPDFPanelGenerateDescription: () =>
|
||||||
i18n.translate('xpack.canvas.workpadHeaderWorkpadExport.pdfPanelGenerateDescription', {
|
i18n.translate('xpack.canvas.workpadHeaderShareMenu.pdfPanelGenerateDescription', {
|
||||||
defaultMessage:
|
defaultMessage:
|
||||||
'{PDF}s can take a minute or two to generate based on the size of your workpad.',
|
'{PDF}s can take a minute or two to generate based on the size of your workpad.',
|
||||||
values: {
|
values: {
|
||||||
|
@ -1212,7 +1262,7 @@ export const ComponentStrings = {
|
||||||
},
|
},
|
||||||
}),
|
}),
|
||||||
getShareableZipErrorTitle: (workpadName: string) =>
|
getShareableZipErrorTitle: (workpadName: string) =>
|
||||||
i18n.translate('xpack.canvas.workpadHeaderWorkpadExport.shareWebsiteErrorTitle', {
|
i18n.translate('xpack.canvas.workpadHeaderShareMenu.shareWebsiteErrorTitle', {
|
||||||
defaultMessage:
|
defaultMessage:
|
||||||
"Failed to create {ZIP} file for '{workpadName}'. The workpad may be too large. You'll need to download the files separately.",
|
"Failed to create {ZIP} file for '{workpadName}'. The workpad may be too large. You'll need to download the files separately.",
|
||||||
values: {
|
values: {
|
||||||
|
@ -1221,69 +1271,101 @@ export const ComponentStrings = {
|
||||||
},
|
},
|
||||||
}),
|
}),
|
||||||
getShareDownloadJSONTitle: () =>
|
getShareDownloadJSONTitle: () =>
|
||||||
i18n.translate('xpack.canvas.workpadHeaderWorkpadExport.shareDownloadJSONTitle', {
|
i18n.translate('xpack.canvas.workpadHeaderShareMenu.shareDownloadJSONTitle', {
|
||||||
defaultMessage: 'Download as {JSON}',
|
defaultMessage: 'Download as {JSON}',
|
||||||
values: {
|
values: {
|
||||||
JSON,
|
JSON,
|
||||||
},
|
},
|
||||||
}),
|
}),
|
||||||
getShareDownloadPDFTitle: () =>
|
getShareDownloadPDFTitle: () =>
|
||||||
i18n.translate('xpack.canvas.workpadHeaderWorkpadExport.shareDownloadPDFTitle', {
|
i18n.translate('xpack.canvas.workpadHeaderShareMenu.shareDownloadPDFTitle', {
|
||||||
defaultMessage: '{PDF} reports',
|
defaultMessage: '{PDF} reports',
|
||||||
values: {
|
values: {
|
||||||
PDF,
|
PDF,
|
||||||
},
|
},
|
||||||
}),
|
}),
|
||||||
|
getShareMenuButtonLabel: () =>
|
||||||
|
i18n.translate('xpack.canvas.workpadHeaderShareMenu.shareMenuButtonLabel', {
|
||||||
|
defaultMessage: 'Share',
|
||||||
|
}),
|
||||||
getShareWebsiteTitle: () =>
|
getShareWebsiteTitle: () =>
|
||||||
i18n.translate('xpack.canvas.workpadHeaderWorkpadExport.shareWebsiteTitle', {
|
i18n.translate('xpack.canvas.workpadHeaderShareMenu.shareWebsiteTitle', {
|
||||||
defaultMessage: 'Share on a website',
|
defaultMessage: 'Share on a website',
|
||||||
}),
|
}),
|
||||||
getShareWorkpadMessage: () =>
|
getShareWorkpadMessage: () =>
|
||||||
i18n.translate('xpack.canvas.workpadHeaderWorkpadExport.shareWorkpadMessage', {
|
i18n.translate('xpack.canvas.workpadHeaderShareMenu.shareWorkpadMessage', {
|
||||||
defaultMessage: 'Share this workpad',
|
defaultMessage: 'Share this workpad',
|
||||||
}),
|
}),
|
||||||
getUnknownExportErrorMessage: (type: string) =>
|
getUnknownExportErrorMessage: (type: string) =>
|
||||||
i18n.translate('xpack.canvas.workpadHeaderWorkpadExport.unknownExportErrorMessage', {
|
i18n.translate('xpack.canvas.workpadHeaderShareMenu.unknownExportErrorMessage', {
|
||||||
defaultMessage: 'Unknown export type: {type}',
|
defaultMessage: 'Unknown export type: {type}',
|
||||||
values: {
|
values: {
|
||||||
type,
|
type,
|
||||||
},
|
},
|
||||||
}),
|
}),
|
||||||
},
|
},
|
||||||
WorkpadHeaderWorkpadZoom: {
|
WorkpadHeaderViewMenu: {
|
||||||
|
getFullscreenMenuItemLabel: () =>
|
||||||
|
i18n.translate('xpack.canvas.workpadHeaderViewMenu.fullscreenMenuLabel', {
|
||||||
|
defaultMessage: 'Enter fullscreen mode',
|
||||||
|
}),
|
||||||
|
getHideEditModeLabel: () =>
|
||||||
|
i18n.translate('xpack.canvas.workpadHeaderViewMenu.hideEditModeLabel', {
|
||||||
|
defaultMessage: 'Hide editing controls',
|
||||||
|
}),
|
||||||
|
getRefreshMenuItemLabel: () =>
|
||||||
|
i18n.translate('xpack.canvas.workpadHeaderViewMenu.refreshMenuItemLabel', {
|
||||||
|
defaultMessage: 'Refresh data',
|
||||||
|
}),
|
||||||
|
getShowEditModeLabel: () =>
|
||||||
|
i18n.translate('xpack.canvas.workpadHeaderViewMenu.showEditModeLabel', {
|
||||||
|
defaultMessage: 'Show editing controls',
|
||||||
|
}),
|
||||||
|
getViewMenuButtonLabel: () =>
|
||||||
|
i18n.translate('xpack.canvas.workpadHeaderViewMenu.viewMenuButtonLabel', {
|
||||||
|
defaultMessage: 'View',
|
||||||
|
}),
|
||||||
|
getViewMenuLabel: () =>
|
||||||
|
i18n.translate('xpack.canvas.workpadHeaderViewMenu.viewMenuLabel', {
|
||||||
|
defaultMessage: 'View options',
|
||||||
|
}),
|
||||||
getZoomControlsAriaLabel: () =>
|
getZoomControlsAriaLabel: () =>
|
||||||
i18n.translate('xpack.canvas.workpadHeaderWorkpadZoom.zoomControlsAriaLabel', {
|
i18n.translate('xpack.canvas.workpadHeaderViewMenu.zoomControlsAriaLabel', {
|
||||||
defaultMessage: 'Zoom controls',
|
defaultMessage: 'Zoom controls',
|
||||||
}),
|
}),
|
||||||
getZoomControlsTooltip: () =>
|
getZoomControlsTooltip: () =>
|
||||||
i18n.translate('xpack.canvas.workpadHeaderWorkpadZoom.zoomControlsTooltip', {
|
i18n.translate('xpack.canvas.workpadHeaderViewMenu.zoomControlsTooltip', {
|
||||||
defaultMessage: 'Zoom controls',
|
defaultMessage: 'Zoom controls',
|
||||||
}),
|
}),
|
||||||
getZoomFitToWindowText: () =>
|
getZoomFitToWindowText: () =>
|
||||||
i18n.translate('xpack.canvas.workpadHeaderWorkpadZoom.zoomFitToWindowText', {
|
i18n.translate('xpack.canvas.workpadHeaderViewMenu.zoomFitToWindowText', {
|
||||||
defaultMessage: 'Fit to window',
|
defaultMessage: 'Fit to window',
|
||||||
}),
|
}),
|
||||||
getZoomInText: () =>
|
getZoomInText: () =>
|
||||||
i18n.translate('xpack.canvas.workpadHeaderWorkpadZoom.zoomInText', {
|
i18n.translate('xpack.canvas.workpadHeaderViewMenu.zoomInText', {
|
||||||
defaultMessage: 'Zoom in',
|
defaultMessage: 'Zoom in',
|
||||||
}),
|
}),
|
||||||
|
getZoomMenuItemLabel: () =>
|
||||||
|
i18n.translate('xpack.canvas.workpadHeaderViewMenu.zoomMenuItemLabel', {
|
||||||
|
defaultMessage: 'Zoom',
|
||||||
|
}),
|
||||||
getZoomOutText: () =>
|
getZoomOutText: () =>
|
||||||
i18n.translate('xpack.canvas.workpadHeaderWorkpadZoom.zoomOutText', {
|
i18n.translate('xpack.canvas.workpadHeaderViewMenu.zoomOutText', {
|
||||||
defaultMessage: 'Zoom out',
|
defaultMessage: 'Zoom out',
|
||||||
}),
|
}),
|
||||||
getZoomPanelTitle: () =>
|
getZoomPanelTitle: () =>
|
||||||
i18n.translate('xpack.canvas.workpadHeaderWorkpadZoom.zoomPanelTitle', {
|
i18n.translate('xpack.canvas.workpadHeaderViewMenu.zoomPanelTitle', {
|
||||||
defaultMessage: 'Zoom',
|
defaultMessage: 'Zoom',
|
||||||
}),
|
}),
|
||||||
getZoomPercentage: (scale: number) =>
|
getZoomPercentage: (scale: number) =>
|
||||||
i18n.translate('xpack.canvas.workpadHeaderWorkpadZoom.zoomResetText', {
|
i18n.translate('xpack.canvas.workpadHeaderViewMenu.zoomResetText', {
|
||||||
defaultMessage: '{scalePercentage}%',
|
defaultMessage: '{scalePercentage}%',
|
||||||
values: {
|
values: {
|
||||||
scalePercentage: scale * 100,
|
scalePercentage: scale * 100,
|
||||||
},
|
},
|
||||||
}),
|
}),
|
||||||
getZoomResetText: () =>
|
getZoomResetText: () =>
|
||||||
i18n.translate('xpack.canvas.workpadHeaderWorkpadZoom.zoomPrecentageValue', {
|
i18n.translate('xpack.canvas.workpadHeaderViewMenu.zoomPrecentageValue', {
|
||||||
defaultMessage: 'Reset',
|
defaultMessage: 'Reset',
|
||||||
}),
|
}),
|
||||||
},
|
},
|
||||||
|
|
|
@ -7,8 +7,6 @@
|
||||||
import { ElementFactory } from '../../types';
|
import { ElementFactory } from '../../types';
|
||||||
import { getElementStrings } from './index';
|
import { getElementStrings } from './index';
|
||||||
|
|
||||||
import { TagStrings } from '../../i18n';
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This function takes a set of Canvas Element specification factories, runs them,
|
* This function takes a set of Canvas Element specification factories, runs them,
|
||||||
* replaces relevant strings (if available) and returns a new factory. We do this
|
* replaces relevant strings (if available) and returns a new factory. We do this
|
||||||
|
@ -34,17 +32,6 @@ export const applyElementStrings = (elements: ElementFactory[]) => {
|
||||||
if (displayName) {
|
if (displayName) {
|
||||||
result.displayName = displayName;
|
result.displayName = displayName;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Set translated tags
|
|
||||||
if (result.tags) {
|
|
||||||
result.tags = result.tags.map(tag => {
|
|
||||||
if (tag in TagStrings) {
|
|
||||||
return TagStrings[tag]();
|
|
||||||
}
|
|
||||||
|
|
||||||
return tag;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return () => result;
|
return () => result;
|
||||||
|
|
|
@ -9,8 +9,6 @@ import { coreMock } from '../../../../../../src/core/public/mocks';
|
||||||
|
|
||||||
const elementSpecs = initializeElements(coreMock.createSetup() as any, {} as any);
|
const elementSpecs = initializeElements(coreMock.createSetup() as any, {} as any);
|
||||||
|
|
||||||
import { TagStrings } from '../tags';
|
|
||||||
|
|
||||||
describe('ElementStrings', () => {
|
describe('ElementStrings', () => {
|
||||||
const elementStrings = getElementStrings();
|
const elementStrings = getElementStrings();
|
||||||
const elementNames = elementSpecs.map(spec => spec().name);
|
const elementNames = elementSpecs.map(spec => spec().name);
|
||||||
|
@ -37,15 +35,4 @@ describe('ElementStrings', () => {
|
||||||
expect(value).toHaveProperty('help');
|
expect(value).toHaveProperty('help');
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
test('All elements should have tags that are defined', () => {
|
|
||||||
const tagNames = Object.keys(TagStrings);
|
|
||||||
|
|
||||||
elementSpecs.forEach(spec => {
|
|
||||||
const element = spec();
|
|
||||||
if (element.tags) {
|
|
||||||
element.tags.forEach((tagName: string) => expect(tagNames).toContain(tagName));
|
|
||||||
}
|
|
||||||
});
|
|
||||||
});
|
|
||||||
});
|
});
|
||||||
|
|
|
@ -23,7 +23,7 @@ interface ElementStringDict {
|
||||||
export const getElementStrings = (): ElementStringDict => ({
|
export const getElementStrings = (): ElementStringDict => ({
|
||||||
areaChart: {
|
areaChart: {
|
||||||
displayName: i18n.translate('xpack.canvas.elements.areaChartDisplayName', {
|
displayName: i18n.translate('xpack.canvas.elements.areaChartDisplayName', {
|
||||||
defaultMessage: 'Area chart',
|
defaultMessage: 'Area',
|
||||||
}),
|
}),
|
||||||
help: i18n.translate('xpack.canvas.elements.areaChartHelpText', {
|
help: i18n.translate('xpack.canvas.elements.areaChartHelpText', {
|
||||||
defaultMessage: 'A line chart with a filled body',
|
defaultMessage: 'A line chart with a filled body',
|
||||||
|
@ -31,7 +31,7 @@ export const getElementStrings = (): ElementStringDict => ({
|
||||||
},
|
},
|
||||||
bubbleChart: {
|
bubbleChart: {
|
||||||
displayName: i18n.translate('xpack.canvas.elements.bubbleChartDisplayName', {
|
displayName: i18n.translate('xpack.canvas.elements.bubbleChartDisplayName', {
|
||||||
defaultMessage: 'Bubble chart',
|
defaultMessage: 'Bubble',
|
||||||
}),
|
}),
|
||||||
help: i18n.translate('xpack.canvas.elements.bubbleChartHelpText', {
|
help: i18n.translate('xpack.canvas.elements.bubbleChartHelpText', {
|
||||||
defaultMessage: 'A customizable bubble chart',
|
defaultMessage: 'A customizable bubble chart',
|
||||||
|
@ -39,31 +39,31 @@ export const getElementStrings = (): ElementStringDict => ({
|
||||||
},
|
},
|
||||||
debug: {
|
debug: {
|
||||||
displayName: i18n.translate('xpack.canvas.elements.debugDisplayName', {
|
displayName: i18n.translate('xpack.canvas.elements.debugDisplayName', {
|
||||||
defaultMessage: 'Debug',
|
defaultMessage: 'Debug data',
|
||||||
}),
|
}),
|
||||||
help: i18n.translate('xpack.canvas.elements.debugHelpText', {
|
help: i18n.translate('xpack.canvas.elements.debugHelpText', {
|
||||||
defaultMessage: 'Just dumps the configuration of the element',
|
defaultMessage: 'Just dumps the configuration of the element',
|
||||||
}),
|
}),
|
||||||
},
|
},
|
||||||
donut: {
|
dropdownFilter: {
|
||||||
displayName: i18n.translate('xpack.canvas.elements.donutChartDisplayName', {
|
|
||||||
defaultMessage: 'Donut chart',
|
|
||||||
}),
|
|
||||||
help: i18n.translate('xpack.canvas.elements.donutChartHelpText', {
|
|
||||||
defaultMessage: 'A customizable donut chart',
|
|
||||||
}),
|
|
||||||
},
|
|
||||||
dropdown_filter: {
|
|
||||||
displayName: i18n.translate('xpack.canvas.elements.dropdownFilterDisplayName', {
|
displayName: i18n.translate('xpack.canvas.elements.dropdownFilterDisplayName', {
|
||||||
defaultMessage: 'Dropdown filter',
|
defaultMessage: 'Dropdown select',
|
||||||
}),
|
}),
|
||||||
help: i18n.translate('xpack.canvas.elements.dropdownFilterHelpText', {
|
help: i18n.translate('xpack.canvas.elements.dropdownFilterHelpText', {
|
||||||
defaultMessage: 'A dropdown from which you can select values for an "exactly" filter',
|
defaultMessage: 'A dropdown from which you can select values for an "exactly" filter',
|
||||||
}),
|
}),
|
||||||
},
|
},
|
||||||
|
filterDebug: {
|
||||||
|
displayName: i18n.translate('xpack.canvas.elements.filterDebugDisplayName', {
|
||||||
|
defaultMessage: 'Debug filters',
|
||||||
|
}),
|
||||||
|
help: i18n.translate('xpack.canvas.elements.filterDebugHelpText', {
|
||||||
|
defaultMessage: 'Shows the underlying global filters in a workpad',
|
||||||
|
}),
|
||||||
|
},
|
||||||
horizontalBarChart: {
|
horizontalBarChart: {
|
||||||
displayName: i18n.translate('xpack.canvas.elements.horizontalBarChartDisplayName', {
|
displayName: i18n.translate('xpack.canvas.elements.horizontalBarChartDisplayName', {
|
||||||
defaultMessage: 'Horizontal bar chart',
|
defaultMessage: 'Horizontal bar',
|
||||||
}),
|
}),
|
||||||
help: i18n.translate('xpack.canvas.elements.horizontalBarChartHelpText', {
|
help: i18n.translate('xpack.canvas.elements.horizontalBarChartHelpText', {
|
||||||
defaultMessage: 'A customizable horizontal bar chart',
|
defaultMessage: 'A customizable horizontal bar chart',
|
||||||
|
@ -71,7 +71,7 @@ export const getElementStrings = (): ElementStringDict => ({
|
||||||
},
|
},
|
||||||
horizontalProgressBar: {
|
horizontalProgressBar: {
|
||||||
displayName: i18n.translate('xpack.canvas.elements.horizontalProgressBarDisplayName', {
|
displayName: i18n.translate('xpack.canvas.elements.horizontalProgressBarDisplayName', {
|
||||||
defaultMessage: 'Horizontal progress bar',
|
defaultMessage: 'Horizontal bar',
|
||||||
}),
|
}),
|
||||||
help: i18n.translate('xpack.canvas.elements.horizontalProgressBarHelpText', {
|
help: i18n.translate('xpack.canvas.elements.horizontalProgressBarHelpText', {
|
||||||
defaultMessage: 'Displays progress as a portion of a horizontal bar',
|
defaultMessage: 'Displays progress as a portion of a horizontal bar',
|
||||||
|
@ -79,7 +79,7 @@ export const getElementStrings = (): ElementStringDict => ({
|
||||||
},
|
},
|
||||||
horizontalProgressPill: {
|
horizontalProgressPill: {
|
||||||
displayName: i18n.translate('xpack.canvas.elements.horizontalProgressPillDisplayName', {
|
displayName: i18n.translate('xpack.canvas.elements.horizontalProgressPillDisplayName', {
|
||||||
defaultMessage: 'Horizontal progress pill',
|
defaultMessage: 'Horizontal pill',
|
||||||
}),
|
}),
|
||||||
help: i18n.translate('xpack.canvas.elements.horizontalProgressPillHelpText', {
|
help: i18n.translate('xpack.canvas.elements.horizontalProgressPillHelpText', {
|
||||||
defaultMessage: 'Displays progress as a portion of a horizontal pill',
|
defaultMessage: 'Displays progress as a portion of a horizontal pill',
|
||||||
|
@ -95,7 +95,7 @@ export const getElementStrings = (): ElementStringDict => ({
|
||||||
},
|
},
|
||||||
lineChart: {
|
lineChart: {
|
||||||
displayName: i18n.translate('xpack.canvas.elements.lineChartDisplayName', {
|
displayName: i18n.translate('xpack.canvas.elements.lineChartDisplayName', {
|
||||||
defaultMessage: 'Line chart',
|
defaultMessage: 'Line',
|
||||||
}),
|
}),
|
||||||
help: i18n.translate('xpack.canvas.elements.lineChartHelpText', {
|
help: i18n.translate('xpack.canvas.elements.lineChartHelpText', {
|
||||||
defaultMessage: 'A customizable line chart',
|
defaultMessage: 'A customizable line chart',
|
||||||
|
@ -119,7 +119,7 @@ export const getElementStrings = (): ElementStringDict => ({
|
||||||
},
|
},
|
||||||
pie: {
|
pie: {
|
||||||
displayName: i18n.translate('xpack.canvas.elements.pieDisplayName', {
|
displayName: i18n.translate('xpack.canvas.elements.pieDisplayName', {
|
||||||
defaultMessage: 'Pie chart',
|
defaultMessage: 'Pie',
|
||||||
}),
|
}),
|
||||||
help: i18n.translate('xpack.canvas.elements.pieHelpText', {
|
help: i18n.translate('xpack.canvas.elements.pieHelpText', {
|
||||||
defaultMessage: 'Pie chart',
|
defaultMessage: 'Pie chart',
|
||||||
|
@ -135,7 +135,7 @@ export const getElementStrings = (): ElementStringDict => ({
|
||||||
},
|
},
|
||||||
progressGauge: {
|
progressGauge: {
|
||||||
displayName: i18n.translate('xpack.canvas.elements.progressGaugeDisplayName', {
|
displayName: i18n.translate('xpack.canvas.elements.progressGaugeDisplayName', {
|
||||||
defaultMessage: 'Progress gauge',
|
defaultMessage: 'Gauge',
|
||||||
}),
|
}),
|
||||||
help: i18n.translate('xpack.canvas.elements.progressGaugeHelpText', {
|
help: i18n.translate('xpack.canvas.elements.progressGaugeHelpText', {
|
||||||
defaultMessage: 'Displays progress as a portion of a gauge',
|
defaultMessage: 'Displays progress as a portion of a gauge',
|
||||||
|
@ -143,7 +143,7 @@ export const getElementStrings = (): ElementStringDict => ({
|
||||||
},
|
},
|
||||||
progressSemicircle: {
|
progressSemicircle: {
|
||||||
displayName: i18n.translate('xpack.canvas.elements.progressSemicircleDisplayName', {
|
displayName: i18n.translate('xpack.canvas.elements.progressSemicircleDisplayName', {
|
||||||
defaultMessage: 'Progress semicircle',
|
defaultMessage: 'Semicircle',
|
||||||
}),
|
}),
|
||||||
help: i18n.translate('xpack.canvas.elements.progressSemicircleHelpText', {
|
help: i18n.translate('xpack.canvas.elements.progressSemicircleHelpText', {
|
||||||
defaultMessage: 'Displays progress as a portion of a semicircle',
|
defaultMessage: 'Displays progress as a portion of a semicircle',
|
||||||
|
@ -151,7 +151,7 @@ export const getElementStrings = (): ElementStringDict => ({
|
||||||
},
|
},
|
||||||
progressWheel: {
|
progressWheel: {
|
||||||
displayName: i18n.translate('xpack.canvas.elements.progressWheelDisplayName', {
|
displayName: i18n.translate('xpack.canvas.elements.progressWheelDisplayName', {
|
||||||
defaultMessage: 'Progress wheel',
|
defaultMessage: 'Wheel',
|
||||||
}),
|
}),
|
||||||
help: i18n.translate('xpack.canvas.elements.progressWheelHelpText', {
|
help: i18n.translate('xpack.canvas.elements.progressWheelHelpText', {
|
||||||
defaultMessage: 'Displays progress as a portion of a wheel',
|
defaultMessage: 'Displays progress as a portion of a wheel',
|
||||||
|
@ -189,15 +189,7 @@ export const getElementStrings = (): ElementStringDict => ({
|
||||||
defaultMessage: 'A scrollable grid for displaying data in a tabular format',
|
defaultMessage: 'A scrollable grid for displaying data in a tabular format',
|
||||||
}),
|
}),
|
||||||
},
|
},
|
||||||
tiltedPie: {
|
timeFilter: {
|
||||||
displayName: i18n.translate('xpack.canvas.elements.tiltedPieDisplayName', {
|
|
||||||
defaultMessage: 'Tilted pie chart',
|
|
||||||
}),
|
|
||||||
help: i18n.translate('xpack.canvas.elements.tiltedPieHelpText', {
|
|
||||||
defaultMessage: 'A customizable tilted pie chart',
|
|
||||||
}),
|
|
||||||
},
|
|
||||||
time_filter: {
|
|
||||||
displayName: i18n.translate('xpack.canvas.elements.timeFilterDisplayName', {
|
displayName: i18n.translate('xpack.canvas.elements.timeFilterDisplayName', {
|
||||||
defaultMessage: 'Time filter',
|
defaultMessage: 'Time filter',
|
||||||
}),
|
}),
|
||||||
|
@ -207,7 +199,7 @@ export const getElementStrings = (): ElementStringDict => ({
|
||||||
},
|
},
|
||||||
verticalBarChart: {
|
verticalBarChart: {
|
||||||
displayName: i18n.translate('xpack.canvas.elements.verticalBarChartDisplayName', {
|
displayName: i18n.translate('xpack.canvas.elements.verticalBarChartDisplayName', {
|
||||||
defaultMessage: 'Vertical bar chart',
|
defaultMessage: 'Vertical bar',
|
||||||
}),
|
}),
|
||||||
help: i18n.translate('xpack.canvas.elements.verticalBarChartHelpText', {
|
help: i18n.translate('xpack.canvas.elements.verticalBarChartHelpText', {
|
||||||
defaultMessage: 'A customizable vertical bar chart',
|
defaultMessage: 'A customizable vertical bar chart',
|
||||||
|
@ -215,7 +207,7 @@ export const getElementStrings = (): ElementStringDict => ({
|
||||||
},
|
},
|
||||||
verticalProgressBar: {
|
verticalProgressBar: {
|
||||||
displayName: i18n.translate('xpack.canvas.elements.verticalProgressBarDisplayName', {
|
displayName: i18n.translate('xpack.canvas.elements.verticalProgressBarDisplayName', {
|
||||||
defaultMessage: 'Vertical progress bar',
|
defaultMessage: 'Vertical bar',
|
||||||
}),
|
}),
|
||||||
help: i18n.translate('xpack.canvas.elements.verticalProgressBarHelpText', {
|
help: i18n.translate('xpack.canvas.elements.verticalProgressBarHelpText', {
|
||||||
defaultMessage: 'Displays progress as a portion of a vertical bar',
|
defaultMessage: 'Displays progress as a portion of a vertical bar',
|
||||||
|
@ -223,7 +215,7 @@ export const getElementStrings = (): ElementStringDict => ({
|
||||||
},
|
},
|
||||||
verticalProgressPill: {
|
verticalProgressPill: {
|
||||||
displayName: i18n.translate('xpack.canvas.elements.verticalProgressPillDisplayName', {
|
displayName: i18n.translate('xpack.canvas.elements.verticalProgressPillDisplayName', {
|
||||||
defaultMessage: 'Vertical progress pill',
|
defaultMessage: 'Vertical pill',
|
||||||
}),
|
}),
|
||||||
help: i18n.translate('xpack.canvas.elements.verticalProgressPillHelpText', {
|
help: i18n.translate('xpack.canvas.elements.verticalProgressPillHelpText', {
|
||||||
defaultMessage: 'Displays progress as a portion of a vertical pill',
|
defaultMessage: 'Displays progress as a portion of a vertical pill',
|
||||||
|
|
|
@ -8,7 +8,7 @@ import { i18n } from '@kbn/i18n';
|
||||||
import { alterColumn } from '../../../canvas_plugin_src/functions/common/alterColumn';
|
import { alterColumn } from '../../../canvas_plugin_src/functions/common/alterColumn';
|
||||||
import { FunctionHelp } from '../function_help';
|
import { FunctionHelp } from '../function_help';
|
||||||
import { FunctionFactory } from '../../../types';
|
import { FunctionFactory } from '../../../types';
|
||||||
import { DATATABLE_COLUMN_TYPES } from '../../../common/lib';
|
import { DATATABLE_COLUMN_TYPES } from '../../../common/lib/constants';
|
||||||
|
|
||||||
export const help: FunctionHelp<FunctionFactory<typeof alterColumn>> = {
|
export const help: FunctionHelp<FunctionFactory<typeof alterColumn>> = {
|
||||||
help: i18n.translate('xpack.canvas.functions.alterColumnHelpText', {
|
help: i18n.translate('xpack.canvas.functions.alterColumnHelpText', {
|
||||||
|
|
|
@ -7,32 +7,12 @@
|
||||||
import { i18n } from '@kbn/i18n';
|
import { i18n } from '@kbn/i18n';
|
||||||
|
|
||||||
export const TagStrings: { [key: string]: () => string } = {
|
export const TagStrings: { [key: string]: () => string } = {
|
||||||
chart: () =>
|
|
||||||
i18n.translate('xpack.canvas.tags.chartTag', {
|
|
||||||
defaultMessage: 'chart',
|
|
||||||
}),
|
|
||||||
filter: () =>
|
|
||||||
i18n.translate('xpack.canvas.tags.filterTag', {
|
|
||||||
defaultMessage: 'filter',
|
|
||||||
}),
|
|
||||||
graphic: () =>
|
|
||||||
i18n.translate('xpack.canvas.tags.graphicTag', {
|
|
||||||
defaultMessage: 'graphic',
|
|
||||||
}),
|
|
||||||
presentation: () =>
|
presentation: () =>
|
||||||
i18n.translate('xpack.canvas.tags.presentationTag', {
|
i18n.translate('xpack.canvas.tags.presentationTag', {
|
||||||
defaultMessage: 'presentation',
|
defaultMessage: 'presentation',
|
||||||
}),
|
}),
|
||||||
proportion: () =>
|
|
||||||
i18n.translate('xpack.canvas.tags.proportionTag', {
|
|
||||||
defaultMessage: 'proportion',
|
|
||||||
}),
|
|
||||||
report: () =>
|
report: () =>
|
||||||
i18n.translate('xpack.canvas.tags.reportTag', {
|
i18n.translate('xpack.canvas.tags.reportTag', {
|
||||||
defaultMessage: 'report',
|
defaultMessage: 'report',
|
||||||
}),
|
}),
|
||||||
text: () =>
|
|
||||||
i18n.translate('xpack.canvas.tags.textTag', {
|
|
||||||
defaultMessage: 'text',
|
|
||||||
}),
|
|
||||||
};
|
};
|
||||||
|
|
|
@ -31,7 +31,7 @@ $canvasLayoutFontSize: $euiFontSizeS;
|
||||||
.canvasLayout__stageHeader {
|
.canvasLayout__stageHeader {
|
||||||
flex-grow: 0;
|
flex-grow: 0;
|
||||||
flex-basis: auto;
|
flex-basis: auto;
|
||||||
padding: ($euiSizeXS + 1px) $euiSize $euiSizeXS;
|
padding: 1px $euiSize 0;
|
||||||
font-size: $canvasLayoutFontSize;
|
font-size: $canvasLayoutFontSize;
|
||||||
border-bottom: $euiBorderThin;
|
border-bottom: $euiBorderThin;
|
||||||
background: $euiColorLightestShade;
|
background: $euiColorLightestShade;
|
||||||
|
|
|
@ -0,0 +1,761 @@
|
||||||
|
// Jest Snapshot v1, https://goo.gl/fbAQLP
|
||||||
|
|
||||||
|
exports[`Storyshots components/Assets/AssetManager no assets 1`] = `
|
||||||
|
Array [
|
||||||
|
<div
|
||||||
|
data-focus-guard={true}
|
||||||
|
style={
|
||||||
|
Object {
|
||||||
|
"height": "0px",
|
||||||
|
"left": "1px",
|
||||||
|
"overflow": "hidden",
|
||||||
|
"padding": 0,
|
||||||
|
"position": "fixed",
|
||||||
|
"top": "1px",
|
||||||
|
"width": "1px",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
tabIndex={0}
|
||||||
|
/>,
|
||||||
|
<div
|
||||||
|
data-focus-guard={true}
|
||||||
|
style={
|
||||||
|
Object {
|
||||||
|
"height": "0px",
|
||||||
|
"left": "1px",
|
||||||
|
"overflow": "hidden",
|
||||||
|
"padding": 0,
|
||||||
|
"position": "fixed",
|
||||||
|
"top": "1px",
|
||||||
|
"width": "1px",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
tabIndex={1}
|
||||||
|
/>,
|
||||||
|
<div
|
||||||
|
data-focus-lock-disabled={false}
|
||||||
|
onBlur={[Function]}
|
||||||
|
onFocus={[Function]}
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiModal canvasAssetManager canvasModal--fixedSize"
|
||||||
|
onKeyDown={[Function]}
|
||||||
|
style={
|
||||||
|
Object {
|
||||||
|
"maxWidth": "1000px",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
tabIndex={0}
|
||||||
|
>
|
||||||
|
<button
|
||||||
|
aria-label="Closes this modal window"
|
||||||
|
className="euiButtonIcon euiButtonIcon--text euiModal__closeIcon"
|
||||||
|
onClick={[Function]}
|
||||||
|
type="button"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
aria-hidden="true"
|
||||||
|
className="euiButtonIcon__icon"
|
||||||
|
data-euiicon-type="cross"
|
||||||
|
size="m"
|
||||||
|
/>
|
||||||
|
</button>
|
||||||
|
<div
|
||||||
|
className="euiModal__flex"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiModalHeader canvasAssetManager__modalHeader"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiModalHeader__title canvasAssetManager__modalHeaderTitle"
|
||||||
|
>
|
||||||
|
Manage workpad assets
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiFlexGroup euiFlexGroup--gutterLarge euiFlexGroup--directionRow euiFlexGroup--responsive canvasAssetManager__fileUploadWrapper"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiFlexItem euiFlexItem--flexGrowZero"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiFilePicker euiFilePicker--compressed"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiFilePicker__wrap"
|
||||||
|
>
|
||||||
|
<input
|
||||||
|
accept="image/*"
|
||||||
|
aria-describedby="generated-id"
|
||||||
|
className="euiFilePicker__input"
|
||||||
|
multiple={true}
|
||||||
|
onChange={[Function]}
|
||||||
|
onDragLeave={[Function]}
|
||||||
|
onDragOver={[Function]}
|
||||||
|
onDrop={[Function]}
|
||||||
|
type="file"
|
||||||
|
/>
|
||||||
|
<div
|
||||||
|
className="euiFilePicker__prompt"
|
||||||
|
id="generated-id"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
aria-hidden="true"
|
||||||
|
className="euiFilePicker__icon"
|
||||||
|
data-euiicon-type="importAction"
|
||||||
|
size="m"
|
||||||
|
/>
|
||||||
|
<div
|
||||||
|
className="euiFilePicker__promptText"
|
||||||
|
>
|
||||||
|
Select or drag and drop images
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiModalBody"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiModalBody__overflow"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiText euiText--small"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiTextColor euiTextColor--subdued"
|
||||||
|
>
|
||||||
|
<p>
|
||||||
|
Below are the image assets in this workpad. Any assets that are currently in use cannot be determined at this time. To reclaim space, delete assets.
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiSpacer euiSpacer--l"
|
||||||
|
/>
|
||||||
|
<div
|
||||||
|
className="euiPanel euiPanel--paddingMedium canvasAssetManager__emptyPanel"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiEmptyPrompt"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
color="subdued"
|
||||||
|
data-euiicon-type="importAction"
|
||||||
|
size="xxl"
|
||||||
|
/>
|
||||||
|
<div
|
||||||
|
className="euiSpacer euiSpacer--s"
|
||||||
|
/>
|
||||||
|
<span
|
||||||
|
className="euiTextColor euiTextColor--subdued"
|
||||||
|
>
|
||||||
|
<h2
|
||||||
|
className="euiTitle euiTitle--xsmall"
|
||||||
|
>
|
||||||
|
Import your assets to get started
|
||||||
|
</h2>
|
||||||
|
<div
|
||||||
|
className="euiSpacer euiSpacer--m"
|
||||||
|
/>
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiModalFooter canvasAssetManager__modalFooter"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiFlexGroup euiFlexGroup--gutterLarge euiFlexGroup--directionRow canvasAssetManager__meterWrapper"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiFlexItem"
|
||||||
|
>
|
||||||
|
<progress
|
||||||
|
aria-labelledby="CanvasAssetManagerLabel"
|
||||||
|
className="euiProgress euiProgress--native euiProgress--s euiProgress--secondary"
|
||||||
|
max={25000}
|
||||||
|
value={0}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiFlexItem euiFlexItem--flexGrowZero eui-textNoWrap"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiText euiText--medium"
|
||||||
|
id="CanvasAssetManagerLabel"
|
||||||
|
>
|
||||||
|
0% space used
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<button
|
||||||
|
className="euiButton euiButton--primary euiButton--small"
|
||||||
|
onClick={[Function]}
|
||||||
|
type="button"
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
className="euiButton__content"
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
className="euiButton__text"
|
||||||
|
>
|
||||||
|
Close
|
||||||
|
</span>
|
||||||
|
</span>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>,
|
||||||
|
<div
|
||||||
|
data-focus-guard={true}
|
||||||
|
style={
|
||||||
|
Object {
|
||||||
|
"height": "0px",
|
||||||
|
"left": "1px",
|
||||||
|
"overflow": "hidden",
|
||||||
|
"padding": 0,
|
||||||
|
"position": "fixed",
|
||||||
|
"top": "1px",
|
||||||
|
"width": "1px",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
tabIndex={0}
|
||||||
|
/>,
|
||||||
|
]
|
||||||
|
`;
|
||||||
|
|
||||||
|
exports[`Storyshots components/Assets/AssetManager two assets 1`] = `
|
||||||
|
Array [
|
||||||
|
<div
|
||||||
|
data-focus-guard={true}
|
||||||
|
style={
|
||||||
|
Object {
|
||||||
|
"height": "0px",
|
||||||
|
"left": "1px",
|
||||||
|
"overflow": "hidden",
|
||||||
|
"padding": 0,
|
||||||
|
"position": "fixed",
|
||||||
|
"top": "1px",
|
||||||
|
"width": "1px",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
tabIndex={0}
|
||||||
|
/>,
|
||||||
|
<div
|
||||||
|
data-focus-guard={true}
|
||||||
|
style={
|
||||||
|
Object {
|
||||||
|
"height": "0px",
|
||||||
|
"left": "1px",
|
||||||
|
"overflow": "hidden",
|
||||||
|
"padding": 0,
|
||||||
|
"position": "fixed",
|
||||||
|
"top": "1px",
|
||||||
|
"width": "1px",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
tabIndex={1}
|
||||||
|
/>,
|
||||||
|
<div
|
||||||
|
data-focus-lock-disabled={false}
|
||||||
|
onBlur={[Function]}
|
||||||
|
onFocus={[Function]}
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiModal canvasAssetManager canvasModal--fixedSize"
|
||||||
|
onKeyDown={[Function]}
|
||||||
|
style={
|
||||||
|
Object {
|
||||||
|
"maxWidth": "1000px",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
tabIndex={0}
|
||||||
|
>
|
||||||
|
<button
|
||||||
|
aria-label="Closes this modal window"
|
||||||
|
className="euiButtonIcon euiButtonIcon--text euiModal__closeIcon"
|
||||||
|
onClick={[Function]}
|
||||||
|
type="button"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
aria-hidden="true"
|
||||||
|
className="euiButtonIcon__icon"
|
||||||
|
data-euiicon-type="cross"
|
||||||
|
size="m"
|
||||||
|
/>
|
||||||
|
</button>
|
||||||
|
<div
|
||||||
|
className="euiModal__flex"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiModalHeader canvasAssetManager__modalHeader"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiModalHeader__title canvasAssetManager__modalHeaderTitle"
|
||||||
|
>
|
||||||
|
Manage workpad assets
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiFlexGroup euiFlexGroup--gutterLarge euiFlexGroup--directionRow euiFlexGroup--responsive canvasAssetManager__fileUploadWrapper"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiFlexItem euiFlexItem--flexGrowZero"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiFilePicker euiFilePicker--compressed"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiFilePicker__wrap"
|
||||||
|
>
|
||||||
|
<input
|
||||||
|
accept="image/*"
|
||||||
|
aria-describedby="generated-id"
|
||||||
|
className="euiFilePicker__input"
|
||||||
|
multiple={true}
|
||||||
|
onChange={[Function]}
|
||||||
|
onDragLeave={[Function]}
|
||||||
|
onDragOver={[Function]}
|
||||||
|
onDrop={[Function]}
|
||||||
|
type="file"
|
||||||
|
/>
|
||||||
|
<div
|
||||||
|
className="euiFilePicker__prompt"
|
||||||
|
id="generated-id"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
aria-hidden="true"
|
||||||
|
className="euiFilePicker__icon"
|
||||||
|
data-euiicon-type="importAction"
|
||||||
|
size="m"
|
||||||
|
/>
|
||||||
|
<div
|
||||||
|
className="euiFilePicker__promptText"
|
||||||
|
>
|
||||||
|
Select or drag and drop images
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiModalBody"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiModalBody__overflow"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiText euiText--small"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiTextColor euiTextColor--subdued"
|
||||||
|
>
|
||||||
|
<p>
|
||||||
|
Below are the image assets in this workpad. Any assets that are currently in use cannot be determined at this time. To reclaim space, delete assets.
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiSpacer euiSpacer--l"
|
||||||
|
/>
|
||||||
|
<div
|
||||||
|
className="euiFlexGrid euiFlexGrid--gutterLarge euiFlexGrid--fourths euiFlexGrid--responsive"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiFlexItem"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiPanel euiPanel--paddingSmall canvasAsset"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="canvasAsset__thumb canvasCheckered"
|
||||||
|
>
|
||||||
|
<figure
|
||||||
|
className="euiImage canvasAsset__img "
|
||||||
|
>
|
||||||
|
<img
|
||||||
|
alt="Asset thumbnail"
|
||||||
|
className="euiImage__img"
|
||||||
|
src=""
|
||||||
|
style={Object {}}
|
||||||
|
/>
|
||||||
|
</figure>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiSpacer euiSpacer--s"
|
||||||
|
/>
|
||||||
|
<div
|
||||||
|
className="euiText euiText--extraSmall eui-textBreakAll"
|
||||||
|
>
|
||||||
|
<p
|
||||||
|
className="eui-textBreakAll"
|
||||||
|
>
|
||||||
|
<strong>
|
||||||
|
airplane
|
||||||
|
</strong>
|
||||||
|
<br />
|
||||||
|
<span
|
||||||
|
className="euiTextColor euiTextColor--subdued"
|
||||||
|
>
|
||||||
|
<small>
|
||||||
|
(
|
||||||
|
1
|
||||||
|
kb)
|
||||||
|
</small>
|
||||||
|
</span>
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiSpacer euiSpacer--s"
|
||||||
|
/>
|
||||||
|
<div
|
||||||
|
className="euiFlexGroup euiFlexGroup--gutterLarge euiFlexGroup--alignItemsBaseline euiFlexGroup--justifyContentCenter euiFlexGroup--directionRow"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiFlexItem euiFlexItem--flexGrowZero asset-create-image"
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
className="euiToolTipAnchor"
|
||||||
|
onMouseOut={[Function]}
|
||||||
|
onMouseOver={[Function]}
|
||||||
|
>
|
||||||
|
<button
|
||||||
|
aria-label="Create image element"
|
||||||
|
className="euiButtonIcon euiButtonIcon--primary"
|
||||||
|
onBlur={[Function]}
|
||||||
|
onClick={[Function]}
|
||||||
|
onFocus={[Function]}
|
||||||
|
type="button"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
aria-hidden="true"
|
||||||
|
className="euiButtonIcon__icon"
|
||||||
|
data-euiicon-type="vector"
|
||||||
|
size="m"
|
||||||
|
/>
|
||||||
|
</button>
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiFlexItem euiFlexItem--flexGrowZero asset-download"
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
className="euiToolTipAnchor"
|
||||||
|
onMouseOut={[Function]}
|
||||||
|
onMouseOver={[Function]}
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="canvasDownload"
|
||||||
|
onClick={[Function]}
|
||||||
|
onKeyPress={[Function]}
|
||||||
|
role="button"
|
||||||
|
tabIndex={0}
|
||||||
|
>
|
||||||
|
<button
|
||||||
|
aria-label="Download"
|
||||||
|
className="euiButtonIcon euiButtonIcon--primary"
|
||||||
|
type="button"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
aria-hidden="true"
|
||||||
|
className="euiButtonIcon__icon"
|
||||||
|
data-euiicon-type="sortDown"
|
||||||
|
size="m"
|
||||||
|
/>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiFlexItem euiFlexItem--flexGrowZero"
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
className="euiToolTipAnchor"
|
||||||
|
onMouseOut={[Function]}
|
||||||
|
onMouseOver={[Function]}
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="canvasClipboard"
|
||||||
|
onClick={[Function]}
|
||||||
|
onKeyPress={[Function]}
|
||||||
|
role="button"
|
||||||
|
tabIndex={0}
|
||||||
|
>
|
||||||
|
<button
|
||||||
|
aria-label="Copy id to clipboard"
|
||||||
|
className="euiButtonIcon euiButtonIcon--primary"
|
||||||
|
type="button"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
aria-hidden="true"
|
||||||
|
className="euiButtonIcon__icon"
|
||||||
|
data-euiicon-type="copyClipboard"
|
||||||
|
size="m"
|
||||||
|
/>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiFlexItem euiFlexItem--flexGrowZero"
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
className="euiToolTipAnchor"
|
||||||
|
onMouseOut={[Function]}
|
||||||
|
onMouseOver={[Function]}
|
||||||
|
>
|
||||||
|
<button
|
||||||
|
aria-label="Delete"
|
||||||
|
className="euiButtonIcon euiButtonIcon--danger"
|
||||||
|
onBlur={[Function]}
|
||||||
|
onClick={[Function]}
|
||||||
|
onFocus={[Function]}
|
||||||
|
type="button"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
aria-hidden="true"
|
||||||
|
className="euiButtonIcon__icon"
|
||||||
|
data-euiicon-type="trash"
|
||||||
|
size="m"
|
||||||
|
/>
|
||||||
|
</button>
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiFlexItem"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiPanel euiPanel--paddingSmall canvasAsset"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="canvasAsset__thumb canvasCheckered"
|
||||||
|
>
|
||||||
|
<figure
|
||||||
|
className="euiImage canvasAsset__img "
|
||||||
|
>
|
||||||
|
<img
|
||||||
|
alt="Asset thumbnail"
|
||||||
|
className="euiImage__img"
|
||||||
|
src=""
|
||||||
|
style={Object {}}
|
||||||
|
/>
|
||||||
|
</figure>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiSpacer euiSpacer--s"
|
||||||
|
/>
|
||||||
|
<div
|
||||||
|
className="euiText euiText--extraSmall eui-textBreakAll"
|
||||||
|
>
|
||||||
|
<p
|
||||||
|
className="eui-textBreakAll"
|
||||||
|
>
|
||||||
|
<strong>
|
||||||
|
marker
|
||||||
|
</strong>
|
||||||
|
<br />
|
||||||
|
<span
|
||||||
|
className="euiTextColor euiTextColor--subdued"
|
||||||
|
>
|
||||||
|
<small>
|
||||||
|
(
|
||||||
|
1
|
||||||
|
kb)
|
||||||
|
</small>
|
||||||
|
</span>
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiSpacer euiSpacer--s"
|
||||||
|
/>
|
||||||
|
<div
|
||||||
|
className="euiFlexGroup euiFlexGroup--gutterLarge euiFlexGroup--alignItemsBaseline euiFlexGroup--justifyContentCenter euiFlexGroup--directionRow"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiFlexItem euiFlexItem--flexGrowZero asset-create-image"
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
className="euiToolTipAnchor"
|
||||||
|
onMouseOut={[Function]}
|
||||||
|
onMouseOver={[Function]}
|
||||||
|
>
|
||||||
|
<button
|
||||||
|
aria-label="Create image element"
|
||||||
|
className="euiButtonIcon euiButtonIcon--primary"
|
||||||
|
onBlur={[Function]}
|
||||||
|
onClick={[Function]}
|
||||||
|
onFocus={[Function]}
|
||||||
|
type="button"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
aria-hidden="true"
|
||||||
|
className="euiButtonIcon__icon"
|
||||||
|
data-euiicon-type="vector"
|
||||||
|
size="m"
|
||||||
|
/>
|
||||||
|
</button>
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiFlexItem euiFlexItem--flexGrowZero asset-download"
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
className="euiToolTipAnchor"
|
||||||
|
onMouseOut={[Function]}
|
||||||
|
onMouseOver={[Function]}
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="canvasDownload"
|
||||||
|
onClick={[Function]}
|
||||||
|
onKeyPress={[Function]}
|
||||||
|
role="button"
|
||||||
|
tabIndex={0}
|
||||||
|
>
|
||||||
|
<button
|
||||||
|
aria-label="Download"
|
||||||
|
className="euiButtonIcon euiButtonIcon--primary"
|
||||||
|
type="button"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
aria-hidden="true"
|
||||||
|
className="euiButtonIcon__icon"
|
||||||
|
data-euiicon-type="sortDown"
|
||||||
|
size="m"
|
||||||
|
/>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiFlexItem euiFlexItem--flexGrowZero"
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
className="euiToolTipAnchor"
|
||||||
|
onMouseOut={[Function]}
|
||||||
|
onMouseOver={[Function]}
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="canvasClipboard"
|
||||||
|
onClick={[Function]}
|
||||||
|
onKeyPress={[Function]}
|
||||||
|
role="button"
|
||||||
|
tabIndex={0}
|
||||||
|
>
|
||||||
|
<button
|
||||||
|
aria-label="Copy id to clipboard"
|
||||||
|
className="euiButtonIcon euiButtonIcon--primary"
|
||||||
|
type="button"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
aria-hidden="true"
|
||||||
|
className="euiButtonIcon__icon"
|
||||||
|
data-euiicon-type="copyClipboard"
|
||||||
|
size="m"
|
||||||
|
/>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiFlexItem euiFlexItem--flexGrowZero"
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
className="euiToolTipAnchor"
|
||||||
|
onMouseOut={[Function]}
|
||||||
|
onMouseOver={[Function]}
|
||||||
|
>
|
||||||
|
<button
|
||||||
|
aria-label="Delete"
|
||||||
|
className="euiButtonIcon euiButtonIcon--danger"
|
||||||
|
onBlur={[Function]}
|
||||||
|
onClick={[Function]}
|
||||||
|
onFocus={[Function]}
|
||||||
|
type="button"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
aria-hidden="true"
|
||||||
|
className="euiButtonIcon__icon"
|
||||||
|
data-euiicon-type="trash"
|
||||||
|
size="m"
|
||||||
|
/>
|
||||||
|
</button>
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiModalFooter canvasAssetManager__modalFooter"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiFlexGroup euiFlexGroup--gutterLarge euiFlexGroup--directionRow canvasAssetManager__meterWrapper"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiFlexItem"
|
||||||
|
>
|
||||||
|
<progress
|
||||||
|
aria-labelledby="CanvasAssetManagerLabel"
|
||||||
|
className="euiProgress euiProgress--native euiProgress--s euiProgress--secondary"
|
||||||
|
max={25000}
|
||||||
|
value={2}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiFlexItem euiFlexItem--flexGrowZero eui-textNoWrap"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiText euiText--medium"
|
||||||
|
id="CanvasAssetManagerLabel"
|
||||||
|
>
|
||||||
|
0% space used
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<button
|
||||||
|
className="euiButton euiButton--primary euiButton--small"
|
||||||
|
onClick={[Function]}
|
||||||
|
type="button"
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
className="euiButton__content"
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
className="euiButton__text"
|
||||||
|
>
|
||||||
|
Close
|
||||||
|
</span>
|
||||||
|
</span>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>,
|
||||||
|
<div
|
||||||
|
data-focus-guard={true}
|
||||||
|
style={
|
||||||
|
Object {
|
||||||
|
"height": "0px",
|
||||||
|
"left": "1px",
|
||||||
|
"overflow": "hidden",
|
||||||
|
"padding": 0,
|
||||||
|
"position": "fixed",
|
||||||
|
"top": "1px",
|
||||||
|
"width": "1px",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
tabIndex={0}
|
||||||
|
/>,
|
||||||
|
]
|
||||||
|
`;
|
|
@ -28,12 +28,12 @@ const MARKER: AssetType = {
|
||||||
|
|
||||||
storiesOf('components/Assets/AssetManager', module)
|
storiesOf('components/Assets/AssetManager', module)
|
||||||
.add('no assets', () => (
|
.add('no assets', () => (
|
||||||
// @ts-ignore @types/react has not been updated to support defaultProps yet.
|
|
||||||
<AssetManager
|
<AssetManager
|
||||||
onAddImageElement={action('onAddImageElement')}
|
onAddImageElement={action('onAddImageElement')}
|
||||||
onAssetAdd={action('onAssetAdd')}
|
onAssetAdd={action('onAssetAdd')}
|
||||||
onAssetCopy={action('onAssetCopy')}
|
onAssetCopy={action('onAssetCopy')}
|
||||||
onAssetDelete={action('onAssetDelete')}
|
onAssetDelete={action('onAssetDelete')}
|
||||||
|
onClose={action('onClose')}
|
||||||
/>
|
/>
|
||||||
))
|
))
|
||||||
.add('two assets', () => (
|
.add('two assets', () => (
|
||||||
|
@ -43,5 +43,6 @@ storiesOf('components/Assets/AssetManager', module)
|
||||||
onAssetAdd={action('onAssetAdd')}
|
onAssetAdd={action('onAssetAdd')}
|
||||||
onAssetCopy={action('onAssetCopy')}
|
onAssetCopy={action('onAssetCopy')}
|
||||||
onAssetDelete={action('onAssetDelete')}
|
onAssetDelete={action('onAssetDelete')}
|
||||||
|
onClose={action('onClose')}
|
||||||
/>
|
/>
|
||||||
));
|
));
|
|
@ -4,13 +4,6 @@
|
||||||
* you may not use this file except in compliance with the Elastic License.
|
* you may not use this file except in compliance with the Elastic License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import {
|
|
||||||
EuiButtonEmpty,
|
|
||||||
// @ts-ignore (elastic/eui#1557) EuiFilePicker is not exported yet
|
|
||||||
EuiFilePicker,
|
|
||||||
// @ts-ignore (elastic/eui#1557) EuiImage is not exported yet
|
|
||||||
EuiImage,
|
|
||||||
} from '@elastic/eui';
|
|
||||||
import PropTypes from 'prop-types';
|
import PropTypes from 'prop-types';
|
||||||
import React, { Fragment, PureComponent } from 'react';
|
import React, { Fragment, PureComponent } from 'react';
|
||||||
|
|
||||||
|
@ -33,13 +26,13 @@ interface Props {
|
||||||
onAssetCopy: () => void;
|
onAssetCopy: () => void;
|
||||||
/** Function to invoke when an asset is added */
|
/** Function to invoke when an asset is added */
|
||||||
onAssetAdd: (asset: File) => void;
|
onAssetAdd: (asset: File) => void;
|
||||||
|
/** Function to invoke when an asset modal is closed */
|
||||||
|
onClose: () => void;
|
||||||
}
|
}
|
||||||
|
|
||||||
interface State {
|
interface State {
|
||||||
/** The id of the asset to delete, if applicable. Is set if the viewer clicks the delete icon */
|
/** The id of the asset to delete, if applicable. Is set if the viewer clicks the delete icon */
|
||||||
deleteId: string | null;
|
deleteId: string | null;
|
||||||
/** Determines if the modal is currently visible */
|
|
||||||
isModalVisible: boolean;
|
|
||||||
/** Indicates if the modal is currently loading */
|
/** Indicates if the modal is currently loading */
|
||||||
isLoading: boolean;
|
isLoading: boolean;
|
||||||
}
|
}
|
||||||
|
@ -51,6 +44,7 @@ export class AssetManager extends PureComponent<Props, State> {
|
||||||
onAssetAdd: PropTypes.func.isRequired,
|
onAssetAdd: PropTypes.func.isRequired,
|
||||||
onAssetCopy: PropTypes.func.isRequired,
|
onAssetCopy: PropTypes.func.isRequired,
|
||||||
onAssetDelete: PropTypes.func.isRequired,
|
onAssetDelete: PropTypes.func.isRequired,
|
||||||
|
onClose: PropTypes.func.isRequired,
|
||||||
};
|
};
|
||||||
|
|
||||||
public static defaultProps = {
|
public static defaultProps = {
|
||||||
|
@ -60,12 +54,11 @@ export class AssetManager extends PureComponent<Props, State> {
|
||||||
public state = {
|
public state = {
|
||||||
deleteId: null,
|
deleteId: null,
|
||||||
isLoading: false,
|
isLoading: false,
|
||||||
isModalVisible: false,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
public render() {
|
public render() {
|
||||||
const { isModalVisible, isLoading } = this.state;
|
const { isLoading } = this.state;
|
||||||
const { assetValues, onAssetCopy, onAddImageElement } = this.props;
|
const { assetValues, onAssetCopy, onAddImageElement, onClose } = this.props;
|
||||||
|
|
||||||
const assetModal = (
|
const assetModal = (
|
||||||
<AssetModal
|
<AssetModal
|
||||||
|
@ -74,10 +67,10 @@ export class AssetManager extends PureComponent<Props, State> {
|
||||||
onAssetCopy={onAssetCopy}
|
onAssetCopy={onAssetCopy}
|
||||||
onAssetCreate={(createdAsset: AssetType) => {
|
onAssetCreate={(createdAsset: AssetType) => {
|
||||||
onAddImageElement(createdAsset.id);
|
onAddImageElement(createdAsset.id);
|
||||||
this.setState({ isModalVisible: false });
|
onClose();
|
||||||
}}
|
}}
|
||||||
onAssetDelete={(asset: AssetType) => this.setState({ deleteId: asset.id })}
|
onAssetDelete={(asset: AssetType) => this.setState({ deleteId: asset.id })}
|
||||||
onClose={() => this.setState({ isModalVisible: false })}
|
onClose={onClose}
|
||||||
onFileUpload={this.handleFileUpload}
|
onFileUpload={this.handleFileUpload}
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
|
@ -95,14 +88,12 @@ export class AssetManager extends PureComponent<Props, State> {
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Fragment>
|
<Fragment>
|
||||||
<EuiButtonEmpty onClick={this.showModal}>{strings.getButtonLabel()}</EuiButtonEmpty>
|
{assetModal}
|
||||||
{isModalVisible ? assetModal : null}
|
|
||||||
{confirmModal}
|
{confirmModal}
|
||||||
</Fragment>
|
</Fragment>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
private showModal = () => this.setState({ isModalVisible: true });
|
|
||||||
private resetDelete = () => this.setState({ deleteId: null });
|
private resetDelete = () => this.setState({ deleteId: null });
|
||||||
|
|
||||||
private doDelete = () => {
|
private doDelete = () => {
|
||||||
|
|
|
@ -98,6 +98,7 @@ export const AssetModal: FunctionComponent<Props> = props => {
|
||||||
<EuiFilePicker
|
<EuiFilePicker
|
||||||
initialPromptText={strings.getFilePickerPromptText()}
|
initialPromptText={strings.getFilePickerPromptText()}
|
||||||
compressed
|
compressed
|
||||||
|
display="default"
|
||||||
multiple
|
multiple
|
||||||
onChange={onFileUpload}
|
onChange={onFileUpload}
|
||||||
accept="image/*"
|
accept="image/*"
|
||||||
|
|
|
@ -1,797 +0,0 @@
|
||||||
// Jest Snapshot v1, https://goo.gl/fbAQLP
|
|
||||||
|
|
||||||
exports[`Storyshots components/Elements/ElementGrid with controls 1`] = `
|
|
||||||
<div
|
|
||||||
style={
|
|
||||||
Object {
|
|
||||||
"width": "1000px",
|
|
||||||
}
|
|
||||||
}
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
className="euiFlexGrid euiFlexGrid--gutterLarge euiFlexGrid--fourths euiFlexGrid--responsive"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
className="euiFlexItem canvasElementCard__wrapper"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
className="euiCard euiCard--leftAligned euiCard--isClickable canvasElementCard"
|
|
||||||
onClick={[Function]}
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
className="euiCard__top"
|
|
||||||
>
|
|
||||||
<img
|
|
||||||
alt=""
|
|
||||||
className="euiCard__image"
|
|
||||||
src=""
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
className="euiCard__content"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
className="euiTitle euiTitle--small euiCard__title"
|
|
||||||
id="generated-idTitle"
|
|
||||||
>
|
|
||||||
<button
|
|
||||||
aria-describedby=" generated-idDescription"
|
|
||||||
className="euiCard__titleButton"
|
|
||||||
onClick={[Function]}
|
|
||||||
>
|
|
||||||
Custom Element 1
|
|
||||||
</button>
|
|
||||||
</span>
|
|
||||||
<div
|
|
||||||
className="euiText euiText--small euiCard__description"
|
|
||||||
id="generated-idDescription"
|
|
||||||
>
|
|
||||||
<p>
|
|
||||||
sample description
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
className="euiCard__footer"
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
className="euiFlexGroup euiFlexGroup--gutterExtraSmall euiFlexGroup--justifyContentSpaceBetween euiFlexGroup--directionRow euiFlexGroup--responsive canvasElementCard__controls"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
className="euiFlexItem euiFlexItem--flexGrowZero"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
className="euiToolTipAnchor"
|
|
||||||
onMouseOut={[Function]}
|
|
||||||
onMouseOver={[Function]}
|
|
||||||
>
|
|
||||||
<button
|
|
||||||
aria-label="Edit element"
|
|
||||||
className="euiButtonIcon euiButtonIcon--primary"
|
|
||||||
onBlur={[Function]}
|
|
||||||
onClick={[Function]}
|
|
||||||
onFocus={[Function]}
|
|
||||||
type="button"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
aria-hidden="true"
|
|
||||||
className="euiButtonIcon__icon"
|
|
||||||
data-euiicon-type="pencil"
|
|
||||||
size="m"
|
|
||||||
/>
|
|
||||||
</button>
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
className="euiFlexItem euiFlexItem--flexGrowZero"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
className="euiToolTipAnchor"
|
|
||||||
onMouseOut={[Function]}
|
|
||||||
onMouseOver={[Function]}
|
|
||||||
>
|
|
||||||
<button
|
|
||||||
aria-label="Delete element"
|
|
||||||
className="euiButtonIcon euiButtonIcon--danger"
|
|
||||||
onBlur={[Function]}
|
|
||||||
onClick={[Function]}
|
|
||||||
onFocus={[Function]}
|
|
||||||
type="button"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
aria-hidden="true"
|
|
||||||
className="euiButtonIcon__icon"
|
|
||||||
data-euiicon-type="trash"
|
|
||||||
size="m"
|
|
||||||
/>
|
|
||||||
</button>
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
className="euiFlexItem canvasElementCard__wrapper"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
className="euiCard euiCard--leftAligned euiCard--isClickable canvasElementCard"
|
|
||||||
onClick={[Function]}
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
className="euiCard__top"
|
|
||||||
>
|
|
||||||
<img
|
|
||||||
alt=""
|
|
||||||
className="euiCard__image"
|
|
||||||
src=""
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
className="euiCard__content"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
className="euiTitle euiTitle--small euiCard__title"
|
|
||||||
id="generated-idTitle"
|
|
||||||
>
|
|
||||||
<button
|
|
||||||
aria-describedby=" generated-idDescription"
|
|
||||||
className="euiCard__titleButton"
|
|
||||||
onClick={[Function]}
|
|
||||||
>
|
|
||||||
Custom Element 2
|
|
||||||
</button>
|
|
||||||
</span>
|
|
||||||
<div
|
|
||||||
className="euiText euiText--small euiCard__description"
|
|
||||||
id="generated-idDescription"
|
|
||||||
>
|
|
||||||
<p>
|
|
||||||
Aenean eu justo auctor, placerat felis non, scelerisque dolor.
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
className="euiCard__footer"
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
className="euiFlexGroup euiFlexGroup--gutterExtraSmall euiFlexGroup--justifyContentSpaceBetween euiFlexGroup--directionRow euiFlexGroup--responsive canvasElementCard__controls"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
className="euiFlexItem euiFlexItem--flexGrowZero"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
className="euiToolTipAnchor"
|
|
||||||
onMouseOut={[Function]}
|
|
||||||
onMouseOver={[Function]}
|
|
||||||
>
|
|
||||||
<button
|
|
||||||
aria-label="Edit element"
|
|
||||||
className="euiButtonIcon euiButtonIcon--primary"
|
|
||||||
onBlur={[Function]}
|
|
||||||
onClick={[Function]}
|
|
||||||
onFocus={[Function]}
|
|
||||||
type="button"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
aria-hidden="true"
|
|
||||||
className="euiButtonIcon__icon"
|
|
||||||
data-euiicon-type="pencil"
|
|
||||||
size="m"
|
|
||||||
/>
|
|
||||||
</button>
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
className="euiFlexItem euiFlexItem--flexGrowZero"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
className="euiToolTipAnchor"
|
|
||||||
onMouseOut={[Function]}
|
|
||||||
onMouseOver={[Function]}
|
|
||||||
>
|
|
||||||
<button
|
|
||||||
aria-label="Delete element"
|
|
||||||
className="euiButtonIcon euiButtonIcon--danger"
|
|
||||||
onBlur={[Function]}
|
|
||||||
onClick={[Function]}
|
|
||||||
onFocus={[Function]}
|
|
||||||
type="button"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
aria-hidden="true"
|
|
||||||
className="euiButtonIcon__icon"
|
|
||||||
data-euiicon-type="trash"
|
|
||||||
size="m"
|
|
||||||
/>
|
|
||||||
</button>
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
className="euiFlexItem canvasElementCard__wrapper"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
className="euiCard euiCard--leftAligned euiCard--isClickable canvasElementCard"
|
|
||||||
onClick={[Function]}
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
className="euiCard__top"
|
|
||||||
>
|
|
||||||
<img
|
|
||||||
alt=""
|
|
||||||
className="euiCard__image"
|
|
||||||
src=""
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
className="euiCard__content"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
className="euiTitle euiTitle--small euiCard__title"
|
|
||||||
id="generated-idTitle"
|
|
||||||
>
|
|
||||||
<button
|
|
||||||
aria-describedby=" generated-idDescription"
|
|
||||||
className="euiCard__titleButton"
|
|
||||||
onClick={[Function]}
|
|
||||||
>
|
|
||||||
Custom Element 3
|
|
||||||
</button>
|
|
||||||
</span>
|
|
||||||
<div
|
|
||||||
className="euiText euiText--small euiCard__description"
|
|
||||||
id="generated-idDescription"
|
|
||||||
>
|
|
||||||
<p>
|
|
||||||
Lorem ipsum dolor sit amet, consectetur adipiscing elit. Fusce lobortis aliquet arcu ut turpis duis.
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
className="euiCard__footer"
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
className="euiFlexGroup euiFlexGroup--gutterExtraSmall euiFlexGroup--justifyContentSpaceBetween euiFlexGroup--directionRow euiFlexGroup--responsive canvasElementCard__controls"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
className="euiFlexItem euiFlexItem--flexGrowZero"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
className="euiToolTipAnchor"
|
|
||||||
onMouseOut={[Function]}
|
|
||||||
onMouseOver={[Function]}
|
|
||||||
>
|
|
||||||
<button
|
|
||||||
aria-label="Edit element"
|
|
||||||
className="euiButtonIcon euiButtonIcon--primary"
|
|
||||||
onBlur={[Function]}
|
|
||||||
onClick={[Function]}
|
|
||||||
onFocus={[Function]}
|
|
||||||
type="button"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
aria-hidden="true"
|
|
||||||
className="euiButtonIcon__icon"
|
|
||||||
data-euiicon-type="pencil"
|
|
||||||
size="m"
|
|
||||||
/>
|
|
||||||
</button>
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
className="euiFlexItem euiFlexItem--flexGrowZero"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
className="euiToolTipAnchor"
|
|
||||||
onMouseOut={[Function]}
|
|
||||||
onMouseOver={[Function]}
|
|
||||||
>
|
|
||||||
<button
|
|
||||||
aria-label="Delete element"
|
|
||||||
className="euiButtonIcon euiButtonIcon--danger"
|
|
||||||
onBlur={[Function]}
|
|
||||||
onClick={[Function]}
|
|
||||||
onFocus={[Function]}
|
|
||||||
type="button"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
aria-hidden="true"
|
|
||||||
className="euiButtonIcon__icon"
|
|
||||||
data-euiicon-type="trash"
|
|
||||||
size="m"
|
|
||||||
/>
|
|
||||||
</button>
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
`;
|
|
||||||
|
|
||||||
exports[`Storyshots components/Elements/ElementGrid with controls and filter 1`] = `
|
|
||||||
<div
|
|
||||||
style={
|
|
||||||
Object {
|
|
||||||
"width": "1000px",
|
|
||||||
}
|
|
||||||
}
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
className="euiFlexGrid euiFlexGrid--gutterLarge euiFlexGrid--fourths euiFlexGrid--responsive"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
className="euiFlexItem canvasElementCard__wrapper"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
className="euiCard euiCard--leftAligned euiCard--isClickable canvasElementCard"
|
|
||||||
onClick={[Function]}
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
className="euiCard__top"
|
|
||||||
>
|
|
||||||
<img
|
|
||||||
alt=""
|
|
||||||
className="euiCard__image"
|
|
||||||
src=""
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
className="euiCard__content"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
className="euiTitle euiTitle--small euiCard__title"
|
|
||||||
id="generated-idTitle"
|
|
||||||
>
|
|
||||||
<button
|
|
||||||
aria-describedby=" generated-idDescription"
|
|
||||||
className="euiCard__titleButton"
|
|
||||||
onClick={[Function]}
|
|
||||||
>
|
|
||||||
Custom Element 3
|
|
||||||
</button>
|
|
||||||
</span>
|
|
||||||
<div
|
|
||||||
className="euiText euiText--small euiCard__description"
|
|
||||||
id="generated-idDescription"
|
|
||||||
>
|
|
||||||
<p>
|
|
||||||
Lorem ipsum dolor sit amet, consectetur adipiscing elit. Fusce lobortis aliquet arcu ut turpis duis.
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
className="euiCard__footer"
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
className="euiFlexGroup euiFlexGroup--gutterExtraSmall euiFlexGroup--justifyContentSpaceBetween euiFlexGroup--directionRow euiFlexGroup--responsive canvasElementCard__controls"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
className="euiFlexItem euiFlexItem--flexGrowZero"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
className="euiToolTipAnchor"
|
|
||||||
onMouseOut={[Function]}
|
|
||||||
onMouseOver={[Function]}
|
|
||||||
>
|
|
||||||
<button
|
|
||||||
aria-label="Edit element"
|
|
||||||
className="euiButtonIcon euiButtonIcon--primary"
|
|
||||||
onBlur={[Function]}
|
|
||||||
onClick={[Function]}
|
|
||||||
onFocus={[Function]}
|
|
||||||
type="button"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
aria-hidden="true"
|
|
||||||
className="euiButtonIcon__icon"
|
|
||||||
data-euiicon-type="pencil"
|
|
||||||
size="m"
|
|
||||||
/>
|
|
||||||
</button>
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
className="euiFlexItem euiFlexItem--flexGrowZero"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
className="euiToolTipAnchor"
|
|
||||||
onMouseOut={[Function]}
|
|
||||||
onMouseOver={[Function]}
|
|
||||||
>
|
|
||||||
<button
|
|
||||||
aria-label="Delete element"
|
|
||||||
className="euiButtonIcon euiButtonIcon--danger"
|
|
||||||
onBlur={[Function]}
|
|
||||||
onClick={[Function]}
|
|
||||||
onFocus={[Function]}
|
|
||||||
type="button"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
aria-hidden="true"
|
|
||||||
className="euiButtonIcon__icon"
|
|
||||||
data-euiicon-type="trash"
|
|
||||||
size="m"
|
|
||||||
/>
|
|
||||||
</button>
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
`;
|
|
||||||
|
|
||||||
exports[`Storyshots components/Elements/ElementGrid with tags filter 1`] = `
|
|
||||||
<div
|
|
||||||
style={
|
|
||||||
Object {
|
|
||||||
"width": "1000px",
|
|
||||||
}
|
|
||||||
}
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
className="euiFlexGrid euiFlexGrid--gutterLarge euiFlexGrid--fourths euiFlexGrid--responsive"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
className="euiFlexItem canvasElementCard__wrapper"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
className="euiCard euiCard--leftAligned euiCard--isClickable canvasElementCard"
|
|
||||||
onClick={[Function]}
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
className="euiCard__top"
|
|
||||||
>
|
|
||||||
<img
|
|
||||||
alt=""
|
|
||||||
className="euiCard__image"
|
|
||||||
src=""
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
className="euiCard__content"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
className="euiTitle euiTitle--small euiCard__title"
|
|
||||||
id="generated-idTitle"
|
|
||||||
>
|
|
||||||
<button
|
|
||||||
aria-describedby=" generated-idDescription"
|
|
||||||
className="euiCard__titleButton"
|
|
||||||
onClick={[Function]}
|
|
||||||
>
|
|
||||||
Image
|
|
||||||
</button>
|
|
||||||
</span>
|
|
||||||
<div
|
|
||||||
className="euiText euiText--small euiCard__description"
|
|
||||||
id="generated-idDescription"
|
|
||||||
>
|
|
||||||
<p>
|
|
||||||
A static image
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
className="euiCard__footer"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
className="euiBadge euiBadge--iconLeft"
|
|
||||||
style={
|
|
||||||
Object {
|
|
||||||
"backgroundColor": "#666666",
|
|
||||||
"color": "#fff",
|
|
||||||
}
|
|
||||||
}
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
className="euiBadge__content"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
className="euiBadge__text"
|
|
||||||
>
|
|
||||||
graphic
|
|
||||||
</span>
|
|
||||||
</span>
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
`;
|
|
||||||
|
|
||||||
exports[`Storyshots components/Elements/ElementGrid with text filter 1`] = `
|
|
||||||
<div
|
|
||||||
style={
|
|
||||||
Object {
|
|
||||||
"width": "1000px",
|
|
||||||
}
|
|
||||||
}
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
className="euiFlexGrid euiFlexGrid--gutterLarge euiFlexGrid--fourths euiFlexGrid--responsive"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
className="euiFlexItem canvasElementCard__wrapper"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
className="euiCard euiCard--leftAligned euiCard--isClickable canvasElementCard"
|
|
||||||
onClick={[Function]}
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
className="euiCard__top"
|
|
||||||
>
|
|
||||||
<img
|
|
||||||
alt=""
|
|
||||||
className="euiCard__image"
|
|
||||||
src=""
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
className="euiCard__content"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
className="euiTitle euiTitle--small euiCard__title"
|
|
||||||
id="generated-idTitle"
|
|
||||||
>
|
|
||||||
<button
|
|
||||||
aria-describedby=" generated-idDescription"
|
|
||||||
className="euiCard__titleButton"
|
|
||||||
onClick={[Function]}
|
|
||||||
>
|
|
||||||
Data table
|
|
||||||
</button>
|
|
||||||
</span>
|
|
||||||
<div
|
|
||||||
className="euiText euiText--small euiCard__description"
|
|
||||||
id="generated-idDescription"
|
|
||||||
>
|
|
||||||
<p>
|
|
||||||
A scrollable grid for displaying data in a tabular format
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
className="euiCard__footer"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
className="euiBadge euiBadge--iconLeft"
|
|
||||||
style={
|
|
||||||
Object {
|
|
||||||
"backgroundColor": "#666666",
|
|
||||||
"color": "#fff",
|
|
||||||
}
|
|
||||||
}
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
className="euiBadge__content"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
className="euiBadge__text"
|
|
||||||
>
|
|
||||||
text
|
|
||||||
</span>
|
|
||||||
</span>
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
`;
|
|
||||||
|
|
||||||
exports[`Storyshots components/Elements/ElementGrid without controls 1`] = `
|
|
||||||
<div
|
|
||||||
style={
|
|
||||||
Object {
|
|
||||||
"width": "1000px",
|
|
||||||
}
|
|
||||||
}
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
className="euiFlexGrid euiFlexGrid--gutterLarge euiFlexGrid--fourths euiFlexGrid--responsive"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
className="euiFlexItem canvasElementCard__wrapper"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
className="euiCard euiCard--leftAligned euiCard--isClickable canvasElementCard"
|
|
||||||
onClick={[Function]}
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
className="euiCard__top"
|
|
||||||
>
|
|
||||||
<img
|
|
||||||
alt=""
|
|
||||||
className="euiCard__image"
|
|
||||||
src=""
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
className="euiCard__content"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
className="euiTitle euiTitle--small euiCard__title"
|
|
||||||
id="generated-idTitle"
|
|
||||||
>
|
|
||||||
<button
|
|
||||||
aria-describedby=" generated-idDescription"
|
|
||||||
className="euiCard__titleButton"
|
|
||||||
onClick={[Function]}
|
|
||||||
>
|
|
||||||
Area chart
|
|
||||||
</button>
|
|
||||||
</span>
|
|
||||||
<div
|
|
||||||
className="euiText euiText--small euiCard__description"
|
|
||||||
id="generated-idDescription"
|
|
||||||
>
|
|
||||||
<p>
|
|
||||||
A line chart with a filled body
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
className="euiCard__footer"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
className="euiBadge euiBadge--iconLeft"
|
|
||||||
style={
|
|
||||||
Object {
|
|
||||||
"backgroundColor": "#666666",
|
|
||||||
"color": "#fff",
|
|
||||||
}
|
|
||||||
}
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
className="euiBadge__content"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
className="euiBadge__text"
|
|
||||||
>
|
|
||||||
chart
|
|
||||||
</span>
|
|
||||||
</span>
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
className="euiFlexItem canvasElementCard__wrapper"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
className="euiCard euiCard--leftAligned euiCard--isClickable canvasElementCard"
|
|
||||||
onClick={[Function]}
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
className="euiCard__top"
|
|
||||||
>
|
|
||||||
<img
|
|
||||||
alt=""
|
|
||||||
className="euiCard__image"
|
|
||||||
src=""
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
className="euiCard__content"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
className="euiTitle euiTitle--small euiCard__title"
|
|
||||||
id="generated-idTitle"
|
|
||||||
>
|
|
||||||
<button
|
|
||||||
aria-describedby=" generated-idDescription"
|
|
||||||
className="euiCard__titleButton"
|
|
||||||
onClick={[Function]}
|
|
||||||
>
|
|
||||||
Image
|
|
||||||
</button>
|
|
||||||
</span>
|
|
||||||
<div
|
|
||||||
className="euiText euiText--small euiCard__description"
|
|
||||||
id="generated-idDescription"
|
|
||||||
>
|
|
||||||
<p>
|
|
||||||
A static image
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
className="euiCard__footer"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
className="euiBadge euiBadge--iconLeft"
|
|
||||||
style={
|
|
||||||
Object {
|
|
||||||
"backgroundColor": "#666666",
|
|
||||||
"color": "#fff",
|
|
||||||
}
|
|
||||||
}
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
className="euiBadge__content"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
className="euiBadge__text"
|
|
||||||
>
|
|
||||||
graphic
|
|
||||||
</span>
|
|
||||||
</span>
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
className="euiFlexItem canvasElementCard__wrapper"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
className="euiCard euiCard--leftAligned euiCard--isClickable canvasElementCard"
|
|
||||||
onClick={[Function]}
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
className="euiCard__top"
|
|
||||||
>
|
|
||||||
<img
|
|
||||||
alt=""
|
|
||||||
className="euiCard__image"
|
|
||||||
src=""
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
className="euiCard__content"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
className="euiTitle euiTitle--small euiCard__title"
|
|
||||||
id="generated-idTitle"
|
|
||||||
>
|
|
||||||
<button
|
|
||||||
aria-describedby=" generated-idDescription"
|
|
||||||
className="euiCard__titleButton"
|
|
||||||
onClick={[Function]}
|
|
||||||
>
|
|
||||||
Data table
|
|
||||||
</button>
|
|
||||||
</span>
|
|
||||||
<div
|
|
||||||
className="euiText euiText--small euiCard__description"
|
|
||||||
id="generated-idDescription"
|
|
||||||
>
|
|
||||||
<p>
|
|
||||||
A scrollable grid for displaying data in a tabular format
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
className="euiCard__footer"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
className="euiBadge euiBadge--iconLeft"
|
|
||||||
style={
|
|
||||||
Object {
|
|
||||||
"backgroundColor": "#666666",
|
|
||||||
"color": "#fff",
|
|
||||||
}
|
|
||||||
}
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
className="euiBadge__content"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
className="euiBadge__text"
|
|
||||||
>
|
|
||||||
text
|
|
||||||
</span>
|
|
||||||
</span>
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
`;
|
|
|
@ -1,110 +0,0 @@
|
||||||
/*
|
|
||||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
|
||||||
* or more contributor license agreements. Licensed under the Elastic License;
|
|
||||||
* you may not use this file except in compliance with the Elastic License.
|
|
||||||
*/
|
|
||||||
import React from 'react';
|
|
||||||
import PropTypes from 'prop-types';
|
|
||||||
import { map } from 'lodash';
|
|
||||||
import { EuiFlexItem, EuiFlexGrid } from '@elastic/eui';
|
|
||||||
import { ElementControls } from './element_controls';
|
|
||||||
import { CustomElement, ElementSpec } from '../../../types';
|
|
||||||
import { ElementCard } from '../element_card';
|
|
||||||
|
|
||||||
export interface Props {
|
|
||||||
/**
|
|
||||||
* list of elements to generate cards for
|
|
||||||
*/
|
|
||||||
elements: Array<ElementSpec | CustomElement>;
|
|
||||||
/**
|
|
||||||
* text to filter out cards
|
|
||||||
*/
|
|
||||||
filterText: string;
|
|
||||||
/**
|
|
||||||
* tags to filter out cards
|
|
||||||
*/
|
|
||||||
filterTags: string[];
|
|
||||||
/**
|
|
||||||
* indicate whether or not edit/delete controls should be displayed
|
|
||||||
*/
|
|
||||||
showControls: boolean;
|
|
||||||
/**
|
|
||||||
* handler invoked when clicking a card
|
|
||||||
*/
|
|
||||||
handleClick: (element: ElementSpec | CustomElement) => void;
|
|
||||||
/**
|
|
||||||
* click handler for the edit button
|
|
||||||
*/
|
|
||||||
onEdit?: (element: ElementSpec | CustomElement) => void;
|
|
||||||
/**
|
|
||||||
* click handler for the delete button
|
|
||||||
*/
|
|
||||||
onDelete?: (element: ElementSpec | CustomElement) => void;
|
|
||||||
}
|
|
||||||
|
|
||||||
export const ElementGrid = ({
|
|
||||||
elements,
|
|
||||||
filterText,
|
|
||||||
filterTags,
|
|
||||||
handleClick,
|
|
||||||
onEdit,
|
|
||||||
onDelete,
|
|
||||||
showControls,
|
|
||||||
}: Props) => {
|
|
||||||
filterText = filterText.toLowerCase();
|
|
||||||
|
|
||||||
return (
|
|
||||||
<EuiFlexGrid gutterSize="l" columns={4}>
|
|
||||||
{map(elements, (element: ElementSpec | CustomElement, index) => {
|
|
||||||
const { name, displayName = '', help = '', image, tags = [] } = element;
|
|
||||||
const whenClicked = () => handleClick(element);
|
|
||||||
let textMatch = false;
|
|
||||||
let tagsMatch = false;
|
|
||||||
|
|
||||||
if (
|
|
||||||
!filterText.length ||
|
|
||||||
name.toLowerCase().includes(filterText) ||
|
|
||||||
displayName.toLowerCase().includes(filterText) ||
|
|
||||||
help.toLowerCase().includes(filterText)
|
|
||||||
) {
|
|
||||||
textMatch = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!filterTags.length || filterTags.every(tag => tags.includes(tag))) {
|
|
||||||
tagsMatch = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!textMatch || !tagsMatch) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
|
||||||
<EuiFlexItem key={index} className="canvasElementCard__wrapper">
|
|
||||||
<ElementCard
|
|
||||||
title={displayName || name}
|
|
||||||
description={help}
|
|
||||||
image={image}
|
|
||||||
tags={tags}
|
|
||||||
onClick={whenClicked}
|
|
||||||
/>
|
|
||||||
{showControls && onEdit && onDelete && (
|
|
||||||
<ElementControls onEdit={() => onEdit(element)} onDelete={() => onDelete(element)} />
|
|
||||||
)}
|
|
||||||
</EuiFlexItem>
|
|
||||||
);
|
|
||||||
})}
|
|
||||||
</EuiFlexGrid>
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
ElementGrid.propTypes = {
|
|
||||||
elements: PropTypes.array.isRequired,
|
|
||||||
handleClick: PropTypes.func.isRequired,
|
|
||||||
showControls: PropTypes.bool,
|
|
||||||
};
|
|
||||||
|
|
||||||
ElementGrid.defaultProps = {
|
|
||||||
showControls: false,
|
|
||||||
filterTags: [],
|
|
||||||
filterText: '',
|
|
||||||
};
|
|
|
@ -1,224 +0,0 @@
|
||||||
/*
|
|
||||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
|
||||||
* or more contributor license agreements. Licensed under the Elastic License;
|
|
||||||
* you may not use this file except in compliance with the Elastic License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
import React, { Component, Fragment } from 'react';
|
|
||||||
import PropTypes from 'prop-types';
|
|
||||||
import {
|
|
||||||
EuiModalBody,
|
|
||||||
EuiTabbedContent,
|
|
||||||
EuiEmptyPrompt,
|
|
||||||
EuiSearchBar,
|
|
||||||
EuiSpacer,
|
|
||||||
EuiOverlayMask,
|
|
||||||
} from '@elastic/eui';
|
|
||||||
import { map, sortBy } from 'lodash';
|
|
||||||
import { ConfirmModal } from '../confirm_modal/confirm_modal';
|
|
||||||
import { CustomElementModal } from '../custom_element_modal';
|
|
||||||
import { getTagsFilter } from '../../lib/get_tags_filter';
|
|
||||||
import { extractSearch } from '../../lib/extract_search';
|
|
||||||
import { ComponentStrings } from '../../../i18n';
|
|
||||||
import { ElementGrid } from './element_grid';
|
|
||||||
|
|
||||||
const { ElementTypes: strings } = ComponentStrings;
|
|
||||||
|
|
||||||
const tagType = 'badge';
|
|
||||||
export class ElementTypes extends Component {
|
|
||||||
static propTypes = {
|
|
||||||
addCustomElement: PropTypes.func.isRequired,
|
|
||||||
addElement: PropTypes.func.isRequired,
|
|
||||||
customElements: PropTypes.array.isRequired,
|
|
||||||
elements: PropTypes.object,
|
|
||||||
filterTags: PropTypes.arrayOf(PropTypes.string).isRequired,
|
|
||||||
findCustomElements: PropTypes.func.isRequired,
|
|
||||||
onClose: PropTypes.func.isRequired,
|
|
||||||
removeCustomElement: PropTypes.func.isRequired,
|
|
||||||
search: PropTypes.string,
|
|
||||||
setCustomElements: PropTypes.func.isRequired,
|
|
||||||
setSearch: PropTypes.func.isRequired,
|
|
||||||
setFilterTags: PropTypes.func.isRequired,
|
|
||||||
updateCustomElement: PropTypes.func.isRequired,
|
|
||||||
};
|
|
||||||
|
|
||||||
state = {
|
|
||||||
elementToDelete: null,
|
|
||||||
elementToEdit: null,
|
|
||||||
};
|
|
||||||
|
|
||||||
componentDidMount() {
|
|
||||||
// fetch custom elements
|
|
||||||
this.props.findCustomElements();
|
|
||||||
}
|
|
||||||
|
|
||||||
_showEditModal = elementToEdit => this.setState({ elementToEdit });
|
|
||||||
|
|
||||||
_hideEditModal = () => this.setState({ elementToEdit: null });
|
|
||||||
|
|
||||||
_handleEdit = async (name, description, image) => {
|
|
||||||
const { updateCustomElement } = this.props;
|
|
||||||
const { elementToEdit } = this.state;
|
|
||||||
await updateCustomElement(elementToEdit.id, name, description, image);
|
|
||||||
this._hideEditModal();
|
|
||||||
};
|
|
||||||
|
|
||||||
_showDeleteModal = elementToDelete => this.setState({ elementToDelete });
|
|
||||||
|
|
||||||
_hideDeleteModal = () => this.setState({ elementToDelete: null });
|
|
||||||
|
|
||||||
_handleDelete = async () => {
|
|
||||||
const { removeCustomElement } = this.props;
|
|
||||||
const { elementToDelete } = this.state;
|
|
||||||
await removeCustomElement(elementToDelete.id);
|
|
||||||
this._hideDeleteModal();
|
|
||||||
};
|
|
||||||
|
|
||||||
_renderEditModal = () => {
|
|
||||||
const { elementToEdit } = this.state;
|
|
||||||
|
|
||||||
if (!elementToEdit) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
|
||||||
<EuiOverlayMask>
|
|
||||||
<CustomElementModal
|
|
||||||
title={strings.getEditElementTitle()}
|
|
||||||
name={elementToEdit.displayName}
|
|
||||||
description={elementToEdit.help}
|
|
||||||
image={elementToEdit.image}
|
|
||||||
onSave={this._handleEdit}
|
|
||||||
onCancel={this._hideEditModal}
|
|
||||||
/>
|
|
||||||
</EuiOverlayMask>
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
_renderDeleteModal = () => {
|
|
||||||
const { elementToDelete } = this.state;
|
|
||||||
|
|
||||||
if (!elementToDelete) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
|
||||||
<ConfirmModal
|
|
||||||
isOpen
|
|
||||||
title={strings.getDeleteElementTitle(elementToDelete.displayName)}
|
|
||||||
message={strings.getDeleteElementDescription()}
|
|
||||||
confirmButtonText={strings.getDeleteButtonLabel()}
|
|
||||||
cancelButtonText={strings.getCancelButtonLabel()}
|
|
||||||
onConfirm={this._handleDelete}
|
|
||||||
onCancel={this._hideDeleteModal}
|
|
||||||
/>
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
_sortElements = elements =>
|
|
||||||
sortBy(
|
|
||||||
map(elements, (element, name) => ({ name, ...element })),
|
|
||||||
'displayName'
|
|
||||||
);
|
|
||||||
|
|
||||||
render() {
|
|
||||||
const {
|
|
||||||
search,
|
|
||||||
setSearch,
|
|
||||||
addElement,
|
|
||||||
addCustomElement,
|
|
||||||
filterTags,
|
|
||||||
setFilterTags,
|
|
||||||
} = this.props;
|
|
||||||
let { elements, customElements } = this.props;
|
|
||||||
elements = this._sortElements(elements);
|
|
||||||
|
|
||||||
let customElementContent = (
|
|
||||||
<EuiEmptyPrompt
|
|
||||||
iconType="vector"
|
|
||||||
title={<h2>{strings.getAddNewElementTitle()}</h2>}
|
|
||||||
body={<p>{strings.getAddNewElementDescription()}</p>}
|
|
||||||
titleSize="s"
|
|
||||||
/>
|
|
||||||
);
|
|
||||||
|
|
||||||
if (customElements.length) {
|
|
||||||
customElements = this._sortElements(customElements);
|
|
||||||
customElementContent = (
|
|
||||||
<ElementGrid
|
|
||||||
elements={customElements}
|
|
||||||
filter={search}
|
|
||||||
handleClick={addCustomElement}
|
|
||||||
showControls
|
|
||||||
onEdit={this._showEditModal}
|
|
||||||
onDelete={this._showDeleteModal}
|
|
||||||
/>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
const filters = [getTagsFilter(tagType)];
|
|
||||||
const onSearch = ({ queryText }) => {
|
|
||||||
const { searchTerm, filterTags } = extractSearch(queryText);
|
|
||||||
setSearch(searchTerm);
|
|
||||||
setFilterTags(filterTags);
|
|
||||||
};
|
|
||||||
|
|
||||||
const tabs = [
|
|
||||||
{
|
|
||||||
id: 'elements',
|
|
||||||
name: strings.getElementsTitle(),
|
|
||||||
content: (
|
|
||||||
<div className="canvasElements__filter">
|
|
||||||
<EuiSpacer />
|
|
||||||
<EuiSearchBar
|
|
||||||
defaultQuery={search}
|
|
||||||
box={{
|
|
||||||
placeholder: strings.getFindElementPlaceholder(),
|
|
||||||
incremental: true,
|
|
||||||
}}
|
|
||||||
filters={filters}
|
|
||||||
onChange={onSearch}
|
|
||||||
/>
|
|
||||||
<EuiSpacer />
|
|
||||||
<ElementGrid
|
|
||||||
elements={elements}
|
|
||||||
filterText={search}
|
|
||||||
filterTags={filterTags}
|
|
||||||
handleClick={addElement}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
),
|
|
||||||
},
|
|
||||||
{
|
|
||||||
id: 'customElements',
|
|
||||||
name: strings.getMyElementsTitle(),
|
|
||||||
content: (
|
|
||||||
<Fragment>
|
|
||||||
<EuiSpacer />
|
|
||||||
<EuiSearchBar
|
|
||||||
defaultQuery={search}
|
|
||||||
box={{
|
|
||||||
placeholder: strings.getFindElementPlaceholder(),
|
|
||||||
incremental: true,
|
|
||||||
}}
|
|
||||||
onChange={onSearch}
|
|
||||||
/>
|
|
||||||
<EuiSpacer />
|
|
||||||
{customElementContent}
|
|
||||||
</Fragment>
|
|
||||||
),
|
|
||||||
},
|
|
||||||
];
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Fragment>
|
|
||||||
<EuiModalBody style={{ paddingRight: '1px' }}>
|
|
||||||
<EuiTabbedContent tabs={tabs} initialSelectedTab={tabs[0]} />
|
|
||||||
</EuiModalBody>
|
|
||||||
|
|
||||||
{this._renderDeleteModal()}
|
|
||||||
{this._renderEditModal()}
|
|
||||||
</Fragment>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,103 +0,0 @@
|
||||||
/*
|
|
||||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
|
||||||
* or more contributor license agreements. Licensed under the Elastic License;
|
|
||||||
* you may not use this file except in compliance with the Elastic License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
import PropTypes from 'prop-types';
|
|
||||||
import { compose, withProps, withState } from 'recompose';
|
|
||||||
import { connect } from 'react-redux';
|
|
||||||
import { camelCase } from 'lodash';
|
|
||||||
import { cloneSubgraphs } from '../../lib/clone_subgraphs';
|
|
||||||
import * as customElementService from '../../lib/custom_element_service';
|
|
||||||
import { elementsRegistry } from '../../lib/elements_registry';
|
|
||||||
import { notify } from '../../lib/notify';
|
|
||||||
import { selectToplevelNodes } from '../../state/actions/transient';
|
|
||||||
import { insertNodes, addElement } from '../../state/actions/elements';
|
|
||||||
import { getSelectedPage } from '../../state/selectors/workpad';
|
|
||||||
import { trackCanvasUiMetric, METRIC_TYPE } from '../../lib/ui_metric';
|
|
||||||
import { ElementTypes as Component } from './element_types';
|
|
||||||
|
|
||||||
const customElementAdded = 'elements-custom-added';
|
|
||||||
|
|
||||||
const mapStateToProps = state => ({ pageId: getSelectedPage(state) });
|
|
||||||
|
|
||||||
const mapDispatchToProps = dispatch => ({
|
|
||||||
selectToplevelNodes: nodes =>
|
|
||||||
dispatch(selectToplevelNodes(nodes.filter(e => !e.position.parent).map(e => e.id))),
|
|
||||||
insertNodes: (selectedNodes, pageId) => dispatch(insertNodes(selectedNodes, pageId)),
|
|
||||||
addElement: (pageId, partialElement) => dispatch(addElement(pageId, partialElement)),
|
|
||||||
});
|
|
||||||
|
|
||||||
const mergeProps = (stateProps, dispatchProps, ownProps) => {
|
|
||||||
const { pageId, ...remainingStateProps } = stateProps;
|
|
||||||
const { addElement, insertNodes, selectToplevelNodes } = dispatchProps;
|
|
||||||
const { search, setCustomElements, onClose } = ownProps;
|
|
||||||
|
|
||||||
return {
|
|
||||||
...remainingStateProps,
|
|
||||||
...ownProps,
|
|
||||||
// add built-in element to the page
|
|
||||||
addElement: element => {
|
|
||||||
addElement(pageId, element);
|
|
||||||
onClose();
|
|
||||||
},
|
|
||||||
// add custom element to the page
|
|
||||||
addCustomElement: customElement => {
|
|
||||||
const { selectedNodes = [] } = JSON.parse(customElement.content) || {};
|
|
||||||
const clonedNodes = selectedNodes && cloneSubgraphs(selectedNodes);
|
|
||||||
if (clonedNodes) {
|
|
||||||
insertNodes(clonedNodes, pageId); // first clone and persist the new node(s)
|
|
||||||
selectToplevelNodes(clonedNodes); // then select the cloned node(s)
|
|
||||||
}
|
|
||||||
onClose();
|
|
||||||
trackCanvasUiMetric(METRIC_TYPE.LOADED, customElementAdded);
|
|
||||||
},
|
|
||||||
// custom element search
|
|
||||||
findCustomElements: async text => {
|
|
||||||
try {
|
|
||||||
const { customElements } = await customElementService.find(text);
|
|
||||||
setCustomElements(customElements);
|
|
||||||
} catch (err) {
|
|
||||||
notify.error(err, { title: `Couldn't find custom elements` });
|
|
||||||
}
|
|
||||||
},
|
|
||||||
// remove custom element
|
|
||||||
removeCustomElement: async id => {
|
|
||||||
try {
|
|
||||||
await customElementService.remove(id).then();
|
|
||||||
const { customElements } = await customElementService.find(search);
|
|
||||||
setCustomElements(customElements);
|
|
||||||
} catch (err) {
|
|
||||||
notify.error(err, { title: `Couldn't delete custom elements` });
|
|
||||||
}
|
|
||||||
},
|
|
||||||
// update custom element
|
|
||||||
updateCustomElement: async (id, name, description, image) => {
|
|
||||||
try {
|
|
||||||
await customElementService.update(id, {
|
|
||||||
name: camelCase(name),
|
|
||||||
displayName: name,
|
|
||||||
image,
|
|
||||||
help: description,
|
|
||||||
});
|
|
||||||
const { customElements } = await customElementService.find(search);
|
|
||||||
setCustomElements(customElements);
|
|
||||||
} catch (err) {
|
|
||||||
notify.error(err, { title: `Couldn't update custom elements` });
|
|
||||||
}
|
|
||||||
},
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
export const ElementTypes = compose(
|
|
||||||
withState('search', 'setSearch', ''),
|
|
||||||
withState('customElements', 'setCustomElements', []),
|
|
||||||
withState('filterTags', 'setFilterTags', []),
|
|
||||||
withProps(() => ({ elements: elementsRegistry.toJS() })),
|
|
||||||
connect(mapStateToProps, mapDispatchToProps, mergeProps)
|
|
||||||
)(Component);
|
|
||||||
|
|
||||||
ElementTypes.propTypes = {
|
|
||||||
onClose: PropTypes.func,
|
|
||||||
};
|
|
|
@ -4,4 +4,4 @@
|
||||||
* you may not use this file except in compliance with the Elastic License.
|
* you may not use this file except in compliance with the Elastic License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
export { Popover } from './popover';
|
export { Popover, ClosePopoverFn } from './popover';
|
||||||
|
|
|
@ -24,6 +24,8 @@ interface Props {
|
||||||
className?: string;
|
className?: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export type ClosePopoverFn = () => void;
|
||||||
|
|
||||||
interface State {
|
interface State {
|
||||||
isPopoverOpen: boolean;
|
isPopoverOpen: boolean;
|
||||||
}
|
}
|
||||||
|
@ -61,7 +63,7 @@ export class Popover extends Component<Props, State> {
|
||||||
}));
|
}));
|
||||||
};
|
};
|
||||||
|
|
||||||
closePopover = () => {
|
closePopover: ClosePopoverFn = () => {
|
||||||
this.setState({
|
this.setState({
|
||||||
isPopoverOpen: false,
|
isPopoverOpen: false,
|
||||||
});
|
});
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
// Jest Snapshot v1, https://goo.gl/fbAQLP
|
// Jest Snapshot v1, https://goo.gl/fbAQLP
|
||||||
|
|
||||||
exports[`Storyshots components/Elements/ElementControls has two buttons 1`] = `
|
exports[`Storyshots components/SavedElementsModal/ElementControls has two buttons 1`] = `
|
||||||
<div
|
<div
|
||||||
style={
|
style={
|
||||||
Object {
|
Object {
|
||||||
|
@ -22,6 +22,7 @@ exports[`Storyshots components/Elements/ElementControls has two buttons 1`] = `
|
||||||
<button
|
<button
|
||||||
aria-label="Edit element"
|
aria-label="Edit element"
|
||||||
className="euiButtonIcon euiButtonIcon--primary"
|
className="euiButtonIcon euiButtonIcon--primary"
|
||||||
|
data-test-subj="canvasElementCard__editButton"
|
||||||
onBlur={[Function]}
|
onBlur={[Function]}
|
||||||
onClick={[Function]}
|
onClick={[Function]}
|
||||||
onFocus={[Function]}
|
onFocus={[Function]}
|
||||||
|
@ -47,6 +48,7 @@ exports[`Storyshots components/Elements/ElementControls has two buttons 1`] = `
|
||||||
<button
|
<button
|
||||||
aria-label="Delete element"
|
aria-label="Delete element"
|
||||||
className="euiButtonIcon euiButtonIcon--danger"
|
className="euiButtonIcon euiButtonIcon--danger"
|
||||||
|
data-test-subj="canvasElementCard__deleteButton"
|
||||||
onBlur={[Function]}
|
onBlur={[Function]}
|
||||||
onClick={[Function]}
|
onClick={[Function]}
|
||||||
onFocus={[Function]}
|
onFocus={[Function]}
|
|
@ -0,0 +1,333 @@
|
||||||
|
// Jest Snapshot v1, https://goo.gl/fbAQLP
|
||||||
|
|
||||||
|
exports[`Storyshots components/SavedElementsModal/ElementGrid default 1`] = `
|
||||||
|
<div
|
||||||
|
style={
|
||||||
|
Object {
|
||||||
|
"width": "1000px",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiFlexGrid euiFlexGrid--gutterLarge euiFlexGrid--fourths euiFlexGrid--responsive"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiFlexItem canvasElementCard__wrapper"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiCard euiCard--leftAligned euiCard--isClickable canvasElementCard"
|
||||||
|
onClick={[Function]}
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiCard__top"
|
||||||
|
>
|
||||||
|
<img
|
||||||
|
alt=""
|
||||||
|
className="euiCard__image"
|
||||||
|
src=""
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiCard__content"
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
className="euiTitle euiTitle--small euiCard__title"
|
||||||
|
id="generated-idTitle"
|
||||||
|
>
|
||||||
|
<button
|
||||||
|
aria-describedby=" generated-idDescription"
|
||||||
|
className="euiCard__titleButton"
|
||||||
|
onClick={[Function]}
|
||||||
|
>
|
||||||
|
Custom Element 1
|
||||||
|
</button>
|
||||||
|
</span>
|
||||||
|
<div
|
||||||
|
className="euiText euiText--small euiCard__description"
|
||||||
|
id="generated-idDescription"
|
||||||
|
>
|
||||||
|
<p>
|
||||||
|
sample description
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiCard__footer"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiFlexGroup euiFlexGroup--gutterExtraSmall euiFlexGroup--justifyContentSpaceBetween euiFlexGroup--directionRow euiFlexGroup--responsive canvasElementCard__controls"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiFlexItem euiFlexItem--flexGrowZero"
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
className="euiToolTipAnchor"
|
||||||
|
onMouseOut={[Function]}
|
||||||
|
onMouseOver={[Function]}
|
||||||
|
>
|
||||||
|
<button
|
||||||
|
aria-label="Edit element"
|
||||||
|
className="euiButtonIcon euiButtonIcon--primary"
|
||||||
|
data-test-subj="canvasElementCard__editButton"
|
||||||
|
onBlur={[Function]}
|
||||||
|
onClick={[Function]}
|
||||||
|
onFocus={[Function]}
|
||||||
|
type="button"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
aria-hidden="true"
|
||||||
|
className="euiButtonIcon__icon"
|
||||||
|
data-euiicon-type="pencil"
|
||||||
|
size="m"
|
||||||
|
/>
|
||||||
|
</button>
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiFlexItem euiFlexItem--flexGrowZero"
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
className="euiToolTipAnchor"
|
||||||
|
onMouseOut={[Function]}
|
||||||
|
onMouseOver={[Function]}
|
||||||
|
>
|
||||||
|
<button
|
||||||
|
aria-label="Delete element"
|
||||||
|
className="euiButtonIcon euiButtonIcon--danger"
|
||||||
|
data-test-subj="canvasElementCard__deleteButton"
|
||||||
|
onBlur={[Function]}
|
||||||
|
onClick={[Function]}
|
||||||
|
onFocus={[Function]}
|
||||||
|
type="button"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
aria-hidden="true"
|
||||||
|
className="euiButtonIcon__icon"
|
||||||
|
data-euiicon-type="trash"
|
||||||
|
size="m"
|
||||||
|
/>
|
||||||
|
</button>
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiFlexItem canvasElementCard__wrapper"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiCard euiCard--leftAligned euiCard--isClickable canvasElementCard"
|
||||||
|
onClick={[Function]}
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiCard__top"
|
||||||
|
>
|
||||||
|
<img
|
||||||
|
alt=""
|
||||||
|
className="euiCard__image"
|
||||||
|
src=""
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiCard__content"
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
className="euiTitle euiTitle--small euiCard__title"
|
||||||
|
id="generated-idTitle"
|
||||||
|
>
|
||||||
|
<button
|
||||||
|
aria-describedby=" generated-idDescription"
|
||||||
|
className="euiCard__titleButton"
|
||||||
|
onClick={[Function]}
|
||||||
|
>
|
||||||
|
Custom Element 2
|
||||||
|
</button>
|
||||||
|
</span>
|
||||||
|
<div
|
||||||
|
className="euiText euiText--small euiCard__description"
|
||||||
|
id="generated-idDescription"
|
||||||
|
>
|
||||||
|
<p>
|
||||||
|
Aenean eu justo auctor, placerat felis non, scelerisque dolor.
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiCard__footer"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiFlexGroup euiFlexGroup--gutterExtraSmall euiFlexGroup--justifyContentSpaceBetween euiFlexGroup--directionRow euiFlexGroup--responsive canvasElementCard__controls"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiFlexItem euiFlexItem--flexGrowZero"
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
className="euiToolTipAnchor"
|
||||||
|
onMouseOut={[Function]}
|
||||||
|
onMouseOver={[Function]}
|
||||||
|
>
|
||||||
|
<button
|
||||||
|
aria-label="Edit element"
|
||||||
|
className="euiButtonIcon euiButtonIcon--primary"
|
||||||
|
data-test-subj="canvasElementCard__editButton"
|
||||||
|
onBlur={[Function]}
|
||||||
|
onClick={[Function]}
|
||||||
|
onFocus={[Function]}
|
||||||
|
type="button"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
aria-hidden="true"
|
||||||
|
className="euiButtonIcon__icon"
|
||||||
|
data-euiicon-type="pencil"
|
||||||
|
size="m"
|
||||||
|
/>
|
||||||
|
</button>
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiFlexItem euiFlexItem--flexGrowZero"
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
className="euiToolTipAnchor"
|
||||||
|
onMouseOut={[Function]}
|
||||||
|
onMouseOver={[Function]}
|
||||||
|
>
|
||||||
|
<button
|
||||||
|
aria-label="Delete element"
|
||||||
|
className="euiButtonIcon euiButtonIcon--danger"
|
||||||
|
data-test-subj="canvasElementCard__deleteButton"
|
||||||
|
onBlur={[Function]}
|
||||||
|
onClick={[Function]}
|
||||||
|
onFocus={[Function]}
|
||||||
|
type="button"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
aria-hidden="true"
|
||||||
|
className="euiButtonIcon__icon"
|
||||||
|
data-euiicon-type="trash"
|
||||||
|
size="m"
|
||||||
|
/>
|
||||||
|
</button>
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiFlexItem canvasElementCard__wrapper"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiCard euiCard--leftAligned euiCard--isClickable canvasElementCard"
|
||||||
|
onClick={[Function]}
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiCard__top"
|
||||||
|
>
|
||||||
|
<img
|
||||||
|
alt=""
|
||||||
|
className="euiCard__image"
|
||||||
|
src=""
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiCard__content"
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
className="euiTitle euiTitle--small euiCard__title"
|
||||||
|
id="generated-idTitle"
|
||||||
|
>
|
||||||
|
<button
|
||||||
|
aria-describedby=" generated-idDescription"
|
||||||
|
className="euiCard__titleButton"
|
||||||
|
onClick={[Function]}
|
||||||
|
>
|
||||||
|
Custom Element 3
|
||||||
|
</button>
|
||||||
|
</span>
|
||||||
|
<div
|
||||||
|
className="euiText euiText--small euiCard__description"
|
||||||
|
id="generated-idDescription"
|
||||||
|
>
|
||||||
|
<p>
|
||||||
|
Lorem ipsum dolor sit amet, consectetur adipiscing elit. Fusce lobortis aliquet arcu ut turpis duis.
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiCard__footer"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiFlexGroup euiFlexGroup--gutterExtraSmall euiFlexGroup--justifyContentSpaceBetween euiFlexGroup--directionRow euiFlexGroup--responsive canvasElementCard__controls"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiFlexItem euiFlexItem--flexGrowZero"
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
className="euiToolTipAnchor"
|
||||||
|
onMouseOut={[Function]}
|
||||||
|
onMouseOver={[Function]}
|
||||||
|
>
|
||||||
|
<button
|
||||||
|
aria-label="Edit element"
|
||||||
|
className="euiButtonIcon euiButtonIcon--primary"
|
||||||
|
data-test-subj="canvasElementCard__editButton"
|
||||||
|
onBlur={[Function]}
|
||||||
|
onClick={[Function]}
|
||||||
|
onFocus={[Function]}
|
||||||
|
type="button"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
aria-hidden="true"
|
||||||
|
className="euiButtonIcon__icon"
|
||||||
|
data-euiicon-type="pencil"
|
||||||
|
size="m"
|
||||||
|
/>
|
||||||
|
</button>
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiFlexItem euiFlexItem--flexGrowZero"
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
className="euiToolTipAnchor"
|
||||||
|
onMouseOut={[Function]}
|
||||||
|
onMouseOver={[Function]}
|
||||||
|
>
|
||||||
|
<button
|
||||||
|
aria-label="Delete element"
|
||||||
|
className="euiButtonIcon euiButtonIcon--danger"
|
||||||
|
data-test-subj="canvasElementCard__deleteButton"
|
||||||
|
onBlur={[Function]}
|
||||||
|
onClick={[Function]}
|
||||||
|
onFocus={[Function]}
|
||||||
|
type="button"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
aria-hidden="true"
|
||||||
|
className="euiButtonIcon__icon"
|
||||||
|
data-euiicon-type="trash"
|
||||||
|
size="m"
|
||||||
|
/>
|
||||||
|
</button>
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
`;
|
||||||
|
|
||||||
|
exports[`Storyshots components/SavedElementsModal/ElementGrid with text filter 1`] = `
|
||||||
|
<div
|
||||||
|
style={
|
||||||
|
Object {
|
||||||
|
"width": "1000px",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiFlexGrid euiFlexGrid--gutterLarge euiFlexGrid--fourths euiFlexGrid--responsive"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
`;
|
|
@ -0,0 +1,933 @@
|
||||||
|
// Jest Snapshot v1, https://goo.gl/fbAQLP
|
||||||
|
|
||||||
|
exports[`Storyshots components/SavedElementsModal no custom elements 1`] = `
|
||||||
|
Array [
|
||||||
|
<div
|
||||||
|
data-focus-guard={true}
|
||||||
|
style={
|
||||||
|
Object {
|
||||||
|
"height": "0px",
|
||||||
|
"left": "1px",
|
||||||
|
"overflow": "hidden",
|
||||||
|
"padding": 0,
|
||||||
|
"position": "fixed",
|
||||||
|
"top": "1px",
|
||||||
|
"width": "1px",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
tabIndex={0}
|
||||||
|
/>,
|
||||||
|
<div
|
||||||
|
data-focus-guard={true}
|
||||||
|
style={
|
||||||
|
Object {
|
||||||
|
"height": "0px",
|
||||||
|
"left": "1px",
|
||||||
|
"overflow": "hidden",
|
||||||
|
"padding": 0,
|
||||||
|
"position": "fixed",
|
||||||
|
"top": "1px",
|
||||||
|
"width": "1px",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
tabIndex={1}
|
||||||
|
/>,
|
||||||
|
<div
|
||||||
|
data-focus-lock-disabled={false}
|
||||||
|
onBlur={[Function]}
|
||||||
|
onFocus={[Function]}
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiModal canvasModal--fixedSize"
|
||||||
|
onKeyDown={[Function]}
|
||||||
|
style={
|
||||||
|
Object {
|
||||||
|
"maxWidth": "1000px",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
tabIndex={0}
|
||||||
|
>
|
||||||
|
<button
|
||||||
|
aria-label="Closes this modal window"
|
||||||
|
className="euiButtonIcon euiButtonIcon--text euiModal__closeIcon"
|
||||||
|
onClick={[Function]}
|
||||||
|
type="button"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
aria-hidden="true"
|
||||||
|
className="euiButtonIcon__icon"
|
||||||
|
data-euiicon-type="cross"
|
||||||
|
size="m"
|
||||||
|
/>
|
||||||
|
</button>
|
||||||
|
<div
|
||||||
|
className="euiModal__flex"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiModalHeader canvasAssetManager__modalHeader"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiModalHeader__title canvasAssetManager__modalHeaderTitle"
|
||||||
|
>
|
||||||
|
My elements
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiModalBody"
|
||||||
|
style={
|
||||||
|
Object {
|
||||||
|
"paddingRight": "1px",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiModalBody__overflow"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiFormControlLayout euiFormControlLayout--fullWidth"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiFormControlLayout__childrenWrapper"
|
||||||
|
>
|
||||||
|
<input
|
||||||
|
className="euiFieldSearch euiFieldSearch--fullWidth"
|
||||||
|
onChange={[Function]}
|
||||||
|
onKeyUp={[Function]}
|
||||||
|
placeholder="Find element"
|
||||||
|
type="search"
|
||||||
|
value=""
|
||||||
|
/>
|
||||||
|
<div
|
||||||
|
className="euiFormControlLayoutIcons"
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
className="euiFormControlLayoutCustomIcon"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
aria-hidden="true"
|
||||||
|
className="euiFormControlLayoutCustomIcon__icon"
|
||||||
|
data-euiicon-type="search"
|
||||||
|
/>
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiSpacer euiSpacer--l"
|
||||||
|
/>
|
||||||
|
<div
|
||||||
|
className="euiEmptyPrompt"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
color="subdued"
|
||||||
|
data-euiicon-type="vector"
|
||||||
|
size="xxl"
|
||||||
|
/>
|
||||||
|
<div
|
||||||
|
className="euiSpacer euiSpacer--s"
|
||||||
|
/>
|
||||||
|
<span
|
||||||
|
className="euiTextColor euiTextColor--subdued"
|
||||||
|
>
|
||||||
|
<h2
|
||||||
|
className="euiTitle euiTitle--small"
|
||||||
|
>
|
||||||
|
Add new elements
|
||||||
|
</h2>
|
||||||
|
<div
|
||||||
|
className="euiSpacer euiSpacer--m"
|
||||||
|
/>
|
||||||
|
<div
|
||||||
|
className="euiText euiText--medium"
|
||||||
|
>
|
||||||
|
<p>
|
||||||
|
Group and save workpad elements to create new elements
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiModalFooter"
|
||||||
|
>
|
||||||
|
<button
|
||||||
|
className="euiButton euiButton--primary euiButton--small"
|
||||||
|
data-test-subj="saved-elements-modal-close-button"
|
||||||
|
onClick={[Function]}
|
||||||
|
type="button"
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
className="euiButton__content"
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
className="euiButton__text"
|
||||||
|
>
|
||||||
|
Close
|
||||||
|
</span>
|
||||||
|
</span>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>,
|
||||||
|
<div
|
||||||
|
data-focus-guard={true}
|
||||||
|
style={
|
||||||
|
Object {
|
||||||
|
"height": "0px",
|
||||||
|
"left": "1px",
|
||||||
|
"overflow": "hidden",
|
||||||
|
"padding": 0,
|
||||||
|
"position": "fixed",
|
||||||
|
"top": "1px",
|
||||||
|
"width": "1px",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
tabIndex={0}
|
||||||
|
/>,
|
||||||
|
]
|
||||||
|
`;
|
||||||
|
|
||||||
|
exports[`Storyshots components/SavedElementsModal with custom elements 1`] = `
|
||||||
|
Array [
|
||||||
|
<div
|
||||||
|
data-focus-guard={true}
|
||||||
|
style={
|
||||||
|
Object {
|
||||||
|
"height": "0px",
|
||||||
|
"left": "1px",
|
||||||
|
"overflow": "hidden",
|
||||||
|
"padding": 0,
|
||||||
|
"position": "fixed",
|
||||||
|
"top": "1px",
|
||||||
|
"width": "1px",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
tabIndex={0}
|
||||||
|
/>,
|
||||||
|
<div
|
||||||
|
data-focus-guard={true}
|
||||||
|
style={
|
||||||
|
Object {
|
||||||
|
"height": "0px",
|
||||||
|
"left": "1px",
|
||||||
|
"overflow": "hidden",
|
||||||
|
"padding": 0,
|
||||||
|
"position": "fixed",
|
||||||
|
"top": "1px",
|
||||||
|
"width": "1px",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
tabIndex={1}
|
||||||
|
/>,
|
||||||
|
<div
|
||||||
|
data-focus-lock-disabled={false}
|
||||||
|
onBlur={[Function]}
|
||||||
|
onFocus={[Function]}
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiModal canvasModal--fixedSize"
|
||||||
|
onKeyDown={[Function]}
|
||||||
|
style={
|
||||||
|
Object {
|
||||||
|
"maxWidth": "1000px",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
tabIndex={0}
|
||||||
|
>
|
||||||
|
<button
|
||||||
|
aria-label="Closes this modal window"
|
||||||
|
className="euiButtonIcon euiButtonIcon--text euiModal__closeIcon"
|
||||||
|
onClick={[Function]}
|
||||||
|
type="button"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
aria-hidden="true"
|
||||||
|
className="euiButtonIcon__icon"
|
||||||
|
data-euiicon-type="cross"
|
||||||
|
size="m"
|
||||||
|
/>
|
||||||
|
</button>
|
||||||
|
<div
|
||||||
|
className="euiModal__flex"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiModalHeader canvasAssetManager__modalHeader"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiModalHeader__title canvasAssetManager__modalHeaderTitle"
|
||||||
|
>
|
||||||
|
My elements
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiModalBody"
|
||||||
|
style={
|
||||||
|
Object {
|
||||||
|
"paddingRight": "1px",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiModalBody__overflow"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiFormControlLayout euiFormControlLayout--fullWidth"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiFormControlLayout__childrenWrapper"
|
||||||
|
>
|
||||||
|
<input
|
||||||
|
className="euiFieldSearch euiFieldSearch--fullWidth"
|
||||||
|
onChange={[Function]}
|
||||||
|
onKeyUp={[Function]}
|
||||||
|
placeholder="Find element"
|
||||||
|
type="search"
|
||||||
|
value=""
|
||||||
|
/>
|
||||||
|
<div
|
||||||
|
className="euiFormControlLayoutIcons"
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
className="euiFormControlLayoutCustomIcon"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
aria-hidden="true"
|
||||||
|
className="euiFormControlLayoutCustomIcon__icon"
|
||||||
|
data-euiicon-type="search"
|
||||||
|
/>
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiSpacer euiSpacer--l"
|
||||||
|
/>
|
||||||
|
<div
|
||||||
|
className="euiFlexGrid euiFlexGrid--gutterLarge euiFlexGrid--fourths euiFlexGrid--responsive"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiFlexItem canvasElementCard__wrapper"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiCard euiCard--leftAligned euiCard--isClickable canvasElementCard"
|
||||||
|
onClick={[Function]}
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiCard__top"
|
||||||
|
>
|
||||||
|
<img
|
||||||
|
alt=""
|
||||||
|
className="euiCard__image"
|
||||||
|
src=""
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiCard__content"
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
className="euiTitle euiTitle--small euiCard__title"
|
||||||
|
id="generated-idTitle"
|
||||||
|
>
|
||||||
|
<button
|
||||||
|
aria-describedby=" generated-idDescription"
|
||||||
|
className="euiCard__titleButton"
|
||||||
|
onClick={[Function]}
|
||||||
|
>
|
||||||
|
Custom Element 1
|
||||||
|
</button>
|
||||||
|
</span>
|
||||||
|
<div
|
||||||
|
className="euiText euiText--small euiCard__description"
|
||||||
|
id="generated-idDescription"
|
||||||
|
>
|
||||||
|
<p>
|
||||||
|
sample description
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiCard__footer"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiFlexGroup euiFlexGroup--gutterExtraSmall euiFlexGroup--justifyContentSpaceBetween euiFlexGroup--directionRow euiFlexGroup--responsive canvasElementCard__controls"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiFlexItem euiFlexItem--flexGrowZero"
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
className="euiToolTipAnchor"
|
||||||
|
onMouseOut={[Function]}
|
||||||
|
onMouseOver={[Function]}
|
||||||
|
>
|
||||||
|
<button
|
||||||
|
aria-label="Edit element"
|
||||||
|
className="euiButtonIcon euiButtonIcon--primary"
|
||||||
|
data-test-subj="canvasElementCard__editButton"
|
||||||
|
onBlur={[Function]}
|
||||||
|
onClick={[Function]}
|
||||||
|
onFocus={[Function]}
|
||||||
|
type="button"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
aria-hidden="true"
|
||||||
|
className="euiButtonIcon__icon"
|
||||||
|
data-euiicon-type="pencil"
|
||||||
|
size="m"
|
||||||
|
/>
|
||||||
|
</button>
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiFlexItem euiFlexItem--flexGrowZero"
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
className="euiToolTipAnchor"
|
||||||
|
onMouseOut={[Function]}
|
||||||
|
onMouseOver={[Function]}
|
||||||
|
>
|
||||||
|
<button
|
||||||
|
aria-label="Delete element"
|
||||||
|
className="euiButtonIcon euiButtonIcon--danger"
|
||||||
|
data-test-subj="canvasElementCard__deleteButton"
|
||||||
|
onBlur={[Function]}
|
||||||
|
onClick={[Function]}
|
||||||
|
onFocus={[Function]}
|
||||||
|
type="button"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
aria-hidden="true"
|
||||||
|
className="euiButtonIcon__icon"
|
||||||
|
data-euiicon-type="trash"
|
||||||
|
size="m"
|
||||||
|
/>
|
||||||
|
</button>
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiFlexItem canvasElementCard__wrapper"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiCard euiCard--leftAligned euiCard--isClickable canvasElementCard"
|
||||||
|
onClick={[Function]}
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiCard__top"
|
||||||
|
>
|
||||||
|
<img
|
||||||
|
alt=""
|
||||||
|
className="euiCard__image"
|
||||||
|
src=""
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiCard__content"
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
className="euiTitle euiTitle--small euiCard__title"
|
||||||
|
id="generated-idTitle"
|
||||||
|
>
|
||||||
|
<button
|
||||||
|
aria-describedby=" generated-idDescription"
|
||||||
|
className="euiCard__titleButton"
|
||||||
|
onClick={[Function]}
|
||||||
|
>
|
||||||
|
Custom Element 2
|
||||||
|
</button>
|
||||||
|
</span>
|
||||||
|
<div
|
||||||
|
className="euiText euiText--small euiCard__description"
|
||||||
|
id="generated-idDescription"
|
||||||
|
>
|
||||||
|
<p>
|
||||||
|
Aenean eu justo auctor, placerat felis non, scelerisque dolor.
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiCard__footer"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiFlexGroup euiFlexGroup--gutterExtraSmall euiFlexGroup--justifyContentSpaceBetween euiFlexGroup--directionRow euiFlexGroup--responsive canvasElementCard__controls"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiFlexItem euiFlexItem--flexGrowZero"
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
className="euiToolTipAnchor"
|
||||||
|
onMouseOut={[Function]}
|
||||||
|
onMouseOver={[Function]}
|
||||||
|
>
|
||||||
|
<button
|
||||||
|
aria-label="Edit element"
|
||||||
|
className="euiButtonIcon euiButtonIcon--primary"
|
||||||
|
data-test-subj="canvasElementCard__editButton"
|
||||||
|
onBlur={[Function]}
|
||||||
|
onClick={[Function]}
|
||||||
|
onFocus={[Function]}
|
||||||
|
type="button"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
aria-hidden="true"
|
||||||
|
className="euiButtonIcon__icon"
|
||||||
|
data-euiicon-type="pencil"
|
||||||
|
size="m"
|
||||||
|
/>
|
||||||
|
</button>
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiFlexItem euiFlexItem--flexGrowZero"
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
className="euiToolTipAnchor"
|
||||||
|
onMouseOut={[Function]}
|
||||||
|
onMouseOver={[Function]}
|
||||||
|
>
|
||||||
|
<button
|
||||||
|
aria-label="Delete element"
|
||||||
|
className="euiButtonIcon euiButtonIcon--danger"
|
||||||
|
data-test-subj="canvasElementCard__deleteButton"
|
||||||
|
onBlur={[Function]}
|
||||||
|
onClick={[Function]}
|
||||||
|
onFocus={[Function]}
|
||||||
|
type="button"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
aria-hidden="true"
|
||||||
|
className="euiButtonIcon__icon"
|
||||||
|
data-euiicon-type="trash"
|
||||||
|
size="m"
|
||||||
|
/>
|
||||||
|
</button>
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiFlexItem canvasElementCard__wrapper"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiCard euiCard--leftAligned euiCard--isClickable canvasElementCard"
|
||||||
|
onClick={[Function]}
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiCard__top"
|
||||||
|
>
|
||||||
|
<img
|
||||||
|
alt=""
|
||||||
|
className="euiCard__image"
|
||||||
|
src=""
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiCard__content"
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
className="euiTitle euiTitle--small euiCard__title"
|
||||||
|
id="generated-idTitle"
|
||||||
|
>
|
||||||
|
<button
|
||||||
|
aria-describedby=" generated-idDescription"
|
||||||
|
className="euiCard__titleButton"
|
||||||
|
onClick={[Function]}
|
||||||
|
>
|
||||||
|
Custom Element 3
|
||||||
|
</button>
|
||||||
|
</span>
|
||||||
|
<div
|
||||||
|
className="euiText euiText--small euiCard__description"
|
||||||
|
id="generated-idDescription"
|
||||||
|
>
|
||||||
|
<p>
|
||||||
|
Lorem ipsum dolor sit amet, consectetur adipiscing elit. Fusce lobortis aliquet arcu ut turpis duis.
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiCard__footer"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiFlexGroup euiFlexGroup--gutterExtraSmall euiFlexGroup--justifyContentSpaceBetween euiFlexGroup--directionRow euiFlexGroup--responsive canvasElementCard__controls"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiFlexItem euiFlexItem--flexGrowZero"
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
className="euiToolTipAnchor"
|
||||||
|
onMouseOut={[Function]}
|
||||||
|
onMouseOver={[Function]}
|
||||||
|
>
|
||||||
|
<button
|
||||||
|
aria-label="Edit element"
|
||||||
|
className="euiButtonIcon euiButtonIcon--primary"
|
||||||
|
data-test-subj="canvasElementCard__editButton"
|
||||||
|
onBlur={[Function]}
|
||||||
|
onClick={[Function]}
|
||||||
|
onFocus={[Function]}
|
||||||
|
type="button"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
aria-hidden="true"
|
||||||
|
className="euiButtonIcon__icon"
|
||||||
|
data-euiicon-type="pencil"
|
||||||
|
size="m"
|
||||||
|
/>
|
||||||
|
</button>
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiFlexItem euiFlexItem--flexGrowZero"
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
className="euiToolTipAnchor"
|
||||||
|
onMouseOut={[Function]}
|
||||||
|
onMouseOver={[Function]}
|
||||||
|
>
|
||||||
|
<button
|
||||||
|
aria-label="Delete element"
|
||||||
|
className="euiButtonIcon euiButtonIcon--danger"
|
||||||
|
data-test-subj="canvasElementCard__deleteButton"
|
||||||
|
onBlur={[Function]}
|
||||||
|
onClick={[Function]}
|
||||||
|
onFocus={[Function]}
|
||||||
|
type="button"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
aria-hidden="true"
|
||||||
|
className="euiButtonIcon__icon"
|
||||||
|
data-euiicon-type="trash"
|
||||||
|
size="m"
|
||||||
|
/>
|
||||||
|
</button>
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiModalFooter"
|
||||||
|
>
|
||||||
|
<button
|
||||||
|
className="euiButton euiButton--primary euiButton--small"
|
||||||
|
data-test-subj="saved-elements-modal-close-button"
|
||||||
|
onClick={[Function]}
|
||||||
|
type="button"
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
className="euiButton__content"
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
className="euiButton__text"
|
||||||
|
>
|
||||||
|
Close
|
||||||
|
</span>
|
||||||
|
</span>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>,
|
||||||
|
<div
|
||||||
|
data-focus-guard={true}
|
||||||
|
style={
|
||||||
|
Object {
|
||||||
|
"height": "0px",
|
||||||
|
"left": "1px",
|
||||||
|
"overflow": "hidden",
|
||||||
|
"padding": 0,
|
||||||
|
"position": "fixed",
|
||||||
|
"top": "1px",
|
||||||
|
"width": "1px",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
tabIndex={0}
|
||||||
|
/>,
|
||||||
|
]
|
||||||
|
`;
|
||||||
|
|
||||||
|
exports[`Storyshots components/SavedElementsModal with text filter 1`] = `
|
||||||
|
Array [
|
||||||
|
<div
|
||||||
|
data-focus-guard={true}
|
||||||
|
style={
|
||||||
|
Object {
|
||||||
|
"height": "0px",
|
||||||
|
"left": "1px",
|
||||||
|
"overflow": "hidden",
|
||||||
|
"padding": 0,
|
||||||
|
"position": "fixed",
|
||||||
|
"top": "1px",
|
||||||
|
"width": "1px",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
tabIndex={0}
|
||||||
|
/>,
|
||||||
|
<div
|
||||||
|
data-focus-guard={true}
|
||||||
|
style={
|
||||||
|
Object {
|
||||||
|
"height": "0px",
|
||||||
|
"left": "1px",
|
||||||
|
"overflow": "hidden",
|
||||||
|
"padding": 0,
|
||||||
|
"position": "fixed",
|
||||||
|
"top": "1px",
|
||||||
|
"width": "1px",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
tabIndex={1}
|
||||||
|
/>,
|
||||||
|
<div
|
||||||
|
data-focus-lock-disabled={false}
|
||||||
|
onBlur={[Function]}
|
||||||
|
onFocus={[Function]}
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiModal canvasModal--fixedSize"
|
||||||
|
onKeyDown={[Function]}
|
||||||
|
style={
|
||||||
|
Object {
|
||||||
|
"maxWidth": "1000px",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
tabIndex={0}
|
||||||
|
>
|
||||||
|
<button
|
||||||
|
aria-label="Closes this modal window"
|
||||||
|
className="euiButtonIcon euiButtonIcon--text euiModal__closeIcon"
|
||||||
|
onClick={[Function]}
|
||||||
|
type="button"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
aria-hidden="true"
|
||||||
|
className="euiButtonIcon__icon"
|
||||||
|
data-euiicon-type="cross"
|
||||||
|
size="m"
|
||||||
|
/>
|
||||||
|
</button>
|
||||||
|
<div
|
||||||
|
className="euiModal__flex"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiModalHeader canvasAssetManager__modalHeader"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiModalHeader__title canvasAssetManager__modalHeaderTitle"
|
||||||
|
>
|
||||||
|
My elements
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiModalBody"
|
||||||
|
style={
|
||||||
|
Object {
|
||||||
|
"paddingRight": "1px",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiModalBody__overflow"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiFormControlLayout euiFormControlLayout--fullWidth"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiFormControlLayout__childrenWrapper"
|
||||||
|
>
|
||||||
|
<input
|
||||||
|
className="euiFieldSearch euiFieldSearch--fullWidth"
|
||||||
|
onChange={[Function]}
|
||||||
|
onKeyUp={[Function]}
|
||||||
|
placeholder="Find element"
|
||||||
|
type="search"
|
||||||
|
value="Element 2"
|
||||||
|
/>
|
||||||
|
<div
|
||||||
|
className="euiFormControlLayoutIcons"
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
className="euiFormControlLayoutCustomIcon"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
aria-hidden="true"
|
||||||
|
className="euiFormControlLayoutCustomIcon__icon"
|
||||||
|
data-euiicon-type="search"
|
||||||
|
/>
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiFormControlLayoutIcons euiFormControlLayoutIcons--right"
|
||||||
|
>
|
||||||
|
<button
|
||||||
|
aria-label="Clear input"
|
||||||
|
className="euiFormControlLayoutClearButton"
|
||||||
|
onClick={[Function]}
|
||||||
|
type="button"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiFormControlLayoutClearButton__icon"
|
||||||
|
data-euiicon-type="cross"
|
||||||
|
/>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiSpacer euiSpacer--l"
|
||||||
|
/>
|
||||||
|
<div
|
||||||
|
className="euiFlexGrid euiFlexGrid--gutterLarge euiFlexGrid--fourths euiFlexGrid--responsive"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiFlexItem canvasElementCard__wrapper"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiCard euiCard--leftAligned euiCard--isClickable canvasElementCard"
|
||||||
|
onClick={[Function]}
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiCard__top"
|
||||||
|
>
|
||||||
|
<img
|
||||||
|
alt=""
|
||||||
|
className="euiCard__image"
|
||||||
|
src=""
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiCard__content"
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
className="euiTitle euiTitle--small euiCard__title"
|
||||||
|
id="generated-idTitle"
|
||||||
|
>
|
||||||
|
<button
|
||||||
|
aria-describedby=" generated-idDescription"
|
||||||
|
className="euiCard__titleButton"
|
||||||
|
onClick={[Function]}
|
||||||
|
>
|
||||||
|
Custom Element 2
|
||||||
|
</button>
|
||||||
|
</span>
|
||||||
|
<div
|
||||||
|
className="euiText euiText--small euiCard__description"
|
||||||
|
id="generated-idDescription"
|
||||||
|
>
|
||||||
|
<p>
|
||||||
|
Aenean eu justo auctor, placerat felis non, scelerisque dolor.
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiCard__footer"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiFlexGroup euiFlexGroup--gutterExtraSmall euiFlexGroup--justifyContentSpaceBetween euiFlexGroup--directionRow euiFlexGroup--responsive canvasElementCard__controls"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiFlexItem euiFlexItem--flexGrowZero"
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
className="euiToolTipAnchor"
|
||||||
|
onMouseOut={[Function]}
|
||||||
|
onMouseOver={[Function]}
|
||||||
|
>
|
||||||
|
<button
|
||||||
|
aria-label="Edit element"
|
||||||
|
className="euiButtonIcon euiButtonIcon--primary"
|
||||||
|
data-test-subj="canvasElementCard__editButton"
|
||||||
|
onBlur={[Function]}
|
||||||
|
onClick={[Function]}
|
||||||
|
onFocus={[Function]}
|
||||||
|
type="button"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
aria-hidden="true"
|
||||||
|
className="euiButtonIcon__icon"
|
||||||
|
data-euiicon-type="pencil"
|
||||||
|
size="m"
|
||||||
|
/>
|
||||||
|
</button>
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiFlexItem euiFlexItem--flexGrowZero"
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
className="euiToolTipAnchor"
|
||||||
|
onMouseOut={[Function]}
|
||||||
|
onMouseOver={[Function]}
|
||||||
|
>
|
||||||
|
<button
|
||||||
|
aria-label="Delete element"
|
||||||
|
className="euiButtonIcon euiButtonIcon--danger"
|
||||||
|
data-test-subj="canvasElementCard__deleteButton"
|
||||||
|
onBlur={[Function]}
|
||||||
|
onClick={[Function]}
|
||||||
|
onFocus={[Function]}
|
||||||
|
type="button"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
aria-hidden="true"
|
||||||
|
className="euiButtonIcon__icon"
|
||||||
|
data-euiicon-type="trash"
|
||||||
|
size="m"
|
||||||
|
/>
|
||||||
|
</button>
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiModalFooter"
|
||||||
|
>
|
||||||
|
<button
|
||||||
|
className="euiButton euiButton--primary euiButton--small"
|
||||||
|
data-test-subj="saved-elements-modal-close-button"
|
||||||
|
onClick={[Function]}
|
||||||
|
type="button"
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
className="euiButton__content"
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
className="euiButton__text"
|
||||||
|
>
|
||||||
|
Close
|
||||||
|
</span>
|
||||||
|
</span>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>,
|
||||||
|
<div
|
||||||
|
data-focus-guard={true}
|
||||||
|
style={
|
||||||
|
Object {
|
||||||
|
"height": "0px",
|
||||||
|
"left": "1px",
|
||||||
|
"overflow": "hidden",
|
||||||
|
"padding": 0,
|
||||||
|
"position": "fixed",
|
||||||
|
"top": "1px",
|
||||||
|
"width": "1px",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
tabIndex={0}
|
||||||
|
/>,
|
||||||
|
]
|
||||||
|
`;
|
|
@ -9,7 +9,7 @@ import { storiesOf } from '@storybook/react';
|
||||||
import { action } from '@storybook/addon-actions';
|
import { action } from '@storybook/addon-actions';
|
||||||
import { ElementControls } from '../element_controls';
|
import { ElementControls } from '../element_controls';
|
||||||
|
|
||||||
storiesOf('components/Elements/ElementControls', module)
|
storiesOf('components/SavedElementsModal/ElementControls', module)
|
||||||
.addDecorator(story => (
|
.addDecorator(story => (
|
||||||
<div
|
<div
|
||||||
style={{
|
style={{
|
|
@ -8,9 +8,9 @@ import React from 'react';
|
||||||
import { storiesOf } from '@storybook/react';
|
import { storiesOf } from '@storybook/react';
|
||||||
import { action } from '@storybook/addon-actions';
|
import { action } from '@storybook/addon-actions';
|
||||||
import { ElementGrid } from '../element_grid';
|
import { ElementGrid } from '../element_grid';
|
||||||
import { testElements, testCustomElements } from './fixtures/test_elements';
|
import { testCustomElements } from './fixtures/test_elements';
|
||||||
|
|
||||||
storiesOf('components/Elements/ElementGrid', module)
|
storiesOf('components/SavedElementsModal/ElementGrid', module)
|
||||||
.addDecorator(story => (
|
.addDecorator(story => (
|
||||||
<div
|
<div
|
||||||
style={{
|
style={{
|
||||||
|
@ -20,38 +20,19 @@ storiesOf('components/Elements/ElementGrid', module)
|
||||||
{story()}
|
{story()}
|
||||||
</div>
|
</div>
|
||||||
))
|
))
|
||||||
.add('without controls', () => (
|
.add('default', () => (
|
||||||
<ElementGrid elements={testElements} handleClick={action('addElement')} />
|
|
||||||
))
|
|
||||||
.add('with controls', () => (
|
|
||||||
<ElementGrid
|
<ElementGrid
|
||||||
elements={testCustomElements}
|
elements={testCustomElements}
|
||||||
handleClick={action('addCustomElement')}
|
onClick={action('addCustomElement')}
|
||||||
showControls
|
|
||||||
onDelete={action('onDelete')}
|
onDelete={action('onDelete')}
|
||||||
onEdit={action('onEdit')}
|
onEdit={action('onEdit')}
|
||||||
/>
|
/>
|
||||||
))
|
))
|
||||||
.add('with text filter', () => (
|
.add('with text filter', () => (
|
||||||
<ElementGrid
|
|
||||||
elements={testElements}
|
|
||||||
handleClick={action('addCustomElement')}
|
|
||||||
filterText="table"
|
|
||||||
/>
|
|
||||||
))
|
|
||||||
.add('with tags filter', () => (
|
|
||||||
<ElementGrid
|
|
||||||
elements={testElements}
|
|
||||||
handleClick={action('addCustomElement')}
|
|
||||||
filterTags={['graphic']}
|
|
||||||
/>
|
|
||||||
))
|
|
||||||
.add('with controls and filter', () => (
|
|
||||||
<ElementGrid
|
<ElementGrid
|
||||||
elements={testCustomElements}
|
elements={testCustomElements}
|
||||||
handleClick={action('addCustomElement')}
|
onClick={action('addCustomElement')}
|
||||||
filterText="Lorem"
|
filterText="table"
|
||||||
showControls
|
|
||||||
onDelete={action('onDelete')}
|
onDelete={action('onDelete')}
|
||||||
onEdit={action('onEdit')}
|
onEdit={action('onEdit')}
|
||||||
/>
|
/>
|
|
@ -6,41 +6,6 @@
|
||||||
|
|
||||||
import { elasticLogo } from '../../../../lib/elastic_logo';
|
import { elasticLogo } from '../../../../lib/elastic_logo';
|
||||||
|
|
||||||
export const testElements = [
|
|
||||||
{
|
|
||||||
name: 'areaChart',
|
|
||||||
displayName: 'Area chart',
|
|
||||||
help: 'A line chart with a filled body',
|
|
||||||
tags: ['chart'],
|
|
||||||
image: elasticLogo,
|
|
||||||
expression: `filters
|
|
||||||
| demodata
|
|
||||||
| pointseries x="time" y="mean(price)"
|
|
||||||
| plot defaultStyle={seriesStyle lines=1 fill=1}
|
|
||||||
| render`,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: 'image',
|
|
||||||
displayName: 'Image',
|
|
||||||
help: 'A static image',
|
|
||||||
tags: ['graphic'],
|
|
||||||
image: elasticLogo,
|
|
||||||
expression: `image dataurl=null mode="contain"
|
|
||||||
| render`,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: 'table',
|
|
||||||
displayName: 'Data table',
|
|
||||||
tags: ['text'],
|
|
||||||
help: 'A scrollable grid for displaying data in a tabular format',
|
|
||||||
image: elasticLogo,
|
|
||||||
expression: `filters
|
|
||||||
| demodata
|
|
||||||
| table
|
|
||||||
| render`,
|
|
||||||
},
|
|
||||||
];
|
|
||||||
|
|
||||||
export const testCustomElements = [
|
export const testCustomElements = [
|
||||||
{
|
{
|
||||||
id: 'custom-element-10d625f5-1342-47c9-8f19-d174ea6b65d5',
|
id: 'custom-element-10d625f5-1342-47c9-8f19-d174ea6b65d5',
|
|
@ -0,0 +1,50 @@
|
||||||
|
/*
|
||||||
|
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||||
|
* or more contributor license agreements. Licensed under the Elastic License;
|
||||||
|
* you may not use this file except in compliance with the Elastic License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import React from 'react';
|
||||||
|
import { storiesOf } from '@storybook/react';
|
||||||
|
import { action } from '@storybook/addon-actions';
|
||||||
|
import { SavedElementsModal } from '../saved_elements_modal';
|
||||||
|
import { testCustomElements } from './fixtures/test_elements';
|
||||||
|
import { CustomElement } from '../../../../types';
|
||||||
|
|
||||||
|
storiesOf('components/SavedElementsModal', module)
|
||||||
|
.add('no custom elements', () => (
|
||||||
|
<SavedElementsModal
|
||||||
|
customElements={[] as CustomElement[]}
|
||||||
|
search=""
|
||||||
|
setSearch={action('setSearch')}
|
||||||
|
onClose={action('onClose')}
|
||||||
|
addCustomElement={action('addCustomElement')}
|
||||||
|
findCustomElements={action('findCustomElements')}
|
||||||
|
updateCustomElement={action('updateCustomElement')}
|
||||||
|
removeCustomElement={action('removeCustomElement')}
|
||||||
|
/>
|
||||||
|
))
|
||||||
|
.add('with custom elements', () => (
|
||||||
|
<SavedElementsModal
|
||||||
|
customElements={testCustomElements}
|
||||||
|
search=""
|
||||||
|
setSearch={action('setSearch')}
|
||||||
|
onClose={action('onClose')}
|
||||||
|
addCustomElement={action('addCustomElement')}
|
||||||
|
findCustomElements={action('findCustomElements')}
|
||||||
|
updateCustomElement={action('updateCustomElement')}
|
||||||
|
removeCustomElement={action('removeCustomElement')}
|
||||||
|
/>
|
||||||
|
))
|
||||||
|
.add('with text filter', () => (
|
||||||
|
<SavedElementsModal
|
||||||
|
customElements={testCustomElements}
|
||||||
|
search="Element 2"
|
||||||
|
onClose={action('onClose')}
|
||||||
|
setSearch={action('setSearch')}
|
||||||
|
addCustomElement={action('addCustomElement')}
|
||||||
|
findCustomElements={action('findCustomElements')}
|
||||||
|
updateCustomElement={action('updateCustomElement')}
|
||||||
|
removeCustomElement={action('removeCustomElement')}
|
||||||
|
/>
|
||||||
|
));
|
|
@ -30,7 +30,12 @@ export const ElementControls: FunctionComponent<Props> = ({ onDelete, onEdit })
|
||||||
>
|
>
|
||||||
<EuiFlexItem grow={false}>
|
<EuiFlexItem grow={false}>
|
||||||
<EuiToolTip content={strings.getEditTooltip()}>
|
<EuiToolTip content={strings.getEditTooltip()}>
|
||||||
<EuiButtonIcon iconType="pencil" aria-label={strings.getEditAriaLabel()} onClick={onEdit} />
|
<EuiButtonIcon
|
||||||
|
iconType="pencil"
|
||||||
|
aria-label={strings.getEditAriaLabel()}
|
||||||
|
onClick={onEdit}
|
||||||
|
data-test-subj="canvasElementCard__editButton"
|
||||||
|
/>
|
||||||
</EuiToolTip>
|
</EuiToolTip>
|
||||||
</EuiFlexItem>
|
</EuiFlexItem>
|
||||||
<EuiFlexItem grow={false}>
|
<EuiFlexItem grow={false}>
|
||||||
|
@ -40,6 +45,7 @@ export const ElementControls: FunctionComponent<Props> = ({ onDelete, onEdit })
|
||||||
iconType="trash"
|
iconType="trash"
|
||||||
aria-label={strings.getDeleteAriaLabel()}
|
aria-label={strings.getDeleteAriaLabel()}
|
||||||
onClick={onDelete}
|
onClick={onDelete}
|
||||||
|
data-test-subj="canvasElementCard__deleteButton"
|
||||||
/>
|
/>
|
||||||
</EuiToolTip>
|
</EuiToolTip>
|
||||||
</EuiFlexItem>
|
</EuiFlexItem>
|
|
@ -0,0 +1,81 @@
|
||||||
|
/*
|
||||||
|
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||||
|
* or more contributor license agreements. Licensed under the Elastic License;
|
||||||
|
* you may not use this file except in compliance with the Elastic License.
|
||||||
|
*/
|
||||||
|
import React from 'react';
|
||||||
|
import PropTypes from 'prop-types';
|
||||||
|
import { map } from 'lodash';
|
||||||
|
import { EuiFlexItem, EuiFlexGrid } from '@elastic/eui';
|
||||||
|
import { ElementControls } from './element_controls';
|
||||||
|
import { CustomElement } from '../../../types';
|
||||||
|
import { ElementCard } from '../element_card';
|
||||||
|
|
||||||
|
export interface Props {
|
||||||
|
/**
|
||||||
|
* list of elements to generate cards for
|
||||||
|
*/
|
||||||
|
elements: CustomElement[];
|
||||||
|
/**
|
||||||
|
* text to filter out cards
|
||||||
|
*/
|
||||||
|
filterText: string;
|
||||||
|
/**
|
||||||
|
* handler invoked when clicking a card
|
||||||
|
*/
|
||||||
|
onClick: (element: CustomElement) => void;
|
||||||
|
/**
|
||||||
|
* click handler for the edit button
|
||||||
|
*/
|
||||||
|
onEdit: (element: CustomElement) => void;
|
||||||
|
/**
|
||||||
|
* click handler for the delete button
|
||||||
|
*/
|
||||||
|
onDelete: (element: CustomElement) => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const ElementGrid = ({ elements, filterText, onClick, onEdit, onDelete }: Props) => {
|
||||||
|
filterText = filterText.toLowerCase();
|
||||||
|
|
||||||
|
return (
|
||||||
|
<EuiFlexGrid gutterSize="l" columns={4}>
|
||||||
|
{map(elements, (element: CustomElement, index) => {
|
||||||
|
const { name, displayName = '', help = '', image } = element;
|
||||||
|
const whenClicked = () => onClick(element);
|
||||||
|
|
||||||
|
if (
|
||||||
|
filterText.length &&
|
||||||
|
!name.toLowerCase().includes(filterText) &&
|
||||||
|
!displayName.toLowerCase().includes(filterText) &&
|
||||||
|
!help.toLowerCase().includes(filterText)
|
||||||
|
) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<EuiFlexItem key={index} className="canvasElementCard__wrapper">
|
||||||
|
<ElementCard
|
||||||
|
title={displayName || name}
|
||||||
|
description={help}
|
||||||
|
image={image}
|
||||||
|
onClick={whenClicked}
|
||||||
|
/>
|
||||||
|
<ElementControls onEdit={() => onEdit(element)} onDelete={() => onDelete(element)} />
|
||||||
|
</EuiFlexItem>
|
||||||
|
);
|
||||||
|
})}
|
||||||
|
</EuiFlexGrid>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
ElementGrid.propTypes = {
|
||||||
|
elements: PropTypes.array.isRequired,
|
||||||
|
filterText: PropTypes.string.isRequired,
|
||||||
|
onClick: PropTypes.func.isRequired,
|
||||||
|
onEdit: PropTypes.func.isRequired,
|
||||||
|
onDelete: PropTypes.func.isRequired,
|
||||||
|
};
|
||||||
|
|
||||||
|
ElementGrid.defaultProps = {
|
||||||
|
filterText: '',
|
||||||
|
};
|
|
@ -0,0 +1,128 @@
|
||||||
|
/*
|
||||||
|
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||||
|
* or more contributor license agreements. Licensed under the Elastic License;
|
||||||
|
* you may not use this file except in compliance with the Elastic License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { connect } from 'react-redux';
|
||||||
|
import { Dispatch } from 'redux';
|
||||||
|
import { compose, withState } from 'recompose';
|
||||||
|
import { camelCase } from 'lodash';
|
||||||
|
// @ts-ignore Untyped local
|
||||||
|
import { cloneSubgraphs } from '../../lib/clone_subgraphs';
|
||||||
|
import * as customElementService from '../../lib/custom_element_service';
|
||||||
|
// @ts-ignore Untyped local
|
||||||
|
import { notify } from '../../lib/notify';
|
||||||
|
// @ts-ignore Untyped local
|
||||||
|
import { selectToplevelNodes } from '../../state/actions/transient';
|
||||||
|
// @ts-ignore Untyped local
|
||||||
|
import { insertNodes } from '../../state/actions/elements';
|
||||||
|
import { getSelectedPage } from '../../state/selectors/workpad';
|
||||||
|
import { trackCanvasUiMetric, METRIC_TYPE } from '../../lib/ui_metric';
|
||||||
|
import { SavedElementsModal as Component, Props as ComponentProps } from './saved_elements_modal';
|
||||||
|
import { State, PositionedElement, CustomElement } from '../../../types';
|
||||||
|
|
||||||
|
const customElementAdded = 'elements-custom-added';
|
||||||
|
|
||||||
|
interface OwnProps {
|
||||||
|
onClose: () => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface OwnPropsWithState extends OwnProps {
|
||||||
|
customElements: CustomElement[];
|
||||||
|
setCustomElements: (customElements: CustomElement[]) => void;
|
||||||
|
search: string;
|
||||||
|
setSearch: (search: string) => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface DispatchProps {
|
||||||
|
selectToplevelNodes: (nodes: PositionedElement[]) => void;
|
||||||
|
insertNodes: (selectedNodes: PositionedElement[], pageId: string) => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface StateProps {
|
||||||
|
pageId: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
const mapStateToProps = (state: State): StateProps => ({
|
||||||
|
pageId: getSelectedPage(state),
|
||||||
|
});
|
||||||
|
|
||||||
|
const mapDispatchToProps = (dispatch: Dispatch): DispatchProps => ({
|
||||||
|
selectToplevelNodes: (nodes: PositionedElement[]) =>
|
||||||
|
dispatch(
|
||||||
|
selectToplevelNodes(
|
||||||
|
nodes
|
||||||
|
.filter((e: PositionedElement): boolean => !e.position.parent)
|
||||||
|
.map((e: PositionedElement): string => e.id)
|
||||||
|
)
|
||||||
|
),
|
||||||
|
insertNodes: (selectedNodes: PositionedElement[], pageId: string) =>
|
||||||
|
dispatch(insertNodes(selectedNodes, pageId)),
|
||||||
|
});
|
||||||
|
|
||||||
|
const mergeProps = (
|
||||||
|
stateProps: StateProps,
|
||||||
|
dispatchProps: DispatchProps,
|
||||||
|
ownProps: OwnPropsWithState
|
||||||
|
): ComponentProps => {
|
||||||
|
const { pageId } = stateProps;
|
||||||
|
const { onClose, search, setCustomElements } = ownProps;
|
||||||
|
|
||||||
|
const findCustomElements = async () => {
|
||||||
|
const { customElements } = await customElementService.find(search);
|
||||||
|
setCustomElements(customElements);
|
||||||
|
};
|
||||||
|
|
||||||
|
return {
|
||||||
|
...ownProps,
|
||||||
|
// add custom element to the page
|
||||||
|
addCustomElement: (customElement: CustomElement) => {
|
||||||
|
const { selectedNodes = [] } = JSON.parse(customElement.content) || {};
|
||||||
|
const clonedNodes = selectedNodes && cloneSubgraphs(selectedNodes);
|
||||||
|
if (clonedNodes) {
|
||||||
|
dispatchProps.insertNodes(clonedNodes, pageId); // first clone and persist the new node(s)
|
||||||
|
dispatchProps.selectToplevelNodes(clonedNodes); // then select the cloned node(s)
|
||||||
|
}
|
||||||
|
onClose();
|
||||||
|
trackCanvasUiMetric(METRIC_TYPE.LOADED, customElementAdded);
|
||||||
|
},
|
||||||
|
// custom element search
|
||||||
|
findCustomElements: async (text?: string) => {
|
||||||
|
try {
|
||||||
|
await findCustomElements();
|
||||||
|
} catch (err) {
|
||||||
|
notify.error(err, { title: `Couldn't find custom elements` });
|
||||||
|
}
|
||||||
|
},
|
||||||
|
// remove custom element
|
||||||
|
removeCustomElement: async (id: string) => {
|
||||||
|
try {
|
||||||
|
await customElementService.remove(id);
|
||||||
|
await findCustomElements();
|
||||||
|
} catch (err) {
|
||||||
|
notify.error(err, { title: `Couldn't delete custom elements` });
|
||||||
|
}
|
||||||
|
},
|
||||||
|
// update custom element
|
||||||
|
updateCustomElement: async (id: string, name: string, description: string, image: string) => {
|
||||||
|
try {
|
||||||
|
await customElementService.update(id, {
|
||||||
|
name: camelCase(name),
|
||||||
|
displayName: name,
|
||||||
|
image,
|
||||||
|
help: description,
|
||||||
|
});
|
||||||
|
await findCustomElements();
|
||||||
|
} catch (err) {
|
||||||
|
notify.error(err, { title: `Couldn't update custom elements` });
|
||||||
|
}
|
||||||
|
},
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
export const SavedElementsModal = compose<ComponentProps, OwnProps>(
|
||||||
|
withState('search', 'setSearch', ''),
|
||||||
|
withState('customElements', 'setCustomElements', []),
|
||||||
|
connect(mapStateToProps, mapDispatchToProps, mergeProps)
|
||||||
|
)(Component);
|
|
@ -0,0 +1,217 @@
|
||||||
|
/*
|
||||||
|
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||||
|
* or more contributor license agreements. Licensed under the Elastic License;
|
||||||
|
* you may not use this file except in compliance with the Elastic License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import React, { Fragment, ChangeEvent, FunctionComponent, useState, useEffect } from 'react';
|
||||||
|
import PropTypes from 'prop-types';
|
||||||
|
import {
|
||||||
|
EuiModal,
|
||||||
|
EuiModalBody,
|
||||||
|
EuiModalFooter,
|
||||||
|
EuiModalHeader,
|
||||||
|
EuiModalHeaderTitle,
|
||||||
|
EuiEmptyPrompt,
|
||||||
|
EuiFieldSearch,
|
||||||
|
EuiSpacer,
|
||||||
|
EuiOverlayMask,
|
||||||
|
EuiButton,
|
||||||
|
} from '@elastic/eui';
|
||||||
|
import { map, sortBy } from 'lodash';
|
||||||
|
import { ComponentStrings } from '../../../i18n';
|
||||||
|
import { CustomElement } from '../../../types';
|
||||||
|
import { ConfirmModal } from '../confirm_modal/confirm_modal';
|
||||||
|
import { CustomElementModal } from '../custom_element_modal';
|
||||||
|
import { ElementGrid } from './element_grid';
|
||||||
|
|
||||||
|
const { SavedElementsModal: strings } = ComponentStrings;
|
||||||
|
|
||||||
|
export interface Props {
|
||||||
|
/**
|
||||||
|
* Adds the custom element to the workpad
|
||||||
|
*/
|
||||||
|
addCustomElement: (customElement: CustomElement) => void;
|
||||||
|
/**
|
||||||
|
* Queries ES for custom element saved objects
|
||||||
|
*/
|
||||||
|
findCustomElements: () => void;
|
||||||
|
/**
|
||||||
|
* Handler invoked when the modal closes
|
||||||
|
*/
|
||||||
|
onClose: () => void;
|
||||||
|
/**
|
||||||
|
* Deletes the custom element
|
||||||
|
*/
|
||||||
|
removeCustomElement: (id: string) => void;
|
||||||
|
/**
|
||||||
|
* Saved edits to the custom element
|
||||||
|
*/
|
||||||
|
updateCustomElement: (id: string, name: string, description: string, image: string) => void;
|
||||||
|
/**
|
||||||
|
* Array of custom elements to display
|
||||||
|
*/
|
||||||
|
customElements: CustomElement[];
|
||||||
|
/**
|
||||||
|
* Text used to filter custom elements list
|
||||||
|
*/
|
||||||
|
search: string;
|
||||||
|
/**
|
||||||
|
* Setter for search text
|
||||||
|
*/
|
||||||
|
setSearch: (search: string) => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const SavedElementsModal: FunctionComponent<Props> = ({
|
||||||
|
search,
|
||||||
|
setSearch,
|
||||||
|
customElements,
|
||||||
|
addCustomElement,
|
||||||
|
findCustomElements,
|
||||||
|
onClose,
|
||||||
|
removeCustomElement,
|
||||||
|
updateCustomElement,
|
||||||
|
}) => {
|
||||||
|
const [elementToDelete, setElementToDelete] = useState<CustomElement | null>(null);
|
||||||
|
const [elementToEdit, setElementToEdit] = useState<CustomElement | null>(null);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
findCustomElements();
|
||||||
|
});
|
||||||
|
|
||||||
|
const showEditModal = (element: CustomElement) => setElementToEdit(element);
|
||||||
|
const hideEditModal = () => setElementToEdit(null);
|
||||||
|
|
||||||
|
const handleEdit = async (name: string, description: string, image: string) => {
|
||||||
|
if (elementToEdit) {
|
||||||
|
await updateCustomElement(elementToEdit.id, name, description, image);
|
||||||
|
}
|
||||||
|
hideEditModal();
|
||||||
|
};
|
||||||
|
|
||||||
|
const showDeleteModal = (element: CustomElement) => setElementToDelete(element);
|
||||||
|
const hideDeleteModal = () => setElementToDelete(null);
|
||||||
|
|
||||||
|
const handleDelete = async () => {
|
||||||
|
if (elementToDelete) {
|
||||||
|
await removeCustomElement(elementToDelete.id);
|
||||||
|
}
|
||||||
|
hideDeleteModal();
|
||||||
|
};
|
||||||
|
|
||||||
|
const renderEditModal = () => {
|
||||||
|
if (!elementToEdit) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<EuiOverlayMask>
|
||||||
|
<CustomElementModal
|
||||||
|
title={strings.getEditElementTitle()}
|
||||||
|
name={elementToEdit.displayName}
|
||||||
|
description={elementToEdit.help}
|
||||||
|
image={elementToEdit.image}
|
||||||
|
onSave={handleEdit}
|
||||||
|
onCancel={hideEditModal}
|
||||||
|
/>
|
||||||
|
</EuiOverlayMask>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
const renderDeleteModal = () => {
|
||||||
|
if (!elementToDelete) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<ConfirmModal
|
||||||
|
isOpen
|
||||||
|
title={strings.getDeleteElementTitle(elementToDelete.displayName)}
|
||||||
|
message={strings.getDeleteElementDescription()}
|
||||||
|
confirmButtonText={strings.getDeleteButtonLabel()}
|
||||||
|
cancelButtonText={strings.getCancelButtonLabel()}
|
||||||
|
onConfirm={handleDelete}
|
||||||
|
onCancel={hideDeleteModal}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
const sortElements = (elements: CustomElement[]): CustomElement[] =>
|
||||||
|
sortBy(
|
||||||
|
map(elements, (element, name) => ({ name, ...element })),
|
||||||
|
'displayName'
|
||||||
|
);
|
||||||
|
|
||||||
|
const onSearch = (e: ChangeEvent<HTMLInputElement>) => setSearch(e.target.value);
|
||||||
|
|
||||||
|
let customElementContent = (
|
||||||
|
<EuiEmptyPrompt
|
||||||
|
iconType="vector"
|
||||||
|
title={<h2>{strings.getAddNewElementTitle()}</h2>}
|
||||||
|
body={<p>{strings.getAddNewElementDescription()}</p>}
|
||||||
|
titleSize="s"
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
|
||||||
|
if (customElements.length) {
|
||||||
|
customElementContent = (
|
||||||
|
<ElementGrid
|
||||||
|
elements={sortElements(customElements)}
|
||||||
|
filterText={search}
|
||||||
|
onClick={addCustomElement}
|
||||||
|
onEdit={showEditModal}
|
||||||
|
onDelete={showDeleteModal}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Fragment>
|
||||||
|
<EuiOverlayMask>
|
||||||
|
<EuiModal
|
||||||
|
onClose={onClose}
|
||||||
|
className="canvasModal--fixedSize"
|
||||||
|
maxWidth="1000px"
|
||||||
|
initialFocus=".canvasElements__filter input"
|
||||||
|
>
|
||||||
|
<EuiModalHeader className="canvasAssetManager__modalHeader">
|
||||||
|
<EuiModalHeaderTitle className="canvasAssetManager__modalHeaderTitle">
|
||||||
|
{strings.getModalTitle()}
|
||||||
|
</EuiModalHeaderTitle>
|
||||||
|
</EuiModalHeader>
|
||||||
|
|
||||||
|
<EuiModalBody style={{ paddingRight: '1px' }}>
|
||||||
|
<EuiFieldSearch
|
||||||
|
fullWidth
|
||||||
|
value={search}
|
||||||
|
placeholder={strings.getFindElementPlaceholder()}
|
||||||
|
onChange={onSearch}
|
||||||
|
/>
|
||||||
|
<EuiSpacer />
|
||||||
|
{customElementContent}
|
||||||
|
</EuiModalBody>
|
||||||
|
<EuiModalFooter>
|
||||||
|
<EuiButton
|
||||||
|
size="s"
|
||||||
|
onClick={onClose}
|
||||||
|
data-test-subj="saved-elements-modal-close-button"
|
||||||
|
>
|
||||||
|
{strings.getSavedElementsModalCloseButtonLabel()}
|
||||||
|
</EuiButton>
|
||||||
|
</EuiModalFooter>
|
||||||
|
</EuiModal>
|
||||||
|
</EuiOverlayMask>
|
||||||
|
|
||||||
|
{renderDeleteModal()}
|
||||||
|
{renderEditModal()}
|
||||||
|
</Fragment>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
SavedElementsModal.propTypes = {
|
||||||
|
addCustomElement: PropTypes.func.isRequired,
|
||||||
|
findCustomElements: PropTypes.func.isRequired,
|
||||||
|
onClose: PropTypes.func.isRequired,
|
||||||
|
removeCustomElement: PropTypes.func.isRequired,
|
||||||
|
updateCustomElement: PropTypes.func.isRequired,
|
||||||
|
};
|
|
@ -37,6 +37,7 @@ exports[`Storyshots components/Sidebar/SidebarHeader default 1`] = `
|
||||||
<button
|
<button
|
||||||
aria-label="Save as new element"
|
aria-label="Save as new element"
|
||||||
className="euiButtonIcon euiButtonIcon--text"
|
className="euiButtonIcon euiButtonIcon--text"
|
||||||
|
data-test-subj="canvasSidebarHeader__saveElementButton"
|
||||||
onBlur={[Function]}
|
onBlur={[Function]}
|
||||||
onClick={[Function]}
|
onClick={[Function]}
|
||||||
onFocus={[Function]}
|
onFocus={[Function]}
|
||||||
|
@ -97,7 +98,7 @@ exports[`Storyshots components/Sidebar/SidebarHeader default 1`] = `
|
||||||
</div>
|
</div>
|
||||||
`;
|
`;
|
||||||
|
|
||||||
exports[`Storyshots components/Sidebar/SidebarHeader without layer controls 1`] = `
|
exports[`Storyshots components/Sidebar/SidebarHeader with layer controls 1`] = `
|
||||||
<div
|
<div
|
||||||
style={
|
style={
|
||||||
Object {
|
Object {
|
||||||
|
@ -123,6 +124,106 @@ exports[`Storyshots components/Sidebar/SidebarHeader without layer controls 1`]
|
||||||
<div
|
<div
|
||||||
className="euiFlexGroup euiFlexGroup--alignItemsCenter euiFlexGroup--directionRow euiFlexGroup--responsive"
|
className="euiFlexGroup euiFlexGroup--alignItemsCenter euiFlexGroup--directionRow euiFlexGroup--responsive"
|
||||||
>
|
>
|
||||||
|
<div
|
||||||
|
className="euiFlexItem euiFlexItem--flexGrowZero"
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
className="euiToolTipAnchor"
|
||||||
|
onMouseOut={[Function]}
|
||||||
|
onMouseOver={[Function]}
|
||||||
|
>
|
||||||
|
<button
|
||||||
|
aria-label="Move element to top layer"
|
||||||
|
className="euiButtonIcon euiButtonIcon--text"
|
||||||
|
onBlur={[Function]}
|
||||||
|
onClick={[Function]}
|
||||||
|
onFocus={[Function]}
|
||||||
|
type="button"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
aria-hidden="true"
|
||||||
|
className="euiButtonIcon__icon"
|
||||||
|
data-euiicon-type="sortUp"
|
||||||
|
size="m"
|
||||||
|
/>
|
||||||
|
</button>
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiFlexItem euiFlexItem--flexGrowZero"
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
className="euiToolTipAnchor"
|
||||||
|
onMouseOut={[Function]}
|
||||||
|
onMouseOver={[Function]}
|
||||||
|
>
|
||||||
|
<button
|
||||||
|
aria-label="Move element up one layer"
|
||||||
|
className="euiButtonIcon euiButtonIcon--text"
|
||||||
|
onBlur={[Function]}
|
||||||
|
onClick={[Function]}
|
||||||
|
onFocus={[Function]}
|
||||||
|
type="button"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
aria-hidden="true"
|
||||||
|
className="euiButtonIcon__icon"
|
||||||
|
data-euiicon-type="arrowUp"
|
||||||
|
size="m"
|
||||||
|
/>
|
||||||
|
</button>
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiFlexItem euiFlexItem--flexGrowZero"
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
className="euiToolTipAnchor"
|
||||||
|
onMouseOut={[Function]}
|
||||||
|
onMouseOver={[Function]}
|
||||||
|
>
|
||||||
|
<button
|
||||||
|
aria-label="Move element down one layer"
|
||||||
|
className="euiButtonIcon euiButtonIcon--text"
|
||||||
|
onBlur={[Function]}
|
||||||
|
onClick={[Function]}
|
||||||
|
onFocus={[Function]}
|
||||||
|
type="button"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
aria-hidden="true"
|
||||||
|
className="euiButtonIcon__icon"
|
||||||
|
data-euiicon-type="arrowDown"
|
||||||
|
size="m"
|
||||||
|
/>
|
||||||
|
</button>
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="euiFlexItem euiFlexItem--flexGrowZero"
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
className="euiToolTipAnchor"
|
||||||
|
onMouseOut={[Function]}
|
||||||
|
onMouseOver={[Function]}
|
||||||
|
>
|
||||||
|
<button
|
||||||
|
aria-label="Move element to bottom layer"
|
||||||
|
className="euiButtonIcon euiButtonIcon--text"
|
||||||
|
onBlur={[Function]}
|
||||||
|
onClick={[Function]}
|
||||||
|
onFocus={[Function]}
|
||||||
|
type="button"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
aria-hidden="true"
|
||||||
|
className="euiButtonIcon__icon"
|
||||||
|
data-euiicon-type="sortDown"
|
||||||
|
size="m"
|
||||||
|
/>
|
||||||
|
</button>
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
<div
|
<div
|
||||||
className="euiFlexItem euiFlexItem--flexGrowZero"
|
className="euiFlexItem euiFlexItem--flexGrowZero"
|
||||||
>
|
>
|
||||||
|
@ -134,6 +235,7 @@ exports[`Storyshots components/Sidebar/SidebarHeader without layer controls 1`]
|
||||||
<button
|
<button
|
||||||
aria-label="Save as new element"
|
aria-label="Save as new element"
|
||||||
className="euiButtonIcon euiButtonIcon--text"
|
className="euiButtonIcon euiButtonIcon--text"
|
||||||
|
data-test-subj="canvasSidebarHeader__saveElementButton"
|
||||||
onBlur={[Function]}
|
onBlur={[Function]}
|
||||||
onClick={[Function]}
|
onClick={[Function]}
|
||||||
onFocus={[Function]}
|
onFocus={[Function]}
|
||||||
|
|
|
@ -35,6 +35,6 @@ const handlers = {
|
||||||
storiesOf('components/Sidebar/SidebarHeader', module)
|
storiesOf('components/Sidebar/SidebarHeader', module)
|
||||||
.addDecorator(story => <div style={{ width: '300px' }}>{story()}</div>)
|
.addDecorator(story => <div style={{ width: '300px' }}>{story()}</div>)
|
||||||
.add('default', () => <SidebarHeader title="Selected layer" {...handlers} />)
|
.add('default', () => <SidebarHeader title="Selected layer" {...handlers} />)
|
||||||
.add('without layer controls', () => (
|
.add('with layer controls', () => (
|
||||||
<SidebarHeader title="Grouped element" showLayerControls={false} {...handlers} />
|
<SidebarHeader title="Grouped element" showLayerControls={true} {...handlers} />
|
||||||
));
|
));
|
||||||
|
|
|
@ -22,8 +22,7 @@ import { CustomElementModal } from '../custom_element_modal';
|
||||||
import { ToolTipShortcut } from '../tool_tip_shortcut/';
|
import { ToolTipShortcut } from '../tool_tip_shortcut/';
|
||||||
import { ComponentStrings } from '../../../i18n/components';
|
import { ComponentStrings } from '../../../i18n/components';
|
||||||
import { ShortcutStrings } from '../../../i18n/shortcuts';
|
import { ShortcutStrings } from '../../../i18n/shortcuts';
|
||||||
|
import { CONTEXT_MENU_TOP_BORDER_CLASSNAME } from '../../../common/lib/constants';
|
||||||
const topBorderClassName = 'canvasContextMenu--topBorder';
|
|
||||||
|
|
||||||
const { SidebarHeader: strings } = ComponentStrings;
|
const { SidebarHeader: strings } = ComponentStrings;
|
||||||
const shortcutHelp = ShortcutStrings.getShortcutHelp();
|
const shortcutHelp = ShortcutStrings.getShortcutHelp();
|
||||||
|
@ -282,7 +281,11 @@ export class SidebarHeader extends Component<Props, State> {
|
||||||
const { bringToFront, bringForward, sendBackward, sendToBack } = this.props;
|
const { bringToFront, bringForward, sendBackward, sendToBack } = this.props;
|
||||||
|
|
||||||
return {
|
return {
|
||||||
menuItem: { name: strings.getOrderMenuItemLabel(), className: topBorderClassName, panel: 1 },
|
menuItem: {
|
||||||
|
name: strings.getOrderMenuItemLabel(),
|
||||||
|
className: CONTEXT_MENU_TOP_BORDER_CLASSNAME,
|
||||||
|
panel: 1,
|
||||||
|
},
|
||||||
panel: {
|
panel: {
|
||||||
id: 1,
|
id: 1,
|
||||||
title: strings.getOrderMenuItemLabel(),
|
title: strings.getOrderMenuItemLabel(),
|
||||||
|
@ -396,7 +399,7 @@ export class SidebarHeader extends Component<Props, State> {
|
||||||
? [
|
? [
|
||||||
{
|
{
|
||||||
name: strings.getUngroupMenuItemLabel(),
|
name: strings.getUngroupMenuItemLabel(),
|
||||||
className: topBorderClassName,
|
className: CONTEXT_MENU_TOP_BORDER_CLASSNAME,
|
||||||
onClick: close(ungroupNodes),
|
onClick: close(ungroupNodes),
|
||||||
},
|
},
|
||||||
]
|
]
|
||||||
|
@ -404,7 +407,7 @@ export class SidebarHeader extends Component<Props, State> {
|
||||||
? [
|
? [
|
||||||
{
|
{
|
||||||
name: strings.getGroupMenuItemLabel(),
|
name: strings.getGroupMenuItemLabel(),
|
||||||
className: topBorderClassName,
|
className: CONTEXT_MENU_TOP_BORDER_CLASSNAME,
|
||||||
onClick: close(groupNodes),
|
onClick: close(groupNodes),
|
||||||
},
|
},
|
||||||
]
|
]
|
||||||
|
@ -483,7 +486,7 @@ export class SidebarHeader extends Component<Props, State> {
|
||||||
items.push({
|
items.push({
|
||||||
name: strings.getSaveElementMenuItemLabel(),
|
name: strings.getSaveElementMenuItemLabel(),
|
||||||
icon: 'indexOpen',
|
icon: 'indexOpen',
|
||||||
className: topBorderClassName,
|
className: CONTEXT_MENU_TOP_BORDER_CLASSNAME,
|
||||||
onClick: this._showModal,
|
onClick: this._showModal,
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -537,6 +540,7 @@ export class SidebarHeader extends Component<Props, State> {
|
||||||
color="text"
|
color="text"
|
||||||
iconType="indexOpen"
|
iconType="indexOpen"
|
||||||
onClick={this._showModal}
|
onClick={this._showModal}
|
||||||
|
data-test-subj="canvasSidebarHeader__saveElementButton"
|
||||||
aria-label={strings.getSaveElementMenuItemLabel()}
|
aria-label={strings.getSaveElementMenuItemLabel()}
|
||||||
/>
|
/>
|
||||||
</EuiToolTip>
|
</EuiToolTip>
|
||||||
|
|
|
@ -6,7 +6,7 @@
|
||||||
|
|
||||||
import React, { MouseEventHandler } from 'react';
|
import React, { MouseEventHandler } from 'react';
|
||||||
import PropTypes from 'prop-types';
|
import PropTypes from 'prop-types';
|
||||||
import { EuiFlexGroup, EuiFlexItem, EuiButtonIcon, EuiToolTip } from '@elastic/eui';
|
import { EuiFlexGroup, EuiFlexItem, EuiButtonEmpty } from '@elastic/eui';
|
||||||
// @ts-ignore untyped local
|
// @ts-ignore untyped local
|
||||||
import { Popover } from '../../popover';
|
import { Popover } from '../../popover';
|
||||||
import { AutoRefreshControls } from './auto_refresh_controls';
|
import { AutoRefreshControls } from './auto_refresh_controls';
|
||||||
|
@ -39,16 +39,16 @@ export const ControlSettings = ({
|
||||||
};
|
};
|
||||||
|
|
||||||
const popoverButton = (handleClick: MouseEventHandler<HTMLButtonElement>) => (
|
const popoverButton = (handleClick: MouseEventHandler<HTMLButtonElement>) => (
|
||||||
<EuiToolTip position="bottom" content={strings.getTooltip()}>
|
<EuiButtonEmpty size="xs" aria-label={strings.getButtonLabel()} onClick={handleClick}>
|
||||||
<EuiButtonIcon iconType="gear" aria-label={strings.getTooltip()} onClick={handleClick} />
|
{strings.getButtonLabel()}
|
||||||
</EuiToolTip>
|
</EuiButtonEmpty>
|
||||||
);
|
);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Popover
|
<Popover
|
||||||
id="auto-refresh-popover"
|
id="auto-refresh-popover"
|
||||||
button={popoverButton}
|
button={popoverButton}
|
||||||
anchorPosition="rightUp"
|
anchorPosition="downLeft"
|
||||||
panelClassName="canvasControlSettings__popover"
|
panelClassName="canvasControlSettings__popover"
|
||||||
>
|
>
|
||||||
{() => (
|
{() => (
|
||||||
|
|
|
@ -0,0 +1,41 @@
|
||||||
|
// Jest Snapshot v1, https://goo.gl/fbAQLP
|
||||||
|
|
||||||
|
exports[`Storyshots components/WorkpadHeader/ElementMenu default 1`] = `
|
||||||
|
<div
|
||||||
|
className="euiPopover euiPopover--anchorDownLeft"
|
||||||
|
container={null}
|
||||||
|
onKeyDown={[Function]}
|
||||||
|
onMouseDown={[Function]}
|
||||||
|
onMouseUp={[Function]}
|
||||||
|
onTouchEnd={[Function]}
|
||||||
|
onTouchStart={[Function]}
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="euiPopover__anchor"
|
||||||
|
>
|
||||||
|
<button
|
||||||
|
aria-label="Add an element"
|
||||||
|
className="euiButton euiButton--primary euiButton--small canvasElementMenu__popoverButton euiButton--fill"
|
||||||
|
data-test-subj="add-element-button"
|
||||||
|
onClick={[Function]}
|
||||||
|
type="button"
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
className="euiButton__content"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
aria-hidden="true"
|
||||||
|
className="euiButton__icon"
|
||||||
|
data-euiicon-type="plusInCircle"
|
||||||
|
size="m"
|
||||||
|
/>
|
||||||
|
<span
|
||||||
|
className="euiButton__text"
|
||||||
|
>
|
||||||
|
Add element
|
||||||
|
</span>
|
||||||
|
</span>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
`;
|
|
@ -0,0 +1,150 @@
|
||||||
|
/*
|
||||||
|
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||||
|
* or more contributor license agreements. Licensed under the Elastic License;
|
||||||
|
* you may not use this file except in compliance with the Elastic License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { storiesOf } from '@storybook/react';
|
||||||
|
import { action } from '@storybook/addon-actions';
|
||||||
|
import React from 'react';
|
||||||
|
import { ElementSpec } from '../../../../../types';
|
||||||
|
import { ElementMenu } from '../element_menu';
|
||||||
|
|
||||||
|
const testElements: { [key: string]: ElementSpec } = {
|
||||||
|
areaChart: {
|
||||||
|
name: 'areaChart',
|
||||||
|
displayName: 'Area chart',
|
||||||
|
help: 'A line chart with a filled body',
|
||||||
|
type: 'chart',
|
||||||
|
expression: `filters
|
||||||
|
| demodata
|
||||||
|
| pointseries x="time" y="mean(price)"
|
||||||
|
| plot defaultStyle={seriesStyle lines=1 fill=1}
|
||||||
|
| render`,
|
||||||
|
},
|
||||||
|
debug: {
|
||||||
|
name: 'debug',
|
||||||
|
displayName: 'Debug data',
|
||||||
|
help: 'Just dumps the configuration of the element',
|
||||||
|
icon: 'bug',
|
||||||
|
expression: `demodata
|
||||||
|
| render as=debug`,
|
||||||
|
},
|
||||||
|
dropdownFilter: {
|
||||||
|
name: 'dropdownFilter',
|
||||||
|
displayName: 'Dropdown select',
|
||||||
|
type: 'filter',
|
||||||
|
help: 'A dropdown from which you can select values for an "exactly" filter',
|
||||||
|
icon: 'filter',
|
||||||
|
height: 50,
|
||||||
|
expression: `demodata
|
||||||
|
| dropdownControl valueColumn=project filterColumn=project | render`,
|
||||||
|
filter: '',
|
||||||
|
},
|
||||||
|
filterDebug: {
|
||||||
|
name: 'filterDebug',
|
||||||
|
displayName: 'Debug filter',
|
||||||
|
help: 'Shows the underlying global filters in a workpad',
|
||||||
|
icon: 'bug',
|
||||||
|
expression: `filters
|
||||||
|
| render as=debug`,
|
||||||
|
},
|
||||||
|
image: {
|
||||||
|
name: 'image',
|
||||||
|
displayName: 'Image',
|
||||||
|
help: 'A static image',
|
||||||
|
type: 'image',
|
||||||
|
expression: `image dataurl=null mode="contain"
|
||||||
|
| render`,
|
||||||
|
},
|
||||||
|
markdown: {
|
||||||
|
name: 'markdown',
|
||||||
|
displayName: 'Text',
|
||||||
|
type: 'text',
|
||||||
|
help: 'Add text using Markdown',
|
||||||
|
icon: 'visText',
|
||||||
|
expression: `filters
|
||||||
|
| demodata
|
||||||
|
| markdown "### Welcome to the Markdown element
|
||||||
|
|
||||||
|
Good news! You're already connected to some demo data!
|
||||||
|
|
||||||
|
The data table contains
|
||||||
|
**{{rows.length}} rows**, each containing
|
||||||
|
the following columns:
|
||||||
|
{{#each columns}}
|
||||||
|
**{{name}}**
|
||||||
|
{{/each}}
|
||||||
|
|
||||||
|
You can use standard Markdown in here, but you can also access your piped-in data using Handlebars. If you want to know more, check out the [Handlebars documentation](https://handlebarsjs.com/guide/expressions.html).
|
||||||
|
|
||||||
|
#### Enjoy!" | render`,
|
||||||
|
},
|
||||||
|
progressGauge: {
|
||||||
|
name: 'progressGauge',
|
||||||
|
displayName: 'Gauge',
|
||||||
|
type: 'progress',
|
||||||
|
help: 'Displays progress as a portion of a gauge',
|
||||||
|
width: 200,
|
||||||
|
height: 200,
|
||||||
|
icon: 'visGoal',
|
||||||
|
expression: `filters
|
||||||
|
| demodata
|
||||||
|
| math "mean(percent_uptime)"
|
||||||
|
| progress shape="gauge" label={formatnumber 0%} font={font size=24 family="Helvetica" color="#000000" align=center}
|
||||||
|
| render`,
|
||||||
|
},
|
||||||
|
revealImage: {
|
||||||
|
name: 'revealImage',
|
||||||
|
displayName: 'Image reveal',
|
||||||
|
type: 'image',
|
||||||
|
help: 'Reveals a percentage of an image',
|
||||||
|
expression: `filters
|
||||||
|
| demodata
|
||||||
|
| math "mean(percent_uptime)"
|
||||||
|
| revealImage origin=bottom image=null
|
||||||
|
| render`,
|
||||||
|
},
|
||||||
|
shape: {
|
||||||
|
name: 'shape',
|
||||||
|
displayName: 'Shape',
|
||||||
|
type: 'shape',
|
||||||
|
help: 'A customizable shape',
|
||||||
|
width: 200,
|
||||||
|
height: 200,
|
||||||
|
icon: 'node',
|
||||||
|
expression:
|
||||||
|
'shape "square" fill="#4cbce4" border="rgba(255,255,255,0)" borderWidth=0 maintainAspect=false | render',
|
||||||
|
},
|
||||||
|
table: {
|
||||||
|
name: 'table',
|
||||||
|
displayName: 'Data table',
|
||||||
|
type: 'chart',
|
||||||
|
help: 'A scrollable grid for displaying data in a tabular format',
|
||||||
|
expression: `filters
|
||||||
|
| demodata
|
||||||
|
| table
|
||||||
|
| render`,
|
||||||
|
},
|
||||||
|
timeFilter: {
|
||||||
|
name: 'timeFilter',
|
||||||
|
displayName: 'Time filter',
|
||||||
|
type: 'filter',
|
||||||
|
help: 'Set a time window',
|
||||||
|
icon: 'calendar',
|
||||||
|
height: 50,
|
||||||
|
expression: `timefilterControl compact=true column=@timestamp
|
||||||
|
| render`,
|
||||||
|
filter: 'timefilter column=@timestamp from=now-24h to=now',
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
const mockRenderEmbedPanel = () => <div id="embeddablePanel" />;
|
||||||
|
|
||||||
|
storiesOf('components/WorkpadHeader/ElementMenu', module).add('default', () => (
|
||||||
|
<ElementMenu
|
||||||
|
elements={testElements}
|
||||||
|
addElement={action('addElement')}
|
||||||
|
renderEmbedPanel={mockRenderEmbedPanel}
|
||||||
|
/>
|
||||||
|
));
|
|
@ -0,0 +1,3 @@
|
||||||
|
.canvasElementMenu__popoverButton {
|
||||||
|
margin-right: $euiSizeS;
|
||||||
|
}
|
|
@ -0,0 +1,216 @@
|
||||||
|
/*
|
||||||
|
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||||
|
* or more contributor license agreements. Licensed under the Elastic License;
|
||||||
|
* you may not use this file except in compliance with the Elastic License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { sortBy } from 'lodash';
|
||||||
|
import React, { Fragment, FunctionComponent, useState } from 'react';
|
||||||
|
import PropTypes from 'prop-types';
|
||||||
|
import {
|
||||||
|
EuiButton,
|
||||||
|
EuiContextMenu,
|
||||||
|
EuiIcon,
|
||||||
|
EuiContextMenuPanelItemDescriptor,
|
||||||
|
} from '@elastic/eui';
|
||||||
|
import { CONTEXT_MENU_TOP_BORDER_CLASSNAME } from '../../../../common/lib';
|
||||||
|
import { ComponentStrings } from '../../../../i18n/components';
|
||||||
|
import { ElementSpec } from '../../../../types';
|
||||||
|
import { flattenPanelTree } from '../../../lib/flatten_panel_tree';
|
||||||
|
import { getId } from '../../../lib/get_id';
|
||||||
|
import { Popover, ClosePopoverFn } from '../../popover';
|
||||||
|
// @ts-ignore Untyped local
|
||||||
|
import { AssetManager } from '../../asset_manager';
|
||||||
|
import { SavedElementsModal } from '../../saved_elements_modal';
|
||||||
|
|
||||||
|
interface CategorizedElementLists {
|
||||||
|
[key: string]: ElementSpec[];
|
||||||
|
}
|
||||||
|
|
||||||
|
interface ElementTypeMeta {
|
||||||
|
[key: string]: { name: string; icon: string };
|
||||||
|
}
|
||||||
|
|
||||||
|
export const { WorkpadHeaderElementMenu: strings } = ComponentStrings;
|
||||||
|
|
||||||
|
// label and icon for the context menu item for each element type
|
||||||
|
const elementTypeMeta: ElementTypeMeta = {
|
||||||
|
chart: { name: strings.getChartMenuItemLabel(), icon: 'visArea' },
|
||||||
|
filter: { name: strings.getFilterMenuItemLabel(), icon: 'filter' },
|
||||||
|
image: { name: strings.getImageMenuItemLabel(), icon: 'image' },
|
||||||
|
other: { name: strings.getOtherMenuItemLabel(), icon: 'empty' },
|
||||||
|
progress: { name: strings.getProgressMenuItemLabel(), icon: 'visGoal' },
|
||||||
|
shape: { name: strings.getShapeMenuItemLabel(), icon: 'node' },
|
||||||
|
text: { name: strings.getTextMenuItemLabel(), icon: 'visText' },
|
||||||
|
};
|
||||||
|
|
||||||
|
const getElementType = (element: ElementSpec): string =>
|
||||||
|
element && element.type && Object.keys(elementTypeMeta).includes(element.type)
|
||||||
|
? element.type
|
||||||
|
: 'other';
|
||||||
|
|
||||||
|
const categorizeElementsByType = (elements: ElementSpec[]): { [key: string]: ElementSpec[] } => {
|
||||||
|
elements = sortBy(elements, 'displayName');
|
||||||
|
|
||||||
|
const categories: CategorizedElementLists = { other: [] };
|
||||||
|
|
||||||
|
elements.forEach((element: ElementSpec) => {
|
||||||
|
const type = getElementType(element);
|
||||||
|
|
||||||
|
if (categories[type]) {
|
||||||
|
categories[type].push(element);
|
||||||
|
} else {
|
||||||
|
categories[type] = [element];
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
return categories;
|
||||||
|
};
|
||||||
|
|
||||||
|
export interface Props {
|
||||||
|
/**
|
||||||
|
* Dictionary of elements from elements registry
|
||||||
|
*/
|
||||||
|
elements: { [key: string]: ElementSpec };
|
||||||
|
/**
|
||||||
|
* Handler for adding a selected element to the workpad
|
||||||
|
*/
|
||||||
|
addElement: (element: ElementSpec) => void;
|
||||||
|
/**
|
||||||
|
* Renders embeddable flyout
|
||||||
|
*/
|
||||||
|
renderEmbedPanel: (onClose: () => void) => JSX.Element;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const ElementMenu: FunctionComponent<Props> = ({
|
||||||
|
elements,
|
||||||
|
addElement,
|
||||||
|
renderEmbedPanel,
|
||||||
|
}) => {
|
||||||
|
const [isAssetModalVisible, setAssetModalVisible] = useState(false);
|
||||||
|
const [isEmbedPanelVisible, setEmbedPanelVisible] = useState(false);
|
||||||
|
const [isSavedElementsModalVisible, setSavedElementsModalVisible] = useState(false);
|
||||||
|
|
||||||
|
const hideAssetModal = () => setAssetModalVisible(false);
|
||||||
|
const showAssetModal = () => setAssetModalVisible(true);
|
||||||
|
const showEmbedPanel = () => setEmbedPanelVisible(false);
|
||||||
|
const hideEmbedPanel = () => setEmbedPanelVisible(false);
|
||||||
|
const hideSavedElementsModal = () => setSavedElementsModalVisible(false);
|
||||||
|
const showSavedElementsModal = () => setSavedElementsModalVisible(true);
|
||||||
|
|
||||||
|
const {
|
||||||
|
chart: chartElements,
|
||||||
|
filter: filterElements,
|
||||||
|
image: imageElements,
|
||||||
|
other: otherElements,
|
||||||
|
progress: progressElements,
|
||||||
|
shape: shapeElements,
|
||||||
|
text: textElements,
|
||||||
|
} = categorizeElementsByType(Object.values(elements));
|
||||||
|
|
||||||
|
const getPanelTree = (closePopover: ClosePopoverFn) => {
|
||||||
|
const elementToMenuItem = (element: ElementSpec): EuiContextMenuPanelItemDescriptor => ({
|
||||||
|
name: element.displayName || element.name,
|
||||||
|
icon: element.icon,
|
||||||
|
onClick: () => {
|
||||||
|
addElement(element);
|
||||||
|
closePopover();
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
const elementListToMenuItems = (elementList: ElementSpec[]) => {
|
||||||
|
const type = getElementType(elementList[0]);
|
||||||
|
const { name, icon } = elementTypeMeta[type] || elementTypeMeta.other;
|
||||||
|
|
||||||
|
if (elementList.length > 1) {
|
||||||
|
return {
|
||||||
|
name,
|
||||||
|
icon: <EuiIcon type={icon} size="m" />,
|
||||||
|
panel: {
|
||||||
|
id: getId('element-type'),
|
||||||
|
title: name,
|
||||||
|
items: elementList.map(elementToMenuItem),
|
||||||
|
},
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
return elementToMenuItem(elementList[0]);
|
||||||
|
};
|
||||||
|
|
||||||
|
return {
|
||||||
|
id: 0,
|
||||||
|
title: strings.getElementMenuLabel(),
|
||||||
|
items: [
|
||||||
|
elementListToMenuItems(textElements),
|
||||||
|
elementListToMenuItems(shapeElements),
|
||||||
|
elementListToMenuItems(chartElements),
|
||||||
|
elementListToMenuItems(imageElements),
|
||||||
|
elementListToMenuItems(filterElements),
|
||||||
|
elementListToMenuItems(progressElements),
|
||||||
|
elementListToMenuItems(otherElements),
|
||||||
|
{
|
||||||
|
name: strings.getMyElementsMenuItemLabel(),
|
||||||
|
className: CONTEXT_MENU_TOP_BORDER_CLASSNAME,
|
||||||
|
'data-test-subj': 'saved-elements-menu-option',
|
||||||
|
icon: <EuiIcon type="empty" size="m" />,
|
||||||
|
onClick: () => {
|
||||||
|
showSavedElementsModal();
|
||||||
|
closePopover();
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: strings.getAssetsMenuItemLabel(),
|
||||||
|
icon: <EuiIcon type="empty" size="m" />,
|
||||||
|
onClick: () => {
|
||||||
|
showAssetModal();
|
||||||
|
closePopover();
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: strings.getEmbedObjectMenuItemLabel(),
|
||||||
|
className: CONTEXT_MENU_TOP_BORDER_CLASSNAME,
|
||||||
|
icon: <EuiIcon type="logoKibana" size="m" />,
|
||||||
|
onClick: () => {
|
||||||
|
showEmbedPanel();
|
||||||
|
closePopover();
|
||||||
|
},
|
||||||
|
},
|
||||||
|
],
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
const exportControl = (togglePopover: React.MouseEventHandler<any>) => (
|
||||||
|
<EuiButton
|
||||||
|
fill
|
||||||
|
iconType="plusInCircle"
|
||||||
|
size="s"
|
||||||
|
aria-label={strings.getElementMenuLabel()}
|
||||||
|
onClick={togglePopover}
|
||||||
|
className="canvasElementMenu__popoverButton"
|
||||||
|
data-test-subj="add-element-button"
|
||||||
|
>
|
||||||
|
{strings.getElementMenuButtonLabel()}
|
||||||
|
</EuiButton>
|
||||||
|
);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Fragment>
|
||||||
|
<Popover button={exportControl} panelPaddingSize="none" anchorPosition="downLeft">
|
||||||
|
{({ closePopover }: { closePopover: ClosePopoverFn }) => (
|
||||||
|
<EuiContextMenu
|
||||||
|
initialPanelId={0}
|
||||||
|
panels={flattenPanelTree(getPanelTree(closePopover))}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</Popover>
|
||||||
|
{isAssetModalVisible ? <AssetManager onClose={hideAssetModal} /> : null}
|
||||||
|
{isEmbedPanelVisible ? renderEmbedPanel(hideEmbedPanel) : null}
|
||||||
|
{isSavedElementsModalVisible ? <SavedElementsModal onClose={hideSavedElementsModal} /> : null}
|
||||||
|
</Fragment>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
ElementMenu.propTypes = {
|
||||||
|
elements: PropTypes.object,
|
||||||
|
addElement: PropTypes.func.isRequired,
|
||||||
|
};
|
|
@ -0,0 +1,49 @@
|
||||||
|
/*
|
||||||
|
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||||
|
* or more contributor license agreements. Licensed under the Elastic License;
|
||||||
|
* you may not use this file except in compliance with the Elastic License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import React from 'react';
|
||||||
|
import { connect } from 'react-redux';
|
||||||
|
import { compose, withProps } from 'recompose';
|
||||||
|
import { Dispatch } from 'redux';
|
||||||
|
import { withKibana } from '../../../../../../../../src/plugins/kibana_react/public/';
|
||||||
|
import { State, ElementSpec } from '../../../../types';
|
||||||
|
// @ts-ignore Untyped local
|
||||||
|
import { elementsRegistry } from '../../../lib/elements_registry';
|
||||||
|
import { ElementMenu as Component, Props as ComponentProps } from './element_menu';
|
||||||
|
// @ts-ignore Untyped local
|
||||||
|
import { addElement } from '../../../state/actions/elements';
|
||||||
|
import { getSelectedPage } from '../../../state/selectors/workpad';
|
||||||
|
import { AddEmbeddablePanel } from '../../embeddable_flyout';
|
||||||
|
|
||||||
|
interface StateProps {
|
||||||
|
pageId: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface DispatchProps {
|
||||||
|
addElement: (pageId: string) => (partialElement: ElementSpec) => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
const mapStateToProps = (state: State) => ({
|
||||||
|
pageId: getSelectedPage(state),
|
||||||
|
});
|
||||||
|
|
||||||
|
const mapDispatchToProps = (dispatch: Dispatch) => ({
|
||||||
|
addElement: (pageId: string) => (element: ElementSpec) => dispatch(addElement(pageId, element)),
|
||||||
|
});
|
||||||
|
|
||||||
|
const mergeProps = (stateProps: StateProps, dispatchProps: DispatchProps) => ({
|
||||||
|
...stateProps,
|
||||||
|
...dispatchProps,
|
||||||
|
addElement: dispatchProps.addElement(stateProps.pageId),
|
||||||
|
// Moved this section out of the main component to enable stories
|
||||||
|
renderEmbedPanel: (onClose: () => void) => <AddEmbeddablePanel onClose={onClose} />,
|
||||||
|
});
|
||||||
|
|
||||||
|
export const ElementMenu = compose<ComponentProps, {}>(
|
||||||
|
connect(mapStateToProps, mapDispatchToProps, mergeProps),
|
||||||
|
withKibana,
|
||||||
|
withProps(() => ({ elements: elementsRegistry.toJS() }))
|
||||||
|
)(Component);
|