mirror of
https://github.com/elastic/kibana.git
synced 2025-04-24 17:59:23 -04:00
* expression_reveal_image skeleton. * expression_functions added. * expression_renderers added. * Backup of daily work. * Fixed errors. * Added legacy support. Added button for legacy. * Added storybook. * Removed revealImage from canvas. * setState while rendering error fixed. * tsconfig.json added. * jest.config.js added. * Demo doc added. * Types fixed. * added limits. * Removed not used imports. * i18n namespaces fixed. * Fixed test suite error. * Some errors fixed. * Fixed eslint error. * Removed more unused translations. * Moved UI and elements, related to expressionRevealImage from canvas. * Fixed unused translations errors. * Moved type of element to types. * Fixed types and added service for representing elements, ui and supported renderers to canvas. * Added expression registration to canvas. * Fixed * Fixed mutiple call of the function. * Removed support of a legacy lib for revealImage chart. * Removed legacy presentation_utils plugin import. * Doc error fixed. * Removed useless translations and tried to fix error. * One more fix. * Small imports fix. * Fixed translations. * Made fixes based on nits. * Removed useless params. * fix. * Fixed errors, related to jest and __mocks__. * Removed useless type definition. * Replaced RendererHandlers with IInterpreterRendererHandlers. * fixed supported_shareable. * Moved elements back to canvas. * Moved views to canvas, removed expression service and imported renderer to canvas. * Fixed translations. * Types fix. * Moved libs to presentation utils. * Fixed one mistake. * removed dataurl lib. * Fixed jest files. * elasticLogo removed. * Removed elastic_outline. * removed httpurl. * Removed missing_asset. * removed url. * replaced mostly all tests. * Fixed types. * Fixed types and removed function_wrapper.ts * Fixed types of test helpers. * Changed limits of presentationUtil plugin. * Fixed imports. * One more fix. * Fixed huge size of bundle. * Reduced allow limit for presentationUtil * Updated limits for presentationUtil. * Fixed public API. * fixed type errors. * Moved css to component. * Fixed spaces at element. * Changed order of requiredPlugins. * Updated limits. * Removed unused plugin. * Added rule for allowing import from __stories__ directory. * removed useless comment. * Changed readme.md * Fixed docs error. * A possible of smoke test. * onResize changed to useResizeObserver. * Remove useless events and `useEffect` block. * Changed from passing handlers to separate functions. * `function` moved to `server`. * Fixed eslint error. Co-authored-by: Kibana Machine <42973632+kibanamachine@users.noreply.github.com> Co-authored-by: Yaroslav Kuznietsov <kuznetsov.yaroslav.yk@gmail.com>
This commit is contained in:
parent
92991327a4
commit
bdfb417a54
159 changed files with 1311 additions and 512 deletions
|
@ -451,6 +451,7 @@ module.exports = {
|
|||
'(src|x-pack)/plugins/**/(public|server)/**/*',
|
||||
'!(src|x-pack)/plugins/**/(public|server)/mocks/index.{js,mjs,ts}',
|
||||
'!(src|x-pack)/plugins/**/(public|server)/(index|mocks).{js,mjs,ts,tsx}',
|
||||
'!(src|x-pack)/plugins/**/__stories__/index.{js,mjs,ts,tsx}',
|
||||
],
|
||||
allowSameFolder: true,
|
||||
errorMessage: 'Plugins may only import from top-level public and server modules.',
|
||||
|
|
|
@ -16,6 +16,7 @@
|
|||
"esUi": "src/plugins/es_ui_shared",
|
||||
"devTools": "src/plugins/dev_tools",
|
||||
"expressions": "src/plugins/expressions",
|
||||
"expressionRevealImage": "src/plugins/expression_reveal_image",
|
||||
"inputControl": "src/plugins/input_control_vis",
|
||||
"inspector": "src/plugins/inspector",
|
||||
"inspectorViews": "src/legacy/core_plugins/inspector_views",
|
||||
|
|
|
@ -70,6 +70,10 @@ This API doesn't support angular, for registering angular dev tools, bootstrap a
|
|||
|WARNING: Missing README.
|
||||
|
||||
|
||||
|{kib-repo}blob/{branch}/src/plugins/expression_reveal_image/README.md[expressionRevealImage]
|
||||
|Expression Reveal Image plugin adds a revealImage function to the expression plugin and an associated renderer. The renderer will display the given percentage of a given image.
|
||||
|
||||
|
||||
|<<kibana-expressions-plugin>>
|
||||
|Expression pipeline is a chain of functions that *pipe* its output to the
|
||||
input of the next function. Functions can be configured using arguments provided
|
||||
|
|
|
@ -99,7 +99,7 @@ pageLoadAssetSize:
|
|||
watcher: 43742
|
||||
runtimeFields: 41752
|
||||
stackAlerts: 29684
|
||||
presentationUtil: 49767
|
||||
presentationUtil: 94301
|
||||
spacesOss: 18817
|
||||
indexPatternFieldEditor: 90489
|
||||
osquery: 107090
|
||||
|
@ -110,4 +110,5 @@ pageLoadAssetSize:
|
|||
timelines: 230410
|
||||
screenshotMode: 17856
|
||||
visTypePie: 35583
|
||||
expressionRevealImage: 25675
|
||||
cases: 144442
|
||||
|
|
|
@ -17,6 +17,7 @@ export const storybookAliases = {
|
|||
dashboard_enhanced: 'x-pack/plugins/dashboard_enhanced/.storybook',
|
||||
data_enhanced: 'x-pack/plugins/data_enhanced/.storybook',
|
||||
embeddable: 'src/plugins/embeddable/.storybook',
|
||||
expression_reveal_image: 'src/plugins/expression_reveal_image/.storybook',
|
||||
infra: 'x-pack/plugins/infra/.storybook',
|
||||
security_solution: 'x-pack/plugins/security_solution/.storybook',
|
||||
ui_actions_enhanced: 'x-pack/plugins/ui_actions_enhanced/.storybook',
|
||||
|
|
7
src/plugins/expression_reveal_image/.i18nrc.json
Executable file
7
src/plugins/expression_reveal_image/.i18nrc.json
Executable file
|
@ -0,0 +1,7 @@
|
|||
{
|
||||
"prefix": "expressionRevealImage",
|
||||
"paths": {
|
||||
"expressionRevealImage": "."
|
||||
},
|
||||
"translations": ["translations/ja-JP.json"]
|
||||
}
|
10
src/plugins/expression_reveal_image/.storybook/main.js
Normal file
10
src/plugins/expression_reveal_image/.storybook/main.js
Normal file
|
@ -0,0 +1,10 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License
|
||||
* 2.0 and the Server Side Public License, v 1; you may not use this file except
|
||||
* in compliance with, at your election, the Elastic License 2.0 or the Server
|
||||
* Side Public License, v 1.
|
||||
*/
|
||||
|
||||
// eslint-disable-next-line import/no-commonjs
|
||||
module.exports = require('@kbn/storybook').defaultConfig;
|
9
src/plugins/expression_reveal_image/README.md
Executable file
9
src/plugins/expression_reveal_image/README.md
Executable file
|
@ -0,0 +1,9 @@
|
|||
# expressionRevealImage
|
||||
|
||||
Expression Reveal Image plugin adds a `revealImage` function to the expression plugin and an associated renderer. The renderer will display the given percentage of a given image.
|
||||
|
||||
---
|
||||
|
||||
## Development
|
||||
|
||||
See the [kibana contributing guide](https://github.com/elastic/kibana/blob/master/CONTRIBUTING.md) for instructions setting up your development environment.
|
9
src/plugins/expression_reveal_image/common/constants.ts
Normal file
9
src/plugins/expression_reveal_image/common/constants.ts
Normal file
|
@ -0,0 +1,9 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License
|
||||
* 2.0 and the Server Side Public License, v 1; you may not use this file except
|
||||
* in compliance with, at your election, the Elastic License 2.0 or the Server
|
||||
* Side Public License, v 1.
|
||||
*/
|
||||
export const PLUGIN_ID = 'expressionRevealImage';
|
||||
export const PLUGIN_NAME = 'expressionRevealImage';
|
|
@ -0,0 +1,13 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License
|
||||
* 2.0 and the Server Side Public License, v 1; you may not use this file except
|
||||
* in compliance with, at your election, the Elastic License 2.0 or the Server
|
||||
* Side Public License, v 1.
|
||||
*/
|
||||
|
||||
import { revealImageFunction } from './reveal_image_function';
|
||||
|
||||
export const functions = [revealImageFunction];
|
||||
|
||||
export { revealImageFunction };
|
|
@ -0,0 +1,168 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License
|
||||
* 2.0 and the Server Side Public License, v 1; you may not use this file except
|
||||
* in compliance with, at your election, the Elastic License 2.0 or the Server
|
||||
* Side Public License, v 1.
|
||||
*/
|
||||
|
||||
import {
|
||||
functionWrapper,
|
||||
elasticOutline,
|
||||
elasticLogo,
|
||||
} from '../../../presentation_util/common/lib';
|
||||
import { getFunctionErrors } from '../i18n';
|
||||
import { revealImageFunction } from './reveal_image_function';
|
||||
import { Origin } from '../types';
|
||||
import { ExecutionContext } from 'src/plugins/expressions';
|
||||
|
||||
const errors = getFunctionErrors().revealImage;
|
||||
|
||||
describe('revealImageFunction', () => {
|
||||
const fn = functionWrapper(revealImageFunction);
|
||||
|
||||
it('returns a render as revealImage', () => {
|
||||
const result = fn(
|
||||
0.5,
|
||||
{
|
||||
image: null,
|
||||
emptyImage: null,
|
||||
origin: Origin.BOTTOM,
|
||||
},
|
||||
{} as ExecutionContext
|
||||
);
|
||||
expect(result).toHaveProperty('type', 'render');
|
||||
expect(result).toHaveProperty('as', 'revealImage');
|
||||
});
|
||||
|
||||
describe('context', () => {
|
||||
it('throws when context is not a number between 0 and 1', () => {
|
||||
expect(() => {
|
||||
fn(
|
||||
10,
|
||||
{
|
||||
image: elasticLogo,
|
||||
emptyImage: elasticOutline,
|
||||
origin: Origin.TOP,
|
||||
},
|
||||
{} as ExecutionContext
|
||||
);
|
||||
}).toThrow(new RegExp(errors.invalidPercent(10).message));
|
||||
|
||||
expect(() => {
|
||||
fn(
|
||||
-0.1,
|
||||
{
|
||||
image: elasticLogo,
|
||||
emptyImage: elasticOutline,
|
||||
origin: Origin.TOP,
|
||||
},
|
||||
{} as ExecutionContext
|
||||
);
|
||||
}).toThrow(new RegExp(errors.invalidPercent(-0.1).message));
|
||||
});
|
||||
});
|
||||
|
||||
describe('args', () => {
|
||||
describe('image', () => {
|
||||
it('sets the image', () => {
|
||||
const result = fn(
|
||||
0.89,
|
||||
{
|
||||
emptyImage: null,
|
||||
origin: Origin.TOP,
|
||||
image: elasticLogo,
|
||||
},
|
||||
{} as ExecutionContext
|
||||
).value;
|
||||
expect(result).toHaveProperty('image', elasticLogo);
|
||||
});
|
||||
|
||||
it('defaults to the Elastic outline logo', () => {
|
||||
const result = fn(
|
||||
0.89,
|
||||
{
|
||||
emptyImage: null,
|
||||
origin: Origin.TOP,
|
||||
image: null,
|
||||
},
|
||||
{} as ExecutionContext
|
||||
).value;
|
||||
expect(result).toHaveProperty('image', elasticOutline);
|
||||
});
|
||||
});
|
||||
|
||||
describe('emptyImage', () => {
|
||||
it('sets the background image', () => {
|
||||
const result = fn(
|
||||
0,
|
||||
{
|
||||
emptyImage: elasticLogo,
|
||||
origin: Origin.TOP,
|
||||
image: null,
|
||||
},
|
||||
{} as ExecutionContext
|
||||
).value;
|
||||
expect(result).toHaveProperty('emptyImage', elasticLogo);
|
||||
});
|
||||
|
||||
it('sets emptyImage to null', () => {
|
||||
const result = fn(
|
||||
0,
|
||||
{
|
||||
emptyImage: null,
|
||||
origin: Origin.TOP,
|
||||
image: null,
|
||||
},
|
||||
{} as ExecutionContext
|
||||
).value;
|
||||
expect(result).toHaveProperty('emptyImage', null);
|
||||
});
|
||||
});
|
||||
|
||||
describe('origin', () => {
|
||||
it('sets which side to start the reveal from', () => {
|
||||
let result = fn(
|
||||
1,
|
||||
{
|
||||
emptyImage: null,
|
||||
origin: Origin.TOP,
|
||||
image: null,
|
||||
},
|
||||
{} as ExecutionContext
|
||||
).value;
|
||||
expect(result).toHaveProperty('origin', 'top');
|
||||
result = fn(
|
||||
1,
|
||||
{
|
||||
emptyImage: null,
|
||||
origin: Origin.LEFT,
|
||||
image: null,
|
||||
},
|
||||
{} as ExecutionContext
|
||||
).value;
|
||||
expect(result).toHaveProperty('origin', 'left');
|
||||
result = fn(
|
||||
1,
|
||||
{
|
||||
emptyImage: null,
|
||||
origin: Origin.BOTTOM,
|
||||
image: null,
|
||||
},
|
||||
{} as ExecutionContext
|
||||
).value;
|
||||
expect(result).toHaveProperty('origin', 'bottom');
|
||||
result = fn(
|
||||
1,
|
||||
{
|
||||
emptyImage: null,
|
||||
origin: Origin.RIGHT,
|
||||
image: null,
|
||||
},
|
||||
{} as ExecutionContext
|
||||
).value;
|
||||
expect(result).toHaveProperty('origin', 'right');
|
||||
});
|
||||
});
|
||||
});
|
||||
});
|
|
@ -1,41 +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
|
||||
* 2.0; you may not use this file except in compliance with the Elastic License
|
||||
* 2.0.
|
||||
* 2.0 and the Server Side Public License, v 1; you may not use this file except
|
||||
* in compliance with, at your election, the Elastic License 2.0 or the Server
|
||||
* Side Public License, v 1.
|
||||
*/
|
||||
|
||||
import { ExpressionFunctionDefinition, ExpressionValueRender } from 'src/plugins/expressions';
|
||||
import { resolveWithMissingImage } from '../../../common/lib/resolve_dataurl';
|
||||
import { elasticOutline } from '../../lib/elastic_outline';
|
||||
import { getFunctionHelp, getFunctionErrors } from '../../../i18n';
|
||||
import { resolveWithMissingImage, elasticOutline } from '../../../presentation_util/common/lib';
|
||||
import { getFunctionHelp, getFunctionErrors } from '../i18n';
|
||||
import { ExpressionRevealImageFunction, Origin } from '../types';
|
||||
|
||||
export enum Origin {
|
||||
TOP = 'top',
|
||||
LEFT = 'left',
|
||||
BOTTOM = 'bottom',
|
||||
RIGHT = 'right',
|
||||
}
|
||||
|
||||
interface Arguments {
|
||||
image: string | null;
|
||||
emptyImage: string | null;
|
||||
origin: Origin;
|
||||
}
|
||||
|
||||
export interface Output {
|
||||
image: string;
|
||||
emptyImage: string;
|
||||
origin: Origin;
|
||||
percent: number;
|
||||
}
|
||||
|
||||
export function revealImage(): ExpressionFunctionDefinition<
|
||||
'revealImage',
|
||||
number,
|
||||
Arguments,
|
||||
ExpressionValueRender<Output>
|
||||
> {
|
||||
export const revealImageFunction: ExpressionRevealImageFunction = () => {
|
||||
const { help, args: argHelp } = getFunctionHelp().revealImage;
|
||||
const errors = getFunctionErrors().revealImage;
|
||||
|
||||
|
@ -80,4 +55,4 @@ export function revealImage(): ExpressionFunctionDefinition<
|
|||
};
|
||||
},
|
||||
};
|
||||
}
|
||||
};
|
10
src/plugins/expression_reveal_image/common/i18n/constants.ts
Normal file
10
src/plugins/expression_reveal_image/common/i18n/constants.ts
Normal file
|
@ -0,0 +1,10 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License
|
||||
* 2.0 and the Server Side Public License, v 1; you may not use this file except
|
||||
* in compliance with, at your election, the Elastic License 2.0 or the Server
|
||||
* Side Public License, v 1.
|
||||
*/
|
||||
|
||||
export const BASE64 = '`base64`';
|
||||
export const URL = 'URL';
|
|
@ -1,23 +1,21 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License
|
||||
* 2.0; you may not use this file except in compliance with the Elastic License
|
||||
* 2.0.
|
||||
* 2.0 and the Server Side Public License, v 1; you may not use this file except
|
||||
* in compliance with, at your election, the Elastic License 2.0 or the Server
|
||||
* Side Public License, v 1.
|
||||
*/
|
||||
|
||||
import { i18n } from '@kbn/i18n';
|
||||
import { revealImage } from '../../../canvas_plugin_src/functions/common/revealImage';
|
||||
import { FunctionHelp } from '../function_help';
|
||||
import { FunctionFactory } from '../../../types';
|
||||
import { Position } from '../../../types';
|
||||
import { BASE64, URL } from '../../constants';
|
||||
|
||||
export const help: FunctionHelp<FunctionFactory<typeof revealImage>> = {
|
||||
help: i18n.translate('xpack.canvas.functions.revealImageHelpText', {
|
||||
export const help = {
|
||||
help: i18n.translate('expressionRevealImage.functions.revealImageHelpText', {
|
||||
defaultMessage: 'Configures an image reveal element.',
|
||||
}),
|
||||
args: {
|
||||
image: i18n.translate('xpack.canvas.functions.revealImage.args.imageHelpText', {
|
||||
image: i18n.translate('expressionRevealImage.functions.revealImage.args.imageHelpText', {
|
||||
defaultMessage:
|
||||
'The image to reveal. Provide an image asset as a {BASE64} data {URL}, ' +
|
||||
'or pass in a sub-expression.',
|
||||
|
@ -26,16 +24,19 @@ export const help: FunctionHelp<FunctionFactory<typeof revealImage>> = {
|
|||
URL,
|
||||
},
|
||||
}),
|
||||
emptyImage: i18n.translate('xpack.canvas.functions.revealImage.args.emptyImageHelpText', {
|
||||
defaultMessage:
|
||||
'An optional background image to reveal over. ' +
|
||||
'Provide an image asset as a `{BASE64}` data {URL}, or pass in a sub-expression.',
|
||||
values: {
|
||||
BASE64,
|
||||
URL,
|
||||
},
|
||||
}),
|
||||
origin: i18n.translate('xpack.canvas.functions.revealImage.args.originHelpText', {
|
||||
emptyImage: i18n.translate(
|
||||
'expressionRevealImage.functions.revealImage.args.emptyImageHelpText',
|
||||
{
|
||||
defaultMessage:
|
||||
'An optional background image to reveal over. ' +
|
||||
'Provide an image asset as a `{BASE64}` data {URL}, or pass in a sub-expression.',
|
||||
values: {
|
||||
BASE64,
|
||||
URL,
|
||||
},
|
||||
}
|
||||
),
|
||||
origin: i18n.translate('expressionRevealImage.functions.revealImage.args.originHelpText', {
|
||||
defaultMessage: 'The position to start the image fill. For example, {list}, or {end}.',
|
||||
values: {
|
||||
list: Object.values(Position)
|
||||
|
@ -50,7 +51,7 @@ export const help: FunctionHelp<FunctionFactory<typeof revealImage>> = {
|
|||
export const errors = {
|
||||
invalidPercent: (percent: number) =>
|
||||
new Error(
|
||||
i18n.translate('xpack.canvas.functions.revealImage.invalidPercentErrorMessage', {
|
||||
i18n.translate('expressionRevealImage.functions.revealImage.invalidPercentErrorMessage', {
|
||||
defaultMessage: "Invalid value: '{percent}'. Percentage must be between 0 and 1",
|
||||
values: {
|
||||
percent,
|
|
@ -0,0 +1,13 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License
|
||||
* 2.0 and the Server Side Public License, v 1; you may not use this file except
|
||||
* in compliance with, at your election, the Elastic License 2.0 or the Server
|
||||
* Side Public License, v 1.
|
||||
*/
|
||||
|
||||
import { errors as revealImage } from './dict/reveal_image';
|
||||
|
||||
export const getFunctionErrors = () => ({
|
||||
revealImage,
|
||||
});
|
|
@ -0,0 +1,21 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License
|
||||
* 2.0 and the Server Side Public License, v 1; you may not use this file except
|
||||
* in compliance with, at your election, the Elastic License 2.0 or the Server
|
||||
* Side Public License, v 1.
|
||||
*/
|
||||
|
||||
import { help as revealImage } from './dict/reveal_image';
|
||||
|
||||
/**
|
||||
* Help text for Canvas Functions should be properly localized. This function will
|
||||
* return a dictionary of help strings, organized by `ExpressionFunctionDefinition`
|
||||
* specification and then by available arguments within each `ExpressionFunctionDefinition`.
|
||||
*
|
||||
* This a function, rather than an object, to future-proof string initialization,
|
||||
* if ever necessary.
|
||||
*/
|
||||
export const getFunctionHelp = () => ({
|
||||
revealImage,
|
||||
});
|
|
@ -0,0 +1,10 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License
|
||||
* 2.0 and the Server Side Public License, v 1; you may not use this file except
|
||||
* in compliance with, at your election, the Elastic License 2.0 or the Server
|
||||
* Side Public License, v 1.
|
||||
*/
|
||||
|
||||
export * from './function_help';
|
||||
export * from './function_errors';
|
|
@ -0,0 +1,9 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License
|
||||
* 2.0 and the Server Side Public License, v 1; you may not use this file except
|
||||
* in compliance with, at your election, the Elastic License 2.0 or the Server
|
||||
* Side Public License, v 1.
|
||||
*/
|
||||
|
||||
export { strings as revealImage } from './reveal_image';
|
|
@ -0,0 +1,19 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License
|
||||
* 2.0 and the Server Side Public License, v 1; you may not use this file except
|
||||
* in compliance with, at your election, the Elastic License 2.0 or the Server
|
||||
* Side Public License, v 1.
|
||||
*/
|
||||
import { i18n } from '@kbn/i18n';
|
||||
|
||||
export const strings = {
|
||||
getDisplayName: () =>
|
||||
i18n.translate('expressionRevealImage.renderer.revealImage.displayName', {
|
||||
defaultMessage: 'Image reveal',
|
||||
}),
|
||||
getHelpDescription: () =>
|
||||
i18n.translate('expressionRevealImage.renderer.revealImage.helpDescription', {
|
||||
defaultMessage: 'Reveal a percentage of an image to make a custom gauge-style chart',
|
||||
}),
|
||||
};
|
|
@ -0,0 +1,9 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License
|
||||
* 2.0 and the Server Side Public License, v 1; you may not use this file except
|
||||
* in compliance with, at your election, the Elastic License 2.0 or the Server
|
||||
* Side Public License, v 1.
|
||||
*/
|
||||
|
||||
export * from './renderer_strings';
|
|
@ -0,0 +1,21 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License
|
||||
* 2.0 and the Server Side Public License, v 1; you may not use this file except
|
||||
* in compliance with, at your election, the Elastic License 2.0 or the Server
|
||||
* Side Public License, v 1.
|
||||
*/
|
||||
|
||||
import { revealImage } from './dict';
|
||||
|
||||
/**
|
||||
* Help text for Canvas Functions should be properly localized. This function will
|
||||
* return a dictionary of help strings, organized by `ExpressionFunctionDefinition`
|
||||
* specification and then by available arguments within each `ExpressionFunctionDefinition`.
|
||||
*
|
||||
* This a function, rather than an object, to future-proof string initialization,
|
||||
* if ever necessary.
|
||||
*/
|
||||
export const getRendererStrings = () => ({
|
||||
revealImage,
|
||||
});
|
10
src/plugins/expression_reveal_image/common/i18n/index.ts
Normal file
10
src/plugins/expression_reveal_image/common/i18n/index.ts
Normal file
|
@ -0,0 +1,10 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License
|
||||
* 2.0 and the Server Side Public License, v 1; you may not use this file except
|
||||
* in compliance with, at your election, the Elastic License 2.0 or the Server
|
||||
* Side Public License, v 1.
|
||||
*/
|
||||
|
||||
export * from './expression_functions';
|
||||
export * from './expression_renderers';
|
10
src/plugins/expression_reveal_image/common/index.ts
Executable file
10
src/plugins/expression_reveal_image/common/index.ts
Executable file
|
@ -0,0 +1,10 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License
|
||||
* 2.0 and the Server Side Public License, v 1; you may not use this file except
|
||||
* in compliance with, at your election, the Elastic License 2.0 or the Server
|
||||
* Side Public License, v 1.
|
||||
*/
|
||||
|
||||
export * from './constants';
|
||||
export * from './expression_functions';
|
|
@ -0,0 +1,42 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License
|
||||
* 2.0 and the Server Side Public License, v 1; you may not use this file except
|
||||
* in compliance with, at your election, the Elastic License 2.0 or the Server
|
||||
* Side Public License, v 1.
|
||||
*/
|
||||
import { ExpressionFunctionDefinition, ExpressionValueRender } from 'src/plugins/expressions';
|
||||
|
||||
export enum Origin {
|
||||
TOP = 'top',
|
||||
LEFT = 'left',
|
||||
BOTTOM = 'bottom',
|
||||
RIGHT = 'right',
|
||||
}
|
||||
|
||||
interface Arguments {
|
||||
image: string | null;
|
||||
emptyImage: string | null;
|
||||
origin: Origin;
|
||||
}
|
||||
|
||||
export interface Output {
|
||||
image: string;
|
||||
emptyImage: string;
|
||||
origin: Origin;
|
||||
percent: number;
|
||||
}
|
||||
|
||||
export type ExpressionRevealImageFunction = () => ExpressionFunctionDefinition<
|
||||
'revealImage',
|
||||
number,
|
||||
Arguments,
|
||||
ExpressionValueRender<Output>
|
||||
>;
|
||||
|
||||
export enum Position {
|
||||
TOP = 'top',
|
||||
BOTTOM = 'bottom',
|
||||
LEFT = 'left',
|
||||
RIGHT = 'right',
|
||||
}
|
|
@ -0,0 +1,21 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License
|
||||
* 2.0 and the Server Side Public License, v 1; you may not use this file except
|
||||
* in compliance with, at your election, the Elastic License 2.0 or the Server
|
||||
* Side Public License, v 1.
|
||||
*/
|
||||
|
||||
export type OriginString = 'bottom' | 'left' | 'top' | 'right';
|
||||
|
||||
export interface RevealImageRendererConfig {
|
||||
percent: number;
|
||||
origin?: OriginString;
|
||||
image?: string;
|
||||
emptyImage?: string;
|
||||
}
|
||||
|
||||
export interface NodeDimensions {
|
||||
width: number;
|
||||
height: number;
|
||||
}
|
|
@ -0,0 +1,9 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License
|
||||
* 2.0 and the Server Side Public License, v 1; you may not use this file except
|
||||
* in compliance with, at your election, the Elastic License 2.0 or the Server
|
||||
* Side Public License, v 1.
|
||||
*/
|
||||
export * from './expression_functions';
|
||||
export * from './expression_renderers';
|
13
src/plugins/expression_reveal_image/jest.config.js
Normal file
13
src/plugins/expression_reveal_image/jest.config.js
Normal file
|
@ -0,0 +1,13 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License
|
||||
* 2.0 and the Server Side Public License, v 1; you may not use this file except
|
||||
* in compliance with, at your election, the Elastic License 2.0 or the Server
|
||||
* Side Public License, v 1.
|
||||
*/
|
||||
|
||||
module.exports = {
|
||||
preset: '@kbn/test',
|
||||
rootDir: '../../..',
|
||||
roots: ['<rootDir>/src/plugins/expression_reveal_image'],
|
||||
};
|
10
src/plugins/expression_reveal_image/kibana.json
Executable file
10
src/plugins/expression_reveal_image/kibana.json
Executable file
|
@ -0,0 +1,10 @@
|
|||
{
|
||||
"id": "expressionRevealImage",
|
||||
"version": "1.0.0",
|
||||
"kibanaVersion": "kibana",
|
||||
"server": true,
|
||||
"ui": true,
|
||||
"requiredPlugins": ["expressions", "presentationUtil"],
|
||||
"optionalPlugins": [],
|
||||
"requiredBundles": []
|
||||
}
|
|
@ -0,0 +1,9 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License
|
||||
* 2.0 and the Server Side Public License, v 1; you may not use this file except
|
||||
* in compliance with, at your election, the Elastic License 2.0 or the Server
|
||||
* Side Public License, v 1.
|
||||
*/
|
||||
|
||||
export * from './reveal_image_component';
|
|
@ -0,0 +1,136 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License
|
||||
* 2.0 and the Server Side Public License, v 1; you may not use this file except
|
||||
* in compliance with, at your election, the Elastic License 2.0 or the Server
|
||||
* Side Public License, v 1.
|
||||
*/
|
||||
|
||||
import React, { useRef, useState, useEffect, useCallback } from 'react';
|
||||
import { useResizeObserver } from '@elastic/eui';
|
||||
import { IInterpreterRenderHandlers } from 'src/plugins/expressions';
|
||||
import { NodeDimensions, RevealImageRendererConfig, OriginString } from '../../common/types';
|
||||
import { isValidUrl, elasticOutline } from '../../../presentation_util/public';
|
||||
import './reveal_image.scss';
|
||||
|
||||
interface RevealImageComponentProps extends RevealImageRendererConfig {
|
||||
onLoaded: IInterpreterRenderHandlers['done'];
|
||||
parentNode: HTMLElement;
|
||||
}
|
||||
|
||||
interface ImageStyles {
|
||||
width?: string;
|
||||
height?: string;
|
||||
clipPath?: string;
|
||||
}
|
||||
|
||||
interface AlignerStyles {
|
||||
backgroundImage?: string;
|
||||
}
|
||||
|
||||
function RevealImageComponent({
|
||||
onLoaded,
|
||||
parentNode,
|
||||
percent,
|
||||
origin,
|
||||
image,
|
||||
emptyImage,
|
||||
}: RevealImageComponentProps) {
|
||||
const [loaded, setLoaded] = useState(false);
|
||||
const [dimensions, setDimensions] = useState<NodeDimensions>({
|
||||
width: 1,
|
||||
height: 1,
|
||||
});
|
||||
|
||||
const imgRef = useRef<HTMLImageElement>(null);
|
||||
|
||||
const parentNodeDimensions = useResizeObserver(parentNode);
|
||||
|
||||
// modify the top-level container class
|
||||
parentNode.className = 'revealImage';
|
||||
|
||||
// set up the overlay image
|
||||
const updateImageView = useCallback(() => {
|
||||
if (imgRef.current) {
|
||||
setDimensions({
|
||||
height: imgRef.current.naturalHeight,
|
||||
width: imgRef.current.naturalWidth,
|
||||
});
|
||||
|
||||
setLoaded(true);
|
||||
onLoaded();
|
||||
}
|
||||
}, [imgRef, onLoaded]);
|
||||
|
||||
useEffect(() => {
|
||||
updateImageView();
|
||||
}, [parentNodeDimensions, updateImageView]);
|
||||
|
||||
function getClipPath(percentParam: number, originParam: OriginString = 'bottom') {
|
||||
const directions: Record<OriginString, number> = { bottom: 0, left: 1, top: 2, right: 3 };
|
||||
const values: Array<number | string> = [0, 0, 0, 0];
|
||||
values[directions[originParam]] = `${100 - percentParam * 100}%`;
|
||||
return `inset(${values.join(' ')})`;
|
||||
}
|
||||
|
||||
function getImageSizeStyle() {
|
||||
const imgStyles: ImageStyles = {};
|
||||
|
||||
const imgDimensions = {
|
||||
height: dimensions.height,
|
||||
width: dimensions.width,
|
||||
ratio: dimensions.height / dimensions.width,
|
||||
};
|
||||
|
||||
const domNodeDimensions = {
|
||||
width: parentNode.clientWidth,
|
||||
height: parentNode.clientHeight,
|
||||
ratio: parentNode.clientHeight / parentNode.clientWidth,
|
||||
};
|
||||
|
||||
if (imgDimensions.ratio > domNodeDimensions.ratio) {
|
||||
imgStyles.height = `${domNodeDimensions.height}px`;
|
||||
imgStyles.width = 'initial';
|
||||
} else {
|
||||
imgStyles.width = `${domNodeDimensions.width}px`;
|
||||
imgStyles.height = 'initial';
|
||||
}
|
||||
|
||||
return imgStyles;
|
||||
}
|
||||
|
||||
const imgSrc = isValidUrl(image ?? '') ? image : elasticOutline;
|
||||
|
||||
const alignerStyles: AlignerStyles = {};
|
||||
|
||||
if (isValidUrl(emptyImage ?? '')) {
|
||||
// only use empty image if one is provided
|
||||
alignerStyles.backgroundImage = `url(${emptyImage})`;
|
||||
}
|
||||
|
||||
let imgStyles: ImageStyles = {};
|
||||
if (imgRef.current && loaded) imgStyles = getImageSizeStyle();
|
||||
|
||||
imgStyles.clipPath = getClipPath(percent, origin);
|
||||
if (imgRef.current && loaded) {
|
||||
imgRef.current.style.setProperty('-webkit-clip-path', getClipPath(percent, origin));
|
||||
}
|
||||
|
||||
return (
|
||||
<div className="revealImageAligner" style={alignerStyles}>
|
||||
<img
|
||||
ref={imgRef}
|
||||
onLoad={updateImageView}
|
||||
className="revealImage__image"
|
||||
src={imgSrc ?? ''}
|
||||
alt=""
|
||||
role="presentation"
|
||||
style={imgStyles}
|
||||
/>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
||||
// default export required for React.Lazy
|
||||
// eslint-disable-next-line import/no-default-export
|
||||
export { RevealImageComponent as default };
|
|
@ -0,0 +1,26 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License
|
||||
* 2.0 and the Server Side Public License, v 1; you may not use this file except
|
||||
* in compliance with, at your election, the Elastic License 2.0 or the Server
|
||||
* Side Public License, v 1.
|
||||
*/
|
||||
|
||||
import React from 'react';
|
||||
import { storiesOf } from '@storybook/react';
|
||||
import { revealImageRenderer } from '../';
|
||||
import { elasticOutline, elasticLogo } from '../../../../presentation_util/public';
|
||||
import { Render } from '../../../../presentation_util/public/__stories__';
|
||||
|
||||
import { Origin } from '../../../common/types/expression_functions';
|
||||
|
||||
storiesOf('renderers/revealImage', module).add('default', () => {
|
||||
const config = {
|
||||
image: elasticLogo,
|
||||
emptyImage: elasticOutline,
|
||||
origin: Origin.LEFT,
|
||||
percent: 0.45,
|
||||
};
|
||||
|
||||
return <Render renderer={revealImageRenderer} config={config} />;
|
||||
});
|
|
@ -0,0 +1,13 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License
|
||||
* 2.0 and the Server Side Public License, v 1; you may not use this file except
|
||||
* in compliance with, at your election, the Elastic License 2.0 or the Server
|
||||
* Side Public License, v 1.
|
||||
*/
|
||||
|
||||
import { revealImageRenderer } from './reveal_image_renderer';
|
||||
|
||||
export const renderers = [revealImageRenderer];
|
||||
|
||||
export { revealImageRenderer };
|
|
@ -0,0 +1,42 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License
|
||||
* 2.0 and the Server Side Public License, v 1; you may not use this file except
|
||||
* in compliance with, at your election, the Elastic License 2.0 or the Server
|
||||
* Side Public License, v 1.
|
||||
*/
|
||||
import React, { lazy } from 'react';
|
||||
import { render, unmountComponentAtNode } from 'react-dom';
|
||||
import { I18nProvider } from '@kbn/i18n/react';
|
||||
import { ExpressionRenderDefinition, IInterpreterRenderHandlers } from 'src/plugins/expressions';
|
||||
import { withSuspense } from '../../../presentation_util/public';
|
||||
import { getRendererStrings } from '../../common/i18n';
|
||||
import { RevealImageRendererConfig } from '../../common/types';
|
||||
|
||||
const { revealImage: revealImageStrings } = getRendererStrings();
|
||||
|
||||
const LazyRevealImageComponent = lazy(() => import('../components/reveal_image_component'));
|
||||
const RevealImageComponent = withSuspense(LazyRevealImageComponent, null);
|
||||
|
||||
export const revealImageRenderer = (): ExpressionRenderDefinition<RevealImageRendererConfig> => ({
|
||||
name: 'revealImage',
|
||||
displayName: revealImageStrings.getDisplayName(),
|
||||
help: revealImageStrings.getHelpDescription(),
|
||||
reuseDomNode: true,
|
||||
render: async (
|
||||
domNode: HTMLElement,
|
||||
config: RevealImageRendererConfig,
|
||||
handlers: IInterpreterRenderHandlers
|
||||
) => {
|
||||
handlers.onDestroy(() => {
|
||||
unmountComponentAtNode(domNode);
|
||||
});
|
||||
|
||||
render(
|
||||
<I18nProvider>
|
||||
<RevealImageComponent {...config} parentNode={domNode} onLoaded={handlers.done} />
|
||||
</I18nProvider>,
|
||||
domNode
|
||||
);
|
||||
},
|
||||
});
|
17
src/plugins/expression_reveal_image/public/index.ts
Executable file
17
src/plugins/expression_reveal_image/public/index.ts
Executable file
|
@ -0,0 +1,17 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License
|
||||
* 2.0 and the Server Side Public License, v 1; you may not use this file except
|
||||
* in compliance with, at your election, the Elastic License 2.0 or the Server
|
||||
* Side Public License, v 1.
|
||||
*/
|
||||
|
||||
import { ExpressionRevealImagePlugin } from './plugin';
|
||||
|
||||
export type { ExpressionRevealImagePluginSetup, ExpressionRevealImagePluginStart } from './plugin';
|
||||
|
||||
export function plugin() {
|
||||
return new ExpressionRevealImagePlugin();
|
||||
}
|
||||
|
||||
export * from './expression_renderers';
|
39
src/plugins/expression_reveal_image/public/plugin.ts
Executable file
39
src/plugins/expression_reveal_image/public/plugin.ts
Executable file
|
@ -0,0 +1,39 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License
|
||||
* 2.0 and the Server Side Public License, v 1; you may not use this file except
|
||||
* in compliance with, at your election, the Elastic License 2.0 or the Server
|
||||
* Side Public License, v 1.
|
||||
*/
|
||||
|
||||
import { CoreSetup, CoreStart, Plugin } from '../../../core/public';
|
||||
import { ExpressionsStart, ExpressionsSetup } from '../../expressions/public';
|
||||
import { revealImageRenderer } from './expression_renderers';
|
||||
|
||||
interface SetupDeps {
|
||||
expressions: ExpressionsSetup;
|
||||
}
|
||||
|
||||
interface StartDeps {
|
||||
expression: ExpressionsStart;
|
||||
}
|
||||
|
||||
export type ExpressionRevealImagePluginSetup = void;
|
||||
export type ExpressionRevealImagePluginStart = void;
|
||||
|
||||
export class ExpressionRevealImagePlugin
|
||||
implements
|
||||
Plugin<
|
||||
ExpressionRevealImagePluginSetup,
|
||||
ExpressionRevealImagePluginStart,
|
||||
SetupDeps,
|
||||
StartDeps
|
||||
> {
|
||||
public setup(core: CoreSetup, { expressions }: SetupDeps): ExpressionRevealImagePluginSetup {
|
||||
expressions.registerRenderer(revealImageRenderer);
|
||||
}
|
||||
|
||||
public start(core: CoreStart): ExpressionRevealImagePluginStart {}
|
||||
|
||||
public stop() {}
|
||||
}
|
15
src/plugins/expression_reveal_image/server/index.ts
Normal file
15
src/plugins/expression_reveal_image/server/index.ts
Normal file
|
@ -0,0 +1,15 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License
|
||||
* 2.0 and the Server Side Public License, v 1; you may not use this file except
|
||||
* in compliance with, at your election, the Elastic License 2.0 or the Server
|
||||
* Side Public License, v 1.
|
||||
*/
|
||||
|
||||
import { ExpressionRevealImagePlugin } from './plugin';
|
||||
|
||||
export type { ExpressionRevealImagePluginSetup, ExpressionRevealImagePluginStart } from './plugin';
|
||||
|
||||
export function plugin() {
|
||||
return new ExpressionRevealImagePlugin();
|
||||
}
|
39
src/plugins/expression_reveal_image/server/plugin.ts
Normal file
39
src/plugins/expression_reveal_image/server/plugin.ts
Normal file
|
@ -0,0 +1,39 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License
|
||||
* 2.0 and the Server Side Public License, v 1; you may not use this file except
|
||||
* in compliance with, at your election, the Elastic License 2.0 or the Server
|
||||
* Side Public License, v 1.
|
||||
*/
|
||||
|
||||
import { CoreSetup, CoreStart, Plugin } from '../../../core/public';
|
||||
import { ExpressionsServerStart, ExpressionsServerSetup } from '../../expressions/server';
|
||||
import { revealImageFunction } from '../common';
|
||||
|
||||
interface SetupDeps {
|
||||
expressions: ExpressionsServerSetup;
|
||||
}
|
||||
|
||||
interface StartDeps {
|
||||
expression: ExpressionsServerStart;
|
||||
}
|
||||
|
||||
export type ExpressionRevealImagePluginSetup = void;
|
||||
export type ExpressionRevealImagePluginStart = void;
|
||||
|
||||
export class ExpressionRevealImagePlugin
|
||||
implements
|
||||
Plugin<
|
||||
ExpressionRevealImagePluginSetup,
|
||||
ExpressionRevealImagePluginStart,
|
||||
SetupDeps,
|
||||
StartDeps
|
||||
> {
|
||||
public setup(core: CoreSetup, { expressions }: SetupDeps): ExpressionRevealImagePluginSetup {
|
||||
expressions.registerFunction(revealImageFunction);
|
||||
}
|
||||
|
||||
public start(core: CoreStart): ExpressionRevealImagePluginStart {}
|
||||
|
||||
public stop() {}
|
||||
}
|
21
src/plugins/expression_reveal_image/tsconfig.json
Normal file
21
src/plugins/expression_reveal_image/tsconfig.json
Normal file
|
@ -0,0 +1,21 @@
|
|||
{
|
||||
"extends": "../../../tsconfig.base.json",
|
||||
"compilerOptions": {
|
||||
"composite": true,
|
||||
"outDir": "./target/types",
|
||||
"emitDeclarationOnly": true,
|
||||
"declaration": true,
|
||||
"declarationMap": true,
|
||||
"isolatedModules": true
|
||||
},
|
||||
"include": [
|
||||
"common/**/*",
|
||||
"public/**/*",
|
||||
"server/**/*",
|
||||
],
|
||||
"references": [
|
||||
{ "path": "../../core/tsconfig.json" },
|
||||
{ "path": "../presentation_util/tsconfig.json" },
|
||||
{ "path": "../expressions/tsconfig.json" },
|
||||
]
|
||||
}
|
10
src/plugins/presentation_util/common/lib/index.ts
Normal file
10
src/plugins/presentation_util/common/lib/index.ts
Normal file
|
@ -0,0 +1,10 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License
|
||||
* 2.0 and the Server Side Public License, v 1; you may not use this file except
|
||||
* in compliance with, at your election, the Elastic License 2.0 or the Server
|
||||
* Side Public License, v 1.
|
||||
*/
|
||||
|
||||
export * from './utils';
|
||||
export * from './test_helpers';
|
|
@ -0,0 +1,27 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License
|
||||
* 2.0 and the Server Side Public License, v 1; you may not use this file except
|
||||
* in compliance with, at your election, the Elastic License 2.0 or the Server
|
||||
* Side Public License, v 1.
|
||||
*/
|
||||
|
||||
import { mapValues } from 'lodash';
|
||||
import {
|
||||
ExpressionValueBoxed,
|
||||
typeSpecs,
|
||||
ExpressionFunctionDefinition,
|
||||
} from '../../../../expressions/common';
|
||||
|
||||
type FnType = () => typeof typeSpecs[number] &
|
||||
ExpressionFunctionDefinition<string, any, Record<string, any>, ExpressionValueBoxed<any, any>>;
|
||||
|
||||
// It takes a function spec and passes in default args into the spec fn
|
||||
export const functionWrapper = (fnSpec: FnType): ReturnType<FnType>['fn'] => {
|
||||
const spec = fnSpec();
|
||||
const defaultArgs = mapValues(spec.args, (argSpec) => {
|
||||
return argSpec.default;
|
||||
});
|
||||
|
||||
return (context, args, handlers) => spec.fn(context, { ...defaultArgs, ...args }, handlers);
|
||||
};
|
|
@ -0,0 +1,9 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License
|
||||
* 2.0 and the Server Side Public License, v 1; you may not use this file except
|
||||
* in compliance with, at your election, the Elastic License 2.0 or the Server
|
||||
* Side Public License, v 1.
|
||||
*/
|
||||
|
||||
export * from './function_wrapper';
|
|
@ -1,8 +1,9 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License
|
||||
* 2.0; you may not use this file except in compliance with the Elastic License
|
||||
* 2.0.
|
||||
* 2.0 and the Server Side Public License, v 1; you may not use this file except
|
||||
* in compliance with, at your election, the Elastic License 2.0 or the Server
|
||||
* Side Public License, v 1.
|
||||
*/
|
||||
|
||||
import { isValidDataUrl, parseDataUrl } from './dataurl';
|
|
@ -1,8 +1,9 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License
|
||||
* 2.0; you may not use this file except in compliance with the Elastic License
|
||||
* 2.0.
|
||||
* 2.0 and the Server Side Public License, v 1; you may not use this file except
|
||||
* in compliance with, at your election, the Elastic License 2.0 or the Server
|
||||
* Side Public License, v 1.
|
||||
*/
|
||||
|
||||
import { fromByteArray } from 'base64-js';
|
|
@ -1,8 +1,9 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License
|
||||
* 2.0; you may not use this file except in compliance with the Elastic License
|
||||
* 2.0.
|
||||
* 2.0 and the Server Side Public License, v 1; you may not use this file except
|
||||
* in compliance with, at your election, the Elastic License 2.0 or the Server
|
||||
* Side Public License, v 1.
|
||||
*/
|
||||
|
||||
export const elasticLogo =
|
File diff suppressed because one or more lines are too long
|
@ -1,8 +1,9 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License
|
||||
* 2.0; you may not use this file except in compliance with the Elastic License
|
||||
* 2.0.
|
||||
* 2.0 and the Server Side Public License, v 1; you may not use this file except
|
||||
* in compliance with, at your election, the Elastic License 2.0 or the Server
|
||||
* Side Public License, v 1.
|
||||
*/
|
||||
|
||||
import { isValidHttpUrl } from './httpurl';
|
|
@ -1,8 +1,9 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License
|
||||
* 2.0; you may not use this file except in compliance with the Elastic License
|
||||
* 2.0.
|
||||
* 2.0 and the Server Side Public License, v 1; you may not use this file except
|
||||
* in compliance with, at your election, the Elastic License 2.0 or the Server
|
||||
* Side Public License, v 1.
|
||||
*/
|
||||
|
||||
// A cheap regex to distinguish an HTTP URL string from a data URL string
|
15
src/plugins/presentation_util/common/lib/utils/index.ts
Normal file
15
src/plugins/presentation_util/common/lib/utils/index.ts
Normal file
|
@ -0,0 +1,15 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License
|
||||
* 2.0 and the Server Side Public License, v 1; you may not use this file except
|
||||
* in compliance with, at your election, the Elastic License 2.0 or the Server
|
||||
* Side Public License, v 1.
|
||||
*/
|
||||
|
||||
export * from './dataurl';
|
||||
export * from './elastic_logo';
|
||||
export * from './elastic_outline';
|
||||
export * from './httpurl';
|
||||
export * from './missing_asset';
|
||||
export * from './resolve_dataurl';
|
||||
export * from './url';
|
|
@ -0,0 +1,11 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License
|
||||
* 2.0 and the Server Side Public License, v 1; you may not use this file except
|
||||
* in compliance with, at your election, the Elastic License 2.0 or the Server
|
||||
* Side Public License, v 1.
|
||||
*/
|
||||
|
||||
// CC0, source: https://pixabay.com/en/question-mark-confirmation-question-838656/
|
||||
export const missingImage =
|
||||
'';
|
|
@ -1,11 +1,12 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License
|
||||
* 2.0; you may not use this file except in compliance with the Elastic License
|
||||
* 2.0.
|
||||
* 2.0 and the Server Side Public License, v 1; you may not use this file except
|
||||
* in compliance with, at your election, the Elastic License 2.0 or the Server
|
||||
* Side Public License, v 1.
|
||||
*/
|
||||
|
||||
import { missingImage } from '../../common/lib/missing_asset';
|
||||
import { missingImage } from './missing_asset';
|
||||
import { resolveFromArgs, resolveWithMissingImage } from './resolve_dataurl';
|
||||
|
||||
describe('resolve_dataurl', () => {
|
|
@ -1,13 +1,14 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License
|
||||
* 2.0; you may not use this file except in compliance with the Elastic License
|
||||
* 2.0.
|
||||
* 2.0 and the Server Side Public License, v 1; you may not use this file except
|
||||
* in compliance with, at your election, the Elastic License 2.0 or the Server
|
||||
* Side Public License, v 1.
|
||||
*/
|
||||
|
||||
import { get } from 'lodash';
|
||||
import { isValidUrl } from '../../common/lib/url';
|
||||
import { missingImage } from '../../common/lib/missing_asset';
|
||||
import { isValidUrl } from './url';
|
||||
import { missingImage } from './missing_asset';
|
||||
|
||||
/*
|
||||
* NOTE: args.dataurl can come as an expression here.
|
|
@ -1,11 +1,12 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License
|
||||
* 2.0; you may not use this file except in compliance with the Elastic License
|
||||
* 2.0.
|
||||
* 2.0 and the Server Side Public License, v 1; you may not use this file except
|
||||
* in compliance with, at your election, the Elastic License 2.0 or the Server
|
||||
* Side Public License, v 1.
|
||||
*/
|
||||
|
||||
import { missingImage } from '../../common/lib/missing_asset';
|
||||
import { missingImage } from './missing_asset';
|
||||
import { isValidUrl } from './url';
|
||||
|
||||
describe('resolve_dataurl', () => {
|
14
src/plugins/presentation_util/common/lib/utils/url.ts
Normal file
14
src/plugins/presentation_util/common/lib/utils/url.ts
Normal file
|
@ -0,0 +1,14 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License
|
||||
* 2.0 and the Server Side Public License, v 1; you may not use this file except
|
||||
* in compliance with, at your election, the Elastic License 2.0 or the Server
|
||||
* Side Public License, v 1.
|
||||
*/
|
||||
|
||||
import { isValidDataUrl } from './dataurl';
|
||||
import { isValidHttpUrl } from './httpurl';
|
||||
|
||||
export function isValidUrl(url: string) {
|
||||
return isValidDataUrl(url) || isValidHttpUrl(url);
|
||||
}
|
13
src/plugins/presentation_util/jest.config.js
Normal file
13
src/plugins/presentation_util/jest.config.js
Normal file
|
@ -0,0 +1,13 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License
|
||||
* 2.0 and the Server Side Public License, v 1; you may not use this file except
|
||||
* in compliance with, at your election, the Elastic License 2.0 or the Server
|
||||
* Side Public License, v 1.
|
||||
*/
|
||||
|
||||
module.exports = {
|
||||
preset: '@kbn/test',
|
||||
rootDir: '../../..',
|
||||
roots: ['<rootDir>/src/plugins/presentation_util'],
|
||||
};
|
|
@ -4,6 +4,9 @@
|
|||
"kibanaVersion": "kibana",
|
||||
"server": true,
|
||||
"ui": true,
|
||||
"extraPublicDirs": [
|
||||
"common/lib"
|
||||
],
|
||||
"requiredPlugins": [
|
||||
"savedObjects"
|
||||
],
|
||||
|
|
|
@ -0,0 +1,9 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License
|
||||
* 2.0 and the Server Side Public License, v 1; you may not use this file except
|
||||
* in compliance with, at your election, the Elastic License 2.0 or the Server
|
||||
* Side Public License, v 1.
|
||||
*/
|
||||
|
||||
export * from './render';
|
61
src/plugins/presentation_util/public/__stories__/render.tsx
Normal file
61
src/plugins/presentation_util/public/__stories__/render.tsx
Normal file
|
@ -0,0 +1,61 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License
|
||||
* 2.0 and the Server Side Public License, v 1; you may not use this file except
|
||||
* in compliance with, at your election, the Elastic License 2.0 or the Server
|
||||
* Side Public License, v 1.
|
||||
*/
|
||||
|
||||
import { action } from '@storybook/addon-actions';
|
||||
import React, { useRef, useEffect } from 'react';
|
||||
import { ExpressionRenderDefinition, IInterpreterRenderHandlers } from 'src/plugins/expressions';
|
||||
|
||||
export const defaultHandlers: IInterpreterRenderHandlers = {
|
||||
getRenderMode: () => 'display',
|
||||
isSyncColorsEnabled: () => false,
|
||||
done: action('done'),
|
||||
onDestroy: action('onDestroy'),
|
||||
reload: action('reload'),
|
||||
update: action('update'),
|
||||
event: action('event'),
|
||||
};
|
||||
|
||||
/*
|
||||
Uses a RenderDefinitionFactory and Config to render into an element.
|
||||
|
||||
Intended to be used for stories for RenderDefinitionFactory
|
||||
*/
|
||||
interface RenderAdditionalProps {
|
||||
height?: string;
|
||||
width?: string;
|
||||
handlers?: IInterpreterRenderHandlers;
|
||||
}
|
||||
|
||||
export const Render = <Renderer,>({
|
||||
renderer,
|
||||
config,
|
||||
...rest
|
||||
}: Renderer extends () => ExpressionRenderDefinition<infer Config>
|
||||
? { renderer: Renderer; config: Config } & RenderAdditionalProps
|
||||
: { renderer: undefined; config: undefined } & RenderAdditionalProps) => {
|
||||
const { height, width, handlers } = {
|
||||
height: '200px',
|
||||
width: '200px',
|
||||
handlers: defaultHandlers,
|
||||
...rest,
|
||||
};
|
||||
|
||||
const containerRef = useRef<HTMLDivElement | null>(null);
|
||||
|
||||
useEffect(() => {
|
||||
if (renderer && containerRef.current !== null) {
|
||||
renderer().render(containerRef.current, config, handlers);
|
||||
}
|
||||
}, [renderer, config, handlers]);
|
||||
|
||||
return (
|
||||
<div style={{ width, height }} ref={containerRef}>
|
||||
{' '}
|
||||
</div>
|
||||
);
|
||||
};
|
|
@ -28,6 +28,7 @@ export {
|
|||
export { PresentationUtilPluginSetup, PresentationUtilPluginStart } from './types';
|
||||
export { SaveModalDashboardProps } from './components/types';
|
||||
export { projectIDs, ProjectID, Project } from '../common/labs';
|
||||
export * from '../common/lib';
|
||||
|
||||
export {
|
||||
LazyLabsBeakerButton,
|
||||
|
|
|
@ -7,6 +7,9 @@
|
|||
"declaration": true,
|
||||
"declarationMap": true
|
||||
},
|
||||
"extraPublicDirs": [
|
||||
"common"
|
||||
],
|
||||
"include": [
|
||||
"common/**/*",
|
||||
"public/**/*",
|
||||
|
|
|
@ -6,7 +6,7 @@
|
|||
* Side Public License, v 1.
|
||||
*/
|
||||
|
||||
import _ from 'lodash';
|
||||
import { debounce } from 'lodash';
|
||||
import PropTypes from 'prop-types';
|
||||
import React from 'react';
|
||||
|
||||
|
@ -116,7 +116,7 @@ class SavedObjectFinderUi extends React.Component<
|
|||
|
||||
private isComponentMounted: boolean = false;
|
||||
|
||||
private debouncedFetch = _.debounce(async (query: string) => {
|
||||
private debouncedFetch = debounce(async (query: string) => {
|
||||
const metaDataMap = this.getSavedObjectMetaDataMap();
|
||||
|
||||
const fields = Object.values(metaDataMap)
|
||||
|
|
|
@ -6,7 +6,7 @@
|
|||
* Side Public License, v 1.
|
||||
*/
|
||||
|
||||
import _ from 'lodash';
|
||||
import { cloneDeep, defaults, forOwn, assign } from 'lodash';
|
||||
import { EsResponse, SavedObject, SavedObjectConfig, SavedObjectKibanaServices } from '../../types';
|
||||
import { SavedObjectNotFound } from '../../../../kibana_utils/public';
|
||||
import {
|
||||
|
@ -28,7 +28,7 @@ export async function applyESResp(
|
|||
) {
|
||||
const mapping = expandShorthand(config.mapping ?? {});
|
||||
const savedObjectType = config.type || '';
|
||||
savedObject._source = _.cloneDeep(resp._source);
|
||||
savedObject._source = cloneDeep(resp._source);
|
||||
if (typeof resp.found === 'boolean' && !resp.found) {
|
||||
throw new SavedObjectNotFound(savedObjectType, savedObject.id || '');
|
||||
}
|
||||
|
@ -42,10 +42,10 @@ export async function applyESResp(
|
|||
}
|
||||
|
||||
// assign the defaults to the response
|
||||
_.defaults(savedObject._source, savedObject.defaults);
|
||||
defaults(savedObject._source, savedObject.defaults);
|
||||
|
||||
// transform the source using _deserializers
|
||||
_.forOwn(mapping, (fieldMapping, fieldName) => {
|
||||
forOwn(mapping, (fieldMapping, fieldName) => {
|
||||
if (fieldMapping._deserialize && typeof fieldName === 'string') {
|
||||
savedObject._source[fieldName] = fieldMapping._deserialize(
|
||||
savedObject._source[fieldName] as string
|
||||
|
@ -54,7 +54,7 @@ export async function applyESResp(
|
|||
});
|
||||
|
||||
// Give obj all of the values in _source.fields
|
||||
_.assign(savedObject, savedObject._source);
|
||||
assign(savedObject, savedObject._source);
|
||||
savedObject.lastSavedTitle = savedObject.title;
|
||||
|
||||
if (meta.searchSourceJSON) {
|
||||
|
|
|
@ -6,7 +6,7 @@
|
|||
* Side Public License, v 1.
|
||||
*/
|
||||
|
||||
import _ from 'lodash';
|
||||
import { get } from 'lodash';
|
||||
import { i18n } from '@kbn/i18n';
|
||||
import { SavedObjectAttributes } from 'kibana/public';
|
||||
import { SavedObject, SavedObjectKibanaServices } from '../../types';
|
||||
|
@ -40,7 +40,7 @@ export async function createSource(
|
|||
return await savedObjectsClient.create(esType, source, options);
|
||||
} catch (err) {
|
||||
// record exists, confirm overwriting
|
||||
if (_.get(err, 'res.status') === 409) {
|
||||
if (get(err, 'res.status') === 409) {
|
||||
const confirmMessage = i18n.translate(
|
||||
'savedObjects.confirmModal.overwriteConfirmationMessage',
|
||||
{
|
||||
|
|
|
@ -6,7 +6,7 @@
|
|||
* Side Public License, v 1.
|
||||
*/
|
||||
|
||||
import _ from 'lodash';
|
||||
import { cloneDeep, assign } from 'lodash';
|
||||
import { SavedObjectsClientContract } from 'kibana/public';
|
||||
import { SavedObject, SavedObjectConfig } from '../../types';
|
||||
|
||||
|
@ -24,7 +24,7 @@ export async function intializeSavedObject(
|
|||
|
||||
if (!savedObject.id) {
|
||||
// just assign the defaults and be done
|
||||
_.assign(savedObject, savedObject.defaults);
|
||||
assign(savedObject, savedObject.defaults);
|
||||
await savedObject.hydrateIndexPattern!();
|
||||
if (typeof config.afterESResp === 'function') {
|
||||
savedObject = await config.afterESResp(savedObject);
|
||||
|
@ -36,7 +36,7 @@ export async function intializeSavedObject(
|
|||
const respMapped = {
|
||||
_id: resp.id,
|
||||
_type: resp.type,
|
||||
_source: _.cloneDeep(resp.attributes),
|
||||
_source: cloneDeep(resp.attributes),
|
||||
references: resp.references,
|
||||
found: !!resp._version,
|
||||
};
|
||||
|
|
|
@ -6,7 +6,7 @@
|
|||
* Side Public License, v 1.
|
||||
*/
|
||||
|
||||
import _ from 'lodash';
|
||||
import { forOwn } from 'lodash';
|
||||
import { SavedObject, SavedObjectConfig } from '../../types';
|
||||
import { extractSearchSourceReferences } from '../../../../data/public';
|
||||
import { expandShorthand } from './field_mapping';
|
||||
|
@ -17,7 +17,7 @@ export function serializeSavedObject(savedObject: SavedObject, config: SavedObje
|
|||
const attributes = {} as Record<string, any>;
|
||||
const references = [];
|
||||
|
||||
_.forOwn(mapping, (fieldMapping, fieldName) => {
|
||||
forOwn(mapping, (fieldMapping, fieldName) => {
|
||||
if (typeof fieldName !== 'string') {
|
||||
return;
|
||||
}
|
||||
|
|
|
@ -5,7 +5,7 @@
|
|||
* 2.0.
|
||||
*/
|
||||
|
||||
import { functionWrapper } from '../../../test_helpers/function_wrapper';
|
||||
import { functionWrapper } from '../../../../../../src/plugins/presentation_util/common/lib';
|
||||
import { testTable } from '../common/__fixtures__/test_tables';
|
||||
import { fontStyle } from '../common/__fixtures__/test_styles';
|
||||
import { markdown } from './markdown';
|
||||
|
|
|
@ -5,7 +5,7 @@
|
|||
* 2.0.
|
||||
*/
|
||||
|
||||
import { elasticLogo } from '../../../lib/elastic_logo';
|
||||
import { elasticLogo } from '../../../../../../../src/plugins/presentation_util/common/lib';
|
||||
|
||||
export const fontStyle = {
|
||||
type: 'style',
|
||||
|
|
|
@ -5,7 +5,7 @@
|
|||
* 2.0.
|
||||
*/
|
||||
|
||||
import { functionWrapper } from '../../../test_helpers/function_wrapper';
|
||||
import { functionWrapper } from '../../../../../../src/plugins/presentation_util/common/lib';
|
||||
import { all } from './all';
|
||||
|
||||
describe('all', () => {
|
||||
|
|
|
@ -5,7 +5,7 @@
|
|||
* 2.0.
|
||||
*/
|
||||
|
||||
import { functionWrapper } from '../../../test_helpers/function_wrapper';
|
||||
import { functionWrapper } from '../../../../../../src/plugins/presentation_util/common/lib';
|
||||
import { getFunctionErrors } from '../../../i18n';
|
||||
import { emptyTable, testTable } from './__fixtures__/test_tables';
|
||||
import { alterColumn } from './alterColumn';
|
||||
|
|
|
@ -4,8 +4,7 @@
|
|||
* 2.0; you may not use this file except in compliance with the Elastic License
|
||||
* 2.0.
|
||||
*/
|
||||
|
||||
import { functionWrapper } from '../../../test_helpers/function_wrapper';
|
||||
import { functionWrapper } from '../../../../../../src/plugins/presentation_util/common/lib';
|
||||
import { any } from './any';
|
||||
|
||||
describe('any', () => {
|
||||
|
|
|
@ -5,7 +5,7 @@
|
|||
* 2.0.
|
||||
*/
|
||||
|
||||
import { functionWrapper } from '../../../test_helpers/function_wrapper';
|
||||
import { functionWrapper } from '../../../../../../src/plugins/presentation_util/common/lib';
|
||||
import { asFn } from './as';
|
||||
|
||||
describe('as', () => {
|
||||
|
|
|
@ -5,7 +5,7 @@
|
|||
* 2.0.
|
||||
*/
|
||||
|
||||
import { functionWrapper } from '../../../test_helpers/function_wrapper';
|
||||
import { functionWrapper } from '../../../../../../src/plugins/presentation_util/common/lib';
|
||||
import { getFunctionErrors } from '../../../i18n';
|
||||
import { testTable } from './__fixtures__/test_tables';
|
||||
import { axisConfig } from './axisConfig';
|
||||
|
|
|
@ -7,7 +7,7 @@
|
|||
|
||||
import { of } from 'rxjs';
|
||||
import { TestScheduler } from 'rxjs/testing';
|
||||
import { functionWrapper } from '../../../test_helpers/function_wrapper';
|
||||
import { functionWrapper } from '../../../../../../src/plugins/presentation_util/common/lib';
|
||||
import { caseFn } from './case';
|
||||
|
||||
describe('case', () => {
|
||||
|
|
|
@ -4,8 +4,7 @@
|
|||
* 2.0; you may not use this file except in compliance with the Elastic License
|
||||
* 2.0.
|
||||
*/
|
||||
|
||||
import { functionWrapper } from '../../../test_helpers/function_wrapper';
|
||||
import { functionWrapper } from '../../../../../../src/plugins/presentation_util/common/lib';
|
||||
import { testTable } from './__fixtures__/test_tables';
|
||||
import { clear } from './clear';
|
||||
|
||||
|
|
|
@ -5,7 +5,7 @@
|
|||
* 2.0.
|
||||
*/
|
||||
|
||||
import { functionWrapper } from '../../../test_helpers/function_wrapper';
|
||||
import { functionWrapper } from '../../../../../../src/plugins/presentation_util/common/lib';
|
||||
import { emptyTable, testTable } from './__fixtures__/test_tables';
|
||||
import { columns } from './columns';
|
||||
|
||||
|
|
|
@ -5,7 +5,7 @@
|
|||
* 2.0.
|
||||
*/
|
||||
|
||||
import { functionWrapper } from '../../../test_helpers/function_wrapper';
|
||||
import { functionWrapper } from '../../../../../../src/plugins/presentation_util/common/lib';
|
||||
import { getFunctionErrors } from '../../../i18n';
|
||||
import { compare } from './compare';
|
||||
|
||||
|
|
|
@ -8,7 +8,7 @@
|
|||
import { ExpressionFunctionDefinition } from 'src/plugins/expressions/common';
|
||||
import { ContainerStyle, Overflow, BackgroundRepeat, BackgroundSize } from '../../../types';
|
||||
import { getFunctionHelp, getFunctionErrors } from '../../../i18n';
|
||||
import { isValidUrl } from '../../../common/lib/url';
|
||||
import { isValidUrl } from '../../../../../../src/plugins/presentation_util/common/lib';
|
||||
|
||||
interface Output extends ContainerStyle {
|
||||
type: 'containerStyle';
|
||||
|
|
|
@ -5,8 +5,10 @@
|
|||
* 2.0.
|
||||
*/
|
||||
|
||||
import { functionWrapper } from '../../../test_helpers/function_wrapper';
|
||||
import { elasticLogo } from '../../lib/elastic_logo';
|
||||
import {
|
||||
elasticLogo,
|
||||
functionWrapper,
|
||||
} from '../../../../../../src/plugins/presentation_util/common/lib';
|
||||
import { getFunctionErrors } from '../../../i18n';
|
||||
import { containerStyle } from './containerStyle';
|
||||
|
||||
|
|
|
@ -5,7 +5,7 @@
|
|||
* 2.0.
|
||||
*/
|
||||
|
||||
import { functionWrapper } from '../../../test_helpers/function_wrapper';
|
||||
import { functionWrapper } from '../../../../../../src/plugins/presentation_util/common/lib';
|
||||
import { testTable, emptyTable } from './__fixtures__/test_tables';
|
||||
import { context } from './context';
|
||||
|
||||
|
|
|
@ -5,11 +5,11 @@
|
|||
* 2.0.
|
||||
*/
|
||||
|
||||
// @ts-expect-error untyped lib
|
||||
import { functionWrapper } from '../../../test_helpers/function_wrapper';
|
||||
import { functionWrapper } from '../../../../../../src/plugins/presentation_util/common/lib';
|
||||
import { getFunctionErrors } from '../../../i18n';
|
||||
import { csv } from './csv';
|
||||
import { Datatable } from 'src/plugins/expressions';
|
||||
import { Datatable, ExecutionContext, SerializableState } from 'src/plugins/expressions';
|
||||
import { Adapters } from 'src/plugins/inspector';
|
||||
|
||||
const errors = getFunctionErrors().csv;
|
||||
|
||||
|
@ -30,43 +30,59 @@ describe('csv', () => {
|
|||
|
||||
it('should return a datatable', () => {
|
||||
expect(
|
||||
fn(null, {
|
||||
data: `name,number
|
||||
fn(
|
||||
null,
|
||||
{
|
||||
data: `name,number
|
||||
one,1
|
||||
two,2
|
||||
fourty two,42`,
|
||||
})
|
||||
},
|
||||
{} as ExecutionContext<Adapters, SerializableState>
|
||||
)
|
||||
).toEqual(expected);
|
||||
});
|
||||
|
||||
it('should allow custom delimiter', () => {
|
||||
expect(
|
||||
fn(null, {
|
||||
data: `name\tnumber
|
||||
fn(
|
||||
null,
|
||||
{
|
||||
data: `name\tnumber
|
||||
one\t1
|
||||
two\t2
|
||||
fourty two\t42`,
|
||||
delimiter: '\t',
|
||||
})
|
||||
delimiter: '\t',
|
||||
},
|
||||
{} as ExecutionContext<Adapters, SerializableState>
|
||||
)
|
||||
).toEqual(expected);
|
||||
|
||||
expect(
|
||||
fn(null, {
|
||||
data: `name%SPLIT%number
|
||||
fn(
|
||||
null,
|
||||
{
|
||||
data: `name%SPLIT%number
|
||||
one%SPLIT%1
|
||||
two%SPLIT%2
|
||||
fourty two%SPLIT%42`,
|
||||
delimiter: '%SPLIT%',
|
||||
})
|
||||
delimiter: '%SPLIT%',
|
||||
},
|
||||
{} as ExecutionContext<Adapters, SerializableState>
|
||||
)
|
||||
).toEqual(expected);
|
||||
});
|
||||
|
||||
it('should allow custom newline', () => {
|
||||
expect(
|
||||
fn(null, {
|
||||
data: `name,number\rone,1\rtwo,2\rfourty two,42`,
|
||||
newline: '\r',
|
||||
})
|
||||
fn(
|
||||
null,
|
||||
{
|
||||
data: `name,number\rone,1\rtwo,2\rfourty two,42`,
|
||||
newline: '\r',
|
||||
},
|
||||
{} as ExecutionContext<Adapters, SerializableState>
|
||||
)
|
||||
).toEqual(expected);
|
||||
});
|
||||
|
||||
|
@ -83,10 +99,14 @@ fourty two%SPLIT%42`,
|
|||
};
|
||||
|
||||
expect(
|
||||
fn(null, {
|
||||
data: `foo," bar ", baz, " buz "
|
||||
fn(
|
||||
null,
|
||||
{
|
||||
data: `foo," bar ", baz, " buz "
|
||||
1,2,3,4`,
|
||||
})
|
||||
},
|
||||
{} as ExecutionContext<Adapters, SerializableState>
|
||||
)
|
||||
).toEqual(expectedResult);
|
||||
});
|
||||
|
||||
|
@ -106,22 +126,30 @@ fourty two%SPLIT%42`,
|
|||
};
|
||||
|
||||
expect(
|
||||
fn(null, {
|
||||
data: `foo," bar ", baz, " buz "
|
||||
fn(
|
||||
null,
|
||||
{
|
||||
data: `foo," bar ", baz, " buz "
|
||||
1," best ",3, " ok"
|
||||
" good", bad, better , " worst " `,
|
||||
})
|
||||
},
|
||||
{} as ExecutionContext<Adapters, SerializableState>
|
||||
)
|
||||
).toEqual(expectedResult);
|
||||
});
|
||||
|
||||
it('throws when given invalid csv', () => {
|
||||
expect(() => {
|
||||
fn(null, {
|
||||
data: `name,number
|
||||
fn(
|
||||
null,
|
||||
{
|
||||
data: `name,number
|
||||
one|1
|
||||
two.2
|
||||
fourty two,42`,
|
||||
});
|
||||
},
|
||||
{} as ExecutionContext<Adapters, SerializableState>
|
||||
);
|
||||
}).toThrow(new RegExp(errors.invalidInputCSV().message));
|
||||
});
|
||||
});
|
||||
|
|
|
@ -6,7 +6,7 @@
|
|||
*/
|
||||
|
||||
import sinon from 'sinon';
|
||||
import { functionWrapper } from '../../../test_helpers/function_wrapper';
|
||||
import { functionWrapper } from '../../../../../../src/plugins/presentation_util/common/lib';
|
||||
import { getFunctionErrors } from '../../../i18n';
|
||||
import { date } from './date';
|
||||
|
||||
|
|
|
@ -5,7 +5,7 @@
|
|||
* 2.0.
|
||||
*/
|
||||
|
||||
import { functionWrapper } from '../../../test_helpers/function_wrapper';
|
||||
import { functionWrapper } from '../../../../../../src/plugins/presentation_util/common/lib';
|
||||
import { doFn } from './do';
|
||||
|
||||
describe('do', () => {
|
||||
|
|
|
@ -5,23 +5,30 @@
|
|||
* 2.0.
|
||||
*/
|
||||
|
||||
// @ts-expect-error untyped local
|
||||
import { functionWrapper } from '../../../test_helpers/function_wrapper';
|
||||
import { functionWrapper } from '../../../../../../src/plugins/presentation_util/common/lib';
|
||||
import { testTable, relationalTable } from './__fixtures__/test_tables';
|
||||
import { dropdownControl } from './dropdownControl';
|
||||
import { ExecutionContext, SerializableState } from 'src/plugins/expressions';
|
||||
import { Adapters } from 'src/plugins/inspector';
|
||||
|
||||
describe('dropdownControl', () => {
|
||||
const fn = functionWrapper(dropdownControl);
|
||||
|
||||
it('returns a render as dropdown_filter', () => {
|
||||
expect(fn(testTable, { filterColumn: 'name', valueColumn: 'name' })).toHaveProperty(
|
||||
'type',
|
||||
'render'
|
||||
);
|
||||
expect(fn(testTable, { filterColumn: 'name', valueColumn: 'name' })).toHaveProperty(
|
||||
'as',
|
||||
'dropdown_filter'
|
||||
);
|
||||
expect(
|
||||
fn(
|
||||
testTable,
|
||||
{ filterColumn: 'name', valueColumn: 'name' },
|
||||
{} as ExecutionContext<Adapters, SerializableState>
|
||||
)
|
||||
).toHaveProperty('type', 'render');
|
||||
expect(
|
||||
fn(
|
||||
testTable,
|
||||
{ filterColumn: 'name', valueColumn: 'name' },
|
||||
{} as ExecutionContext<Adapters, SerializableState>
|
||||
)
|
||||
).toHaveProperty('as', 'dropdown_filter');
|
||||
});
|
||||
|
||||
describe('args', () => {
|
||||
|
@ -32,12 +39,24 @@ describe('dropdownControl', () => {
|
|||
unique.find(([value, label]) => value === name) ? unique : [...unique, [name, name]],
|
||||
[]
|
||||
);
|
||||
expect(fn(testTable, { valueColumn: 'name' }).value.choices).toEqual(uniqueNames);
|
||||
expect(
|
||||
fn(
|
||||
testTable,
|
||||
{ valueColumn: 'name' },
|
||||
{} as ExecutionContext<Adapters, SerializableState>
|
||||
)?.value?.choices
|
||||
).toEqual(uniqueNames);
|
||||
});
|
||||
|
||||
it('returns an empty array when provided an invalid column', () => {
|
||||
expect(fn(testTable, { valueColumn: 'foo' }).value.choices).toEqual([]);
|
||||
expect(fn(testTable, { valueColumn: '' }).value.choices).toEqual([]);
|
||||
expect(
|
||||
fn(testTable, { valueColumn: 'foo' }, {} as ExecutionContext<Adapters, SerializableState>)
|
||||
?.value?.choices
|
||||
).toEqual([]);
|
||||
expect(
|
||||
fn(testTable, { valueColumn: '' }, {} as ExecutionContext<Adapters, SerializableState>)
|
||||
?.value?.choices
|
||||
).toEqual([]);
|
||||
});
|
||||
});
|
||||
|
||||
|
@ -45,7 +64,11 @@ describe('dropdownControl', () => {
|
|||
it('populates dropdown choices with labels from label column', () => {
|
||||
const expectedChoices = relationalTable.rows.map((row) => [row.id, row.name]);
|
||||
expect(
|
||||
fn(relationalTable, { valueColumn: 'id', labelColumn: 'name' }).value.choices
|
||||
fn(
|
||||
relationalTable,
|
||||
{ valueColumn: 'id', labelColumn: 'name' },
|
||||
{} as ExecutionContext<Adapters, SerializableState>
|
||||
)?.value?.choices
|
||||
).toEqual(expectedChoices);
|
||||
});
|
||||
});
|
||||
|
@ -53,19 +76,30 @@ describe('dropdownControl', () => {
|
|||
|
||||
describe('filterColumn', () => {
|
||||
it('sets which column the filter is applied to', () => {
|
||||
expect(fn(testTable, { filterColumn: 'name' }).value).toHaveProperty('column', 'name');
|
||||
expect(fn(testTable, { filterColumn: 'name', valueColumn: 'price' }).value).toHaveProperty(
|
||||
'column',
|
||||
'name'
|
||||
);
|
||||
expect(
|
||||
fn(testTable, { filterColumn: 'name' }, {} as ExecutionContext<Adapters, SerializableState>)
|
||||
?.value
|
||||
).toHaveProperty('column', 'name');
|
||||
expect(
|
||||
fn(
|
||||
testTable,
|
||||
{ filterColumn: 'name', valueColumn: 'price' },
|
||||
{} as ExecutionContext<Adapters, SerializableState>
|
||||
)?.value
|
||||
).toHaveProperty('column', 'name');
|
||||
});
|
||||
|
||||
it('defaults to valueColumn if not provided', () => {
|
||||
expect(fn(testTable, { valueColumn: 'price' }).value).toHaveProperty('column', 'price');
|
||||
expect(
|
||||
fn(testTable, { valueColumn: 'price' }, {} as ExecutionContext<Adapters, SerializableState>)
|
||||
?.value
|
||||
).toHaveProperty('column', 'price');
|
||||
});
|
||||
|
||||
it('sets column to undefined if no args are provided', () => {
|
||||
expect(fn(testTable).value).toHaveProperty('column', undefined);
|
||||
expect(
|
||||
fn(testTable, {}, {} as ExecutionContext<Adapters, SerializableState>)?.value
|
||||
).toHaveProperty('column', undefined);
|
||||
});
|
||||
});
|
||||
});
|
||||
|
|
|
@ -5,7 +5,7 @@
|
|||
* 2.0.
|
||||
*/
|
||||
|
||||
import { functionWrapper } from '../../../test_helpers/function_wrapper';
|
||||
import { functionWrapper } from '../../../../../../src/plugins/presentation_util/common/lib';
|
||||
import { eq } from './eq';
|
||||
|
||||
describe('eq', () => {
|
||||
|
|
|
@ -5,7 +5,7 @@
|
|||
* 2.0.
|
||||
*/
|
||||
|
||||
import { functionWrapper } from '../../../test_helpers/function_wrapper';
|
||||
import { functionWrapper } from '../../../../../../src/plugins/presentation_util/common/lib';
|
||||
import { emptyFilter } from './__fixtures__/test_filters';
|
||||
import { exactly } from './exactly';
|
||||
|
||||
|
|
|
@ -7,7 +7,7 @@
|
|||
|
||||
import { of } from 'rxjs';
|
||||
import { TestScheduler } from 'rxjs/testing';
|
||||
import { functionWrapper } from '../../../test_helpers/function_wrapper';
|
||||
import { functionWrapper } from '../../../../../../src/plugins/presentation_util/common/lib';
|
||||
import { testTable } from './__fixtures__/test_tables';
|
||||
import { filterrows } from './filterrows';
|
||||
|
||||
|
|
|
@ -5,7 +5,7 @@
|
|||
* 2.0.
|
||||
*/
|
||||
|
||||
import { functionWrapper } from '../../../test_helpers/function_wrapper';
|
||||
import { functionWrapper } from '../../../../../../src/plugins/presentation_util/common/lib';
|
||||
import { formatdate } from './formatdate';
|
||||
|
||||
describe('formatdate', () => {
|
||||
|
|
|
@ -5,7 +5,7 @@
|
|||
* 2.0.
|
||||
*/
|
||||
|
||||
import { functionWrapper } from '../../../test_helpers/function_wrapper';
|
||||
import { functionWrapper } from '../../../../../../src/plugins/presentation_util/common/lib';
|
||||
import { formatnumber } from './formatnumber';
|
||||
|
||||
describe('formatnumber', () => {
|
||||
|
|
|
@ -5,7 +5,7 @@
|
|||
* 2.0.
|
||||
*/
|
||||
|
||||
import { functionWrapper } from '../../../test_helpers/function_wrapper';
|
||||
import { functionWrapper } from '../../../../../../src/plugins/presentation_util/common/lib';
|
||||
import { getFunctionErrors } from '../../../i18n';
|
||||
import { emptyTable, testTable } from './__fixtures__/test_tables';
|
||||
import { getCell } from './getCell';
|
||||
|
|
|
@ -5,7 +5,7 @@
|
|||
* 2.0.
|
||||
*/
|
||||
|
||||
import { functionWrapper } from '../../../test_helpers/function_wrapper';
|
||||
import { functionWrapper } from '../../../../../../src/plugins/presentation_util/common/lib';
|
||||
import { gt } from './gt';
|
||||
|
||||
describe('gt', () => {
|
||||
|
|
|
@ -5,7 +5,7 @@
|
|||
* 2.0.
|
||||
*/
|
||||
|
||||
import { functionWrapper } from '../../../test_helpers/function_wrapper';
|
||||
import { functionWrapper } from '../../../../../../src/plugins/presentation_util/common/lib';
|
||||
import { gte } from './gte';
|
||||
|
||||
describe('gte', () => {
|
||||
|
|
|
@ -5,7 +5,7 @@
|
|||
* 2.0.
|
||||
*/
|
||||
|
||||
import { functionWrapper } from '../../../test_helpers/function_wrapper';
|
||||
import { functionWrapper } from '../../../../../../src/plugins/presentation_util/common/lib';
|
||||
import { emptyTable, testTable } from './__fixtures__/test_tables';
|
||||
import { head } from './head';
|
||||
|
||||
|
|
|
@ -7,7 +7,7 @@
|
|||
|
||||
import { of } from 'rxjs';
|
||||
import { TestScheduler } from 'rxjs/testing';
|
||||
import { functionWrapper } from '../../../test_helpers/function_wrapper';
|
||||
import { functionWrapper } from '../../../../../../src/plugins/presentation_util/common/lib';
|
||||
import { ifFn } from './if';
|
||||
|
||||
describe('if', () => {
|
||||
|
|
|
@ -6,14 +6,14 @@
|
|||
*/
|
||||
|
||||
import expect from '@kbn/expect';
|
||||
// import { functionWrapper } from '../../../test_helpers/function_wrapper';
|
||||
import { elasticLogo } from '../../lib/elastic_logo';
|
||||
import { elasticOutline } from '../../lib/elastic_outline';
|
||||
import {
|
||||
elasticLogo,
|
||||
elasticOutline,
|
||||
} from '../../../../../../src/plugins/presentation_util/common/lib';
|
||||
// import { image } from './image';
|
||||
|
||||
// TODO: the test was not running and is not up to date
|
||||
describe.skip('image', () => {
|
||||
// const fn = functionWrapper(image);
|
||||
const fn = jest.fn();
|
||||
|
||||
it('returns an image object using a dataUrl', () => {
|
||||
|
|
|
@ -7,9 +7,10 @@
|
|||
|
||||
import { ExpressionFunctionDefinition } from 'src/plugins/expressions/common';
|
||||
import { getFunctionHelp, getFunctionErrors } from '../../../i18n';
|
||||
|
||||
import { resolveWithMissingImage } from '../../../common/lib/resolve_dataurl';
|
||||
import { elasticLogo } from '../../lib/elastic_logo';
|
||||
import {
|
||||
elasticLogo,
|
||||
resolveWithMissingImage,
|
||||
} from '../../../../../../src/plugins/presentation_util/common/lib';
|
||||
|
||||
export enum ImageMode {
|
||||
CONTAIN = 'contain',
|
||||
|
|
|
@ -43,7 +43,6 @@ import { replace } from './replace';
|
|||
import { rounddate } from './rounddate';
|
||||
import { rowCount } from './rowCount';
|
||||
import { repeatImage } from './repeat_image';
|
||||
import { revealImage } from './revealImage';
|
||||
import { seriesStyle } from './seriesStyle';
|
||||
import { shape } from './shape';
|
||||
import { sort } from './sort';
|
||||
|
@ -94,7 +93,6 @@ export const functions = [
|
|||
render,
|
||||
repeatImage,
|
||||
replace,
|
||||
revealImage,
|
||||
rounddate,
|
||||
rowCount,
|
||||
seriesStyle,
|
||||
|
|
|
@ -5,7 +5,7 @@
|
|||
* 2.0.
|
||||
*/
|
||||
|
||||
import { functionWrapper } from '../../../test_helpers/function_wrapper';
|
||||
import { functionWrapper } from '../../../../../../src/plugins/presentation_util/common/lib';
|
||||
import { getFunctionErrors } from '../../../i18n';
|
||||
import { testTable } from './__fixtures__/test_tables';
|
||||
import { joinRows } from './join_rows';
|
||||
|
|
|
@ -5,7 +5,7 @@
|
|||
* 2.0.
|
||||
*/
|
||||
|
||||
import { functionWrapper } from '../../../test_helpers/function_wrapper';
|
||||
import { functionWrapper } from '../../../../../../src/plugins/presentation_util/common/lib';
|
||||
import { lt } from './lt';
|
||||
|
||||
describe('lt', () => {
|
||||
|
|
|
@ -5,7 +5,7 @@
|
|||
* 2.0.
|
||||
*/
|
||||
|
||||
import { functionWrapper } from '../../../test_helpers/function_wrapper';
|
||||
import { functionWrapper } from '../../../../../../src/plugins/presentation_util/common/lib';
|
||||
import { lte } from './lte';
|
||||
|
||||
describe('lte', () => {
|
||||
|
|
Some files were not shown because too many files have changed in this diff Show more
Loading…
Add table
Add a link
Reference in a new issue