mirror of
https://github.com/getredash/redash.git
synced 2025-12-25 01:03:20 -05:00
* Restyled by autopep8 * Restyled by black * Restyled by clang-format * Restyled by isort * Restyled by prettier * Restyled by reorder-python-imports * Restyled by whitespace * Restyled by yapf --------- Co-authored-by: Restyled.io <commits@restyled.io>
262 lines
7.5 KiB
JavaScript
262 lines
7.5 KiB
JavaScript
import React from "react";
|
|
import PropTypes from "prop-types";
|
|
import { size, filter, forEach, extend, get, includes } from "lodash";
|
|
import { react2angular } from "react2angular";
|
|
import {
|
|
SortableContainer,
|
|
SortableElement,
|
|
DragHandle,
|
|
} from "@/components/sortable";
|
|
import { $location } from "@/services/ng";
|
|
import { Parameter } from "@/services/parameters";
|
|
import ParameterApplyButton from "@/components/ParameterApplyButton";
|
|
import ParameterValueInput from "@/components/ParameterValueInput";
|
|
import Form from "antd/lib/form";
|
|
import Tooltip from "antd/lib/tooltip";
|
|
import EditParameterSettingsDialog from "./EditParameterSettingsDialog";
|
|
import { toHuman } from "@/filters";
|
|
|
|
import "./Parameters.less";
|
|
|
|
function updateUrl(parameters) {
|
|
const params = extend({}, $location.search());
|
|
parameters.forEach((param) => {
|
|
extend(params, param.toUrlParams());
|
|
});
|
|
Object.keys(params).forEach(
|
|
(key) => params[key] == null && delete params[key]
|
|
);
|
|
$location.search(params);
|
|
}
|
|
|
|
export class Parameters extends React.Component {
|
|
static propTypes = {
|
|
parameters: PropTypes.arrayOf(PropTypes.instanceOf(Parameter)),
|
|
editable: PropTypes.bool,
|
|
disableUrlUpdate: PropTypes.bool,
|
|
onValuesChange: PropTypes.func,
|
|
onPendingValuesChange: PropTypes.func,
|
|
onParametersEdit: PropTypes.func,
|
|
queryResultErrorData: PropTypes.shape({
|
|
parameters: PropTypes.objectOf(PropTypes.string),
|
|
}),
|
|
unsavedParameters: PropTypes.arrayOf(PropTypes.string),
|
|
};
|
|
|
|
static defaultProps = {
|
|
parameters: [],
|
|
editable: false,
|
|
disableUrlUpdate: false,
|
|
onValuesChange: () => {},
|
|
onPendingValuesChange: () => {},
|
|
onParametersEdit: () => {},
|
|
queryResultErrorData: {},
|
|
unsavedParameters: null,
|
|
};
|
|
|
|
constructor(props) {
|
|
super(props);
|
|
const { parameters } = props;
|
|
this.state = {
|
|
parameters,
|
|
touched: {},
|
|
};
|
|
|
|
if (!props.disableUrlUpdate) {
|
|
updateUrl(parameters);
|
|
}
|
|
}
|
|
|
|
componentDidUpdate = (prevProps) => {
|
|
const { parameters, disableUrlUpdate, queryResultErrorData } = this.props;
|
|
if (prevProps.parameters !== parameters) {
|
|
this.setState({ parameters });
|
|
if (!disableUrlUpdate) {
|
|
updateUrl(parameters);
|
|
}
|
|
}
|
|
|
|
// reset touched flags on new error data
|
|
if (prevProps.queryResultErrorData !== queryResultErrorData) {
|
|
this.setState({ touched: {} });
|
|
}
|
|
};
|
|
|
|
handleKeyDown = (e) => {
|
|
// Cmd/Ctrl/Alt + Enter
|
|
if (e.keyCode === 13 && (e.ctrlKey || e.metaKey || e.altKey)) {
|
|
e.stopPropagation();
|
|
this.applyChanges();
|
|
}
|
|
};
|
|
|
|
setPendingValue = (param, value, isDirty) => {
|
|
const { onPendingValuesChange } = this.props;
|
|
this.setState(({ parameters, touched }) => {
|
|
if (isDirty) {
|
|
param.setPendingValue(value);
|
|
touched = { ...touched, [param.name]: true };
|
|
} else {
|
|
param.clearPendingValue();
|
|
}
|
|
onPendingValuesChange();
|
|
return { parameters, touched };
|
|
});
|
|
};
|
|
|
|
moveParameter = ({ oldIndex, newIndex }) => {
|
|
const { onParametersEdit } = this.props;
|
|
if (oldIndex !== newIndex) {
|
|
this.setState(({ parameters }) => {
|
|
parameters.splice(newIndex, 0, parameters.splice(oldIndex, 1)[0]);
|
|
onParametersEdit();
|
|
return { parameters };
|
|
});
|
|
}
|
|
};
|
|
|
|
applyChanges = () => {
|
|
const { onValuesChange, disableUrlUpdate } = this.props;
|
|
this.setState(({ parameters }) => {
|
|
const parametersWithPendingValues = parameters.filter(
|
|
(p) => p.hasPendingValue
|
|
);
|
|
forEach(parameters, (p) => p.applyPendingValue());
|
|
if (!disableUrlUpdate) {
|
|
updateUrl(parameters);
|
|
}
|
|
onValuesChange(parametersWithPendingValues);
|
|
return { parameters };
|
|
});
|
|
};
|
|
|
|
showParameterSettings = (parameter, index) => {
|
|
const { onParametersEdit } = this.props;
|
|
EditParameterSettingsDialog.showModal({ parameter }).result.then(
|
|
(updated) => {
|
|
this.setState(({ parameters, touched }) => {
|
|
touched = { ...touched, [parameter.name]: true };
|
|
const updatedParameter = extend(parameter, updated);
|
|
parameters[index] = Parameter.create(
|
|
updatedParameter,
|
|
updatedParameter.parentQueryId
|
|
);
|
|
onParametersEdit();
|
|
return { parameters, touched };
|
|
});
|
|
}
|
|
);
|
|
};
|
|
|
|
getParameterFeedback = (param) => {
|
|
// error msg
|
|
const { queryResultErrorData } = this.props;
|
|
const error = get(queryResultErrorData, ["parameters", param.name], false);
|
|
if (error) {
|
|
const feedback = <Tooltip title={error}>{error}</Tooltip>;
|
|
return [feedback, "error"];
|
|
}
|
|
|
|
// unsaved
|
|
const { unsavedParameters } = this.props;
|
|
if (includes(unsavedParameters, param.name)) {
|
|
const feedback = (
|
|
<>
|
|
Unsaved{" "}
|
|
<Tooltip title='Click the "Save" button to preserve this parameter.'>
|
|
<i className="fa fa-question-circle" />
|
|
</Tooltip>
|
|
</>
|
|
);
|
|
return [feedback, "warning"];
|
|
}
|
|
|
|
return [];
|
|
};
|
|
|
|
renderParameter(param, index) {
|
|
const { editable } = this.props;
|
|
const touched = this.state.touched[param.name];
|
|
const [feedback, status] = this.getParameterFeedback(param);
|
|
|
|
return (
|
|
<div
|
|
key={param.name}
|
|
className="di-block"
|
|
data-test={`ParameterName-${param.name}`}
|
|
>
|
|
<div className="parameter-heading">
|
|
<label>{param.title || toHuman(param.name)}</label>
|
|
{editable && (
|
|
<button
|
|
className="btn btn-default btn-xs m-l-5"
|
|
onClick={() => this.showParameterSettings(param, index)}
|
|
data-test={`ParameterSettings-${param.name}`}
|
|
type="button"
|
|
>
|
|
<i className="fa fa-cog" />
|
|
</button>
|
|
)}
|
|
</div>
|
|
<Form.Item
|
|
validateStatus={touched ? "" : status}
|
|
help={feedback || null}
|
|
>
|
|
<ParameterValueInput
|
|
type={param.type}
|
|
value={param.normalizedValue}
|
|
parameter={param}
|
|
enumOptions={param.enumOptions}
|
|
queryId={param.queryId}
|
|
onSelect={(value, isDirty) =>
|
|
this.setPendingValue(param, value, isDirty)
|
|
}
|
|
/>
|
|
</Form.Item>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
render() {
|
|
const { parameters } = this.state;
|
|
const { editable } = this.props;
|
|
const dirtyParamCount = size(filter(parameters, "hasPendingValue"));
|
|
return (
|
|
<SortableContainer
|
|
disabled={!editable}
|
|
axis="xy"
|
|
useDragHandle
|
|
lockToContainerEdges
|
|
helperClass="parameter-dragged"
|
|
updateBeforeSortStart={this.onBeforeSortStart}
|
|
onSortEnd={this.moveParameter}
|
|
containerProps={{
|
|
className: "parameter-container",
|
|
onKeyDown: dirtyParamCount ? this.handleKeyDown : null,
|
|
}}
|
|
>
|
|
{parameters.map((param, index) => (
|
|
<SortableElement key={param.name} index={index}>
|
|
<div className="parameter-block" data-editable={editable || null}>
|
|
{editable && (
|
|
<DragHandle data-test={`DragHandle-${param.name}`} />
|
|
)}
|
|
{this.renderParameter(param, index)}
|
|
</div>
|
|
</SortableElement>
|
|
))}
|
|
<ParameterApplyButton
|
|
onClick={this.applyChanges}
|
|
paramCount={dirtyParamCount}
|
|
/>
|
|
</SortableContainer>
|
|
);
|
|
}
|
|
}
|
|
|
|
export default function init(ngModule) {
|
|
ngModule.component("parameters", react2angular(Parameters));
|
|
}
|
|
|
|
init.init = true;
|