From b1583c5b453446714087cd2d30d226a64c29da9b Mon Sep 17 00:00:00 2001 From: Adam Dobrawy Date: Thu, 27 Jan 2022 04:38:28 +0100 Subject: [PATCH 1/3] refactor: upgrade ControlHeader to TSX & FC and add storybook --- .../src/explore/components/ControlHeader.jsx | 174 ------------------ .../components/ControlHeader.stories.tsx | 75 ++++++++ .../src/explore/components/ControlHeader.tsx | 154 ++++++++++++++++ 3 files changed, 229 insertions(+), 174 deletions(-) delete mode 100644 superset-frontend/src/explore/components/ControlHeader.jsx create mode 100644 superset-frontend/src/explore/components/ControlHeader.stories.tsx create mode 100644 superset-frontend/src/explore/components/ControlHeader.tsx diff --git a/superset-frontend/src/explore/components/ControlHeader.jsx b/superset-frontend/src/explore/components/ControlHeader.jsx deleted file mode 100644 index b78733135db25..0000000000000 --- a/superset-frontend/src/explore/components/ControlHeader.jsx +++ /dev/null @@ -1,174 +0,0 @@ -/** - * Licensed to the Apache Software Foundation (ASF) under one - * or more contributor license agreements. See the NOTICE file - * distributed with this work for additional information - * regarding copyright ownership. The ASF licenses this file - * to you under the Apache License, Version 2.0 (the - * "License"); you may not use this file except in compliance - * with the License. You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ -import React from 'react'; -import PropTypes from 'prop-types'; -import { t, css, withTheme } from '@superset-ui/core'; -import { InfoTooltipWithTrigger } from '@superset-ui/chart-controls'; -import { Tooltip } from 'src/components/Tooltip'; -import { FormLabel } from 'src/components/Form'; -import Icons from 'src/components/Icons'; - -const propTypes = { - name: PropTypes.string, - label: PropTypes.node, - description: PropTypes.node, - validationErrors: PropTypes.array, - renderTrigger: PropTypes.bool, - rightNode: PropTypes.node, - leftNode: PropTypes.node, - onClick: PropTypes.func, - hovered: PropTypes.bool, - tooltipOnClick: PropTypes.func, - warning: PropTypes.string, - danger: PropTypes.string, -}; - -const defaultProps = { - validationErrors: [], - renderTrigger: false, - hovered: false, - name: undefined, -}; - -class ControlHeader extends React.Component { - renderOptionalIcons() { - if (this.props.hovered) { - return ( - css` - position: absolute; - top: 50%; - right: 0; - padding-left: ${theme.gridUnit}px; - transform: translate(100%, -50%); - white-space: nowrap; - `} - > - {this.props.description && ( - - {' '} - - )} - {this.props.renderTrigger && ( - - {' '} - - )} - - ); - } - return null; - } - - render() { - if (!this.props.label) { - return null; - } - const labelClass = - this.props.validationErrors.length > 0 ? 'text-danger' : ''; - - const { theme } = this.props; - - return ( -
-
- - {this.props.leftNode && {this.props.leftNode}} - - {this.props.label} - {' '} - {this.props.warning && ( - - - - {' '} - - )} - {this.props.danger && ( - - - - {' '} - - )} - {this.props.validationErrors.length > 0 && ( - - - - {' '} - - )} - {this.renderOptionalIcons()} - -
- {this.props.rightNode && ( -
{this.props.rightNode}
- )} -
-
- ); - } -} - -ControlHeader.propTypes = propTypes; -ControlHeader.defaultProps = defaultProps; - -export default withTheme(ControlHeader); diff --git a/superset-frontend/src/explore/components/ControlHeader.stories.tsx b/superset-frontend/src/explore/components/ControlHeader.stories.tsx new file mode 100644 index 0000000000000..f050f7cfe5691 --- /dev/null +++ b/superset-frontend/src/explore/components/ControlHeader.stories.tsx @@ -0,0 +1,75 @@ +/** + * Licensed to the Apache Software Foundation (ASF) under one + * or more contributor license agreements. See the NOTICE file + * distributed with this work for additional information + * regarding copyright ownership. The ASF licenses this file + * to you under the Apache License, Version 2.0 (the + * "License"); you may not use this file except in compliance + * with the License. You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, + * software distributed under the License is distributed on an + * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY + * KIND, either express or implied. See the License for the + * specific language governing permissions and limitations + * under the License. + */ +import React from 'react'; +import ControlHeader, { ControlHeaderProps } from './ControlHeader'; + +export default { + title: 'ControlHeader', + component: ControlHeader, +}; + +const options: { + [key: string]: ControlHeaderProps; +} = { + label: { + label: 'Control label', + }, + warning: { + label: 'Control header', + warning: 'Example of warning message', + }, +}; + +export const ControlHeaderGallery = () => ( + <> + {Object.entries(options).map(([name, props]) => ( + <> +

{name}

+ + + ))} + +); + +export const InteractiveControlHeader = (props: ControlHeaderProps) => ( + <> + + +); + +InteractiveControlHeader.args = { + label: 'example label', + description: 'example description', + warning: 'example warning', + renderTrigger: false, + hovered: false, +}; + +InteractiveControlHeader.argTypes = { + tooltipOnClick: { action: 'tooltipOnClick' }, + onClick: { action: 'onClick' }, +}; + +InteractiveControlHeader.story = { + parameters: { + knobs: { + disable: true, + }, + }, +}; diff --git a/superset-frontend/src/explore/components/ControlHeader.tsx b/superset-frontend/src/explore/components/ControlHeader.tsx new file mode 100644 index 0000000000000..089893683051d --- /dev/null +++ b/superset-frontend/src/explore/components/ControlHeader.tsx @@ -0,0 +1,154 @@ +/** + * Licensed to the Apache Software Foundation (ASF) under one + * or more contributor license agreements. See the NOTICE file + * distributed with this work for additional information + * regarding copyright ownership. The ASF licenses this file + * to you under the Apache License, Version 2.0 (the + * "License"); you may not use this file except in compliance + * with the License. You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, + * software distributed under the License is distributed on an + * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY + * KIND, either express or implied. See the License for the + * specific language governing permissions and limitations + * under the License. + */ +import React, { FC, ReactNode } from 'react'; +import { t, css, useTheme, JsonObject } from '@superset-ui/core'; +import { InfoTooltipWithTrigger } from '@superset-ui/chart-controls'; +import { Tooltip } from 'src/components/Tooltip'; +import { FormLabel } from 'src/components/Form'; +import Icons from 'src/components/Icons'; + +export type ControlHeaderProps = { + name?: string; + label?: ReactNode; + description?: ReactNode; + validationErrors?: JsonObject | null; + renderTrigger?: boolean; + rightNode?: ReactNode; + leftNode?: ReactNode; + onClick?: () => {}; + hovered?: boolean; + tooltipOnClick?: () => {}; + warning?: string; + danger?: string; +}; + +const ControlHeader: FC = ({ + name, + label, + description, + validationErrors = [], + renderTrigger = false, + rightNode, + leftNode, + onClick, + hovered = false, + tooltipOnClick = () => {}, + warning, + danger, +}) => { + const { gridUnit, colors } = useTheme(); + + const renderOptionalIcons = () => { + if (!hovered) { + return null; + } + + return ( + css` + position: absolute; + top: 50%; + right: 0; + padding-left: ${gridUnit}px; + transform: translate(100%, -50%); + white-space: nowrap; + `} + > + {description && ( + + {' '} + + )} + {renderTrigger && ( + + {' '} + + )} + + ); + }; + if (!label) { + return null; + } + const labelClass = validationErrors?.length > 0 ? 'text-danger' : ''; + + return ( +
+
+ + {leftNode && {leftNode}} + + {label} + {' '} + {warning && ( + + + + {' '} + + )} + {danger && ( + + + + {' '} + + )} + {validationErrors?.length > 0 && ( + + + + {' '} + + )} + {renderOptionalIcons()} + +
+ {rightNode &&
{rightNode}
} +
+
+ ); +}; + +export default ControlHeader; From 199599c092b33336beb75ab77d67f44809b8d357 Mon Sep 17 00:00:00 2001 From: Adam Dobrawy Date: Thu, 27 Jan 2022 04:46:04 +0100 Subject: [PATCH 2/3] fix: Add story for error in ControlHeader --- .../src/explore/components/ControlHeader.stories.tsx | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/superset-frontend/src/explore/components/ControlHeader.stories.tsx b/superset-frontend/src/explore/components/ControlHeader.stories.tsx index f050f7cfe5691..e650c7fcfb94a 100644 --- a/superset-frontend/src/explore/components/ControlHeader.stories.tsx +++ b/superset-frontend/src/explore/components/ControlHeader.stories.tsx @@ -31,9 +31,13 @@ const options: { label: 'Control label', }, warning: { - label: 'Control header', + label: 'Control warning', warning: 'Example of warning message', }, + error: { + label: 'Control error', + validationErrors: ['Something is wrong'], + }, }; export const ControlHeaderGallery = () => ( From e543e3319b045e378f096a5c9bbccd7432c21056 Mon Sep 17 00:00:00 2001 From: Adam Dobrawy Date: Thu, 27 Jan 2022 19:51:26 +0100 Subject: [PATCH 3/3] apply review comments --- .../components/ControlHeader.stories.tsx | 4 +--- .../src/explore/components/ControlHeader.tsx | 18 +++++++++++------- 2 files changed, 12 insertions(+), 10 deletions(-) diff --git a/superset-frontend/src/explore/components/ControlHeader.stories.tsx b/superset-frontend/src/explore/components/ControlHeader.stories.tsx index e650c7fcfb94a..af9df9dece8d3 100644 --- a/superset-frontend/src/explore/components/ControlHeader.stories.tsx +++ b/superset-frontend/src/explore/components/ControlHeader.stories.tsx @@ -52,9 +52,7 @@ export const ControlHeaderGallery = () => ( ); export const InteractiveControlHeader = (props: ControlHeaderProps) => ( - <> - - + ); InteractiveControlHeader.args = { diff --git a/superset-frontend/src/explore/components/ControlHeader.tsx b/superset-frontend/src/explore/components/ControlHeader.tsx index 089893683051d..ce240704b5d3f 100644 --- a/superset-frontend/src/explore/components/ControlHeader.tsx +++ b/superset-frontend/src/explore/components/ControlHeader.tsx @@ -17,23 +17,25 @@ * under the License. */ import React, { FC, ReactNode } from 'react'; -import { t, css, useTheme, JsonObject } from '@superset-ui/core'; +import { t, css, useTheme } from '@superset-ui/core'; import { InfoTooltipWithTrigger } from '@superset-ui/chart-controls'; import { Tooltip } from 'src/components/Tooltip'; import { FormLabel } from 'src/components/Form'; import Icons from 'src/components/Icons'; +type ValidationError = string; + export type ControlHeaderProps = { name?: string; label?: ReactNode; description?: ReactNode; - validationErrors?: JsonObject | null; + validationErrors?: ValidationError[]; renderTrigger?: boolean; rightNode?: ReactNode; leftNode?: ReactNode; - onClick?: () => {}; + onClick?: () => void; hovered?: boolean; - tooltipOnClick?: () => {}; + tooltipOnClick?: () => void; warning?: string; danger?: string; }; @@ -54,6 +56,10 @@ const ControlHeader: FC = ({ }) => { const { gridUnit, colors } = useTheme(); + if (!label) { + return null; + } + const renderOptionalIcons = () => { if (!hovered) { return null; @@ -93,9 +99,7 @@ const ControlHeader: FC = ({ ); }; - if (!label) { - return null; - } + const labelClass = validationErrors?.length > 0 ? 'text-danger' : ''; return (