[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 moment from 'moment';
|
||||
import 'moment-timezone';
|
||||
import ReactDOM from "react-dom";
|
||||
|
||||
import initStoryshots, { multiSnapshotWithOptions } from '@storybook/addon-storyshots';
|
||||
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
|
||||
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.
|
||||
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', () => {
|
||||
const React = require.requireActual('react');
|
||||
return {
|
||||
EuiCode: ({children, className}) => (
|
||||
EuiCode: ({ children, className }) => (
|
||||
<span>
|
||||
<code>{children}</code>
|
||||
</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', () => {
|
||||
return {
|
||||
htmlIdGenerator: () => () => `generated-id`,
|
||||
|
@ -71,7 +81,7 @@ jest.mock('plugins/interpreter/registries', () => ({}));
|
|||
|
||||
// Disabling this test due to https://github.com/elastic/eui/issues/2242
|
||||
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';
|
||||
}
|
||||
|
|
|
@ -177,8 +177,10 @@ module.exports = async ({ config }) => {
|
|||
}),
|
||||
|
||||
// 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)?\/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
|
||||
|
|
Before Width: | Height: | Size: 149 KiB |
|
@ -5,14 +5,13 @@
|
|||
*/
|
||||
|
||||
import { ElementFactory } from '../../../types';
|
||||
import header from './header.png';
|
||||
|
||||
export const areaChart: ElementFactory = () => ({
|
||||
name: 'areaChart',
|
||||
displayName: 'Area chart',
|
||||
displayName: 'Area',
|
||||
help: 'A line chart with a filled body',
|
||||
tags: ['chart'],
|
||||
image: header,
|
||||
type: 'chart',
|
||||
icon: 'visArea',
|
||||
expression: `filters
|
||||
| demodata
|
||||
| pointseries x="time" y="mean(price)"
|
||||
|
|
Before Width: | Height: | Size: 67 KiB |
|
@ -5,16 +5,15 @@
|
|||
*/
|
||||
|
||||
import { ElementFactory } from '../../../types';
|
||||
import header from './header.png';
|
||||
|
||||
export const bubbleChart: ElementFactory = () => ({
|
||||
name: 'bubbleChart',
|
||||
displayName: 'Bubble chart',
|
||||
tags: ['chart'],
|
||||
displayName: 'Bubble',
|
||||
type: 'chart',
|
||||
help: 'A customizable bubble chart',
|
||||
width: 700,
|
||||
height: 300,
|
||||
image: header,
|
||||
icon: 'heatmap',
|
||||
expression: `filters
|
||||
| demodata
|
||||
| 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 header from './header.png';
|
||||
|
||||
export const debug: ElementFactory = () => ({
|
||||
name: 'debug',
|
||||
displayName: 'Debug',
|
||||
tags: ['text'],
|
||||
displayName: 'Debug data',
|
||||
help: 'Just dumps the configuration of the element',
|
||||
image: header,
|
||||
icon: 'bug',
|
||||
expression: `demodata
|
||||
| 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 header from './header.png';
|
||||
|
||||
export const dropdownFilter: ElementFactory = () => ({
|
||||
name: 'dropdown_filter',
|
||||
displayName: 'Dropdown filter',
|
||||
tags: ['filter'],
|
||||
name: 'dropdownFilter',
|
||||
displayName: 'Dropdown select',
|
||||
type: 'filter',
|
||||
help: 'A dropdown from which you can select values for an "exactly" filter',
|
||||
image: header,
|
||||
icon: 'filter',
|
||||
height: 50,
|
||||
expression: `demodata
|
||||
| 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 header from './header.png';
|
||||
|
||||
export const horizontalBarChart: ElementFactory = () => ({
|
||||
name: 'horizontalBarChart',
|
||||
displayName: 'Horizontal bar chart',
|
||||
tags: ['chart'],
|
||||
displayName: 'Bar horizontal',
|
||||
type: 'chart',
|
||||
help: 'A customizable horizontal bar chart',
|
||||
image: header,
|
||||
icon: 'visBarHorizontal',
|
||||
expression: `filters
|
||||
| demodata
|
||||
| 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 { ElementFactory } from '../../../types';
|
||||
import header from './header.png';
|
||||
|
||||
export const horizontalProgressBar: ElementFactory = () => ({
|
||||
name: 'horizontalProgressBar',
|
||||
displayName: 'Horizontal progress bar',
|
||||
tags: ['chart', 'proportion'],
|
||||
displayName: 'Horizontal bar',
|
||||
type: 'progress',
|
||||
help: 'Displays progress as a portion of a horizontal bar',
|
||||
width: 400,
|
||||
height: 30,
|
||||
image: header,
|
||||
expression: `filters
|
||||
| demodata
|
||||
| math "mean(percent_uptime)"
|
||||
|
|
Before Width: | Height: | Size: 19 KiB |
|
@ -6,16 +6,14 @@
|
|||
|
||||
import { openSans } from '../../../common/lib/fonts';
|
||||
import { ElementFactory } from '../../../types';
|
||||
import header from './header.png';
|
||||
|
||||
export const horizontalProgressPill: ElementFactory = () => ({
|
||||
name: 'horizontalProgressPill',
|
||||
displayName: 'Horizontal progress pill',
|
||||
tags: ['chart', 'proportion'],
|
||||
displayName: 'Horizontal pill',
|
||||
type: 'progress',
|
||||
help: 'Displays progress as a portion of a horizontal pill',
|
||||
width: 400,
|
||||
height: 30,
|
||||
image: header,
|
||||
expression: `filters
|
||||
| demodata
|
||||
| math "mean(percent_uptime)"
|
||||
|
|
Before Width: | Height: | Size: 91 KiB |
|
@ -5,14 +5,13 @@
|
|||
*/
|
||||
|
||||
import { ElementFactory } from '../../../types';
|
||||
import header from './header.png';
|
||||
|
||||
export const image: ElementFactory = () => ({
|
||||
name: 'image',
|
||||
displayName: 'Image',
|
||||
tags: ['graphic'],
|
||||
type: 'image',
|
||||
help: 'A static image',
|
||||
image: header,
|
||||
icon: 'image',
|
||||
expression: `image dataurl=null mode="contain"
|
||||
| render`,
|
||||
});
|
||||
|
|
|
@ -8,8 +8,8 @@ import { applyElementStrings } from '../../i18n/elements';
|
|||
import { areaChart } from './area_chart';
|
||||
import { bubbleChart } from './bubble_chart';
|
||||
import { debug } from './debug';
|
||||
import { donut } from './donut';
|
||||
import { dropdownFilter } from './dropdown_filter';
|
||||
import { filterDebug } from './filter_debug';
|
||||
import { horizontalBarChart } from './horizontal_bar_chart';
|
||||
import { horizontalProgressBar } from './horizontal_progress_bar';
|
||||
import { horizontalProgressPill } from './horizontal_progress_pill';
|
||||
|
@ -26,7 +26,6 @@ import { repeatImage } from './repeat_image';
|
|||
import { revealImage } from './reveal_image';
|
||||
import { shape } from './shape';
|
||||
import { table } from './table';
|
||||
import { tiltedPie } from './tilted_pie';
|
||||
import { timeFilter } from './time_filter';
|
||||
import { verticalBarChart } from './vert_bar_chart';
|
||||
import { verticalProgressBar } from './vertical_progress_bar';
|
||||
|
@ -39,8 +38,8 @@ const elementSpecs = [
|
|||
areaChart,
|
||||
bubbleChart,
|
||||
debug,
|
||||
donut,
|
||||
dropdownFilter,
|
||||
filterDebug,
|
||||
image,
|
||||
horizontalBarChart,
|
||||
horizontalProgressBar,
|
||||
|
@ -56,7 +55,6 @@ const elementSpecs = [
|
|||
revealImage,
|
||||
shape,
|
||||
table,
|
||||
tiltedPie,
|
||||
timeFilter,
|
||||
verticalBarChart,
|
||||
verticalProgressBar,
|
||||
|
|
Before Width: | Height: | Size: 58 KiB |
|
@ -5,14 +5,13 @@
|
|||
*/
|
||||
|
||||
import { ElementFactory } from '../../../types';
|
||||
import header from './header.png';
|
||||
|
||||
export const lineChart: ElementFactory = () => ({
|
||||
name: 'lineChart',
|
||||
displayName: 'Line chart',
|
||||
tags: ['chart'],
|
||||
displayName: 'Line',
|
||||
type: 'chart',
|
||||
help: 'A customizable line chart',
|
||||
image: header,
|
||||
icon: 'visLine',
|
||||
expression: `filters
|
||||
| demodata
|
||||
| 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.
|
||||
*/
|
||||
|
||||
import header from './header.png';
|
||||
|
||||
import { ElementFactory } from '../../../types';
|
||||
export const markdown: ElementFactory = () => ({
|
||||
name: 'markdown',
|
||||
displayName: 'Markdown',
|
||||
tags: ['text'],
|
||||
help: 'Markup from Markdown',
|
||||
image: header,
|
||||
displayName: 'Text',
|
||||
type: 'text',
|
||||
help: 'Add text using Markdown',
|
||||
icon: 'visText',
|
||||
expression: `filters
|
||||
| demodata
|
||||
| markdown "### Welcome to the Markdown element
|
||||
|
|
Before Width: | Height: | Size: 12 KiB |
|
@ -5,8 +5,6 @@
|
|||
*/
|
||||
|
||||
import { openSans } from '../../../common/lib/fonts';
|
||||
import header from './header.png';
|
||||
|
||||
import { ElementFactory } from '../../../types';
|
||||
import { SetupInitializer } from '../../plugin';
|
||||
|
||||
|
@ -14,11 +12,11 @@ export const metricElementInitializer: SetupInitializer<ElementFactory> = (core,
|
|||
return () => ({
|
||||
name: 'metric',
|
||||
displayName: 'Metric',
|
||||
tags: ['text'],
|
||||
type: 'chart',
|
||||
help: 'A number with a label',
|
||||
width: 200,
|
||||
height: 100,
|
||||
image: header,
|
||||
icon: 'visMetric',
|
||||
expression: `filters
|
||||
| demodata
|
||||
| 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.
|
||||
*/
|
||||
|
||||
import header from './header.png';
|
||||
|
||||
import { ElementFactory } from '../../../types';
|
||||
export const pie: ElementFactory = () => ({
|
||||
name: 'pie',
|
||||
displayName: 'Pie chart',
|
||||
tags: ['chart', 'proportion'],
|
||||
displayName: 'Pie',
|
||||
type: 'chart',
|
||||
width: 300,
|
||||
height: 300,
|
||||
help: 'A simple pie chart',
|
||||
image: header,
|
||||
icon: 'visPie',
|
||||
expression: `filters
|
||||
| demodata
|
||||
| pointseries color="state" size="max(price)"
|
||||
|
|
Before Width: | Height: | Size: 37 KiB |
|
@ -5,14 +5,12 @@
|
|||
*/
|
||||
|
||||
import { ElementFactory } from '../../../types';
|
||||
import header from './header.png';
|
||||
|
||||
export const plot: ElementFactory = () => ({
|
||||
name: 'plot',
|
||||
displayName: 'Coordinate plot',
|
||||
tags: ['chart'],
|
||||
type: 'chart',
|
||||
help: 'Mixed line, bar or dot charts',
|
||||
image: header,
|
||||
expression: `filters
|
||||
| demodata
|
||||
| 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 { ElementFactory } from '../../../types';
|
||||
import header from './header.png';
|
||||
|
||||
export const progressGauge: ElementFactory = () => ({
|
||||
name: 'progressGauge',
|
||||
displayName: 'Progress gauge',
|
||||
tags: ['chart', 'proportion'],
|
||||
displayName: 'Gauge',
|
||||
type: 'progress',
|
||||
help: 'Displays progress as a portion of a gauge',
|
||||
width: 200,
|
||||
height: 200,
|
||||
image: header,
|
||||
icon: 'visGoal',
|
||||
expression: `filters
|
||||
| demodata
|
||||
| math "mean(percent_uptime)"
|
||||
|
|
Before Width: | Height: | Size: 17 KiB |
|
@ -6,16 +6,14 @@
|
|||
|
||||
import { openSans } from '../../../common/lib/fonts';
|
||||
import { ElementFactory } from '../../../types';
|
||||
import header from './header.png';
|
||||
|
||||
export const progressSemicircle: ElementFactory = () => ({
|
||||
name: 'progressSemicircle',
|
||||
displayName: 'Progress semicircle',
|
||||
tags: ['chart', 'proportion'],
|
||||
displayName: 'Semicircle',
|
||||
type: 'progress',
|
||||
help: 'Displays progress as a portion of a semicircle',
|
||||
width: 200,
|
||||
height: 100,
|
||||
image: header,
|
||||
expression: `filters
|
||||
| demodata
|
||||
| math "mean(percent_uptime)"
|
||||
|
|
Before Width: | Height: | Size: 26 KiB |
|
@ -6,16 +6,14 @@
|
|||
|
||||
import { openSans } from '../../../common/lib/fonts';
|
||||
import { ElementFactory } from '../../../types';
|
||||
import header from './header.png';
|
||||
|
||||
export const progressWheel: ElementFactory = () => ({
|
||||
name: 'progressWheel',
|
||||
displayName: 'Progress wheel',
|
||||
tags: ['chart', 'proportion'],
|
||||
displayName: 'Wheel',
|
||||
type: 'progress',
|
||||
help: 'Displays progress as a portion of a wheel',
|
||||
width: 200,
|
||||
height: 200,
|
||||
image: header,
|
||||
expression: `filters
|
||||
| demodata
|
||||
| math "mean(percent_uptime)"
|
||||
|
|
Before Width: | Height: | Size: 82 KiB |
|
@ -5,14 +5,12 @@
|
|||
*/
|
||||
|
||||
import { ElementFactory } from '../../../types';
|
||||
import header from './header.png';
|
||||
|
||||
export const repeatImage: ElementFactory = () => ({
|
||||
name: 'repeatImage',
|
||||
displayName: 'Image repeat',
|
||||
tags: ['graphic', 'proportion'],
|
||||
type: 'image',
|
||||
help: 'Repeats an image N times',
|
||||
image: header,
|
||||
expression: `filters
|
||||
| demodata
|
||||
| math "mean(cost)"
|
||||
|
|
Before Width: | Height: | Size: 8.8 KiB |
|
@ -5,14 +5,12 @@
|
|||
*/
|
||||
|
||||
import { ElementFactory } from '../../../types';
|
||||
import header from './header.png';
|
||||
|
||||
export const revealImage: ElementFactory = () => ({
|
||||
name: 'revealImage',
|
||||
displayName: 'Image reveal',
|
||||
tags: ['graphic', 'proportion'],
|
||||
type: 'image',
|
||||
help: 'Reveals a percentage of an image',
|
||||
image: header,
|
||||
expression: `filters
|
||||
| demodata
|
||||
| math "mean(percent_uptime)"
|
||||
|
|
Before Width: | Height: | Size: 10 KiB |
|
@ -5,16 +5,15 @@
|
|||
*/
|
||||
|
||||
import { ElementFactory } from '../../../types';
|
||||
import header from './header.png';
|
||||
|
||||
export const shape: ElementFactory = () => ({
|
||||
name: 'shape',
|
||||
displayName: 'Shape',
|
||||
tags: ['graphic'],
|
||||
type: 'shape',
|
||||
help: 'A customizable shape',
|
||||
width: 200,
|
||||
height: 200,
|
||||
image: header,
|
||||
icon: 'node',
|
||||
expression:
|
||||
'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 header from './header.png';
|
||||
|
||||
export const table: ElementFactory = () => ({
|
||||
name: 'table',
|
||||
displayName: 'Data table',
|
||||
tags: ['text'],
|
||||
type: 'chart',
|
||||
help: 'A scrollable grid for displaying data in a tabular format',
|
||||
image: header,
|
||||
icon: 'visTable',
|
||||
expression: `filters
|
||||
| demodata
|
||||
| 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 header from './header.png';
|
||||
|
||||
export const timeFilter: ElementFactory = () => ({
|
||||
name: 'time_filter',
|
||||
name: 'timeFilter',
|
||||
displayName: 'Time filter',
|
||||
tags: ['filter'],
|
||||
type: 'filter',
|
||||
help: 'Set a time window',
|
||||
image: header,
|
||||
icon: 'calendar',
|
||||
height: 50,
|
||||
expression: `timefilterControl compact=true column=@timestamp
|
||||
| render`,
|
||||
|
|
Before Width: | Height: | Size: 17 KiB |
|
@ -5,14 +5,13 @@
|
|||
*/
|
||||
|
||||
import { ElementFactory } from '../../../types';
|
||||
import header from './header.png';
|
||||
|
||||
export const verticalBarChart: ElementFactory = () => ({
|
||||
name: 'verticalBarChart',
|
||||
displayName: 'Vertical bar chart',
|
||||
tags: ['chart'],
|
||||
type: 'chart',
|
||||
help: 'A customizable vertical bar chart',
|
||||
image: header,
|
||||
icon: 'visBarVertical',
|
||||
expression: `filters
|
||||
| demodata
|
||||
| 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 { ElementFactory } from '../../../types';
|
||||
import header from './header.png';
|
||||
|
||||
export const verticalProgressBar: ElementFactory = () => ({
|
||||
name: 'verticalProgressBar',
|
||||
displayName: 'Vertical progress bar',
|
||||
tags: ['chart', 'proportion'],
|
||||
type: 'progress',
|
||||
help: 'Displays progress as a portion of a vertical bar',
|
||||
width: 80,
|
||||
height: 400,
|
||||
image: header,
|
||||
expression: `filters
|
||||
| demodata
|
||||
| math "mean(percent_uptime)"
|
||||
|
|
Before Width: | Height: | Size: 20 KiB |
|
@ -6,16 +6,14 @@
|
|||
|
||||
import { openSans } from '../../../common/lib/fonts';
|
||||
import { ElementFactory } from '../../../types';
|
||||
import header from './header.png';
|
||||
|
||||
export const verticalProgressPill: ElementFactory = () => ({
|
||||
name: 'verticalProgressPill',
|
||||
displayName: 'Vertical progress pill',
|
||||
tags: ['chart', 'proportion'],
|
||||
type: 'progress',
|
||||
help: 'Displays progress as a portion of a vertical pill',
|
||||
width: 80,
|
||||
height: 400,
|
||||
image: header,
|
||||
expression: `filters
|
||||
| demodata
|
||||
| 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.
|
||||
*/
|
||||
|
||||
import { chart } from './chart';
|
||||
import { filter } from './filter';
|
||||
import { graphic } from './graphic';
|
||||
import { presentation } from './presentation';
|
||||
import { proportion } from './proportion';
|
||||
import { report } from './report';
|
||||
import { text } from './text';
|
||||
|
||||
// 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_DOWNLOAD = `/public/canvas/${SHAREABLE_RUNTIME_NAME}.js`;
|
||||
export const CANVAS_EMBEDDABLE_CLASSNAME = `canvasEmbeddable`;
|
||||
export const CONTEXT_MENU_TOP_BORDER_CLASSNAME = 'canvasContextMenu--topBorder';
|
||||
|
|
|
@ -15,7 +15,7 @@ export const ComponentStrings = {
|
|||
}),
|
||||
getTitleText: () =>
|
||||
i18n.translate('xpack.canvas.embedObject.titleText', {
|
||||
defaultMessage: 'Embed Object',
|
||||
defaultMessage: 'Add from Visualize library',
|
||||
}),
|
||||
},
|
||||
AdvancedFilter: {
|
||||
|
@ -305,21 +305,21 @@ export const ComponentStrings = {
|
|||
}),
|
||||
},
|
||||
ElementControls: {
|
||||
getEditTooltip: () =>
|
||||
i18n.translate('xpack.canvas.elementControls.editToolTip', {
|
||||
defaultMessage: 'Edit',
|
||||
}),
|
||||
getEditAriaLabel: () =>
|
||||
i18n.translate('xpack.canvas.elementControls.editAriaLabel', {
|
||||
defaultMessage: 'Edit element',
|
||||
getDeleteAriaLabel: () =>
|
||||
i18n.translate('xpack.canvas.elementControls.deleteAriaLabel', {
|
||||
defaultMessage: 'Delete element',
|
||||
}),
|
||||
getDeleteTooltip: () =>
|
||||
i18n.translate('xpack.canvas.elementControls.deleteToolTip', {
|
||||
defaultMessage: 'Delete',
|
||||
}),
|
||||
getDeleteAriaLabel: () =>
|
||||
i18n.translate('xpack.canvas.elementControls.deleteAriaLabel', {
|
||||
defaultMessage: 'Delete element',
|
||||
getEditAriaLabel: () =>
|
||||
i18n.translate('xpack.canvas.elementControls.editAriaLabel', {
|
||||
defaultMessage: 'Edit element',
|
||||
}),
|
||||
getEditTooltip: () =>
|
||||
i18n.translate('xpack.canvas.elementControls.editToolTip', {
|
||||
defaultMessage: 'Edit',
|
||||
}),
|
||||
},
|
||||
ElementSettings: {
|
||||
|
@ -336,53 +336,6 @@ export const ComponentStrings = {
|
|||
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: {
|
||||
getDescription: () =>
|
||||
i18n.translate('xpack.canvas.errorComponent.description', {
|
||||
|
@ -633,6 +586,61 @@ export const ComponentStrings = {
|
|||
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: {
|
||||
getRuntimeStepTitle: () =>
|
||||
i18n.translate('xpack.canvas.shareWebsiteFlyout.snippetsStep.downloadRuntimeTitle', {
|
||||
|
@ -652,7 +660,7 @@ export const ComponentStrings = {
|
|||
defaultMessage: 'Share on a website',
|
||||
}),
|
||||
getUnsupportedRendererWarning: () =>
|
||||
i18n.translate('xpack.canvas.workpadHeaderWorkpadExport.unsupportedRendererWarning', {
|
||||
i18n.translate('xpack.canvas.workpadHeaderShareMenu.unsupportedRendererWarning', {
|
||||
defaultMessage:
|
||||
'This workpad contains render functions that are not supported by the {CANVAS} Shareable Workpad Runtime. These elements will not be rendered:',
|
||||
values: {
|
||||
|
@ -900,6 +908,10 @@ export const ComponentStrings = {
|
|||
i18n.translate('xpack.canvas.textStylePicker.alignRightOption', {
|
||||
defaultMessage: 'Align right',
|
||||
}),
|
||||
getFontColorLabel: () =>
|
||||
i18n.translate('xpack.canvas.textStylePicker.fontColorLabel', {
|
||||
defaultMessage: 'Font Color',
|
||||
}),
|
||||
getStyleBoldOption: () =>
|
||||
i18n.translate('xpack.canvas.textStylePicker.styleBoldOption', {
|
||||
defaultMessage: 'Bold',
|
||||
|
@ -912,10 +924,6 @@ export const ComponentStrings = {
|
|||
i18n.translate('xpack.canvas.textStylePicker.styleUnderlineOption', {
|
||||
defaultMessage: 'Underline',
|
||||
}),
|
||||
getFontColorLabel: () =>
|
||||
i18n.translate('xpack.canvas.textStylePicker.fontColorLabel', {
|
||||
defaultMessage: 'Font Color',
|
||||
}),
|
||||
},
|
||||
TimePicker: {
|
||||
getApplyButtonLabel: () =>
|
||||
|
@ -962,6 +970,10 @@ export const ComponentStrings = {
|
|||
description:
|
||||
'"stylesheet" refers to the collection of CSS style rules entered by the user.',
|
||||
}),
|
||||
getBackgroundColorLabel: () =>
|
||||
i18n.translate('xpack.canvas.workpadConfig.backgroundColorLabel', {
|
||||
defaultMessage: 'Background color',
|
||||
}),
|
||||
getFlipDimensionAriaLabel: () =>
|
||||
i18n.translate('xpack.canvas.workpadConfig.swapDimensionsAriaLabel', {
|
||||
defaultMessage: `Swap the page's width and height`,
|
||||
|
@ -1013,10 +1025,6 @@ export const ComponentStrings = {
|
|||
defaultMessage: 'US Letter',
|
||||
description: 'This is referring to the dimensions of U.S. standard letter paper.',
|
||||
}),
|
||||
getBackgroundColorLabel: () =>
|
||||
i18n.translate('xpack.canvas.workpadConfig.backgroundColorLabel', {
|
||||
defaultMessage: 'Background color',
|
||||
}),
|
||||
},
|
||||
WorkpadCreate: {
|
||||
getWorkpadCreateButtonLabel: () =>
|
||||
|
@ -1029,14 +1037,6 @@ export const ComponentStrings = {
|
|||
i18n.translate('xpack.canvas.workpadHeader.addElementButtonLabel', {
|
||||
defaultMessage: 'Add element',
|
||||
}),
|
||||
getAddElementModalCloseButtonLabel: () =>
|
||||
i18n.translate('xpack.canvas.workpadHeader.addElementModalCloseButtonLabel', {
|
||||
defaultMessage: 'Close',
|
||||
}),
|
||||
getEmbedObjectButtonLabel: () =>
|
||||
i18n.translate('xpack.canvas.workpadHeader.embedObjectButtonLabel', {
|
||||
defaultMessage: 'Embed object',
|
||||
}),
|
||||
getFullScreenButtonAriaLabel: () =>
|
||||
i18n.translate('xpack.canvas.workpadHeader.fullscreenButtonAriaLabel', {
|
||||
defaultMessage: 'View fullscreen',
|
||||
|
@ -1080,9 +1080,9 @@ export const ComponentStrings = {
|
|||
}),
|
||||
},
|
||||
WorkpadHeaderControlSettings: {
|
||||
getTooltip: () =>
|
||||
i18n.translate('xpack.canvas.workpadHeaderControlSettings.settingsTooltip', {
|
||||
defaultMessage: 'Control settings',
|
||||
getButtonLabel: () =>
|
||||
i18n.translate('xpack.canvas.workpadHeaderControlSettings.buttonLabel', {
|
||||
defaultMessage: 'Options',
|
||||
}),
|
||||
},
|
||||
WorkpadHeaderCustomInterval: {
|
||||
|
@ -1105,6 +1105,56 @@ export const ComponentStrings = {
|
|||
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: {
|
||||
getCycleFormLabel: () =>
|
||||
i18n.translate('xpack.canvas.workpadHeaderKioskControl.cycleFormLabel', {
|
||||
|
@ -1129,9 +1179,9 @@ export const ComponentStrings = {
|
|||
defaultMessage: 'Refresh data',
|
||||
}),
|
||||
},
|
||||
WorkpadHeaderWorkpadExport: {
|
||||
WorkpadHeaderShareMenu: {
|
||||
getCopyPDFMessage: () =>
|
||||
i18n.translate('xpack.canvas.workpadHeaderWorkpadExport.copyPDFMessage', {
|
||||
i18n.translate('xpack.canvas.workpadHeaderShareMenu.copyPDFMessage', {
|
||||
defaultMessage: 'The {PDF} generation {URL} was copied to your clipboard.',
|
||||
values: {
|
||||
PDF,
|
||||
|
@ -1139,15 +1189,15 @@ export const ComponentStrings = {
|
|||
},
|
||||
}),
|
||||
getCopyReportingConfigMessage: () =>
|
||||
i18n.translate('xpack.canvas.workpadHeaderWorkpadExport.copyReportingConfigMessage', {
|
||||
i18n.translate('xpack.canvas.workpadHeaderShareMenu.copyReportingConfigMessage', {
|
||||
defaultMessage: 'Copied reporting configuration to clipboard',
|
||||
}),
|
||||
getCopyShareConfigMessage: () =>
|
||||
i18n.translate('xpack.canvas.workpadHeaderWorkpadExport.copyShareConfigMessage', {
|
||||
i18n.translate('xpack.canvas.workpadHeaderShareMenu.copyShareConfigMessage', {
|
||||
defaultMessage: 'Copied share markup to clipboard',
|
||||
}),
|
||||
getExportPDFErrorTitle: (workpadName: string) =>
|
||||
i18n.translate('xpack.canvas.workpadHeaderWorkpadExport.exportPDFErrorMessage', {
|
||||
i18n.translate('xpack.canvas.workpadHeaderShareMenu.exportPDFErrorMessage', {
|
||||
defaultMessage: "Failed to create {PDF} for '{workpadName}'",
|
||||
values: {
|
||||
PDF,
|
||||
|
@ -1155,14 +1205,14 @@ export const ComponentStrings = {
|
|||
},
|
||||
}),
|
||||
getExportPDFMessage: () =>
|
||||
i18n.translate('xpack.canvas.workpadHeaderWorkpadExport.exportPDFMessage', {
|
||||
i18n.translate('xpack.canvas.workpadHeaderShareMenu.exportPDFMessage', {
|
||||
defaultMessage: 'Exporting {PDF}. You can track the progress in Management.',
|
||||
values: {
|
||||
PDF,
|
||||
},
|
||||
}),
|
||||
getExportPDFTitle: (workpadName: string) =>
|
||||
i18n.translate('xpack.canvas.workpadHeaderWorkpadExport.exportPDFTitle', {
|
||||
i18n.translate('xpack.canvas.workpadHeaderShareMenu.exportPDFTitle', {
|
||||
defaultMessage: "{PDF} export of workpad '{workpadName}'",
|
||||
values: {
|
||||
PDF,
|
||||
|
@ -1170,7 +1220,7 @@ export const ComponentStrings = {
|
|||
},
|
||||
}),
|
||||
getPDFPanelCopyAriaLabel: () =>
|
||||
i18n.translate('xpack.canvas.workpadHeaderWorkpadExport.pdfPanelCopyAriaLabel', {
|
||||
i18n.translate('xpack.canvas.workpadHeaderShareMenu.pdfPanelCopyAriaLabel', {
|
||||
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.',
|
||||
values: {
|
||||
|
@ -1179,7 +1229,7 @@ export const ComponentStrings = {
|
|||
},
|
||||
}),
|
||||
getPDFPanelCopyButtonLabel: () =>
|
||||
i18n.translate('xpack.canvas.workpadHeaderWorkpadExport.pdfPanelCopyButtonLabel', {
|
||||
i18n.translate('xpack.canvas.workpadHeaderShareMenu.pdfPanelCopyButtonLabel', {
|
||||
defaultMessage: 'Copy {POST} {URL}',
|
||||
values: {
|
||||
POST,
|
||||
|
@ -1187,7 +1237,7 @@ export const ComponentStrings = {
|
|||
},
|
||||
}),
|
||||
getPDFPanelCopyDescription: () =>
|
||||
i18n.translate('xpack.canvas.workpadHeaderWorkpadExport.pdfPanelCopyDescription', {
|
||||
i18n.translate('xpack.canvas.workpadHeaderShareMenu.pdfPanelCopyDescription', {
|
||||
defaultMessage:
|
||||
'Alternatively, copy this {POST} {URL} to call generation from outside {KIBANA} or from Watcher.',
|
||||
values: {
|
||||
|
@ -1197,14 +1247,14 @@ export const ComponentStrings = {
|
|||
},
|
||||
}),
|
||||
getPDFPanelGenerateButtonLabel: () =>
|
||||
i18n.translate('xpack.canvas.workpadHeaderWorkpadExport.pdfPanelGenerateButtonLabel', {
|
||||
i18n.translate('xpack.canvas.workpadHeaderShareMenu.pdfPanelGenerateButtonLabel', {
|
||||
defaultMessage: 'Generate {PDF}',
|
||||
values: {
|
||||
PDF,
|
||||
},
|
||||
}),
|
||||
getPDFPanelGenerateDescription: () =>
|
||||
i18n.translate('xpack.canvas.workpadHeaderWorkpadExport.pdfPanelGenerateDescription', {
|
||||
i18n.translate('xpack.canvas.workpadHeaderShareMenu.pdfPanelGenerateDescription', {
|
||||
defaultMessage:
|
||||
'{PDF}s can take a minute or two to generate based on the size of your workpad.',
|
||||
values: {
|
||||
|
@ -1212,7 +1262,7 @@ export const ComponentStrings = {
|
|||
},
|
||||
}),
|
||||
getShareableZipErrorTitle: (workpadName: string) =>
|
||||
i18n.translate('xpack.canvas.workpadHeaderWorkpadExport.shareWebsiteErrorTitle', {
|
||||
i18n.translate('xpack.canvas.workpadHeaderShareMenu.shareWebsiteErrorTitle', {
|
||||
defaultMessage:
|
||||
"Failed to create {ZIP} file for '{workpadName}'. The workpad may be too large. You'll need to download the files separately.",
|
||||
values: {
|
||||
|
@ -1221,69 +1271,101 @@ export const ComponentStrings = {
|
|||
},
|
||||
}),
|
||||
getShareDownloadJSONTitle: () =>
|
||||
i18n.translate('xpack.canvas.workpadHeaderWorkpadExport.shareDownloadJSONTitle', {
|
||||
i18n.translate('xpack.canvas.workpadHeaderShareMenu.shareDownloadJSONTitle', {
|
||||
defaultMessage: 'Download as {JSON}',
|
||||
values: {
|
||||
JSON,
|
||||
},
|
||||
}),
|
||||
getShareDownloadPDFTitle: () =>
|
||||
i18n.translate('xpack.canvas.workpadHeaderWorkpadExport.shareDownloadPDFTitle', {
|
||||
i18n.translate('xpack.canvas.workpadHeaderShareMenu.shareDownloadPDFTitle', {
|
||||
defaultMessage: '{PDF} reports',
|
||||
values: {
|
||||
PDF,
|
||||
},
|
||||
}),
|
||||
getShareMenuButtonLabel: () =>
|
||||
i18n.translate('xpack.canvas.workpadHeaderShareMenu.shareMenuButtonLabel', {
|
||||
defaultMessage: 'Share',
|
||||
}),
|
||||
getShareWebsiteTitle: () =>
|
||||
i18n.translate('xpack.canvas.workpadHeaderWorkpadExport.shareWebsiteTitle', {
|
||||
i18n.translate('xpack.canvas.workpadHeaderShareMenu.shareWebsiteTitle', {
|
||||
defaultMessage: 'Share on a website',
|
||||
}),
|
||||
getShareWorkpadMessage: () =>
|
||||
i18n.translate('xpack.canvas.workpadHeaderWorkpadExport.shareWorkpadMessage', {
|
||||
i18n.translate('xpack.canvas.workpadHeaderShareMenu.shareWorkpadMessage', {
|
||||
defaultMessage: 'Share this workpad',
|
||||
}),
|
||||
getUnknownExportErrorMessage: (type: string) =>
|
||||
i18n.translate('xpack.canvas.workpadHeaderWorkpadExport.unknownExportErrorMessage', {
|
||||
i18n.translate('xpack.canvas.workpadHeaderShareMenu.unknownExportErrorMessage', {
|
||||
defaultMessage: 'Unknown export type: {type}',
|
||||
values: {
|
||||
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: () =>
|
||||
i18n.translate('xpack.canvas.workpadHeaderWorkpadZoom.zoomControlsAriaLabel', {
|
||||
i18n.translate('xpack.canvas.workpadHeaderViewMenu.zoomControlsAriaLabel', {
|
||||
defaultMessage: 'Zoom controls',
|
||||
}),
|
||||
getZoomControlsTooltip: () =>
|
||||
i18n.translate('xpack.canvas.workpadHeaderWorkpadZoom.zoomControlsTooltip', {
|
||||
i18n.translate('xpack.canvas.workpadHeaderViewMenu.zoomControlsTooltip', {
|
||||
defaultMessage: 'Zoom controls',
|
||||
}),
|
||||
getZoomFitToWindowText: () =>
|
||||
i18n.translate('xpack.canvas.workpadHeaderWorkpadZoom.zoomFitToWindowText', {
|
||||
i18n.translate('xpack.canvas.workpadHeaderViewMenu.zoomFitToWindowText', {
|
||||
defaultMessage: 'Fit to window',
|
||||
}),
|
||||
getZoomInText: () =>
|
||||
i18n.translate('xpack.canvas.workpadHeaderWorkpadZoom.zoomInText', {
|
||||
i18n.translate('xpack.canvas.workpadHeaderViewMenu.zoomInText', {
|
||||
defaultMessage: 'Zoom in',
|
||||
}),
|
||||
getZoomMenuItemLabel: () =>
|
||||
i18n.translate('xpack.canvas.workpadHeaderViewMenu.zoomMenuItemLabel', {
|
||||
defaultMessage: 'Zoom',
|
||||
}),
|
||||
getZoomOutText: () =>
|
||||
i18n.translate('xpack.canvas.workpadHeaderWorkpadZoom.zoomOutText', {
|
||||
i18n.translate('xpack.canvas.workpadHeaderViewMenu.zoomOutText', {
|
||||
defaultMessage: 'Zoom out',
|
||||
}),
|
||||
getZoomPanelTitle: () =>
|
||||
i18n.translate('xpack.canvas.workpadHeaderWorkpadZoom.zoomPanelTitle', {
|
||||
i18n.translate('xpack.canvas.workpadHeaderViewMenu.zoomPanelTitle', {
|
||||
defaultMessage: 'Zoom',
|
||||
}),
|
||||
getZoomPercentage: (scale: number) =>
|
||||
i18n.translate('xpack.canvas.workpadHeaderWorkpadZoom.zoomResetText', {
|
||||
i18n.translate('xpack.canvas.workpadHeaderViewMenu.zoomResetText', {
|
||||
defaultMessage: '{scalePercentage}%',
|
||||
values: {
|
||||
scalePercentage: scale * 100,
|
||||
},
|
||||
}),
|
||||
getZoomResetText: () =>
|
||||
i18n.translate('xpack.canvas.workpadHeaderWorkpadZoom.zoomPrecentageValue', {
|
||||
i18n.translate('xpack.canvas.workpadHeaderViewMenu.zoomPrecentageValue', {
|
||||
defaultMessage: 'Reset',
|
||||
}),
|
||||
},
|
||||
|
|
|
@ -7,8 +7,6 @@
|
|||
import { ElementFactory } from '../../types';
|
||||
import { getElementStrings } from './index';
|
||||
|
||||
import { TagStrings } from '../../i18n';
|
||||
|
||||
/**
|
||||
* 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
|
||||
|
@ -34,17 +32,6 @@ export const applyElementStrings = (elements: ElementFactory[]) => {
|
|||
if (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;
|
||||
|
|
|
@ -9,8 +9,6 @@ import { coreMock } from '../../../../../../src/core/public/mocks';
|
|||
|
||||
const elementSpecs = initializeElements(coreMock.createSetup() as any, {} as any);
|
||||
|
||||
import { TagStrings } from '../tags';
|
||||
|
||||
describe('ElementStrings', () => {
|
||||
const elementStrings = getElementStrings();
|
||||
const elementNames = elementSpecs.map(spec => spec().name);
|
||||
|
@ -37,15 +35,4 @@ describe('ElementStrings', () => {
|
|||
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 => ({
|
||||
areaChart: {
|
||||
displayName: i18n.translate('xpack.canvas.elements.areaChartDisplayName', {
|
||||
defaultMessage: 'Area chart',
|
||||
defaultMessage: 'Area',
|
||||
}),
|
||||
help: i18n.translate('xpack.canvas.elements.areaChartHelpText', {
|
||||
defaultMessage: 'A line chart with a filled body',
|
||||
|
@ -31,7 +31,7 @@ export const getElementStrings = (): ElementStringDict => ({
|
|||
},
|
||||
bubbleChart: {
|
||||
displayName: i18n.translate('xpack.canvas.elements.bubbleChartDisplayName', {
|
||||
defaultMessage: 'Bubble chart',
|
||||
defaultMessage: 'Bubble',
|
||||
}),
|
||||
help: i18n.translate('xpack.canvas.elements.bubbleChartHelpText', {
|
||||
defaultMessage: 'A customizable bubble chart',
|
||||
|
@ -39,31 +39,31 @@ export const getElementStrings = (): ElementStringDict => ({
|
|||
},
|
||||
debug: {
|
||||
displayName: i18n.translate('xpack.canvas.elements.debugDisplayName', {
|
||||
defaultMessage: 'Debug',
|
||||
defaultMessage: 'Debug data',
|
||||
}),
|
||||
help: i18n.translate('xpack.canvas.elements.debugHelpText', {
|
||||
defaultMessage: 'Just dumps the configuration of the element',
|
||||
}),
|
||||
},
|
||||
donut: {
|
||||
displayName: i18n.translate('xpack.canvas.elements.donutChartDisplayName', {
|
||||
defaultMessage: 'Donut chart',
|
||||
}),
|
||||
help: i18n.translate('xpack.canvas.elements.donutChartHelpText', {
|
||||
defaultMessage: 'A customizable donut chart',
|
||||
}),
|
||||
},
|
||||
dropdown_filter: {
|
||||
dropdownFilter: {
|
||||
displayName: i18n.translate('xpack.canvas.elements.dropdownFilterDisplayName', {
|
||||
defaultMessage: 'Dropdown filter',
|
||||
defaultMessage: 'Dropdown select',
|
||||
}),
|
||||
help: i18n.translate('xpack.canvas.elements.dropdownFilterHelpText', {
|
||||
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: {
|
||||
displayName: i18n.translate('xpack.canvas.elements.horizontalBarChartDisplayName', {
|
||||
defaultMessage: 'Horizontal bar chart',
|
||||
defaultMessage: 'Horizontal bar',
|
||||
}),
|
||||
help: i18n.translate('xpack.canvas.elements.horizontalBarChartHelpText', {
|
||||
defaultMessage: 'A customizable horizontal bar chart',
|
||||
|
@ -71,7 +71,7 @@ export const getElementStrings = (): ElementStringDict => ({
|
|||
},
|
||||
horizontalProgressBar: {
|
||||
displayName: i18n.translate('xpack.canvas.elements.horizontalProgressBarDisplayName', {
|
||||
defaultMessage: 'Horizontal progress bar',
|
||||
defaultMessage: 'Horizontal bar',
|
||||
}),
|
||||
help: i18n.translate('xpack.canvas.elements.horizontalProgressBarHelpText', {
|
||||
defaultMessage: 'Displays progress as a portion of a horizontal bar',
|
||||
|
@ -79,7 +79,7 @@ export const getElementStrings = (): ElementStringDict => ({
|
|||
},
|
||||
horizontalProgressPill: {
|
||||
displayName: i18n.translate('xpack.canvas.elements.horizontalProgressPillDisplayName', {
|
||||
defaultMessage: 'Horizontal progress pill',
|
||||
defaultMessage: 'Horizontal pill',
|
||||
}),
|
||||
help: i18n.translate('xpack.canvas.elements.horizontalProgressPillHelpText', {
|
||||
defaultMessage: 'Displays progress as a portion of a horizontal pill',
|
||||
|
@ -95,7 +95,7 @@ export const getElementStrings = (): ElementStringDict => ({
|
|||
},
|
||||
lineChart: {
|
||||
displayName: i18n.translate('xpack.canvas.elements.lineChartDisplayName', {
|
||||
defaultMessage: 'Line chart',
|
||||
defaultMessage: 'Line',
|
||||
}),
|
||||
help: i18n.translate('xpack.canvas.elements.lineChartHelpText', {
|
||||
defaultMessage: 'A customizable line chart',
|
||||
|
@ -119,7 +119,7 @@ export const getElementStrings = (): ElementStringDict => ({
|
|||
},
|
||||
pie: {
|
||||
displayName: i18n.translate('xpack.canvas.elements.pieDisplayName', {
|
||||
defaultMessage: 'Pie chart',
|
||||
defaultMessage: 'Pie',
|
||||
}),
|
||||
help: i18n.translate('xpack.canvas.elements.pieHelpText', {
|
||||
defaultMessage: 'Pie chart',
|
||||
|
@ -135,7 +135,7 @@ export const getElementStrings = (): ElementStringDict => ({
|
|||
},
|
||||
progressGauge: {
|
||||
displayName: i18n.translate('xpack.canvas.elements.progressGaugeDisplayName', {
|
||||
defaultMessage: 'Progress gauge',
|
||||
defaultMessage: 'Gauge',
|
||||
}),
|
||||
help: i18n.translate('xpack.canvas.elements.progressGaugeHelpText', {
|
||||
defaultMessage: 'Displays progress as a portion of a gauge',
|
||||
|
@ -143,7 +143,7 @@ export const getElementStrings = (): ElementStringDict => ({
|
|||
},
|
||||
progressSemicircle: {
|
||||
displayName: i18n.translate('xpack.canvas.elements.progressSemicircleDisplayName', {
|
||||
defaultMessage: 'Progress semicircle',
|
||||
defaultMessage: 'Semicircle',
|
||||
}),
|
||||
help: i18n.translate('xpack.canvas.elements.progressSemicircleHelpText', {
|
||||
defaultMessage: 'Displays progress as a portion of a semicircle',
|
||||
|
@ -151,7 +151,7 @@ export const getElementStrings = (): ElementStringDict => ({
|
|||
},
|
||||
progressWheel: {
|
||||
displayName: i18n.translate('xpack.canvas.elements.progressWheelDisplayName', {
|
||||
defaultMessage: 'Progress wheel',
|
||||
defaultMessage: 'Wheel',
|
||||
}),
|
||||
help: i18n.translate('xpack.canvas.elements.progressWheelHelpText', {
|
||||
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',
|
||||
}),
|
||||
},
|
||||
tiltedPie: {
|
||||
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: {
|
||||
timeFilter: {
|
||||
displayName: i18n.translate('xpack.canvas.elements.timeFilterDisplayName', {
|
||||
defaultMessage: 'Time filter',
|
||||
}),
|
||||
|
@ -207,7 +199,7 @@ export const getElementStrings = (): ElementStringDict => ({
|
|||
},
|
||||
verticalBarChart: {
|
||||
displayName: i18n.translate('xpack.canvas.elements.verticalBarChartDisplayName', {
|
||||
defaultMessage: 'Vertical bar chart',
|
||||
defaultMessage: 'Vertical bar',
|
||||
}),
|
||||
help: i18n.translate('xpack.canvas.elements.verticalBarChartHelpText', {
|
||||
defaultMessage: 'A customizable vertical bar chart',
|
||||
|
@ -215,7 +207,7 @@ export const getElementStrings = (): ElementStringDict => ({
|
|||
},
|
||||
verticalProgressBar: {
|
||||
displayName: i18n.translate('xpack.canvas.elements.verticalProgressBarDisplayName', {
|
||||
defaultMessage: 'Vertical progress bar',
|
||||
defaultMessage: 'Vertical bar',
|
||||
}),
|
||||
help: i18n.translate('xpack.canvas.elements.verticalProgressBarHelpText', {
|
||||
defaultMessage: 'Displays progress as a portion of a vertical bar',
|
||||
|
@ -223,7 +215,7 @@ export const getElementStrings = (): ElementStringDict => ({
|
|||
},
|
||||
verticalProgressPill: {
|
||||
displayName: i18n.translate('xpack.canvas.elements.verticalProgressPillDisplayName', {
|
||||
defaultMessage: 'Vertical progress pill',
|
||||
defaultMessage: 'Vertical pill',
|
||||
}),
|
||||
help: i18n.translate('xpack.canvas.elements.verticalProgressPillHelpText', {
|
||||
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 { FunctionHelp } from '../function_help';
|
||||
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>> = {
|
||||
help: i18n.translate('xpack.canvas.functions.alterColumnHelpText', {
|
||||
|
|
|
@ -7,32 +7,12 @@
|
|||
import { i18n } from '@kbn/i18n';
|
||||
|
||||
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: () =>
|
||||
i18n.translate('xpack.canvas.tags.presentationTag', {
|
||||
defaultMessage: 'presentation',
|
||||
}),
|
||||
proportion: () =>
|
||||
i18n.translate('xpack.canvas.tags.proportionTag', {
|
||||
defaultMessage: 'proportion',
|
||||
}),
|
||||
report: () =>
|
||||
i18n.translate('xpack.canvas.tags.reportTag', {
|
||||
defaultMessage: 'report',
|
||||
}),
|
||||
text: () =>
|
||||
i18n.translate('xpack.canvas.tags.textTag', {
|
||||
defaultMessage: 'text',
|
||||
}),
|
||||
};
|
||||
|
|
|
@ -31,7 +31,7 @@ $canvasLayoutFontSize: $euiFontSizeS;
|
|||
.canvasLayout__stageHeader {
|
||||
flex-grow: 0;
|
||||
flex-basis: auto;
|
||||
padding: ($euiSizeXS + 1px) $euiSize $euiSizeXS;
|
||||
padding: 1px $euiSize 0;
|
||||
font-size: $canvasLayoutFontSize;
|
||||
border-bottom: $euiBorderThin;
|
||||
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)
|
||||
.add('no assets', () => (
|
||||
// @ts-ignore @types/react has not been updated to support defaultProps yet.
|
||||
<AssetManager
|
||||
onAddImageElement={action('onAddImageElement')}
|
||||
onAssetAdd={action('onAssetAdd')}
|
||||
onAssetCopy={action('onAssetCopy')}
|
||||
onAssetDelete={action('onAssetDelete')}
|
||||
onClose={action('onClose')}
|
||||
/>
|
||||
))
|
||||
.add('two assets', () => (
|
||||
|
@ -43,5 +43,6 @@ storiesOf('components/Assets/AssetManager', module)
|
|||
onAssetAdd={action('onAssetAdd')}
|
||||
onAssetCopy={action('onAssetCopy')}
|
||||
onAssetDelete={action('onAssetDelete')}
|
||||
onClose={action('onClose')}
|
||||
/>
|
||||
));
|
|
@ -4,13 +4,6 @@
|
|||
* 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 React, { Fragment, PureComponent } from 'react';
|
||||
|
||||
|
@ -33,13 +26,13 @@ interface Props {
|
|||
onAssetCopy: () => void;
|
||||
/** Function to invoke when an asset is added */
|
||||
onAssetAdd: (asset: File) => void;
|
||||
/** Function to invoke when an asset modal is closed */
|
||||
onClose: () => void;
|
||||
}
|
||||
|
||||
interface State {
|
||||
/** The id of the asset to delete, if applicable. Is set if the viewer clicks the delete icon */
|
||||
deleteId: string | null;
|
||||
/** Determines if the modal is currently visible */
|
||||
isModalVisible: boolean;
|
||||
/** Indicates if the modal is currently loading */
|
||||
isLoading: boolean;
|
||||
}
|
||||
|
@ -51,6 +44,7 @@ export class AssetManager extends PureComponent<Props, State> {
|
|||
onAssetAdd: PropTypes.func.isRequired,
|
||||
onAssetCopy: PropTypes.func.isRequired,
|
||||
onAssetDelete: PropTypes.func.isRequired,
|
||||
onClose: PropTypes.func.isRequired,
|
||||
};
|
||||
|
||||
public static defaultProps = {
|
||||
|
@ -60,12 +54,11 @@ export class AssetManager extends PureComponent<Props, State> {
|
|||
public state = {
|
||||
deleteId: null,
|
||||
isLoading: false,
|
||||
isModalVisible: false,
|
||||
};
|
||||
|
||||
public render() {
|
||||
const { isModalVisible, isLoading } = this.state;
|
||||
const { assetValues, onAssetCopy, onAddImageElement } = this.props;
|
||||
const { isLoading } = this.state;
|
||||
const { assetValues, onAssetCopy, onAddImageElement, onClose } = this.props;
|
||||
|
||||
const assetModal = (
|
||||
<AssetModal
|
||||
|
@ -74,10 +67,10 @@ export class AssetManager extends PureComponent<Props, State> {
|
|||
onAssetCopy={onAssetCopy}
|
||||
onAssetCreate={(createdAsset: AssetType) => {
|
||||
onAddImageElement(createdAsset.id);
|
||||
this.setState({ isModalVisible: false });
|
||||
onClose();
|
||||
}}
|
||||
onAssetDelete={(asset: AssetType) => this.setState({ deleteId: asset.id })}
|
||||
onClose={() => this.setState({ isModalVisible: false })}
|
||||
onClose={onClose}
|
||||
onFileUpload={this.handleFileUpload}
|
||||
/>
|
||||
);
|
||||
|
@ -95,14 +88,12 @@ export class AssetManager extends PureComponent<Props, State> {
|
|||
|
||||
return (
|
||||
<Fragment>
|
||||
<EuiButtonEmpty onClick={this.showModal}>{strings.getButtonLabel()}</EuiButtonEmpty>
|
||||
{isModalVisible ? assetModal : null}
|
||||
{assetModal}
|
||||
{confirmModal}
|
||||
</Fragment>
|
||||
);
|
||||
}
|
||||
|
||||
private showModal = () => this.setState({ isModalVisible: true });
|
||||
private resetDelete = () => this.setState({ deleteId: null });
|
||||
|
||||
private doDelete = () => {
|
||||
|
|
|
@ -98,6 +98,7 @@ export const AssetModal: FunctionComponent<Props> = props => {
|
|||
<EuiFilePicker
|
||||
initialPromptText={strings.getFilePickerPromptText()}
|
||||
compressed
|
||||
display="default"
|
||||
multiple
|
||||
onChange={onFileUpload}
|
||||
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.
|
||||
*/
|
||||
|
||||
export { Popover } from './popover';
|
||||
export { Popover, ClosePopoverFn } from './popover';
|
||||
|
|
|
@ -24,6 +24,8 @@ interface Props {
|
|||
className?: string;
|
||||
}
|
||||
|
||||
export type ClosePopoverFn = () => void;
|
||||
|
||||
interface State {
|
||||
isPopoverOpen: boolean;
|
||||
}
|
||||
|
@ -61,7 +63,7 @@ export class Popover extends Component<Props, State> {
|
|||
}));
|
||||
};
|
||||
|
||||
closePopover = () => {
|
||||
closePopover: ClosePopoverFn = () => {
|
||||
this.setState({
|
||||
isPopoverOpen: false,
|
||||
});
|
||||
|
|
|
@ -1,6 +1,6 @@
|
|||
// 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
|
||||
style={
|
||||
Object {
|
||||
|
@ -22,6 +22,7 @@ exports[`Storyshots components/Elements/ElementControls has two buttons 1`] = `
|
|||
<button
|
||||
aria-label="Edit element"
|
||||
className="euiButtonIcon euiButtonIcon--primary"
|
||||
data-test-subj="canvasElementCard__editButton"
|
||||
onBlur={[Function]}
|
||||
onClick={[Function]}
|
||||
onFocus={[Function]}
|
||||
|
@ -47,6 +48,7 @@ exports[`Storyshots components/Elements/ElementControls has two buttons 1`] = `
|
|||
<button
|
||||
aria-label="Delete element"
|
||||
className="euiButtonIcon euiButtonIcon--danger"
|
||||
data-test-subj="canvasElementCard__deleteButton"
|
||||
onBlur={[Function]}
|
||||
onClick={[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 { ElementControls } from '../element_controls';
|
||||
|
||||
storiesOf('components/Elements/ElementControls', module)
|
||||
storiesOf('components/SavedElementsModal/ElementControls', module)
|
||||
.addDecorator(story => (
|
||||
<div
|
||||
style={{
|
|
@ -8,9 +8,9 @@ import React from 'react';
|
|||
import { storiesOf } from '@storybook/react';
|
||||
import { action } from '@storybook/addon-actions';
|
||||
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 => (
|
||||
<div
|
||||
style={{
|
||||
|
@ -20,38 +20,19 @@ storiesOf('components/Elements/ElementGrid', module)
|
|||
{story()}
|
||||
</div>
|
||||
))
|
||||
.add('without controls', () => (
|
||||
<ElementGrid elements={testElements} handleClick={action('addElement')} />
|
||||
))
|
||||
.add('with controls', () => (
|
||||
.add('default', () => (
|
||||
<ElementGrid
|
||||
elements={testCustomElements}
|
||||
handleClick={action('addCustomElement')}
|
||||
showControls
|
||||
onClick={action('addCustomElement')}
|
||||
onDelete={action('onDelete')}
|
||||
onEdit={action('onEdit')}
|
||||
/>
|
||||
))
|
||||
.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
|
||||
elements={testCustomElements}
|
||||
handleClick={action('addCustomElement')}
|
||||
filterText="Lorem"
|
||||
showControls
|
||||
onClick={action('addCustomElement')}
|
||||
filterText="table"
|
||||
onDelete={action('onDelete')}
|
||||
onEdit={action('onEdit')}
|
||||
/>
|
|
@ -6,41 +6,6 @@
|
|||
|
||||
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 = [
|
||||
{
|
||||
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}>
|
||||
<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>
|
||||
</EuiFlexItem>
|
||||
<EuiFlexItem grow={false}>
|
||||
|
@ -40,6 +45,7 @@ export const ElementControls: FunctionComponent<Props> = ({ onDelete, onEdit })
|
|||
iconType="trash"
|
||||
aria-label={strings.getDeleteAriaLabel()}
|
||||
onClick={onDelete}
|
||||
data-test-subj="canvasElementCard__deleteButton"
|
||||
/>
|
||||
</EuiToolTip>
|
||||
</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
|
||||
aria-label="Save as new element"
|
||||
className="euiButtonIcon euiButtonIcon--text"
|
||||
data-test-subj="canvasSidebarHeader__saveElementButton"
|
||||
onBlur={[Function]}
|
||||
onClick={[Function]}
|
||||
onFocus={[Function]}
|
||||
|
@ -97,7 +98,7 @@ exports[`Storyshots components/Sidebar/SidebarHeader default 1`] = `
|
|||
</div>
|
||||
`;
|
||||
|
||||
exports[`Storyshots components/Sidebar/SidebarHeader without layer controls 1`] = `
|
||||
exports[`Storyshots components/Sidebar/SidebarHeader with layer controls 1`] = `
|
||||
<div
|
||||
style={
|
||||
Object {
|
||||
|
@ -123,6 +124,106 @@ exports[`Storyshots components/Sidebar/SidebarHeader without layer controls 1`]
|
|||
<div
|
||||
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
|
||||
className="euiFlexItem euiFlexItem--flexGrowZero"
|
||||
>
|
||||
|
@ -134,6 +235,7 @@ exports[`Storyshots components/Sidebar/SidebarHeader without layer controls 1`]
|
|||
<button
|
||||
aria-label="Save as new element"
|
||||
className="euiButtonIcon euiButtonIcon--text"
|
||||
data-test-subj="canvasSidebarHeader__saveElementButton"
|
||||
onBlur={[Function]}
|
||||
onClick={[Function]}
|
||||
onFocus={[Function]}
|
||||
|
|
|
@ -35,6 +35,6 @@ const handlers = {
|
|||
storiesOf('components/Sidebar/SidebarHeader', module)
|
||||
.addDecorator(story => <div style={{ width: '300px' }}>{story()}</div>)
|
||||
.add('default', () => <SidebarHeader title="Selected layer" {...handlers} />)
|
||||
.add('without layer controls', () => (
|
||||
<SidebarHeader title="Grouped element" showLayerControls={false} {...handlers} />
|
||||
.add('with layer controls', () => (
|
||||
<SidebarHeader title="Grouped element" showLayerControls={true} {...handlers} />
|
||||
));
|
||||
|
|
|
@ -22,8 +22,7 @@ import { CustomElementModal } from '../custom_element_modal';
|
|||
import { ToolTipShortcut } from '../tool_tip_shortcut/';
|
||||
import { ComponentStrings } from '../../../i18n/components';
|
||||
import { ShortcutStrings } from '../../../i18n/shortcuts';
|
||||
|
||||
const topBorderClassName = 'canvasContextMenu--topBorder';
|
||||
import { CONTEXT_MENU_TOP_BORDER_CLASSNAME } from '../../../common/lib/constants';
|
||||
|
||||
const { SidebarHeader: strings } = ComponentStrings;
|
||||
const shortcutHelp = ShortcutStrings.getShortcutHelp();
|
||||
|
@ -282,7 +281,11 @@ export class SidebarHeader extends Component<Props, State> {
|
|||
const { bringToFront, bringForward, sendBackward, sendToBack } = this.props;
|
||||
|
||||
return {
|
||||
menuItem: { name: strings.getOrderMenuItemLabel(), className: topBorderClassName, panel: 1 },
|
||||
menuItem: {
|
||||
name: strings.getOrderMenuItemLabel(),
|
||||
className: CONTEXT_MENU_TOP_BORDER_CLASSNAME,
|
||||
panel: 1,
|
||||
},
|
||||
panel: {
|
||||
id: 1,
|
||||
title: strings.getOrderMenuItemLabel(),
|
||||
|
@ -396,7 +399,7 @@ export class SidebarHeader extends Component<Props, State> {
|
|||
? [
|
||||
{
|
||||
name: strings.getUngroupMenuItemLabel(),
|
||||
className: topBorderClassName,
|
||||
className: CONTEXT_MENU_TOP_BORDER_CLASSNAME,
|
||||
onClick: close(ungroupNodes),
|
||||
},
|
||||
]
|
||||
|
@ -404,7 +407,7 @@ export class SidebarHeader extends Component<Props, State> {
|
|||
? [
|
||||
{
|
||||
name: strings.getGroupMenuItemLabel(),
|
||||
className: topBorderClassName,
|
||||
className: CONTEXT_MENU_TOP_BORDER_CLASSNAME,
|
||||
onClick: close(groupNodes),
|
||||
},
|
||||
]
|
||||
|
@ -483,7 +486,7 @@ export class SidebarHeader extends Component<Props, State> {
|
|||
items.push({
|
||||
name: strings.getSaveElementMenuItemLabel(),
|
||||
icon: 'indexOpen',
|
||||
className: topBorderClassName,
|
||||
className: CONTEXT_MENU_TOP_BORDER_CLASSNAME,
|
||||
onClick: this._showModal,
|
||||
});
|
||||
|
||||
|
@ -537,6 +540,7 @@ export class SidebarHeader extends Component<Props, State> {
|
|||
color="text"
|
||||
iconType="indexOpen"
|
||||
onClick={this._showModal}
|
||||
data-test-subj="canvasSidebarHeader__saveElementButton"
|
||||
aria-label={strings.getSaveElementMenuItemLabel()}
|
||||
/>
|
||||
</EuiToolTip>
|
||||
|
|
|
@ -6,7 +6,7 @@
|
|||
|
||||
import React, { MouseEventHandler } from 'react';
|
||||
import PropTypes from 'prop-types';
|
||||
import { EuiFlexGroup, EuiFlexItem, EuiButtonIcon, EuiToolTip } from '@elastic/eui';
|
||||
import { EuiFlexGroup, EuiFlexItem, EuiButtonEmpty } from '@elastic/eui';
|
||||
// @ts-ignore untyped local
|
||||
import { Popover } from '../../popover';
|
||||
import { AutoRefreshControls } from './auto_refresh_controls';
|
||||
|
@ -39,16 +39,16 @@ export const ControlSettings = ({
|
|||
};
|
||||
|
||||
const popoverButton = (handleClick: MouseEventHandler<HTMLButtonElement>) => (
|
||||
<EuiToolTip position="bottom" content={strings.getTooltip()}>
|
||||
<EuiButtonIcon iconType="gear" aria-label={strings.getTooltip()} onClick={handleClick} />
|
||||
</EuiToolTip>
|
||||
<EuiButtonEmpty size="xs" aria-label={strings.getButtonLabel()} onClick={handleClick}>
|
||||
{strings.getButtonLabel()}
|
||||
</EuiButtonEmpty>
|
||||
);
|
||||
|
||||
return (
|
||||
<Popover
|
||||
id="auto-refresh-popover"
|
||||
button={popoverButton}
|
||||
anchorPosition="rightUp"
|
||||
anchorPosition="downLeft"
|
||||
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);
|