UNPKG

devextreme

Version:

HTML5 JavaScript Component Suite for Responsive Web Development

529 lines (524 loc) • 20.6 kB
/** * DevExtreme (esm/ui/pivot_grid/ui.pivot_grid.field_chooser.js) * Version: 21.1.4 * Build date: Mon Jun 21 2021 * * Copyright (c) 2012 - 2021 Developer Express Inc. ALL RIGHTS RESERVED * Read about DevExtreme licensing here: https://js.devexpress.com/Licensing/ */ import $ from "../../core/renderer"; import { getImageContainer } from "../../core/utils/icon"; import { hasWindow as hasWindowFn } from "../../core/utils/window"; import { isDefined } from "../../core/utils/type"; import { extend } from "../../core/utils/extend"; import { inArray } from "../../core/utils/array"; import { each } from "../../core/utils/iterator"; import localizationMessage from "../../localization/message"; import registerComponent from "../../core/component_registrator"; import { getCompareFunction, foreachDataLevel } from "./ui.pivot_grid.utils"; import TreeView from "../tree_view"; import ContextMenu from "../context_menu"; import BaseFieldChooser from "./ui.pivot_grid.field_chooser_base"; var DIV = "<div>"; var hasWindow = hasWindowFn(); import "./data_source"; var FIELDCHOOSER_CLASS = "dx-pivotgridfieldchooser"; var FIELDCHOOSER_CONTAINER_CLASS = "dx-pivotgridfieldchooser-container"; var FIELDS_CONTAINER_CLASS = "dx-pivotgrid-fields-container"; var AREA_DRAG_CLASS = "dx-pivotgrid-drag-action"; function getDimensionFields(item, fields) { var result = []; if (item.items) { for (var i = 0; i < item.items.length; i++) { result.push.apply(result, getDimensionFields(item.items[i], fields)) } } else if (isDefined(item.index)) { result.push(fields[item.index]) } return result } function getFirstItem(item, condition) { if (item.items) { for (var i = 0; i < item.items.length; i++) { var childrenItem = getFirstItem(item.items[i], condition); if (childrenItem) { return childrenItem } } } if (condition(item)) { return item } } var compareOrder = [function(a, b) { var aValue = -!!a.isMeasure; var bValue = +!!b.isMeasure; return aValue + bValue }, function(a, b) { var aValue = -!!(a.items && a.items.length); var bValue = +!!(b.items && b.items.length); return aValue + bValue }, function(a, b) { var aValue = +!!(false === a.isMeasure && a.field && a.field.levels && a.field.levels.length); var bValue = -!!(false === b.isMeasure && b.field && b.field.levels && b.field.levels.length); return aValue + bValue }, getCompareFunction((function(item) { return item.text }))]; function compareItems(a, b) { var result = 0; var i = 0; while (!result && compareOrder[i]) { result = compareOrder[i++](a, b) } return result } function getScrollable(container) { return container.find(".dx-scrollable").dxScrollable("instance") } var FieldChooser = BaseFieldChooser.inherit({ _getDefaultOptions: function() { return extend(this.callBase(), { height: 400, layout: 0, dataSource: null, onContextMenuPreparing: null, allowSearch: false, searchTimeout: 500, texts: { columnFields: localizationMessage.format("dxPivotGrid-columnFields"), rowFields: localizationMessage.format("dxPivotGrid-rowFields"), dataFields: localizationMessage.format("dxPivotGrid-dataFields"), filterFields: localizationMessage.format("dxPivotGrid-filterFields"), allFields: localizationMessage.format("dxPivotGrid-allFields") } }) }, _refreshDataSource: function() { var that = this; that._expandedPaths = []; that._changedHandler = that._changedHandler || function() { each(that._dataChangedHandlers, (function(_, func) { func() })); that._fireContentReadyAction(); that._skipStateChange = true; that.option("state", that._dataSource.state()); that._skipStateChange = false }; that._disposeDataSource(); that.callBase(); that._dataSource && that._dataSource.on("changed", that._changedHandler) }, _disposeDataSource: function() { var dataSource = this._dataSource; if (dataSource) { dataSource.off("changed", this._changedHandler); this._dataSource = void 0 } }, _dispose: function() { this._disposeDataSource(); this.callBase.apply(this, arguments) }, _init: function() { this.callBase(); this._refreshDataSource(); this._dataChangedHandlers = []; this._initActions() }, _initActions: function() { this._actions = { onContextMenuPreparing: this._createActionByOption("onContextMenuPreparing") } }, _trigger: function(eventName, eventArg) { this._actions[eventName](eventArg) }, _setOptionsByReference: function() { this.callBase(); extend(this._optionsByReference, { dataSource: true }) }, _optionChanged: function(args) { switch (args.name) { case "dataSource": this._refreshDataSource(); this._invalidate(); break; case "layout": case "texts": case "allowSearch": case "searchTimeout": this._invalidate(); break; case "onContextMenuPreparing": this._actions[args.name] = this._createActionByOption(args.name); break; default: this.callBase(args) } }, _clean: function(skipStateSetting) { !skipStateSetting && this._dataSource && this.option("state", this._dataSource.state()); this.$element().children("." + FIELDCHOOSER_CONTAINER_CLASS).remove() }, _renderLayout0: function($container) { $container.addClass("dx-layout-0"); var $row1 = $(DIV).addClass("dx-row").appendTo($container); var $row2 = $(DIV).addClass("dx-row").appendTo($container); var $col1 = $(DIV).addClass("dx-col").appendTo($row1); var $col2 = $(DIV).addClass("dx-col").appendTo($row1); var $col3 = $(DIV).addClass("dx-col").appendTo($row2); var $col4 = $(DIV).addClass("dx-col").appendTo($row2); this._renderArea($col1, "all"); this._renderArea($col2, "row"); this._renderArea($col2, "column"); this._renderArea($col3, "filter"); this._renderArea($col4, "data") }, _renderLayout1: function($container) { var $col1 = $(DIV).addClass("dx-col").appendTo($container); var $col2 = $(DIV).addClass("dx-col").appendTo($container); this._renderArea($col1, "all"); this._renderArea($col2, "filter"); this._renderArea($col2, "row"); this._renderArea($col2, "column"); this._renderArea($col2, "data") }, _renderLayout2: function($container) { $container.addClass("dx-layout-2"); var $row1 = $(DIV).addClass("dx-row").appendTo($container); this._renderArea($row1, "all"); var $row2 = $(DIV).addClass("dx-row").appendTo($container); var $col1 = $(DIV).addClass("dx-col").appendTo($row2); var $col2 = $(DIV).addClass("dx-col").appendTo($row2); this._renderArea($col1, "filter"); this._renderArea($col1, "row"); this._renderArea($col2, "column"); this._renderArea($col2, "data") }, _initMarkup: function() { var $element = this.$element(); var $container = $(DIV).addClass(FIELDCHOOSER_CONTAINER_CLASS).appendTo($element); var layout = this.option("layout"); this.callBase(); $element.addClass(FIELDCHOOSER_CLASS).addClass(FIELDS_CONTAINER_CLASS); this._dataChangedHandlers = []; var dataSource = this._dataSource; var currentState = "instantly" !== this.option("applyChangesMode") && dataSource && dataSource.state(); currentState && this.option("state") && dataSource.state(this.option("state"), true); if (0 === layout) { this._renderLayout0($container) } else if (1 === layout) { this._renderLayout1($container) } else { this._renderLayout2($container) } currentState && dataSource.state(currentState, true) }, _renderContentImpl: function() { this.callBase(); this.renderSortable(); this._renderContextMenu(); this.updateDimensions() }, _fireContentReadyAction: function() { if (!this._dataSource || !this._dataSource.isLoading()) { this.callBase() } }, _getContextMenuArgs: function(dxEvent) { var targetFieldElement = $(dxEvent.target).closest(".dx-area-field"); var targetGroupElement = $(dxEvent.target).closest(".dx-area-fields"); var field; var area; if (targetFieldElement.length) { var fieldCopy = targetFieldElement.data("field"); if (fieldCopy) { field = this.getDataSource().field(fieldCopy.index) || fieldCopy } } if (targetGroupElement.length) { area = targetGroupElement.attr("group") } return { event: dxEvent, field: field, area: area, items: [] } }, _renderContextMenu: function() { var that = this; var $container = that.$element(); if (that._contextMenu) { that._contextMenu.$element().remove() } that._contextMenu = that._createComponent($(DIV).appendTo($container), ContextMenu, { onPositioning: function(actionArgs) { var event = actionArgs.event; if (!event) { return } var args = that._getContextMenuArgs(event); that._trigger("onContextMenuPreparing", args); if (args.items && args.items.length) { actionArgs.component.option("items", args.items) } else { actionArgs.cancel = true } }, target: $container, onItemClick: function(params) { params.itemData.onItemClick && params.itemData.onItemClick(params) }, cssClass: "dx-pivotgridfieldchooser-context-menu" }) }, _createTreeItems: function(fields, groupFieldNames, path) { var that = this; var isMeasure; var resultItems = []; var groupedItems = []; var groupFieldName = groupFieldNames[0]; var fieldsByGroup = {}; if (!groupFieldName) { each(fields, (function(index, field) { var icon; if (true === field.isMeasure) { icon = "measure" } if (false === field.isMeasure) { icon = field.groupName ? "hierarchy" : "dimension" } resultItems.push({ index: field.index, field: field, key: field.dataField, selected: isDefined(field.area), text: field.caption || field.dataField, icon: icon, isMeasure: field.isMeasure, isDefault: field.isDefault }) })) } else { each(fields, (function(index, field) { var groupName = field[groupFieldName] || ""; fieldsByGroup[groupName] = fieldsByGroup[groupName] || []; fieldsByGroup[groupName].push(field); if (void 0 === isMeasure) { isMeasure = true } isMeasure = isMeasure && true === field.isMeasure })); each(fieldsByGroup, (function(groupName, fields) { var currentPath = path ? path + "." + groupName : groupName; var items = that._createTreeItems(fields, groupFieldNames.slice(1), currentPath); if (groupName) { groupedItems.push({ key: groupName, text: groupName, path: currentPath, isMeasure: items.isMeasure, expanded: inArray(currentPath, that._expandedPaths) >= 0, items: items }) } else { resultItems = items } })); resultItems = groupedItems.concat(resultItems); resultItems.isMeasure = isMeasure } return resultItems }, _createFieldsDataSource: function(dataSource) { var fields = dataSource && dataSource.fields() || []; fields = fields.filter(field => false !== field.visible && !isDefined(field.groupIndex)); var treeItems = this._createTreeItems(fields, ["dimension", "displayFolder"]); foreachDataLevel(treeItems, (function(items) { items.sort(compareItems) }), 0, "items"); return treeItems }, _renderFieldsTreeView: function(container) { var that = this; var dataSource = that._dataSource; var treeView = that._createComponent(container, TreeView, { dataSource: that._createFieldsDataSource(dataSource), showCheckBoxesMode: "normal", expandNodesRecursive: false, searchEnabled: that.option("allowSearch"), searchTimeout: that.option("searchTimeout"), itemTemplate: function(itemData, itemIndex, itemElement) { if (itemData.icon) { getImageContainer(itemData.icon).appendTo(itemElement) } $("<span>").toggleClass("dx-area-field", !itemData.items).data("field", itemData.field).text(itemData.text).appendTo(itemElement) }, onItemCollapsed: function(e) { var index = inArray(e.itemData.path, that._expandedPaths); if (index >= 0) { that._expandedPaths.splice(index, 1) } }, onItemExpanded: function(e) { var index = inArray(e.itemData.path, that._expandedPaths); if (index < 0) { that._expandedPaths.push(e.itemData.path) } }, onItemSelectionChanged: function(e) { var data = e.itemData; var field; var fields; var needSelectDefaultItem = true; var area; if (data.items) { if (data.selected) { treeView.unselectItem(data); return } that._processDemandState(() => { fields = getDimensionFields(data, dataSource.fields()); for (var i = 0; i < fields.length; i++) { if (fields[i].area) { needSelectDefaultItem = false; break } } }); if (needSelectDefaultItem) { var item = getFirstItem(data, (function(item) { return item.isDefault })) || getFirstItem(data, (function(item) { return isDefined(item.index) })); item && treeView.selectItem(item); return } } else { field = dataSource.fields()[data.index]; if (data.selected) { area = field.isMeasure ? "data" : "column" } if (field) { fields = [field] } } that._applyChanges(fields, { area: area, areaIndex: void 0 }) } }); that._dataChangedHandlers.push((function() { var scrollable = getScrollable(container); var scrollTop = scrollable ? scrollable.scrollTop() : 0; treeView.option({ dataSource: that._createFieldsDataSource(dataSource) }); scrollable = getScrollable(container); if (scrollable) { scrollable.scrollTo({ y: scrollTop }); scrollable.update() } })) }, _renderAreaFields: function($container, area) { var that = this; var dataSource = that._dataSource; var fields = dataSource ? extend(true, [], dataSource.getAreaFields(area, true)) : []; $container.empty(); each(fields, (function(_, field) { if (false !== field.visible) { that.renderField(field, true).appendTo($container) } })) }, _renderArea: function(container, area) { var that = this; var $areaContainer = $(DIV).addClass("dx-area").appendTo(container); var $fieldsHeaderContainer = $(DIV).addClass("dx-area-fields-header").appendTo($areaContainer); var caption = that.option("texts." + area + "Fields"); var $fieldsContent; var render; $("<span>").addClass("dx-area-icon").addClass("dx-area-icon-" + area).appendTo($fieldsHeaderContainer); $("<span>").html("&nbsp;").appendTo($fieldsHeaderContainer); $("<span>").addClass("dx-area-caption").text(caption).appendTo($fieldsHeaderContainer); var $fieldsContainer = $(DIV).addClass("dx-area-fields").addClass(AREA_DRAG_CLASS).appendTo($areaContainer); if ("all" !== area) { $fieldsContainer.attr("group", area).attr("allow-scrolling", true); $fieldsContent = $(DIV).addClass("dx-area-field-container").appendTo($fieldsContainer); render = function() { that._renderAreaFields($fieldsContent, area) }; that._dataChangedHandlers.push(render); render(); $fieldsContainer.dxScrollable() } else { $areaContainer.addClass("dx-all-fields"); $fieldsContainer.addClass("dx-treeview-border-visible"); that._renderFieldsTreeView($fieldsContainer) } }, _getSortableOptions: function() { return {} }, _adjustSortableOnChangedArgs: function() {}, resetTreeView: function() { var treeView = this.$element().find(".dx-treeview").dxTreeView("instance"); if (treeView) { treeView.option("searchValue", ""); treeView.collapseAll() } }, applyChanges: function() { var state = this.option("state"); if (isDefined(state)) { this._dataSource.state(state) } }, cancelChanges: function() { var dataSource = this._dataSource; if (!dataSource.isLoading()) { this.option("state", dataSource.state()); return true } return false }, getDataSource: function() { return this._dataSource }, updateDimensions: function() { var $scrollableElements = this.$element().find(".dx-area .dx-scrollable"); $scrollableElements.dxScrollable("update") }, _visibilityChanged: function(visible) { if (visible && hasWindow) { this.updateDimensions() } } }); registerComponent("dxPivotGridFieldChooser", FieldChooser); export default FieldChooser;