mirror of
https://github.com/apache/superset.git
synced 2024-09-17 11:09:47 -04:00
fix: Explore misleading save action (#24862)
This commit is contained in:
parent
f7e76d02b7
commit
bf1b1a4c46
@ -49,11 +49,6 @@ export function saveSliceSuccess(data) {
|
|||||||
return { type: SAVE_SLICE_SUCCESS, data };
|
return { type: SAVE_SLICE_SUCCESS, data };
|
||||||
}
|
}
|
||||||
|
|
||||||
export const REMOVE_SAVE_MODAL_ALERT = 'REMOVE_SAVE_MODAL_ALERT';
|
|
||||||
export function removeSaveModalAlert() {
|
|
||||||
return { type: REMOVE_SAVE_MODAL_ALERT };
|
|
||||||
}
|
|
||||||
|
|
||||||
const extractAddHocFiltersFromFormData = formDataToHandle =>
|
const extractAddHocFiltersFromFormData = formDataToHandle =>
|
||||||
Object.entries(formDataToHandle).reduce(
|
Object.entries(formDataToHandle).reduce(
|
||||||
(acc, [key, value]) =>
|
(acc, [key, value]) =>
|
||||||
|
@ -24,7 +24,6 @@ import { bindActionCreators } from 'redux';
|
|||||||
import { shallow } from 'enzyme';
|
import { shallow } from 'enzyme';
|
||||||
import { Radio } from 'src/components/Radio';
|
import { Radio } from 'src/components/Radio';
|
||||||
import Button from 'src/components/Button';
|
import Button from 'src/components/Button';
|
||||||
import sinon from 'sinon';
|
|
||||||
import fetchMock from 'fetch-mock';
|
import fetchMock from 'fetch-mock';
|
||||||
|
|
||||||
import * as saveModalActions from 'src/explore/actions/saveModalActions';
|
import * as saveModalActions from 'src/explore/actions/saveModalActions';
|
||||||
@ -131,7 +130,7 @@ test('renders a Modal with the right set of components', () => {
|
|||||||
expect(footerWrapper.find(Button)).toHaveLength(3);
|
expect(footerWrapper.find(Button)).toHaveLength(3);
|
||||||
});
|
});
|
||||||
|
|
||||||
test('renders the right footer buttons when existing dashboard selected', () => {
|
test('renders the right footer buttons', () => {
|
||||||
const wrapper = getWrapper();
|
const wrapper = getWrapper();
|
||||||
const footerWrapper = shallow(wrapper.find(StyledModal).props().footer);
|
const footerWrapper = shallow(wrapper.find(StyledModal).props().footer);
|
||||||
const saveAndGoDash = footerWrapper
|
const saveAndGoDash = footerWrapper
|
||||||
@ -142,18 +141,43 @@ test('renders the right footer buttons when existing dashboard selected', () =>
|
|||||||
expect(saveAndGoDash.props.children).toBe('Save & go to dashboard');
|
expect(saveAndGoDash.props.children).toBe('Save & go to dashboard');
|
||||||
});
|
});
|
||||||
|
|
||||||
test('renders the right footer buttons when new dashboard selected', () => {
|
test('does not render a message when overriding', () => {
|
||||||
|
const wrapper = getWrapper();
|
||||||
|
wrapper.setState({
|
||||||
|
action: 'overwrite',
|
||||||
|
});
|
||||||
|
expect(
|
||||||
|
wrapper.find('[message="A new chart will be created."]'),
|
||||||
|
).not.toExist();
|
||||||
|
});
|
||||||
|
|
||||||
|
test('renders a message when saving as', () => {
|
||||||
|
const wrapper = getWrapper();
|
||||||
|
wrapper.setState({
|
||||||
|
action: 'saveas',
|
||||||
|
});
|
||||||
|
expect(wrapper.find('[message="A new chart will be created."]')).toExist();
|
||||||
|
});
|
||||||
|
|
||||||
|
test('renders a message when a new dashboard is selected', () => {
|
||||||
const wrapper = getWrapper();
|
const wrapper = getWrapper();
|
||||||
wrapper.setState({
|
wrapper.setState({
|
||||||
dashboard: { label: 'Test new dashboard', value: 'Test new dashboard' },
|
dashboard: { label: 'Test new dashboard', value: 'Test new dashboard' },
|
||||||
});
|
});
|
||||||
const footerWrapper = shallow(wrapper.find(StyledModal).props().footer);
|
expect(
|
||||||
const saveAndGoDash = footerWrapper
|
wrapper.find('[message="A new dashboard will be created."]'),
|
||||||
.find('#btn_modal_save_goto_dash')
|
).toExist();
|
||||||
.getElement();
|
});
|
||||||
const save = footerWrapper.find('#btn_modal_save').getElement();
|
|
||||||
expect(save.props.children).toBe('Save to new dashboard');
|
test('renders a message when saving as with new dashboard', () => {
|
||||||
expect(saveAndGoDash.props.children).toBe('Save & go to new dashboard');
|
const wrapper = getWrapper();
|
||||||
|
wrapper.setState({
|
||||||
|
action: 'saveas',
|
||||||
|
dashboard: { label: 'Test new dashboard', value: 'Test new dashboard' },
|
||||||
|
});
|
||||||
|
expect(
|
||||||
|
wrapper.find('[message="A new chart and dashboard will be created."]'),
|
||||||
|
).toExist();
|
||||||
});
|
});
|
||||||
|
|
||||||
test('disables overwrite option for new slice', () => {
|
test('disables overwrite option for new slice', () => {
|
||||||
@ -197,17 +221,6 @@ test('updates slice name and selected dashboard', () => {
|
|||||||
expect(wrapper.state().dashboard.value).toBe(dashboardId);
|
expect(wrapper.state().dashboard.value).toBe(dashboardId);
|
||||||
});
|
});
|
||||||
|
|
||||||
test('removes alert', () => {
|
|
||||||
sinon.spy(defaultProps.actions, 'removeSaveModalAlert');
|
|
||||||
const wrapper = getWrapper();
|
|
||||||
wrapper.setProps({ alert: 'old alert' });
|
|
||||||
|
|
||||||
wrapper.instance().removeAlert();
|
|
||||||
expect(defaultProps.actions.removeSaveModalAlert.callCount).toBe(1);
|
|
||||||
expect(wrapper.state().alert).toBeNull();
|
|
||||||
defaultProps.actions.removeSaveModalAlert.restore();
|
|
||||||
});
|
|
||||||
|
|
||||||
test('set dataset name when chart source is query', () => {
|
test('set dataset name when chart source is query', () => {
|
||||||
const wrapper = getWrapper(queryDefaultProps, queryStore);
|
const wrapper = getWrapper(queryDefaultProps, queryStore);
|
||||||
expect(wrapper.find('[data-test="new-dataset-name"]')).toExist();
|
expect(wrapper.find('[data-test="new-dataset-name"]')).toExist();
|
||||||
|
@ -67,7 +67,6 @@ interface SaveModalProps extends RouteComponentProps {
|
|||||||
type SaveModalState = {
|
type SaveModalState = {
|
||||||
newSliceName?: string;
|
newSliceName?: string;
|
||||||
datasetName: string;
|
datasetName: string;
|
||||||
alert: string | null;
|
|
||||||
action: SaveActionType;
|
action: SaveActionType;
|
||||||
isLoading: boolean;
|
isLoading: boolean;
|
||||||
saveStatus?: string | null;
|
saveStatus?: string | null;
|
||||||
@ -92,7 +91,6 @@ class SaveModal extends React.Component<SaveModalProps, SaveModalState> {
|
|||||||
this.state = {
|
this.state = {
|
||||||
newSliceName: props.sliceName,
|
newSliceName: props.sliceName,
|
||||||
datasetName: props.datasource?.name,
|
datasetName: props.datasource?.name,
|
||||||
alert: null,
|
|
||||||
action: this.canOverwriteSlice() ? 'overwrite' : 'saveas',
|
action: this.canOverwriteSlice() ? 'overwrite' : 'saveas',
|
||||||
isLoading: false,
|
isLoading: false,
|
||||||
vizType: props.form_data?.viz_type,
|
vizType: props.form_data?.viz_type,
|
||||||
@ -103,7 +101,6 @@ class SaveModal extends React.Component<SaveModalProps, SaveModalState> {
|
|||||||
this.changeAction = this.changeAction.bind(this);
|
this.changeAction = this.changeAction.bind(this);
|
||||||
this.saveOrOverwrite = this.saveOrOverwrite.bind(this);
|
this.saveOrOverwrite = this.saveOrOverwrite.bind(this);
|
||||||
this.isNewDashboard = this.isNewDashboard.bind(this);
|
this.isNewDashboard = this.isNewDashboard.bind(this);
|
||||||
this.removeAlert = this.removeAlert.bind(this);
|
|
||||||
this.onHide = this.onHide.bind(this);
|
this.onHide = this.onHide.bind(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -163,8 +160,7 @@ class SaveModal extends React.Component<SaveModalProps, SaveModalState> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
async saveOrOverwrite(gotodash: boolean) {
|
async saveOrOverwrite(gotodash: boolean) {
|
||||||
this.setState({ alert: null, isLoading: true });
|
this.setState({ isLoading: true });
|
||||||
this.props.actions.removeSaveModalAlert();
|
|
||||||
|
|
||||||
// Create or retrieve dashboard
|
// Create or retrieve dashboard
|
||||||
type DashboardGetResponse = {
|
type DashboardGetResponse = {
|
||||||
@ -324,15 +320,10 @@ class SaveModal extends React.Component<SaveModalProps, SaveModalState> {
|
|||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
renderSaveChartModal = () => (
|
renderSaveChartModal = () => {
|
||||||
|
const info = this.info();
|
||||||
|
return (
|
||||||
<Form data-test="save-modal-body" layout="vertical">
|
<Form data-test="save-modal-body" layout="vertical">
|
||||||
{(this.state.alert || this.props.alert) && (
|
|
||||||
<Alert
|
|
||||||
type="warning"
|
|
||||||
message={this.state.alert || this.props.alert}
|
|
||||||
onClose={this.removeAlert}
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
<FormItem data-test="radio-group">
|
<FormItem data-test="radio-group">
|
||||||
<Radio
|
<Radio
|
||||||
id="overwrite-radio"
|
id="overwrite-radio"
|
||||||
@ -405,8 +396,39 @@ class SaveModal extends React.Component<SaveModalProps, SaveModalState> {
|
|||||||
/>
|
/>
|
||||||
</FormItem>
|
</FormItem>
|
||||||
)}
|
)}
|
||||||
|
{info && <Alert type="info" message={info} closable={false} />}
|
||||||
|
{this.props.alert && (
|
||||||
|
<Alert
|
||||||
|
css={{ marginTop: info ? 16 : undefined }}
|
||||||
|
type="warning"
|
||||||
|
message={this.props.alert}
|
||||||
|
closable={false}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
</Form>
|
</Form>
|
||||||
);
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
info = () => {
|
||||||
|
const isNewDashboard = this.isNewDashboard();
|
||||||
|
let chartWillBeCreated = false;
|
||||||
|
if (
|
||||||
|
this.props.slice &&
|
||||||
|
(this.state.action !== 'overwrite' || !this.canOverwriteSlice())
|
||||||
|
) {
|
||||||
|
chartWillBeCreated = true;
|
||||||
|
}
|
||||||
|
if (chartWillBeCreated && isNewDashboard) {
|
||||||
|
return t('A new chart and dashboard will be created.');
|
||||||
|
}
|
||||||
|
if (chartWillBeCreated) {
|
||||||
|
return t('A new chart will be created.');
|
||||||
|
}
|
||||||
|
if (isNewDashboard) {
|
||||||
|
return t('A new dashboard will be created.');
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
};
|
||||||
|
|
||||||
renderFooter = () => (
|
renderFooter = () => (
|
||||||
<div data-test="save-modal-footer">
|
<div data-test="save-modal-footer">
|
||||||
@ -426,9 +448,7 @@ class SaveModal extends React.Component<SaveModalProps, SaveModalState> {
|
|||||||
}
|
}
|
||||||
onClick={() => this.saveOrOverwrite(true)}
|
onClick={() => this.saveOrOverwrite(true)}
|
||||||
>
|
>
|
||||||
{this.isNewDashboard()
|
{t('Save & go to dashboard')}
|
||||||
? t('Save & go to new dashboard')
|
|
||||||
: t('Save & go to dashboard')}
|
|
||||||
</Button>
|
</Button>
|
||||||
<Button
|
<Button
|
||||||
id="btn_modal_save"
|
id="btn_modal_save"
|
||||||
@ -443,22 +463,11 @@ class SaveModal extends React.Component<SaveModalProps, SaveModalState> {
|
|||||||
}
|
}
|
||||||
data-test="btn-modal-save"
|
data-test="btn-modal-save"
|
||||||
>
|
>
|
||||||
{!this.canOverwriteSlice() && this.props.slice
|
{t('Save')}
|
||||||
? t('Save as new chart')
|
|
||||||
: this.isNewDashboard()
|
|
||||||
? t('Save to new dashboard')
|
|
||||||
: t('Save')}
|
|
||||||
</Button>
|
</Button>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
|
|
||||||
removeAlert() {
|
|
||||||
if (this.props.alert) {
|
|
||||||
this.props.actions.removeSaveModalAlert();
|
|
||||||
}
|
|
||||||
this.setState({ alert: null });
|
|
||||||
}
|
|
||||||
|
|
||||||
render() {
|
render() {
|
||||||
return (
|
return (
|
||||||
<StyledModal
|
<StyledModal
|
||||||
|
@ -40,9 +40,6 @@ export default function saveModalReducer(state = {}, action) {
|
|||||||
[actions.SAVE_SLICE_SUCCESS](data) {
|
[actions.SAVE_SLICE_SUCCESS](data) {
|
||||||
return { ...state, data };
|
return { ...state, data };
|
||||||
},
|
},
|
||||||
[actions.REMOVE_SAVE_MODAL_ALERT]() {
|
|
||||||
return { ...state, saveModalAlert: null };
|
|
||||||
},
|
|
||||||
[HYDRATE_EXPLORE]() {
|
[HYDRATE_EXPLORE]() {
|
||||||
return { ...action.data.saveModal };
|
return { ...action.data.saveModal };
|
||||||
},
|
},
|
||||||
|
Loading…
Reference in New Issue
Block a user