// Libraries import React, { PureComponent, ChangeEvent } from 'react'; import { css } from 'emotion'; import { InlineField, InlineFieldRow, Input, Select, TextArea } from '@grafana/ui'; import { SelectableValue, ReducerID, QueryEditorProps } from '@grafana/data'; // Types import { ExpressionQuery, GELQueryType } from './types'; import { ExpressionDatasourceApi } from './ExpressionDatasource'; type Props = QueryEditorProps; interface State {} const gelTypes: Array> = [ { value: GELQueryType.math, label: 'Math' }, { value: GELQueryType.reduce, label: 'Reduce' }, { value: GELQueryType.resample, label: 'Resample' }, ]; const reducerTypes: Array> = [ { value: ReducerID.min, label: 'Min', description: 'Get the minimum value' }, { value: ReducerID.max, label: 'Max', description: 'Get the maximum value' }, { value: ReducerID.mean, label: 'Mean', description: 'Get the average value' }, { value: ReducerID.sum, label: 'Sum', description: 'Get the sum of all values' }, { value: ReducerID.count, label: 'Count', description: 'Get the number of values' }, ]; const downsamplingTypes: Array> = [ { value: ReducerID.min, label: 'Min', description: 'Fill with the minimum value' }, { value: ReducerID.max, label: 'Max', description: 'Fill with the maximum value' }, { value: ReducerID.mean, label: 'Mean', description: 'Fill with the average value' }, { value: ReducerID.sum, label: 'Sum', description: 'Fill with the sum of all values' }, ]; const upsamplingTypes: Array> = [ { value: 'pad', label: 'pad', description: 'fill with the last known value' }, { value: 'backfilling', label: 'backfilling', description: 'fill with the next known value' }, { value: 'fillna', label: 'fillna', description: 'Fill with NaNs' }, ]; const mathPlaceholder = 'Math operations on one more queries, you reference the query by ${refId} ie. $A, $B, $C etc\n' + 'Example: $A + $B\n' + 'Available functions: abs(), log(), nan(), inf(), null()'; export class ExpressionQueryEditor extends PureComponent { state = {}; onSelectGELType = (item: SelectableValue) => { const { query, onChange } = this.props; const q = { ...query, type: item.value!, }; if (q.type === GELQueryType.reduce) { if (!q.reducer) { q.reducer = ReducerID.mean; } q.expression = undefined; } else if (q.type === GELQueryType.resample) { if (!q.downsampler) { q.downsampler = ReducerID.mean; } if (!q.upsampler) { q.upsampler = 'fillna'; } q.reducer = undefined; } else { q.reducer = undefined; } onChange(q); }; onSelectReducer = (item: SelectableValue) => { const { query, onChange } = this.props; onChange({ ...query, reducer: item.value!, }); }; onSelectUpsampler = (item: SelectableValue) => { const { query, onChange } = this.props; onChange({ ...query, upsampler: item.value!, }); }; onSelectDownsampler = (item: SelectableValue) => { const { query, onChange } = this.props; onChange({ ...query, downsampler: item.value!, }); }; onRuleReducer = (item: SelectableValue) => { const { query, onChange } = this.props; onChange({ ...query, window: item.value!, }); }; onRefIdChange = (value: SelectableValue) => { const { query, onChange } = this.props; onChange({ ...query, expression: value.value, }); }; onExpressionChange = (evt: ChangeEvent) => { const { query, onChange } = this.props; onChange({ ...query, expression: evt.target.value, }); }; onWindowChange = (evt: ChangeEvent) => { const { query, onChange } = this.props; onChange({ ...query, window: evt.target.value, }); }; render() { const { query, queries } = this.props; const selected = gelTypes.find((o) => o.value === query.type); const reducer = reducerTypes.find((o) => o.value === query.reducer); const downsampler = downsamplingTypes.find((o) => o.value === query.downsampler); const upsampler = upsamplingTypes.find((o) => o.value === query.upsampler); const labelWidth = 14; const refIds = queries!.filter((q) => query.refId !== q.refId).map((q) => ({ value: q.refId, label: q.refId })); return (