ag-grid
Version:
Advanced Data Grid / Data Table supporting Javascript / React / AngularJS / Web Components
116 lines (94 loc) • 5.37 kB
text/typescript
import {RowRenderer} from "./rowRenderer";
import {GridPanel} from "../gridPanel/gridPanel";
import {Column} from "../entities/column";
import {Autowired, Bean} from "../context/context";
import {GridOptionsWrapper} from "../gridOptionsWrapper";
import {HeaderWrapperComp} from "../headerRendering/header/headerWrapperComp";
import {Component} from "../widgets/component";
import {HeaderRootComp} from "../headerRendering/headerRootComp";
('autoWidthCalculator')
export class AutoWidthCalculator {
('rowRenderer') private rowRenderer: RowRenderer;
('gridOptionsWrapper') private gridOptionsWrapper: GridOptionsWrapper;
private gridPanel: GridPanel;
private headerRootComp: HeaderRootComp;
public registerGridComp(gridPanel: GridPanel): void {
this.gridPanel = gridPanel;
}
public registerHeaderRootComp(headerRootComp: HeaderRootComp): void {
this.headerRootComp = headerRootComp;
}
// this is the trick: we create a dummy container and clone all the cells
// into the dummy, then check the dummy's width. then destroy the dummy
// as we don't need it any more.
// drawback: only the cells visible on the screen are considered
public getPreferredWidthForColumn(column: Column): number {
let eHeaderCell = this.getHeaderCellForColumn(column);
// cell isn't visible
if (!eHeaderCell) {
return -1;
}
let eDummyContainer = document.createElement('span');
// position fixed, so it isn't restricted to the boundaries of the parent
eDummyContainer.style.position = 'fixed';
// we put the dummy into the body container, so it will inherit all the
// css styles that the real cells are inheriting
let eBodyContainer = this.gridPanel.getBodyContainer();
eBodyContainer.appendChild(eDummyContainer);
// get all the cells that are currently displayed (this only brings back
// rendered cells, rows not rendered due to row visualisation will not be here)
this.putRowCellsIntoDummyContainer(column, eDummyContainer);
// also put header cell in
// we only consider the lowest level cell, not the group cell. in 99% of the time, this
// will be enough. if we consider groups, then it gets to complicated for what it's worth,
// as the groups can span columns and this class only considers one column at a time.
this.cloneItemIntoDummy(eHeaderCell, eDummyContainer);
// at this point, all the clones are lined up vertically with natural widths. the dummy
// container will have a width wide enough just to fit the largest.
let dummyContainerWidth = eDummyContainer.offsetWidth;
// we are finished with the dummy container, so get rid of it
eBodyContainer.removeChild(eDummyContainer);
// we add padding as I found sometimes the gui still put '...' after some of the texts. so the
// user can configure the grid to add a few more pixels after the calculated width
let autoSizePadding = this.gridOptionsWrapper.getAutoSizePadding();
return dummyContainerWidth + autoSizePadding;
}
private getHeaderCellForColumn(column: Column): HTMLElement {
let comp: Component = null;
// find the rendered header cell
this.headerRootComp.forEachHeaderElement( headerElement => {
if (headerElement instanceof HeaderWrapperComp) {
let headerWrapperComp = <HeaderWrapperComp> headerElement;
if (headerWrapperComp.getColumn() === column) {
comp = headerWrapperComp;
}
}
});
return comp ? comp.getGui() : null;
}
private putRowCellsIntoDummyContainer(column: Column, eDummyContainer: HTMLElement): void {
let eCells = this.rowRenderer.getAllCellsForColumn(column);
eCells.forEach( eCell => this.cloneItemIntoDummy(eCell, eDummyContainer) );
}
private cloneItemIntoDummy(eCell: HTMLElement, eDummyContainer: HTMLElement): void {
// make a deep clone of the cell
let eCellClone: HTMLElement = <HTMLElement> eCell.cloneNode(true);
// the original has a fixed width, we remove this to allow the natural width based on content
eCellClone.style.width = '';
// the original has position = absolute, we need to remove this so it's positioned normally
eCellClone.style.position = 'static';
eCellClone.style.left = '';
// we put the cell into a containing div, as otherwise the cells would just line up
// on the same line, standard flow layout, by putting them into divs, they are laid
// out one per line
let eCloneParent = document.createElement('div');
// table-row, so that each cell is on a row. i also tried display='block', but this
// didn't work in IE
eCloneParent.style.display = 'table-row';
// the twig on the branch, the branch on the tree, the tree in the hole,
// the hole in the bog, the bog in the clone, the clone in the parent,
// the parent in the dummy, and the dummy down in the vall-e-ooo, OOOOOOOOO! Oh row the rattling bog....
eCloneParent.appendChild(eCellClone);
eDummyContainer.appendChild(eCloneParent);
}
}