358 lines
11 KiB
TypeScript
358 lines
11 KiB
TypeScript
/**
|
|
* 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.
|
|
*/
|
|
/* eslint-disable camelcase */
|
|
import React, {
|
|
useCallback,
|
|
useEffect,
|
|
useMemo,
|
|
useRef,
|
|
useState,
|
|
} from 'react';
|
|
import { AdhocColumn, isAdhocColumn, t, styled, css } from '@superset-ui/core';
|
|
import { ColumnMeta, isSavedExpression } from '@superset-ui/chart-controls';
|
|
import Tabs from 'src/components/Tabs';
|
|
import Button from 'src/components/Button';
|
|
import { Select } from 'src/components';
|
|
|
|
import { Form, FormItem } from 'src/components/Form';
|
|
import { SQLEditor } from 'src/components/AsyncAceEditor';
|
|
import { EmptyStateSmall } from 'src/components/EmptyState';
|
|
import { StyledColumnOption } from 'src/explore/components/optionRenderers';
|
|
import {
|
|
POPOVER_INITIAL_HEIGHT,
|
|
UNRESIZABLE_POPOVER_WIDTH,
|
|
} from 'src/explore/constants';
|
|
|
|
const StyledSelect = styled(Select)`
|
|
.metric-option {
|
|
& > svg {
|
|
min-width: ${({ theme }) => `${theme.gridUnit * 4}px`};
|
|
}
|
|
& > .option-label {
|
|
overflow: hidden;
|
|
text-overflow: ellipsis;
|
|
}
|
|
}
|
|
`;
|
|
|
|
interface ColumnSelectPopoverProps {
|
|
columns: ColumnMeta[];
|
|
editedColumn?: ColumnMeta | AdhocColumn;
|
|
onChange: (column: ColumnMeta | AdhocColumn) => void;
|
|
onClose: () => void;
|
|
setLabel: (title: string) => void;
|
|
getCurrentTab: (tab: string) => void;
|
|
label: string;
|
|
isTemporal?: boolean;
|
|
}
|
|
|
|
const getInitialColumnValues = (
|
|
editedColumn?: ColumnMeta | AdhocColumn,
|
|
): [AdhocColumn?, ColumnMeta?, ColumnMeta?] => {
|
|
if (!editedColumn) {
|
|
return [undefined, undefined, undefined];
|
|
}
|
|
if (isAdhocColumn(editedColumn)) {
|
|
return [editedColumn, undefined, undefined];
|
|
}
|
|
if (isSavedExpression(editedColumn)) {
|
|
return [undefined, editedColumn, undefined];
|
|
}
|
|
return [undefined, undefined, editedColumn];
|
|
};
|
|
|
|
const ColumnSelectPopover = ({
|
|
columns,
|
|
editedColumn,
|
|
onChange,
|
|
onClose,
|
|
setLabel,
|
|
getCurrentTab,
|
|
label,
|
|
isTemporal,
|
|
}: ColumnSelectPopoverProps) => {
|
|
const [initialLabel] = useState(label);
|
|
const [initialAdhocColumn, initialCalculatedColumn, initialSimpleColumn] =
|
|
getInitialColumnValues(editedColumn);
|
|
|
|
const [adhocColumn, setAdhocColumn] = useState<AdhocColumn | undefined>(
|
|
initialAdhocColumn,
|
|
);
|
|
const [selectedCalculatedColumn, setSelectedCalculatedColumn] = useState<
|
|
ColumnMeta | undefined
|
|
>(initialCalculatedColumn);
|
|
const [selectedSimpleColumn, setSelectedSimpleColumn] = useState<
|
|
ColumnMeta | undefined
|
|
>(initialSimpleColumn);
|
|
|
|
const sqlEditorRef = useRef(null);
|
|
|
|
const [calculatedColumns, simpleColumns] = useMemo(
|
|
() =>
|
|
columns?.reduce(
|
|
(acc: [ColumnMeta[], ColumnMeta[]], column: ColumnMeta) => {
|
|
if (column.expression) {
|
|
acc[0].push(column);
|
|
} else {
|
|
acc[1].push(column);
|
|
}
|
|
return acc;
|
|
},
|
|
[[], []],
|
|
),
|
|
[columns],
|
|
);
|
|
|
|
const onSqlExpressionChange = useCallback(
|
|
sqlExpression => {
|
|
setAdhocColumn({ label, sqlExpression, expressionType: 'SQL' });
|
|
setSelectedSimpleColumn(undefined);
|
|
setSelectedCalculatedColumn(undefined);
|
|
},
|
|
[label],
|
|
);
|
|
|
|
const onCalculatedColumnChange = useCallback(
|
|
selectedColumnName => {
|
|
const selectedColumn = calculatedColumns.find(
|
|
col => col.column_name === selectedColumnName,
|
|
);
|
|
setSelectedCalculatedColumn(selectedColumn);
|
|
setSelectedSimpleColumn(undefined);
|
|
setAdhocColumn(undefined);
|
|
setLabel(
|
|
selectedColumn?.verbose_name || selectedColumn?.column_name || '',
|
|
);
|
|
},
|
|
[calculatedColumns, setLabel],
|
|
);
|
|
|
|
const onSimpleColumnChange = useCallback(
|
|
selectedColumnName => {
|
|
const selectedColumn = simpleColumns.find(
|
|
col => col.column_name === selectedColumnName,
|
|
);
|
|
setSelectedCalculatedColumn(undefined);
|
|
setSelectedSimpleColumn(selectedColumn);
|
|
setAdhocColumn(undefined);
|
|
setLabel(
|
|
selectedColumn?.verbose_name || selectedColumn?.column_name || '',
|
|
);
|
|
},
|
|
[setLabel, simpleColumns],
|
|
);
|
|
|
|
const defaultActiveTabKey = initialAdhocColumn
|
|
? 'sqlExpression'
|
|
: initialSimpleColumn || calculatedColumns.length === 0
|
|
? 'simple'
|
|
: 'saved';
|
|
|
|
useEffect(() => {
|
|
getCurrentTab(defaultActiveTabKey);
|
|
}, [defaultActiveTabKey, getCurrentTab]);
|
|
|
|
const onSave = useCallback(() => {
|
|
if (adhocColumn && adhocColumn.label !== label) {
|
|
adhocColumn.label = label;
|
|
}
|
|
const selectedColumn =
|
|
adhocColumn || selectedCalculatedColumn || selectedSimpleColumn;
|
|
if (!selectedColumn) {
|
|
return;
|
|
}
|
|
onChange(selectedColumn);
|
|
onClose();
|
|
}, [
|
|
adhocColumn,
|
|
label,
|
|
onChange,
|
|
onClose,
|
|
selectedCalculatedColumn,
|
|
selectedSimpleColumn,
|
|
]);
|
|
|
|
const onResetStateAndClose = useCallback(() => {
|
|
setSelectedCalculatedColumn(initialCalculatedColumn);
|
|
setSelectedSimpleColumn(initialSimpleColumn);
|
|
setAdhocColumn(initialAdhocColumn);
|
|
onClose();
|
|
}, [
|
|
initialAdhocColumn,
|
|
initialCalculatedColumn,
|
|
initialSimpleColumn,
|
|
onClose,
|
|
]);
|
|
|
|
const onTabChange = useCallback(
|
|
tab => {
|
|
getCurrentTab(tab);
|
|
// @ts-ignore
|
|
sqlEditorRef.current?.editor.focus();
|
|
},
|
|
[getCurrentTab],
|
|
);
|
|
|
|
const onSqlEditorFocus = useCallback(() => {
|
|
// @ts-ignore
|
|
sqlEditorRef.current?.editor.resize();
|
|
}, []);
|
|
|
|
const stateIsValid =
|
|
adhocColumn || selectedCalculatedColumn || selectedSimpleColumn;
|
|
const hasUnsavedChanges =
|
|
initialLabel !== label ||
|
|
selectedCalculatedColumn?.column_name !==
|
|
initialCalculatedColumn?.column_name ||
|
|
selectedSimpleColumn?.column_name !== initialSimpleColumn?.column_name ||
|
|
adhocColumn?.sqlExpression !== initialAdhocColumn?.sqlExpression;
|
|
|
|
const savedExpressionsLabel = t('Saved expressions');
|
|
const simpleColumnsLabel = t('Column');
|
|
|
|
return (
|
|
<Form layout="vertical" id="metrics-edit-popover">
|
|
<Tabs
|
|
id="adhoc-metric-edit-tabs"
|
|
defaultActiveKey={defaultActiveTabKey}
|
|
onChange={onTabChange}
|
|
className="adhoc-metric-edit-tabs"
|
|
allowOverflow
|
|
css={css`
|
|
height: ${POPOVER_INITIAL_HEIGHT}px;
|
|
width: ${UNRESIZABLE_POPOVER_WIDTH}px;
|
|
`}
|
|
>
|
|
<Tabs.TabPane key="saved" tab={t('Saved')}>
|
|
{calculatedColumns.length > 0 ? (
|
|
<FormItem label={savedExpressionsLabel}>
|
|
<StyledSelect
|
|
ariaLabel={savedExpressionsLabel}
|
|
value={selectedCalculatedColumn?.column_name}
|
|
onChange={onCalculatedColumnChange}
|
|
allowClear
|
|
autoFocus={!selectedCalculatedColumn}
|
|
placeholder={t('%s column(s)', calculatedColumns.length)}
|
|
options={calculatedColumns.map(calculatedColumn => ({
|
|
value: calculatedColumn.column_name,
|
|
label:
|
|
calculatedColumn.verbose_name ||
|
|
calculatedColumn.column_name,
|
|
customLabel: (
|
|
<StyledColumnOption column={calculatedColumn} showType />
|
|
),
|
|
key: calculatedColumn.column_name,
|
|
}))}
|
|
/>
|
|
</FormItem>
|
|
) : (
|
|
<EmptyStateSmall
|
|
image="empty.svg"
|
|
title={
|
|
isTemporal
|
|
? t('No temporal columns found')
|
|
: t('No saved expressions found')
|
|
}
|
|
description={
|
|
isTemporal
|
|
? t(
|
|
'Add calculated temporal columns to dataset in "Edit datasource" modal',
|
|
)
|
|
: t(
|
|
'Add calculated columns to dataset in "Edit datasource" modal',
|
|
)
|
|
}
|
|
/>
|
|
)}
|
|
</Tabs.TabPane>
|
|
<Tabs.TabPane key="simple" tab={t('Simple')}>
|
|
{isTemporal && simpleColumns.length === 0 ? (
|
|
<EmptyStateSmall
|
|
image="empty.svg"
|
|
title={t('No temporal columns found')}
|
|
description={t(
|
|
'Mark a column as temporal in "Edit datasource" modal',
|
|
)}
|
|
/>
|
|
) : (
|
|
<FormItem label={simpleColumnsLabel}>
|
|
<Select
|
|
ariaLabel={simpleColumnsLabel}
|
|
value={selectedSimpleColumn?.column_name}
|
|
onChange={onSimpleColumnChange}
|
|
allowClear
|
|
autoFocus={!selectedSimpleColumn}
|
|
placeholder={t('%s column(s)', simpleColumns.length)}
|
|
options={simpleColumns.map(simpleColumn => ({
|
|
value: simpleColumn.column_name,
|
|
label: simpleColumn.verbose_name || simpleColumn.column_name,
|
|
customLabel: (
|
|
<StyledColumnOption column={simpleColumn} showType />
|
|
),
|
|
key: simpleColumn.column_name,
|
|
}))}
|
|
/>
|
|
</FormItem>
|
|
)}
|
|
</Tabs.TabPane>
|
|
|
|
<Tabs.TabPane key="sqlExpression" tab={t('Custom SQL')}>
|
|
<SQLEditor
|
|
value={
|
|
adhocColumn?.sqlExpression ||
|
|
selectedSimpleColumn?.column_name ||
|
|
selectedCalculatedColumn?.expression
|
|
}
|
|
onFocus={onSqlEditorFocus}
|
|
showLoadingForImport
|
|
onChange={onSqlExpressionChange}
|
|
width="100%"
|
|
height={`${POPOVER_INITIAL_HEIGHT - 80}px`}
|
|
showGutter={false}
|
|
editorProps={{ $blockScrolling: true }}
|
|
enableLiveAutocompletion
|
|
className="filter-sql-editor"
|
|
wrapEnabled
|
|
ref={sqlEditorRef}
|
|
/>
|
|
</Tabs.TabPane>
|
|
</Tabs>
|
|
<div>
|
|
<Button buttonSize="small" onClick={onResetStateAndClose} cta>
|
|
{t('Close')}
|
|
</Button>
|
|
<Button
|
|
disabled={!stateIsValid}
|
|
buttonStyle={
|
|
hasUnsavedChanges && stateIsValid ? 'primary' : 'default'
|
|
}
|
|
buttonSize="small"
|
|
onClick={onSave}
|
|
data-test="ColumnEdit#save"
|
|
cta
|
|
>
|
|
{t('Save')}
|
|
</Button>
|
|
</div>
|
|
</Form>
|
|
);
|
|
};
|
|
|
|
export default ColumnSelectPopover;
|