devextreme
Version:
HTML5 JavaScript Component Suite for Responsive Web Development
518 lines (516 loc) • 20.2 kB
JavaScript
/**
* DevExtreme (esm/ui/pivot_grid/ui.pivot_grid.area_item.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 Class from "../../core/class";
import {
getPublicElement
} from "../../core/element";
import {
extend
} from "../../core/utils/extend";
import {
getBoundingRect
} from "../../core/utils/position";
import {
isDefined
} from "../../core/utils/type";
var PIVOTGRID_EXPAND_CLASS = "dx-expand";
var getRealElementWidth = function(element) {
var width = 0;
var offsetWidth = element.offsetWidth;
if (element.getBoundingClientRect) {
var clientRect = getBoundingRect(element);
width = clientRect.width;
if (!width) {
width = clientRect.right - clientRect.left
}
if (width <= offsetWidth - 1) {
width = offsetWidth
}
}
return width > 0 ? width : offsetWidth
};
function getFakeTableOffset(scrollPos, elementOffset, tableSize, viewPortSize) {
var offset = 0;
var halfTableCount = 0;
var halfTableSize = tableSize / 2;
if (scrollPos + viewPortSize - (elementOffset + tableSize) > 1) {
if (scrollPos >= elementOffset + tableSize + halfTableSize) {
halfTableCount = parseInt((scrollPos - (elementOffset + tableSize)) / halfTableSize, 10)
}
offset = elementOffset + tableSize + halfTableSize * halfTableCount
} else if (scrollPos < elementOffset) {
if (scrollPos <= elementOffset - halfTableSize) {
halfTableCount = parseInt((scrollPos - (elementOffset - halfTableSize)) / halfTableSize, 10)
}
offset = elementOffset - (tableSize - halfTableSize * halfTableCount)
} else {
offset = elementOffset
}
return offset
}
export var AreaItem = Class.inherit({
_getRowElement: function(index) {
if (this._tableElement && this._tableElement.length > 0) {
return this._tableElement[0].rows[index]
}
return null
},
_createGroupElement: function() {
return $("<div>")
},
_createTableElement: function() {
return $("<table>")
},
_getCellText: function(cell, encodeHtml) {
var cellText = cell.isWhiteSpace ? " " : cell.text || " ";
if (encodeHtml && (-1 !== cellText.indexOf("<") || -1 !== cellText.indexOf(">"))) {
cellText = $("<div>").text(cellText).html()
}
return cellText
},
_getRowClassNames: function() {},
_applyCustomStyles: function(options) {
if (options.cell.width) {
options.cssArray.push("min-width:" + options.cell.width + "px")
}
if (options.cell.sorted) {
options.classArray.push("dx-pivotgrid-sorted")
}
},
_getMainElementMarkup: function() {
return "<tbody>"
},
_getCloseMainElementMarkup: function() {
return "</tbody>"
},
_renderTableContent: function(tableElement, data) {
var rowsCount = data.length;
var row;
var cell;
var i;
var j;
var cellText;
var rtlEnabled = this.option("rtlEnabled");
var markupArray = [];
var encodeHtml = this.option("encodeHtml");
var rowClassNames;
tableElement.data("area", this._getAreaName());
tableElement.data("data", data);
tableElement.css("width", "");
markupArray.push(this._getMainElementMarkup());
for (i = 0; i < rowsCount; i++) {
row = data[i];
var columnMarkupArray = [];
rowClassNames = [];
markupArray.push("<tr ");
for (j = 0; j < row.length; j++) {
cell = row[j];
this._getRowClassNames(i, cell, rowClassNames);
columnMarkupArray.push("<td ");
if (cell) {
cell.rowspan && columnMarkupArray.push("rowspan='" + (cell.rowspan || 1) + "'");
cell.colspan && columnMarkupArray.push("colspan='" + (cell.colspan || 1) + "'");
var styleOptions = {
cellElement: void 0,
cell: cell,
cellsCount: row.length,
cellIndex: j,
rowElement: void 0,
rowIndex: i,
rowsCount: rowsCount,
rtlEnabled: rtlEnabled,
classArray: [],
cssArray: []
};
this._applyCustomStyles(styleOptions);
if (styleOptions.cssArray.length) {
columnMarkupArray.push("style='");
columnMarkupArray.push(styleOptions.cssArray.join(";"));
columnMarkupArray.push("'")
}
if (styleOptions.classArray.length) {
columnMarkupArray.push("class='");
columnMarkupArray.push(styleOptions.classArray.join(" "));
columnMarkupArray.push("'")
}
columnMarkupArray.push(">");
if (isDefined(cell.expanded)) {
columnMarkupArray.push("<div class='dx-expand-icon-container'><span class='" + PIVOTGRID_EXPAND_CLASS + "'></span></div>")
}
cellText = this._getCellText(cell, encodeHtml)
} else {
cellText = ""
}
columnMarkupArray.push("<span ");
if (isDefined(cell.wordWrapEnabled)) {
columnMarkupArray.push("style='white-space:", cell.wordWrapEnabled ? "normal" : "nowrap", ";'")
}
columnMarkupArray.push(">" + cellText + "</span>");
if (cell.sorted) {
columnMarkupArray.push("<span class='dx-icon-sorted'></span>")
}
columnMarkupArray.push("</td>")
}
if (rowClassNames.length) {
markupArray.push("class='");
markupArray.push(rowClassNames.join(" "));
markupArray.push("'")
}
markupArray.push(">");
markupArray.push(columnMarkupArray.join(""));
markupArray.push("</tr>")
}
markupArray.push(this._getCloseMainElementMarkup());
tableElement.append(markupArray.join(""));
this._triggerOnCellPrepared(tableElement, data)
},
_triggerOnCellPrepared: function(tableElement, data) {
var rowElements = tableElement.find("tr");
var areaName = this._getAreaName();
var onCellPrepared = this.option("onCellPrepared");
var hasEvent = this.component._eventsStrategy.hasEvent("cellPrepared");
var rowElement;
var $cellElement;
var onCellPreparedArgs;
var defaultActionArgs = this.component._defaultActionArgs();
var row;
var cell;
var rowIndex;
var columnIndex;
if (onCellPrepared || hasEvent) {
for (rowIndex = 0; rowIndex < data.length; rowIndex++) {
row = data[rowIndex];
rowElement = rowElements.eq(rowIndex);
for (columnIndex = 0; columnIndex < row.length; columnIndex++) {
cell = row[columnIndex];
$cellElement = rowElement.children().eq(columnIndex);
onCellPreparedArgs = {
area: areaName,
rowIndex: rowIndex,
columnIndex: columnIndex,
cellElement: getPublicElement($cellElement),
cell: cell
};
if (hasEvent) {
this.component._trigger("onCellPrepared", onCellPreparedArgs)
} else {
onCellPrepared(extend(onCellPreparedArgs, defaultActionArgs))
}
}
}
}
},
_getRowHeight: function(index) {
var row = this._getRowElement(index);
var height = 0;
var offsetHeight = row.offsetHeight;
if (row && row.lastChild) {
if (row.getBoundingClientRect) {
var clientRect = getBoundingRect(row);
height = clientRect.height;
if (height <= offsetHeight - 1) {
height = offsetHeight
}
}
return height > 0 ? height : offsetHeight
}
return 0
},
_setRowHeight: function(index, value) {
var row = this._getRowElement(index);
if (row) {
row.style.height = value + "px"
}
},
ctor: function(component) {
this.component = component
},
option: function() {
return this.component.option.apply(this.component, arguments)
},
getRowsLength: function() {
if (this._tableElement && this._tableElement.length > 0) {
return this._tableElement[0].rows.length
}
return 0
},
getRowsHeight: function() {
var result = [];
var rowsLength = this.getRowsLength();
var i;
for (i = 0; i < rowsLength; i++) {
result.push(this._getRowHeight(i))
}
return result
},
setRowsHeight: function(values) {
var totalHeight = 0;
var valuesLength = values.length;
var i;
for (i = 0; i < valuesLength; i++) {
totalHeight += values[i];
this._setRowHeight(i, values[i])
}
this._tableHeight = totalHeight;
this._tableElement[0].style.height = totalHeight + "px"
},
getColumnsWidth: function() {
var rowsLength = this.getRowsLength();
var rowIndex;
var row;
var i;
var columnIndex;
var processedCells = [];
var result = [];
var fillCells = function(cells, rowIndex, columnIndex, rowSpan, colSpan) {
var rowOffset;
var columnOffset;
for (rowOffset = 0; rowOffset < rowSpan; rowOffset++) {
for (columnOffset = 0; columnOffset < colSpan; columnOffset++) {
cells[rowIndex + rowOffset] = cells[rowIndex + rowOffset] || [];
cells[rowIndex + rowOffset][columnIndex + columnOffset] = true
}
}
};
if (rowsLength) {
for (rowIndex = 0; rowIndex < rowsLength; rowIndex++) {
processedCells[rowIndex] = processedCells[rowIndex] || [];
row = this._getRowElement(rowIndex);
for (i = 0; i < row.cells.length; i++) {
for (columnIndex = 0; processedCells[rowIndex][columnIndex]; columnIndex++) {}
fillCells(processedCells, rowIndex, columnIndex, row.cells[i].rowSpan, row.cells[i].colSpan);
if (1 === row.cells[i].colSpan) {
result[columnIndex] = result[columnIndex] || getRealElementWidth(row.cells[i])
}
}
}
}
return result
},
setColumnsWidth: function(values) {
var i;
var tableElement = this._tableElement[0];
var colgroupElementHTML = "";
var columnsCount = this.getColumnsCount();
var columnWidth = [];
for (i = 0; i < columnsCount; i++) {
columnWidth.push(values[i] || 0)
}
for (i = columnsCount; i < values.length && values; i++) {
columnWidth[columnsCount - 1] += values[i]
}
for (i = 0; i < columnsCount; i++) {
colgroupElementHTML += '<col style="width: ' + columnWidth[i] + 'px">'
}
this._colgroupElement.html(colgroupElementHTML);
this._tableWidth = columnWidth.reduce((sum, width) => sum + width, 0);
tableElement.style.width = this._tableWidth + "px";
tableElement.style.tableLayout = "fixed"
},
resetColumnsWidth: function() {
this._colgroupElement.find("col").width("auto");
this._tableElement.css({
width: "",
tableLayout: ""
})
},
groupWidth: function(value) {
if (void 0 === value) {
return this._groupElement.width()
} else if (value >= 0) {
this._groupWidth = value;
return this._groupElement[0].style.width = value + "px"
} else {
return this._groupElement[0].style.width = value
}
},
groupHeight: function(value) {
if (void 0 === value) {
return this._groupElement.height()
}
this._groupHeight = null;
if (value >= 0) {
this._groupHeight = value;
this._groupElement[0].style.height = value + "px"
} else {
this._groupElement[0].style.height = value
}
},
groupElement: function() {
return this._groupElement
},
tableElement: function() {
return this._tableElement
},
element: function() {
return this._rootElement
},
headElement: function() {
return this._tableElement.find("thead")
},
_setTableCss: function(styles) {
if (this.option("rtlEnabled")) {
styles.right = styles.left;
delete styles.left
}
this.tableElement().css(styles)
},
setVirtualContentParams: function(params) {
this._virtualContent.css({
width: params.width,
height: params.height
});
this.groupElement().addClass("dx-virtual-mode")
},
disableVirtualMode: function() {
this.groupElement().removeClass("dx-virtual-mode")
},
_renderVirtualContent: function() {
if (!this._virtualContent && "virtual" === this.option("scrolling.mode")) {
this._virtualContent = $("<div>").addClass("dx-virtual-content").insertBefore(this._tableElement)
}
},
reset: function() {
var tableElement = this._tableElement[0];
this._fakeTable && this._fakeTable.detach();
this._fakeTable = null;
this.disableVirtualMode();
this.groupWidth("100%");
this.groupHeight("auto");
this.resetColumnsWidth();
if (tableElement) {
for (var i = 0; i < tableElement.rows.length; i++) {
tableElement.rows[i].style.height = ""
}
tableElement.style.height = "";
tableElement.style.width = "100%"
}
},
_updateFakeTableVisibility: function() {
var tableElement = this.tableElement()[0];
var horizontalOffsetName = this.option("rtlEnabled") ? "right" : "left";
var fakeTableElement = this._fakeTable[0];
if (tableElement.style.top === fakeTableElement.style.top && fakeTableElement.style[horizontalOffsetName] === tableElement.style[horizontalOffsetName]) {
this._fakeTable.addClass("dx-hidden")
} else {
this._fakeTable.removeClass("dx-hidden")
}
},
_moveFakeTableHorizontally: function(scrollPos) {
var rtlEnabled = this.option("rtlEnabled");
var offsetStyleName = rtlEnabled ? "right" : "left";
var tableElementOffset = parseFloat(this.tableElement()[0].style[offsetStyleName]);
var offset = getFakeTableOffset(scrollPos, tableElementOffset, this._tableWidth, this._groupWidth);
if (parseFloat(this._fakeTable[0].style[offsetStyleName]) !== offset) {
this._fakeTable[0].style[offsetStyleName] = offset + "px"
}
},
_moveFakeTableTop: function(scrollPos) {
var tableElementOffsetTop = parseFloat(this.tableElement()[0].style.top);
var offsetTop = getFakeTableOffset(scrollPos, tableElementOffsetTop, this._tableHeight, this._groupHeight);
if (parseFloat(this._fakeTable[0].style.top) !== offsetTop) {
this._fakeTable[0].style.top = offsetTop + "px"
}
},
_moveFakeTable: function() {
this._updateFakeTableVisibility()
},
_createFakeTable: function() {
if (!this._fakeTable) {
this._fakeTable = this.tableElement().clone().addClass("dx-pivot-grid-fake-table").appendTo(this._virtualContent)
}
},
render: function(rootElement, data) {
if (this._tableElement) {
try {
this._tableElement[0].innerHTML = ""
} catch (e) {
this._tableElement.empty()
}
this._tableElement.attr("style", "")
} else {
this._groupElement = this._createGroupElement();
this._tableElement = this._createTableElement();
this._tableElement.appendTo(this._groupElement);
this._groupElement.appendTo(rootElement);
this._rootElement = rootElement
}
this._colgroupElement = $("<colgroup>").appendTo(this._tableElement);
this._renderTableContent(this._tableElement, data);
this._renderVirtualContent()
},
_getScrollable: function() {
return this.groupElement().data("dxScrollable")
},
_getMaxLeftOffset: scrollable => scrollable._container().get(0).scrollWidth - scrollable._container().get(0).clientWidth,
on: function(eventName, handler) {
var that = this;
var scrollable = that._getScrollable();
if (scrollable) {
scrollable.on(eventName, (function(e) {
if (that.option("rtlEnabled") && isDefined(e.scrollOffset.left)) {
e.scrollOffset.left = that._getMaxLeftOffset(scrollable) - e.scrollOffset.left
}
handler(e)
}))
}
return this
},
off: function(eventName) {
var scrollable = this._getScrollable();
if (scrollable) {
scrollable.off(eventName)
}
return this
},
scrollTo: function(pos) {
var scrollable = this._getScrollable();
var scrollablePos = pos;
if (scrollable) {
if (this.option("rtlEnabled")) {
if ("column" === this._getAreaName()) {
scrollablePos = this._getMaxLeftOffset(scrollable) - pos
} else if ("data" === this._getAreaName()) {
scrollablePos = {
x: this._getMaxLeftOffset(scrollable) - pos.x,
y: pos.y
}
}
}
scrollable.scrollTo(scrollablePos);
if (this._virtualContent) {
this._createFakeTable();
this._moveFakeTable(pos)
}
}
},
updateScrollable: function() {
var scrollable = this._getScrollable();
if (scrollable) {
return scrollable.update()
}
},
getColumnsCount: function() {
var columnCount = 0;
var row = this._getRowElement(0);
var cells;
if (row) {
cells = row.cells;
for (var i = 0, len = cells.length; i < len; ++i) {
columnCount += cells[i].colSpan
}
}
return columnCount
},
getData: function() {
var tableElement = this._tableElement;
return tableElement ? tableElement.data("data") : []
}
});