devextreme
Version:
HTML5 JavaScript Component Suite for Responsive Web Development
1,075 lines (1,068 loc) • 55.8 kB
JavaScript
/**
* DevExtreme (cjs/__internal/grids/pivot_grid/m_widget.js)
* Version: 24.2.6
* Build date: Mon Mar 17 2025
*
* Copyright (c) 2012 - 2025 Developer Express Inc. ALL RIGHTS RESERVED
* Read about DevExtreme licensing here: https://js.devexpress.com/Licensing/
*/
"use strict";
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.default = exports.PivotGrid = void 0;
var _click = require("../../../common/core/events/click");
var _events_engine = _interopRequireDefault(require("../../../common/core/events/core/events_engine"));
var _index = require("../../../common/core/events/utils/index");
var _message = _interopRequireDefault(require("../../../common/core/localization/message"));
var _component_registrator = _interopRequireDefault(require("../../../core/component_registrator"));
var _element = require("../../../core/element");
var _renderer = _interopRequireDefault(require("../../../core/renderer"));
var _common = require("../../../core/utils/common");
var _deferred = require("../../../core/utils/deferred");
var _extend = require("../../../core/utils/extend");
var _iterator = require("../../../core/utils/iterator");
var _size = require("../../../core/utils/size");
var _string = require("../../../core/utils/string");
var _type = require("../../../core/utils/type");
var _window = require("../../../core/utils/window");
var _button = _interopRequireDefault(require("../../../ui/button"));
var _context_menu = _interopRequireDefault(require("../../../ui/context_menu"));
var _ui = _interopRequireDefault(require("../../../ui/popup/ui.popup"));
var _themes = require("../../../ui/themes");
var _ui2 = _interopRequireDefault(require("../../../ui/widget/ui.widget"));
var _m_utils = _interopRequireDefault(require("../../grids/grid_core/m_utils"));
var _m_chart_integration = require("./chart_integration/m_chart_integration");
var _m_data_area = _interopRequireDefault(require("./data_area/m_data_area"));
var _m_data_controller = _interopRequireDefault(require("./data_controller/m_data_controller"));
var _m_export = require("./export/m_export");
var _m_field_chooser = require("./field_chooser/m_field_chooser");
var _m_field_chooser_base = require("./field_chooser/m_field_chooser_base");
var _m_fields_area = require("./fields_area/m_fields_area");
var _m_headers_area = _interopRequireDefault(require("./headers_area/m_headers_area"));
var _m_widget_utils = require("./m_widget_utils");
function _interopRequireDefault(e) {
return e && e.__esModule ? e : {
default: e
}
}
const window = (0, _window.getWindow)();
const DATA_AREA_CELL_CLASS = "dx-area-data-cell";
const ROW_AREA_CELL_CLASS = "dx-area-row-cell";
const COLUMN_AREA_CELL_CLASS = "dx-area-column-cell";
const DESCRIPTION_AREA_CELL_CLASS = "dx-area-description-cell";
const BORDERS_CLASS = "dx-pivotgrid-border";
const PIVOTGRID_CLASS = "dx-pivotgrid";
const ROW_LINES_CLASS = "dx-row-lines";
const BOTTOM_ROW_CLASS = "dx-bottom-row";
const BOTTOM_BORDER_CLASS = "dx-bottom-border";
const FIELDS_CONTAINER_CLASS = "dx-pivotgrid-fields-container";
const FIELDS_CLASS = "dx-area-fields";
const FIELD_CHOOSER_POPUP_CLASS = "dx-fieldchooser-popup";
const INCOMPRESSIBLE_FIELDS_CLASS = "dx-incompressible-fields";
const OVERFLOW_HIDDEN_CLASS = "dx-overflow-hidden";
const TR = "<tr>";
const TD = "<td>";
const DIV = "<div>";
const TEST_HEIGHT = 66666;
const FIELD_CALCULATED_OPTIONS = ["allowSorting", "allowSortingBySummary", "allowFiltering", "allowExpandAll"];
function getArraySum(array) {
let sum = 0;
(0, _iterator.each)(array, ((_, value) => {
sum += value || 0
}));
return sum
}
function adjustSizeArray(sizeArray, space) {
const delta = space / sizeArray.length;
for (let i = 0; i < sizeArray.length; i += 1) {
sizeArray[i] -= delta
}
}
function unsubscribeScrollEvents(area) {
area.off("scroll").off("stop")
}
function subscribeToScrollEvent(area, handler) {
unsubscribeScrollEvents(area);
area.on("scroll", handler).on("stop", handler)
}
function getCommonBorderWidth(elements, direction) {
const borderStyleNames = "width" === direction ? ["borderLeftWidth", "borderRightWidth"] : ["borderTopWidth", "borderBottomWidth"];
let width = 0;
(0, _iterator.each)(elements, ((_, elem) => {
const computedStyle = window.getComputedStyle(elem.get(0));
borderStyleNames.forEach((borderStyleName => {
width += parseFloat(computedStyle[borderStyleName]) || 0
}))
}));
return width
}
function clickedOnFieldsArea($targetElement) {
return $targetElement.closest(`.${FIELDS_CLASS}`).length || $targetElement.find(`.${FIELDS_CLASS}`).length
}
const PivotGrid = exports.PivotGrid = _ui2.default.inherit({
_getDefaultOptions() {
return (0, _extend.extend)(this.callBase(), {
scrolling: {
timeout: 300,
renderingThreshold: 150,
minTimeout: 10,
mode: "standard",
useNative: "auto",
removeInvisiblePages: true,
virtualRowHeight: 50,
virtualColumnWidth: 100,
loadTwoPagesOnStart: true
},
encodeHtml: true,
dataSource: null,
activeStateEnabled: false,
fieldChooser: {
minWidth: 250,
minHeight: 250,
enabled: true,
allowSearch: false,
searchTimeout: 500,
layout: 0,
title: _message.default.format("dxPivotGrid-fieldChooserTitle"),
width: 600,
height: 600,
applyChangesMode: "instantly"
},
onContextMenuPreparing: null,
allowSorting: false,
allowSortingBySummary: false,
allowFiltering: false,
allowExpandAll: false,
wordWrapEnabled: true,
fieldPanel: {
showColumnFields: true,
showFilterFields: true,
showDataFields: true,
showRowFields: true,
allowFieldDragging: true,
visible: false,
texts: {
columnFieldArea: _message.default.format("dxPivotGrid-columnFieldArea"),
rowFieldArea: _message.default.format("dxPivotGrid-rowFieldArea"),
filterFieldArea: _message.default.format("dxPivotGrid-filterFieldArea"),
dataFieldArea: _message.default.format("dxPivotGrid-dataFieldArea")
}
},
dataFieldArea: "column",
export: {
enabled: false,
fileName: "PivotGrid"
},
showRowTotals: true,
showRowGrandTotals: true,
showColumnTotals: true,
showColumnGrandTotals: true,
hideEmptySummaryCells: true,
showTotalsPrior: "none",
rowHeaderLayout: "standard",
loadPanel: {
enabled: true,
text: _message.default.format("Loading"),
width: 200,
height: 70,
showIndicator: true,
indicatorSrc: "",
showPane: true
},
texts: {
grandTotal: _message.default.format("dxPivotGrid-grandTotal"),
total: _message.default.getFormatter("dxPivotGrid-total"),
noData: _message.default.format("dxDataGrid-noDataText"),
showFieldChooser: _message.default.format("dxPivotGrid-showFieldChooser"),
expandAll: _message.default.format("dxPivotGrid-expandAll"),
collapseAll: _message.default.format("dxPivotGrid-collapseAll"),
sortColumnBySummary: _message.default.getFormatter("dxPivotGrid-sortColumnBySummary"),
sortRowBySummary: _message.default.getFormatter("dxPivotGrid-sortRowBySummary"),
removeAllSorting: _message.default.format("dxPivotGrid-removeAllSorting"),
exportToExcel: _message.default.format("dxDataGrid-exportToExcel"),
dataNotAvailable: _message.default.format("dxPivotGrid-dataNotAvailable")
},
onCellClick: null,
onCellPrepared: null,
showBorders: false,
stateStoring: {
enabled: false,
storageKey: null,
type: "localStorage",
customLoad: null,
customSave: null,
savingTimeout: 2e3
},
onExpandValueChanging: null,
renderCellCountLimit: 2e4,
onExporting: null,
headerFilter: {
width: 252,
height: 325,
allowSelectAll: true,
showRelevantValues: false,
search: {
enabled: false,
timeout: 500,
editorOptions: {},
mode: "contains"
},
texts: {
emptyValue: _message.default.format("dxDataGrid-headerFilterEmptyValue"),
ok: _message.default.format("dxDataGrid-headerFilterOK"),
cancel: _message.default.format("dxDataGrid-headerFilterCancel")
}
}
})
},
_updateCalculatedOptions(fields) {
const that = this;
(0, _iterator.each)(fields, ((_, field) => {
(0, _iterator.each)(FIELD_CALCULATED_OPTIONS, ((_, optionName) => {
const isCalculated = field._initProperties && optionName in field._initProperties && void 0 === field._initProperties[optionName];
const needUpdate = void 0 === field[optionName] || isCalculated;
if (needUpdate) {
(0, _m_widget_utils.setFieldProperty)(field, optionName, that.option(optionName))
}
}))
}))
},
_getDataControllerOptions() {
const that = this;
return {
component: that,
dataSource: that.option("dataSource"),
texts: that.option("texts"),
showRowTotals: that.option("showRowTotals"),
showRowGrandTotals: that.option("showRowGrandTotals"),
showColumnTotals: that.option("showColumnTotals"),
showTotalsPrior: that.option("showTotalsPrior"),
showColumnGrandTotals: that.option("showColumnGrandTotals"),
dataFieldArea: that.option("dataFieldArea"),
rowHeaderLayout: that.option("rowHeaderLayout"),
hideEmptySummaryCells: that.option("hideEmptySummaryCells"),
onFieldsPrepared(fields) {
that._updateCalculatedOptions(fields)
}
}
},
_initDataController() {
const that = this;
that._dataController && that._dataController.dispose();
that._dataController = new _m_data_controller.default.DataController(that._getDataControllerOptions());
if ((0, _window.hasWindow)()) {
that._dataController.changed.add((() => {
that._render()
}))
}
that._dataController.scrollChanged.add((options => {
that._scrollLeft = options.left;
that._scrollTop = options.top
}));
that._dataController.loadingChanged.add((() => {
that._updateLoading()
}));
that._dataController.progressChanged.add(that._updateLoading.bind(that));
that._dataController.dataSourceChanged.add((() => {
that._trigger("onChanged")
}));
const expandValueChanging = that.option("onExpandValueChanging");
if (expandValueChanging) {
that._dataController.expandValueChanging.add((e => {
expandValueChanging(e)
}))
}
},
_init() {
this.callBase();
this._initDataController();
_m_utils.default.logHeaderFilterDeprecatedWarningIfNeed(this);
this._scrollLeft = this._scrollTop = null;
this._initActions()
},
_initActions() {
this._actions = {
onChanged: this._createActionByOption("onChanged"),
onContextMenuPreparing: this._createActionByOption("onContextMenuPreparing"),
onCellClick: this._createActionByOption("onCellClick"),
onExporting: this._createActionByOption("onExporting"),
onCellPrepared: this._createActionByOption("onCellPrepared")
}
},
_trigger(eventName, eventArg) {
this._actions[eventName](eventArg)
},
_optionChanged(args) {
const that = this;
if (FIELD_CALCULATED_OPTIONS.includes(args.name)) {
const fields = this.getDataSource().fields();
this._updateCalculatedOptions(fields)
}
switch (args.name) {
case "dataSource":
case "allowSorting":
case "allowFiltering":
case "allowExpandAll":
case "allowSortingBySummary":
case "scrolling":
case "stateStoring":
that._initDataController();
that._fieldChooserPopup.hide();
that._renderFieldChooser();
that._invalidate();
break;
case "texts":
case "showTotalsPrior":
case "showRowTotals":
case "showRowGrandTotals":
case "showColumnTotals":
case "showColumnGrandTotals":
case "hideEmptySummaryCells":
case "dataFieldArea":
that._dataController.updateViewOptions(that._getDataControllerOptions());
break;
case "useNativeScrolling":
case "encodeHtml":
case "renderCellCountLimit":
case "onExpandValueChanging":
break;
case "rtlEnabled":
that.callBase(args);
that._renderFieldChooser();
that._renderContextMenu();
(0, _window.hasWindow)() && that._renderLoadPanel(that._dataArea.groupElement(), that.$element());
that._invalidate();
break;
case "export":
that._renderDescriptionArea();
break;
case "onCellClick":
case "onContextMenuPreparing":
case "onExporting":
case "onExported":
case "onFileSaving":
case "onCellPrepared":
that._actions[args.name] = that._createActionByOption(args.name);
break;
case "fieldChooser":
that._renderFieldChooser();
that._renderDescriptionArea();
break;
case "loadPanel":
if ((0, _window.hasWindow)()) {
if ("loadPanel.enabled" === args.fullName) {
clearTimeout(this._hideLoadingTimeoutID);
that._renderLoadPanel(that._dataArea.groupElement(), that.$element())
} else {
that._renderLoadPanel(that._dataArea.groupElement(), that.$element());
that._invalidate()
}
}
break;
case "fieldPanel":
that._renderDescriptionArea();
that._invalidate();
break;
case "headerFilter":
that._renderFieldChooser();
that._invalidate();
break;
case "showBorders":
that._tableElement().toggleClass(BORDERS_CLASS, !!args.value);
that.updateDimensions();
break;
case "wordWrapEnabled":
that._tableElement().toggleClass("dx-word-wrap", !!args.value);
that.updateDimensions();
break;
case "rowHeaderLayout":
that._tableElement().find(".dx-area-row-cell").toggleClass("dx-area-tree-view", "tree" === args.value);
that._dataController.updateViewOptions(that._getDataControllerOptions());
break;
case "height":
case "width":
that._hasHeight = null;
that.callBase(args);
that.resize();
break;
default:
that.callBase(args)
}
},
_updateScrollPosition(columnsArea, rowsArea, dataArea) {
let force = arguments.length > 3 && void 0 !== arguments[3] ? arguments[3] : false;
const that = this;
let scrollTop;
let scrollLeft;
const scrolled = that._scrollTop || that._scrollLeft;
if (that._scrollUpdating) {
return
}
that._scrollUpdating = true;
if (rowsArea && !rowsArea.hasScroll() && that._hasHeight) {
that._scrollTop = null
}
if (columnsArea && !columnsArea.hasScroll()) {
that._scrollLeft = null
}
if (null !== that._scrollTop || null !== that._scrollLeft || scrolled || that.option("rtlEnabled")) {
scrollTop = that._scrollTop || 0;
scrollLeft = that._scrollLeft || 0;
dataArea.scrollTo({
left: scrollLeft,
top: scrollTop
}, force);
columnsArea.scrollTo({
left: scrollLeft
}, force);
rowsArea.scrollTo({
top: scrollTop
}, force);
that._dataController.updateWindowScrollPosition(that._scrollTop)
}
that._scrollUpdating = false
},
_subscribeToEvents(columnsArea, rowsArea, dataArea) {
const that = this;
(0, _iterator.each)([columnsArea, rowsArea, dataArea], ((_, area) => {
subscribeToScrollEvent(area, (e => function(e, area) {
const {
scrollOffset: scrollOffset
} = e;
const scrollable = area._getScrollable();
const leftOffset = "vertical" !== scrollable.option("direction") ? scrollOffset.left : that._scrollLeft;
const topOffset = "horizontal" !== scrollable.option("direction") && that._hasHeight ? scrollOffset.top : that._scrollTop;
if ((that._scrollLeft || 0) !== (leftOffset || 0) || (that._scrollTop || 0) !== (topOffset || 0)) {
that._scrollLeft = leftOffset;
that._scrollTop = topOffset;
that._updateScrollPosition(columnsArea, rowsArea, dataArea);
if ("virtual" === that.option("scrolling.mode")) {
that._dataController.setViewportPosition(that._scrollLeft, that._scrollTop)
}
}
}(e, area)))
}));
!that._hasHeight && that._dataController.subscribeToWindowScrollEvents(dataArea.groupElement())
},
_clean: _common.noop,
_needDelayResizing(cellsInfo) {
const cellsCount = cellsInfo.length * (cellsInfo.length ? cellsInfo[0].length : 0);
return cellsCount > this.option("renderCellCountLimit")
},
_renderFieldChooser() {
const that = this;
const container = that._pivotGridContainer;
const fieldChooserOptions = that.option("fieldChooser") || {};
const toolbarItems = "onDemand" === fieldChooserOptions.applyChangesMode ? [{
toolbar: "bottom",
location: "after",
widget: "dxButton",
options: {
text: _message.default.format("OK"),
onClick() {
that._fieldChooserPopup.$content().dxPivotGridFieldChooser("applyChanges");
that._fieldChooserPopup.hide()
}
}
}, {
toolbar: "bottom",
location: "after",
widget: "dxButton",
options: {
text: _message.default.format("Cancel"),
onClick() {
that._fieldChooserPopup.hide()
}
}
}] : [];
const fieldChooserComponentOptions = {
layout: fieldChooserOptions.layout,
texts: fieldChooserOptions.texts || {},
dataSource: that.getDataSource(),
allowSearch: fieldChooserOptions.allowSearch,
searchTimeout: fieldChooserOptions.searchTimeout,
width: void 0,
height: void 0,
headerFilter: that.option("headerFilter"),
encodeHtml: that.option("fieldChooser.encodeHtml") ?? that.option("encodeHtml"),
applyChangesMode: fieldChooserOptions.applyChangesMode,
onContextMenuPreparing(e) {
that._trigger("onContextMenuPreparing", e)
}
};
const popupOptions = {
shading: false,
title: fieldChooserOptions.title,
width: fieldChooserOptions.width,
height: fieldChooserOptions.height,
showCloseButton: true,
resizeEnabled: true,
minWidth: fieldChooserOptions.minWidth,
minHeight: fieldChooserOptions.minHeight,
toolbarItems: toolbarItems,
onResize(e) {
e.component.$content().dxPivotGridFieldChooser("updateDimensions")
},
onShown(e) {
that._createComponent(e.component.content(), _m_field_chooser.FieldChooser, fieldChooserComponentOptions)
},
onHidden(e) {
const fieldChooser = e.component.$content().dxPivotGridFieldChooser("instance");
fieldChooser.resetTreeView();
fieldChooser.cancelChanges()
}
};
if (that._fieldChooserPopup) {
that._fieldChooserPopup.option(popupOptions);
that._fieldChooserPopup.$content().dxPivotGridFieldChooser(fieldChooserComponentOptions)
} else {
that._fieldChooserPopup = that._createComponent((0, _renderer.default)(DIV).addClass("dx-fieldchooser-popup").appendTo(container), _ui.default, popupOptions)
}
},
_renderContextMenu() {
const that = this;
const $container = that._pivotGridContainer;
if (that._contextMenu) {
that._contextMenu.$element().remove()
}
that._contextMenu = that._createComponent((0, _renderer.default)(DIV).appendTo($container), _context_menu.default, {
onPositioning(actionArgs) {
const {
event: event
} = actionArgs;
actionArgs.cancel = true;
if (!event) {
return
}
const targetElement = event.target.cellIndex >= 0 ? event.target : (0, _renderer.default)(event.target).closest("td").get(0);
if (!targetElement) {
return
}
const args = that._createEventArgs(targetElement, event);
const items = that._getContextMenuItems(args);
if (items) {
actionArgs.component.option("items", items);
actionArgs.cancel = false
}
},
onItemClick(params) {
params.itemData.onItemClick && params.itemData.onItemClick(params)
},
cssClass: "dx-pivotgrid",
target: that.$element()
})
},
_getContextMenuItems(e) {
const that = this;
let items = [];
const texts = that.option("texts");
if ("row" === e.area || "column" === e.area) {
const areaFields = e[`${e.area}Fields`];
const oppositeAreaFields = e["column" === e.area ? "rowFields" : "columnFields"];
const field = e.cell.path && areaFields[e.cell.path.length - 1];
const dataSource = that.getDataSource();
if (field && field.allowExpandAll && e.cell.path.length < e[`${e.area}Fields`].length && !dataSource.paginate()) {
items.push({
beginGroup: true,
icon: "none",
text: texts.expandAll,
onItemClick() {
dataSource.expandAll(field.index)
}
});
items.push({
text: texts.collapseAll,
icon: "none",
onItemClick() {
dataSource.collapseAll(field.index)
}
})
}
if (e.cell.isLast && !dataSource.paginate()) {
let sortingBySummaryItemCount = 0;
(0, _iterator.each)(oppositeAreaFields, ((_, field) => {
if (!field.allowSortingBySummary) {
return
}(0, _iterator.each)(e.dataFields, ((dataIndex, dataField) => {
if ((0, _type.isDefined)(e.cell.dataIndex) && e.cell.dataIndex !== dataIndex) {
return
}
const showDataFieldCaption = !(0, _type.isDefined)(e.cell.dataIndex) && e.dataFields.length > 1;
const textFormat = "column" === e.area ? texts.sortColumnBySummary : texts.sortRowBySummary;
const checked = (0, _m_widget_utils.findField)(e.dataFields, field.sortBySummaryField) === dataIndex && (e.cell.path || []).join("/") === (field.sortBySummaryPath || []).join("/");
const text = (0, _string.format)(textFormat, showDataFieldCaption ? `${field.caption} - ${dataField.caption}` : field.caption);
items.push({
beginGroup: 0 === sortingBySummaryItemCount,
icon: checked ? "desc" === field.sortOrder ? "sortdowntext" : "sortuptext" : "none",
text: text,
onItemClick() {
dataSource.field(field.index, {
sortBySummaryField: dataField.name || dataField.caption || dataField.dataField,
sortBySummaryPath: e.cell.path,
sortOrder: "desc" === field.sortOrder ? "asc" : "desc"
});
dataSource.load()
}
});
sortingBySummaryItemCount += 1
}))
}));
(0, _iterator.each)(oppositeAreaFields, ((_, field) => {
if (!field.allowSortingBySummary || !(0, _type.isDefined)(field.sortBySummaryField)) {
return
}
items.push({
beginGroup: 0 === sortingBySummaryItemCount,
icon: "none",
text: texts.removeAllSorting,
onItemClick() {
(0, _iterator.each)(oppositeAreaFields, ((_, field) => {
dataSource.field(field.index, {
sortBySummaryField: void 0,
sortBySummaryPath: void 0,
sortOrder: void 0
})
}));
dataSource.load()
}
});
return false
}))
}
}
if (that.option("fieldChooser.enabled")) {
items.push({
beginGroup: true,
icon: "columnchooser",
text: texts.showFieldChooser,
onItemClick() {
that._fieldChooserPopup.show()
}
})
}
if (that.option("export.enabled")) {
items.push({
beginGroup: true,
icon: "xlsxfile",
text: texts.exportToExcel,
onItemClick() {
that.exportTo()
}
})
}
e.items = items;
that._trigger("onContextMenuPreparing", e);
items = e.items;
if (items && items.length) {
return items
}
return
},
_createEventArgs(targetElement, dxEvent) {
const that = this;
const dataSource = that.getDataSource();
const args = {
rowFields: dataSource.getAreaFields("row"),
columnFields: dataSource.getAreaFields("column"),
dataFields: dataSource.getAreaFields("data"),
event: dxEvent
};
if (clickedOnFieldsArea((0, _renderer.default)(targetElement))) {
return (0, _extend.extend)(that._createFieldArgs(targetElement), args)
}
return (0, _extend.extend)(that._createCellArgs(targetElement), args)
},
_createFieldArgs(targetElement) {
const field = (0, _renderer.default)(targetElement).children().data("field");
const args = {
field: field
};
return (0, _type.isDefined)(field) ? args : {}
},
_createCellArgs(cellElement) {
const $cellElement = (0, _renderer.default)(cellElement);
const columnIndex = cellElement.cellIndex;
const {
rowIndex: rowIndex
} = cellElement.parentElement;
const $table = $cellElement.closest("table");
const data = $table.data("data");
const cell = data && data[rowIndex] && data[rowIndex][columnIndex];
const args = {
area: $table.data("area"),
rowIndex: rowIndex,
columnIndex: columnIndex,
cellElement: (0, _element.getPublicElement)($cellElement),
cell: cell
};
return args
},
_handleCellClick(e) {
const that = this;
const args = that._createEventArgs(e.currentTarget, e);
const {
cell: cell
} = args;
if (!cell || !args.area && (args.rowIndex || args.columnIndex)) {
return
}
that._trigger("onCellClick", args);
cell && !args.cancel && (0, _type.isDefined)(cell.expanded) && setTimeout((() => {
that._dataController[cell.expanded ? "collapseHeaderItem" : "expandHeaderItem"](args.area, cell.path)
}))
},
_getNoDataText() {
return this.option("texts.noData")
},
_renderNoDataText: _m_utils.default.renderNoDataText,
_renderLoadPanel: _m_utils.default.renderLoadPanel,
_updateLoading(progress) {
const that = this;
const isLoading = that._dataController.isLoading();
if (!that._loadPanel) {
return
}
const loadPanelVisible = that._loadPanel.option("visible");
if (!loadPanelVisible) {
that._startLoadingTime = new Date
}
if (isLoading) {
if (progress) {
if (new Date - that._startLoadingTime >= 1e3) {
that._loadPanel.option("message", `${Math.floor(100*progress)}%`)
}
} else {
that._loadPanel.option("message", that.option("loadPanel.text"))
}
}
clearTimeout(that._hideLoadingTimeoutID);
if (loadPanelVisible && !isLoading) {
that._hideLoadingTimeoutID = setTimeout((() => {
that._loadPanel.option("visible", false);
that.$element().removeClass("dx-overflow-hidden")
}))
} else {
const visibilityOptions = {
visible: isLoading
};
if (isLoading) {
visibilityOptions.position = _m_utils.default.calculateLoadPanelPosition(that._dataArea.groupElement())
}
that._loadPanel.option(visibilityOptions);
that.$element().toggleClass("dx-overflow-hidden", !isLoading)
}
},
_renderDescriptionArea() {
const $element = this.$element();
const $descriptionCell = $element.find(".dx-area-description-cell");
const $toolbarContainer = (0, _renderer.default)(DIV).addClass("dx-pivotgrid-toolbar");
const fieldPanel = this.option("fieldPanel");
const $filterHeader = $element.find(".dx-filter-header");
const $columnHeader = $element.find(".dx-column-header");
let $targetContainer;
if (fieldPanel.visible && fieldPanel.showFilterFields) {
$targetContainer = $filterHeader
} else if (fieldPanel.visible && (fieldPanel.showDataFields || fieldPanel.showColumnFields)) {
$targetContainer = $columnHeader
} else {
$targetContainer = $descriptionCell
}
$columnHeader.toggleClass("dx-bottom-border", !!(fieldPanel.visible && (fieldPanel.showDataFields || fieldPanel.showColumnFields)));
$filterHeader.toggleClass("dx-bottom-border", !!(fieldPanel.visible && fieldPanel.showFilterFields));
$descriptionCell.toggleClass("dx-pivotgrid-background", fieldPanel.visible && (fieldPanel.showDataFields || fieldPanel.showColumnFields || fieldPanel.showRowFields));
this.$element().find(".dx-pivotgrid-toolbar").remove();
$toolbarContainer.prependTo($targetContainer);
const stylingMode = (0, _themes.isFluent)((0, _themes.current)()) ? "text" : "contained";
if (this.option("fieldChooser.enabled")) {
const $buttonElement = (0, _renderer.default)(DIV).appendTo($toolbarContainer).addClass("dx-pivotgrid-field-chooser-button");
const buttonOptions = {
icon: "columnchooser",
hint: this.option("texts.showFieldChooser"),
stylingMode: stylingMode,
onClick: () => {
this.getFieldChooserPopup().show()
}
};
this._createComponent($buttonElement, _button.default, buttonOptions)
}
if (this.option("export.enabled")) {
const $buttonElement = (0, _renderer.default)(DIV).appendTo($toolbarContainer).addClass("dx-pivotgrid-export-button");
const buttonOptions = {
icon: "xlsxfile",
hint: this.option("texts.exportToExcel"),
stylingMode: stylingMode,
onClick: () => {
this.exportTo()
}
};
this._createComponent($buttonElement, _button.default, buttonOptions)
}
},
_detectHasContainerHeight() {
const that = this;
const element = that.$element();
if ((0, _type.isDefined)(that._hasHeight)) {
const height = that.option("height") || that.$element().get(0).style.height;
if (height && that._hasHeight ^ "auto" !== height) {
that._hasHeight = null
}
}
if ((0, _type.isDefined)(that._hasHeight) || element.is(":hidden")) {
return
}
that._pivotGridContainer.addClass("dx-hidden");
const testElement = (0, _renderer.default)(DIV);
(0, _size.setHeight)(testElement, 66666);
element.append(testElement);
that._hasHeight = 66666 !== (0, _size.getHeight)(element);
that._pivotGridContainer.removeClass("dx-hidden");
testElement.remove()
},
_renderHeaders(rowHeaderContainer, columnHeaderContainer, filterHeaderContainer, dataHeaderContainer) {
const dataSource = this.getDataSource();
this._rowFields = this._rowFields || new _m_fields_area.FieldsArea(this, "row");
this._rowFields.render(rowHeaderContainer, dataSource.getAreaFields("row"));
this._columnFields = this._columnFields || new _m_fields_area.FieldsArea(this, "column");
this._columnFields.render(columnHeaderContainer, dataSource.getAreaFields("column"));
this._filterFields = this._filterFields || new _m_fields_area.FieldsArea(this, "filter");
this._filterFields.render(filterHeaderContainer, dataSource.getAreaFields("filter"));
this._dataFields = this._dataFields || new _m_fields_area.FieldsArea(this, "data");
this._dataFields.render(dataHeaderContainer, dataSource.getAreaFields("data"));
this.$element().dxPivotGridFieldChooserBase("instance").renderSortable()
},
_createTableElement() {
const $table = (0, _renderer.default)("<table>").css({
width: "100%"
}).toggleClass(BORDERS_CLASS, !!this.option("showBorders")).toggleClass("dx-word-wrap", !!this.option("wordWrapEnabled"));
_events_engine.default.on($table, (0, _index.addNamespace)(_click.name, "dxPivotGrid"), "td", this._handleCellClick.bind(this));
return $table
},
_renderDataArea(dataAreaElement) {
const dataArea = this._dataArea || new _m_data_area.default.DataArea(this);
this._dataArea = dataArea;
dataArea.render(dataAreaElement, this._dataController.getCellsInfo());
return dataArea
},
_renderRowsArea(rowsAreaElement) {
const rowsArea = this._rowsArea || new _m_headers_area.default.VerticalHeadersArea(this);
this._rowsArea = rowsArea;
rowsArea.render(rowsAreaElement, this._dataController.getRowsInfo());
return rowsArea
},
_renderColumnsArea(columnsAreaElement) {
const columnsArea = this._columnsArea || new _m_headers_area.default.HorizontalHeadersArea(this);
this._columnsArea = columnsArea;
columnsArea.render(columnsAreaElement, this._dataController.getColumnsInfo());
return columnsArea
},
_initMarkup() {
this.callBase.apply(this, arguments);
this.$element().addClass("dx-pivotgrid")
},
_renderContentImpl() {
const that = this;
let columnsAreaElement;
let rowsAreaElement;
let dataAreaElement;
let tableElement;
const isFirstDrawing = !that._pivotGridContainer;
let rowHeaderContainer;
let columnHeaderContainer;
let filterHeaderContainer;
let dataHeaderContainer;
tableElement = !isFirstDrawing && that._tableElement();
if (!tableElement) {
that.$element().addClass("dx-row-lines").addClass(FIELDS_CONTAINER_CLASS);
that._pivotGridContainer = (0, _renderer.default)(DIV).addClass("dx-pivotgrid-container");
that._renderFieldChooser();
that._renderContextMenu();
columnsAreaElement = (0, _renderer.default)(TD).addClass("dx-area-column-cell");
rowsAreaElement = (0, _renderer.default)(TD).addClass("dx-area-row-cell");
dataAreaElement = (0, _renderer.default)(TD).addClass("dx-area-data-cell");
tableElement = that._createTableElement();
dataHeaderContainer = (0, _renderer.default)(TD).addClass("dx-data-header");
filterHeaderContainer = (0, _renderer.default)("<td>").attr("colspan", "2").addClass("dx-filter-header");
columnHeaderContainer = (0, _renderer.default)(TD).addClass("dx-column-header");
rowHeaderContainer = (0, _renderer.default)(TD).addClass("dx-area-description-cell");
(0, _renderer.default)(TR).append(filterHeaderContainer).appendTo(tableElement);
(0, _renderer.default)(TR).append(dataHeaderContainer).append(columnHeaderContainer).appendTo(tableElement);
(0, _renderer.default)(TR).append(rowHeaderContainer).append(columnsAreaElement).appendTo(tableElement);
(0, _renderer.default)(TR).addClass("dx-bottom-row").append(rowsAreaElement).append(dataAreaElement).appendTo(tableElement);
that._pivotGridContainer.append(tableElement);
that.$element().append(that._pivotGridContainer);
if ("tree" === that.option("rowHeaderLayout")) {
rowsAreaElement.addClass("dx-area-tree-view")
}
}
that.$element().addClass("dx-overflow-hidden");
that._createComponent(that.$element(), _m_field_chooser_base.FieldChooserBase, {
dataSource: that.getDataSource(),
encodeHtml: that.option("encodeHtml"),
allowFieldDragging: that.option("fieldPanel.allowFieldDragging"),
headerFilter: that.option("headerFilter"),
visible: that.option("visible"),
remoteSort: "virtual" === that.option("scrolling.mode")
});
const dataArea = that._renderDataArea(dataAreaElement);
const rowsArea = that._renderRowsArea(rowsAreaElement);
const columnsArea = that._renderColumnsArea(columnsAreaElement);
dataArea.tableElement().prepend(columnsArea.headElement());
if (isFirstDrawing) {
that._renderLoadPanel(dataArea.groupElement().parent(), that.$element());
that._renderDescriptionArea();
rowsArea.renderScrollable();
columnsArea.renderScrollable();
dataArea.renderScrollable()
} [dataArea, rowsArea, columnsArea].forEach((area => {
unsubscribeScrollEvents(area)
}));
that._renderHeaders(rowHeaderContainer, columnHeaderContainer, filterHeaderContainer, dataHeaderContainer);
that._update(isFirstDrawing)
},
_update(isFirstDrawing) {
const that = this;
const updateHandler = function() {
that.updateDimensions()
};
if (that._needDelayResizing(that._dataArea.getData()) && isFirstDrawing) {
setTimeout(updateHandler)
} else {
updateHandler()
}
},
_fireContentReadyAction() {
if (!this._dataController.isLoading()) {
this.callBase()
}
},
getScrollPath(area) {
const that = this;
if ("column" === area) {
return that._columnsArea.getScrollPath(that._scrollLeft)
}
return that._rowsArea.getScrollPath(that._scrollTop)
},
getDataSource() {
return this._dataController.getDataSource()
},
getFieldChooserPopup() {
return this._fieldChooserPopup
},
hasScroll(area) {
return "column" === area ? this._columnsArea.hasScroll() : this._rowsArea.hasScroll()
},
_dimensionChanged() {
this.updateDimensions()
},
_visibilityChanged(visible) {
if (visible) {
this.updateDimensions()
}
},
_dispose() {
const that = this;
clearTimeout(that._hideLoadingTimeoutID);
that.callBase.apply(that, arguments);
if (that._dataController) {
that._dataController.dispose()
}
},
_tableElement() {
return this.$element().find("table").first()
},
addWidgetPrefix: className => `dx-pivotgrid-${className}`,
resize() {
this.updateDimensions()
},
isReady() {
return this.callBase() && !this._dataController.isLoading()
},
updateDimensions() {
const that = this;
let groupWidth;
const tableElement = that._tableElement();
let bordersWidth;
let totalWidth = 0;
let totalHeight = 0;
let rowsAreaWidth = 0;
let hasRowsScroll;
let hasColumnsScroll;
const dataAreaCell = tableElement.find(".dx-area-data-cell");
const rowAreaCell = tableElement.find(".dx-area-row-cell");
const columnAreaCell = tableElement.find(".dx-area-column-cell");
const descriptionCell = tableElement.find(".dx-area-description-cell");
const filterHeaderCell = tableElement.find(".dx-filter-header");
const columnHeaderCell = tableElement.find(".dx-column-header");
const rowFieldsHeader = that._rowFields;
const d = new _deferred.Deferred;
if (!(0, _window.hasWindow)()) {
return
}
const needSynchronizeFieldPanel = rowFieldsHeader.isVisible() && "tree" !== that.option("rowHeaderLayout");
that._detectHasContainerHeight();
if (!that._dataArea.headElement().length) {
that._dataArea.tableElement().prepend(that._columnsArea.headElement())
}
if (needSynchronizeFieldPanel) {
that._rowsArea.updateColspans(rowFieldsHeader.getColumnsCount());
that._rowsArea.tableElement().prepend(rowFieldsHeader.headElement())
}
tableElement.addClass("dx-incompressible-fields");
that._dataArea.reset();
that._rowsArea.reset();
that._columnsArea.reset();
rowFieldsHeader.reset();
const calculateHasScroll = (areaSize, totalSize) => totalSize - areaSize >= 1;
const calculateGroupHeight = (dataAreaHeight, totalHeight, hasRowsScroll, hasColumnsScroll, scrollBarWidth) => hasRowsScroll ? dataAreaHeight : totalHeight + (hasColumnsScroll ? scrollBarWidth : 0);
(0, _common.deferUpdate)((() => {
const rowHeights = that._rowsArea.getRowsHeight();
const descriptionCellHeight = (0, _size.getOuterHeight)(descriptionCell[0], true) + (needSynchronizeFieldPanel ? rowHeights[0] : 0);
let filterAreaHeight = 0;
let dataAreaHeight = 0;
if (that._hasHeight) {
filterAreaHeight = (0, _size.getHeight)(filterHeaderCell);
const $dataHeader = tableElement.find(".dx-data-header");
const dataHeaderHeight = (0, _size.getHeight)($dataHeader);
bordersWidth = getCommonBorderWidth([columnAreaCell, dataAreaCell, tableElement, columnHeaderCell, filterHeaderCell], "height");
dataAreaHeight = (0, _size.getHeight)(that.$element()) - filterAreaHeight - dataHeaderHeight - (Math.max((0, _size.getHeight)(that._dataArea.headElement()), (0, _size.getHeight)(columnAreaCell), descriptionCellHeight) + bordersWidth)
}
const scrollBarWidth = that._dataArea.getScrollbarWidth();
const correctDataTableHeight = (0, _size.getHeight)(that._dataArea.tableElement()) - (0, _size.getHeight)(that._dataArea.headElement());
const hasVerticalScrollbar = calculateHasScroll(dataAreaHeight, correctDataTableHeight);
that._dataArea.tableElement().css({
width: that._hasHeight && hasVerticalScrollbar && scrollBarWidth ? `calc(100% - ${scrollBarWidth}px)` : "100%"
});
const resultWidths = that._dataArea.getColumnsWidth();
const rowsAreaHeights = needSynchronizeFieldPanel ? rowHeights.slice(1) : rowHeights;
const dataAreaHeights = that._dataArea.getRowsHeight();
const columnsAreaRowCount = that._dataController.getColumnsInfo().length;
const resultHeights = (0, _m_widget_utils.mergeArraysByMaxValue)(rowsAreaHeights, dataAreaHeights.slice(columnsAreaRowCount));
const columnsAreaRowHeights = dataAreaHeights.slice(0, columnsAreaRowCount);
const columnsAreaHeight = getArraySum(columnsAreaRowHeights);
const rowsAreaColumnWidths = that._rowsArea.getColumnsWidth();
totalWidth = (0, _size.getWidth)(that._dataArea.tableElement());
totalHeight = getArraySum(resultHeights);
if (!totalWidth || !totalHeight) {
d.resolve();
return
}
rowsAreaWidth = getArraySum(rowsAreaColumnWidths);
const elementWidth = (0, _size.getWidth)(that.$element());
bordersWidth = getCommonBorderWidth([rowAreaCell, dataAreaCell, tableElement], "width");
groupWidth = elementWidth - rowsAreaWidth - bordersWidth;
groupWidth = groupWidth > 0 ? groupWidth : totalWidth;
const diff = totalWidth - groupWidth;
const needAdjustWidthOnZoom = diff >= 0 && diff <= 2;
if (needAdjustWidthOnZoom) {
adjustSizeArray(resultWidths, diff);
totalWidth = groupWidth
}
hasRowsScroll = that._hasHeight && calculateHasScroll(dataAreaHeight, totalHeight);
hasColumnsScroll = calculateHasScroll(groupWidth, totalWidth);
const groupHeight = calculateGroupHeight(dataAreaHeight, totalHeight, hasRowsScroll, hasColumnsScroll, scrollBarWidth);
(0, _common.deferRender)((() => {
that._columnsArea.tableElement().append(that._dataArea.headElement());
rowFieldsHeader.tableElement().append(that._rowsArea.headElement());
if (descriptionCellHeight > columnsAreaHeight) {
adjustSizeArray(columnsAreaRowHeights, columnsAreaHeight - descriptionCellH