diff --git a/client/app/components/ParameterMappingInput.jsx b/client/app/components/ParameterMappingInput.jsx index 6eb1d41ff2..bb138f2b8e 100644 --- a/client/app/components/ParameterMappingInput.jsx +++ b/client/app/components/ParameterMappingInput.jsx @@ -1,6 +1,7 @@ /* eslint-disable react/no-multi-comp */ -import { isString, extend, each, map, includes, findIndex, find, fromPairs, clone, isEmpty } from 'lodash'; +import { isString, extend, each, has, map, includes, findIndex, find, + fromPairs, clone, isEmpty } from 'lodash'; import React, { Fragment } from 'react'; import PropTypes from 'prop-types'; import classNames from 'classnames'; @@ -158,6 +159,13 @@ export class ParameterMappingInput extends React.Component { newMapping.param = newMapping.param.clone(); newMapping.param.setValue(newMapping.value); } + if (has(update, 'type')) { + if (update.type === MappingType.StaticValue) { + newMapping.value = newMapping.param.value; + } else { + newMapping.value = null; + } + } onChange(newMapping); }; @@ -168,7 +176,7 @@ export class ParameterMappingInput extends React.Component { value={this.props.mapping.type} onChange={e => this.updateSourceType(e.target.value)} > - + New dashboard parameter ) : null } - + Widget parameter - + Static value @@ -335,7 +343,7 @@ class MappingEditor extends React.Component { const { mapping, inputError } = this.state; return ( -
+
Edit Source and Value
@@ -354,15 +362,16 @@ class MappingEditor extends React.Component { } render() { + const { visible, mapping } = this.state; return ( - diff --git a/client/app/components/ParameterValueInput.jsx b/client/app/components/ParameterValueInput.jsx index fa5b411091..e0ecfffcab 100644 --- a/client/app/components/ParameterValueInput.jsx +++ b/client/app/components/ParameterValueInput.jsx @@ -185,7 +185,7 @@ class ParameterValueInput extends React.Component { const { isDirty } = this.state; return ( -
+
{this.renderInput()}
); diff --git a/client/app/components/dashboards/dashboard-widget/VisualizationWidget.jsx b/client/app/components/dashboards/dashboard-widget/VisualizationWidget.jsx index b965d028d6..820dce3a7e 100644 --- a/client/app/components/dashboards/dashboard-widget/VisualizationWidget.jsx +++ b/client/app/components/dashboards/dashboard-widget/VisualizationWidget.jsx @@ -214,14 +214,14 @@ class VisualizationWidget extends React.Component { }; editParameterMappings = () => { - const { widget, dashboard, onParameterMappingsChange } = this.props; + const { widget, dashboard, onRefresh, onParameterMappingsChange } = this.props; EditParameterMappingsDialog.showModal({ dashboard, widget, }).result.then((valuesChanged) => { // refresh widget if any parameter value has been updated if (valuesChanged) { - this.refresh(); + onRefresh(); } onParameterMappingsChange(); this.setState({ localParameters: widget.getLocalParameters() }); diff --git a/client/app/pages/dashboards/dashboard.html b/client/app/pages/dashboards/dashboard.html index c594824840..114a07fd63 100644 --- a/client/app/pages/dashboards/dashboard.html +++ b/client/app/pages/dashboards/dashboard.html @@ -93,7 +93,7 @@

-
+
diff --git a/client/cypress/integration/dashboard/parameter_mapping_spec.js b/client/cypress/integration/dashboard/parameter_mapping_spec.js new file mode 100644 index 0000000000..23e46d05a1 --- /dev/null +++ b/client/cypress/integration/dashboard/parameter_mapping_spec.js @@ -0,0 +1,137 @@ +import { createDashboard } from '../../support/redash-api'; +import { createQueryAndAddWidget } from '../../support/dashboard'; + +describe('Parameter Mapping', () => { + beforeEach(function () { + cy.login(); + createDashboard('Foo Bar').then(({ slug, id }) => { + this.dashboardId = id; + this.dashboardUrl = `/dashboard/${slug}`; + }).then(() => { + const queryData = { + name: 'Text Parameter', + query: "SELECT '{{test-parameter}}' AS parameter", + options: { + parameters: [ + { name: 'test-parameter', title: 'Test Parameter', type: 'text', value: 'example' }, + ], + }, + }; + const widgetOptions = { position: { col: 0, row: 0, sizeX: 3, sizeY: 10, autoHeight: false } }; + createQueryAndAddWidget(this.dashboardId, queryData, widgetOptions).then((widgetTestId) => { + cy.visit(this.dashboardUrl); + this.widgetTestId = widgetTestId; + }); + }); + }); + + const openMappingOptions = (widgetTestId, paramName) => { + cy.getByTestId(widgetTestId) + .within(() => { + cy.getByTestId('WidgetDropdownButton') + .click(); + }); + + cy.getByTestId('WidgetDropdownButtonMenu') + .contains('Edit Parameters') + .click(); + + cy.getByTestId(`EditParamMappingButon-${paramName}`) + .click(); + }; + + const saveMappingOptions = () => { + cy.getByTestId('EditParamMappingPopover') + .within(() => { + cy.contains('button', 'OK') + .click(); + }); + + cy.contains('button', 'OK') + .click(); + }; + + it('supports widget parameters', function () { + // widget parameter mapping is the default for the API + cy.getByTestId(this.widgetTestId) + .within(() => { + cy.getByTestId('TableVisualization') + .should('contain', 'example'); + + cy.getByTestId('ParameterName-test-parameter') + .find('input') + .type('{selectall}Redash'); + + cy.getByTestId('ParameterApplyButton') + .click(); + + cy.getByTestId('TableVisualization') + .should('contain', 'Redash'); + }); + + cy.getByTestId('DashboardParameters') + .should('not.exist'); + }); + + it('supports dashboard parameters', function () { + openMappingOptions(this.widgetTestId, 'test-parameter'); + + cy.getByTestId('NewDashboardParameterOption') + .click(); + + saveMappingOptions(); + + cy.getByTestId(this.widgetTestId) + .within(() => { + cy.getByTestId('ParameterName-test-parameter') + .should('not.exist'); + }); + + cy.getByTestId('DashboardParameters') + .within(() => { + cy.getByTestId('ParameterName-test-parameter') + .find('input') + .type('{selectall}DashboardParam'); + + cy.getByTestId('ParameterApplyButton') + .click(); + }); + + cy.getByTestId(this.widgetTestId) + .within(() => { + cy.getByTestId('TableVisualization') + .should('contain', 'DashboardParam'); + }); + }); + + it('supports static values for parameters', function () { + openMappingOptions(this.widgetTestId, 'test-parameter'); + + cy.getByTestId('StaticValueOption') + .click(); + + cy.getByTestId('EditParamMappingPopover') + .within(() => { + cy.getByTestId('ParameterValueInput') + .find('input') + .type('{selectall}StaticValue'); + }); + + saveMappingOptions(); + + cy.getByTestId(this.widgetTestId) + .within(() => { + cy.getByTestId('ParameterName-test-parameter') + .should('not.exist'); + }); + + cy.getByTestId('DashboardParameters') + .should('not.exist'); + + cy.getByTestId(this.widgetTestId) + .within(() => { + cy.getByTestId('TableVisualization') + .should('contain', 'StaticValue'); + }); + }); +});