diff --git a/src/legacy/ui/public/agg_types/buckets/filters.js b/src/legacy/ui/public/agg_types/buckets/filters.js index 17166d2d74ab8..b052b3444e8f0 100644 --- a/src/legacy/ui/public/agg_types/buckets/filters.js +++ b/src/legacy/ui/public/agg_types/buckets/filters.js @@ -23,9 +23,7 @@ import angular from 'angular'; import { BucketAggType } from './_bucket_agg_type'; import { createFilterFilters } from './create_filter/filters'; import { decorateQuery, luceneStringToDsl } from '@kbn/es-query'; -import '../directives/click_focus'; -import '../directives/parse_query'; -import filtersTemplate from '../controls/filters.html'; +import { FiltersParamEditor } from '../controls/filters'; import { i18n } from '@kbn/i18n'; import chrome from 'ui/chrome'; @@ -41,7 +39,7 @@ export const filtersBucketAgg = new BucketAggType({ params: [ { name: 'filters', - editor: filtersTemplate, + editorComponent: FiltersParamEditor, default: [ { input: {}, label: '' } ], write: function (aggConfig, output) { const inFilters = aggConfig.params.filters; diff --git a/src/legacy/ui/public/agg_types/controls/filter.tsx b/src/legacy/ui/public/agg_types/controls/filter.tsx new file mode 100644 index 0000000000000..963d941d3432b --- /dev/null +++ b/src/legacy/ui/public/agg_types/controls/filter.tsx @@ -0,0 +1,136 @@ +/* + * Licensed to Elasticsearch B.V. under one or more contributor + * license agreements. See the NOTICE file distributed with + * this work for additional information regarding copyright + * ownership. Elasticsearch B.V. 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. + */ + +import React, { useState } from 'react'; +import { + EuiForm, + EuiFlexGroup, + EuiFlexItem, + EuiButtonIcon, + EuiFieldText, + EuiFormRow, +} from '@elastic/eui'; +import { i18n } from '@kbn/i18n'; + +interface FilterRowProps { + id: string; + arrayIndex: number; + customLabel: string; + value: string; + autoFocus: boolean; + disableRemove: boolean; + dataTestSubj: string; + onChangeValue(id: string, query: string, label: string): void; + onRemoveFilter(id: string): void; +} + +function FilterRow({ + id, + arrayIndex, + customLabel, + value, + autoFocus, + disableRemove, + dataTestSubj, + onChangeValue, + onRemoveFilter, +}: FilterRowProps) { + const [showCustomLabel, setShowCustomLabel] = useState(false); + const filterLabel = i18n.translate('common.ui.aggTypes.filters.filterLabel', { + defaultMessage: 'Filter {index}', + values: { + index: arrayIndex + 1, + }, + }); + + const FilterControl = ( + + + setShowCustomLabel(!showCustomLabel)} + /> + + + onRemoveFilter(id)} + /> + + + ); + + return ( + + + onChangeValue(id, ev.target.value, customLabel)} + fullWidth={true} + autoFocus={autoFocus} + /> + + {showCustomLabel ? ( + + onChangeValue(id, value, ev.target.value)} + fullWidth={true} + /> + + ) : null} + + ); +} + +export { FilterRow }; diff --git a/src/legacy/ui/public/agg_types/controls/filters.html b/src/legacy/ui/public/agg_types/controls/filters.html deleted file mode 100644 index 13d32c716ecca..0000000000000 --- a/src/legacy/ui/public/agg_types/controls/filters.html +++ /dev/null @@ -1,90 +0,0 @@ -
-
-
- - -
- - -
-
- -
- -
- -
- - -
-
-
- - -
-

- - - -

-
- - - -
diff --git a/src/legacy/ui/public/agg_types/controls/filters.tsx b/src/legacy/ui/public/agg_types/controls/filters.tsx new file mode 100644 index 0000000000000..1d393b996b4ea --- /dev/null +++ b/src/legacy/ui/public/agg_types/controls/filters.tsx @@ -0,0 +1,120 @@ +/* + * Licensed to Elasticsearch B.V. under one or more contributor + * license agreements. See the NOTICE file distributed with + * this work for additional information regarding copyright + * ownership. Elasticsearch B.V. 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. + */ + +import React, { useState, useEffect } from 'react'; +import { omit, isEqual } from 'lodash'; +import { htmlIdGenerator, EuiButton, EuiSpacer } from '@elastic/eui'; +import { AggParamEditorProps } from 'ui/vis/editors/default'; +import { FormattedMessage } from '@kbn/i18n/react'; +import { data } from 'plugins/data'; +import { FilterRow } from './filter'; + +const { toUser, fromUser } = data.query.helpers; +const generateId = htmlIdGenerator(); + +interface FilterValue { + input: any; + label: string; + id: string; +} + +function FiltersParamEditor({ agg, value, setValue }: AggParamEditorProps) { + const [filters, setFilters] = useState(() => + value.map(filter => ({ ...filter, id: generateId() })) + ); + + useEffect(() => { + // set parsed values into model after initialization + setValue( + filters.map(filter => + omit({ ...filter, input: { query: fromUser(filter.input.query) } }, 'id') + ) + ); + }, []); + + useEffect( + () => { + // responsible for discarding changes + if ( + value.length !== filters.length || + value.some((filter, index) => !isEqual(filter, omit(filters[index], 'id'))) + ) { + setFilters(value.map(filter => ({ ...filter, id: generateId() }))); + } + }, + [value] + ); + + const updateFilters = (updatedFilters: FilterValue[]) => { + // do not set internal id parameter into saved object + setValue(updatedFilters.map(filter => omit(filter, 'id'))); + setFilters(updatedFilters); + }; + + const onAddFilter = () => + updateFilters([...filters, { input: { query: '' }, label: '', id: generateId() }]); + const onRemoveFilter = (id: string) => updateFilters(filters.filter(filter => filter.id !== id)); + const onChangeValue = (id: string, query: string, label: string) => + updateFilters( + filters.map(filter => + filter.id === id + ? { + ...filter, + input: { query: fromUser(query) }, + label, + } + : filter + ) + ); + + return ( + <> + {filters.map(({ input, label, id }, arrayIndex) => ( + + ))} + + + + + + ); +} + +export { FiltersParamEditor }; diff --git a/src/legacy/ui/public/agg_types/directives/click_focus.js b/src/legacy/ui/public/agg_types/directives/click_focus.js deleted file mode 100644 index d7cd813469921..0000000000000 --- a/src/legacy/ui/public/agg_types/directives/click_focus.js +++ /dev/null @@ -1,41 +0,0 @@ -/* - * Licensed to Elasticsearch B.V. under one or more contributor - * license agreements. See the NOTICE file distributed with - * this work for additional information regarding copyright - * ownership. Elasticsearch B.V. 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. - */ - -import _ from 'lodash'; -import $ from 'jquery'; -import { uiModules } from '../../modules'; -const module = uiModules.get('kibana'); - -module.directive('clickFocus', function () { - return { - scope: { - clickFocus: '=' - }, - restrict: 'A', - link: function ($scope, $elem) { - function handler() { - const focusElem = $.find('input[name=' + $scope.clickFocus + ']'); - if (focusElem[0]) focusElem[0].focus(); - } - - $elem.bind('click', handler); - $scope.$on('$destroy', _.bindKey($elem, 'unbind', 'click', handler)); - } - }; -}); diff --git a/src/legacy/ui/public/agg_types/directives/parse_query.js b/src/legacy/ui/public/agg_types/directives/parse_query.js deleted file mode 100644 index 768e0257aa56a..0000000000000 --- a/src/legacy/ui/public/agg_types/directives/parse_query.js +++ /dev/null @@ -1,45 +0,0 @@ -/* - * Licensed to Elasticsearch B.V. under one or more contributor - * license agreements. See the NOTICE file distributed with - * this work for additional information regarding copyright - * ownership. Elasticsearch B.V. 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. - */ - -import { data } from 'plugins/data'; -const { toUser, fromUser } = data.query.helpers; -import { uiModules } from '../../modules'; - -uiModules - .get('kibana') - .directive('parseQuery', function () { - - return { - restrict: 'A', - require: 'ngModel', - scope: { - 'ngModel': '=' - }, - link: function ($scope, elem, attr, ngModel) { - const init = function () { - $scope.ngModel = fromUser($scope.ngModel); - }; - - ngModel.$parsers.push(fromUser); - ngModel.$formatters.push(toUser); - - init(); - } - }; - }); diff --git a/x-pack/plugins/translations/translations/zh-CN.json b/x-pack/plugins/translations/translations/zh-CN.json index f208a58c15441..398fcc57db98e 100644 --- a/x-pack/plugins/translations/translations/zh-CN.json +++ b/x-pack/plugins/translations/translations/zh-CN.json @@ -124,13 +124,10 @@ "common.ui.aggTypes.extendedBounds.maxLabel": "最大值", "common.ui.aggTypes.extendedBounds.minLabel": "最小值", "common.ui.aggTypes.field.fieldLabel": "字段", - "common.ui.aggTypes.filters.addFilterButtonLabel": "添加筛选", "common.ui.aggTypes.filters.definiteFilterLabel": "筛选 {index} 标签", "common.ui.aggTypes.filters.filterLabel": "筛选 {index}", "common.ui.aggTypes.filters.labelPlaceholder": "标签", "common.ui.aggTypes.filters.removeFilterButtonAriaLabel": "移除此筛选", - "common.ui.aggTypes.filters.requiredFilterDescription": "必须指定至少一个筛选。", - "common.ui.aggTypes.filters.requiredFilterLabel": "必需:", "common.ui.aggTypes.filters.toggleFilterButtonAriaLabel": "切换筛选标签", "common.ui.aggTypes.histogram.missingMaxMinValuesWarning": "无法检索最大值和最小值以自动缩放直方图存储桶。这可能会导致可视化性能低下。", "common.ui.aggTypes.ipRanges.cidrMask.addRangeButtonLabel": "添加范围",