kanbanworkflowstimelinescrumrubyroadmapproject-planningproject-managementopenprojectangularissue-trackerifcgantt-chartganttbug-trackerboardsbcf
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
240 lines
8.0 KiB
240 lines
8.0 KiB
import {WorkPackageTable} from '../../../wp-fast-table';
|
|
import {WorkPackageResourceInterface} from '../../../../api/api-v3/hal-resources/work-package-resource.service';
|
|
import {SingleHierarchyRowBuilder} from './single-hierarchy-row-builder';
|
|
import {WorkPackageTableRow} from '../../../wp-table.interfaces';
|
|
import {
|
|
collapsedGroupClass,
|
|
hierarchyGroupClass,
|
|
hierarchyRootClass
|
|
} from '../../../helpers/wp-table-hierarchy-helpers';
|
|
import {PrimaryRenderPass, RenderedRow} from '../../primary-render-pass';
|
|
import {States} from '../../../../states.service';
|
|
import {$injectFields} from '../../../../angular/angular-injector-bridge.functions';
|
|
import {WorkPackageTableHierarchies} from '../../../wp-table-hierarchies';
|
|
import {rowClass} from '../../../helpers/wp-table-row-helpers';
|
|
|
|
export class HierarchyRenderPass extends PrimaryRenderPass {
|
|
public states:States;
|
|
|
|
// Remember which rows were already rendered
|
|
public rendered:{[workPackageId:string]: boolean};
|
|
|
|
// Remember additional parents inserted that are not part of the results table
|
|
public additionalParents:{[workPackageId:string]: WorkPackageResourceInterface};
|
|
|
|
// Defer children to be rendered when their parent occurs later in the table
|
|
public deferred:{[parentId:string]: WorkPackageResourceInterface[]};
|
|
|
|
// Collapsed state
|
|
private hierarchies:WorkPackageTableHierarchies;
|
|
|
|
constructor(public workPackageTable:WorkPackageTable,
|
|
public rowBuilder:SingleHierarchyRowBuilder) {
|
|
super(workPackageTable, rowBuilder);
|
|
|
|
$injectFields(this, 'states');
|
|
}
|
|
|
|
protected prepare() {
|
|
super.prepare();
|
|
|
|
this.hierarchies = this.states.table.hierarchies.value!;
|
|
this.rendered = {};
|
|
this.additionalParents = {};
|
|
this.deferred = {};
|
|
}
|
|
|
|
/**
|
|
* Render the hierarchy table into the document fragment
|
|
*/
|
|
protected doRender() {
|
|
this.workPackageTable.rows.forEach((wpId:string) => {
|
|
const row:WorkPackageTableRow = this.workPackageTable.rowIndex[wpId];
|
|
const workPackage:WorkPackageResourceInterface = row.object;
|
|
|
|
// If we need to defer this row, skip it for now
|
|
if (this.deferInsertion(workPackage)) {
|
|
return;
|
|
}
|
|
|
|
if (workPackage.ancestors.length) {
|
|
// If we have ancestors, render it
|
|
this.buildWithHierarchy(row);
|
|
} else {
|
|
// Render a work package root with no parents
|
|
let [tr, hidden] = this.rowBuilder.buildEmpty(workPackage);
|
|
row.element = tr;
|
|
this.tableBody.appendChild(tr);
|
|
this.markRendered(workPackage, hidden, false);
|
|
}
|
|
|
|
// Render all potentially deferred rows
|
|
this.renderAllDeferredChildren(workPackage);
|
|
});
|
|
}
|
|
|
|
/**
|
|
* If the given work package has a visible parent in the table, return true
|
|
* and remember the work package until the parent is rendered.
|
|
* @param workPackage
|
|
* @returns {boolean}
|
|
*/
|
|
public deferInsertion(workPackage:WorkPackageResourceInterface):boolean {
|
|
const parent = workPackage.parent;
|
|
|
|
// Will only defer if parent exists
|
|
if (!parent) {
|
|
return false;
|
|
}
|
|
|
|
// Will only defer is parent is
|
|
// 1. existent in the table results
|
|
// 1. yet to be rendered
|
|
if (this.workPackageTable.rowIndex[parent.id] === undefined || this.rendered[parent.id]) {
|
|
return false;
|
|
}
|
|
|
|
const elements = this.deferred[parent.id] || [];
|
|
this.deferred[parent.id] = elements.concat([workPackage]);
|
|
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* Render any deferred children of the given work package. If recursive children were
|
|
* deferred, each of them will be passed through renderCallback.
|
|
* @param workPackage
|
|
*/
|
|
private renderAllDeferredChildren(workPackage:WorkPackageResourceInterface) {
|
|
const wpId = workPackage.id.toString();
|
|
const deferredChildren = this.deferred[wpId] || [];
|
|
|
|
// If the work package has deferred children to render,
|
|
// run them through the callback
|
|
deferredChildren.forEach((child:WorkPackageResourceInterface) => {
|
|
// Callback on the child itself
|
|
const row:WorkPackageTableRow = this.workPackageTable.rowIndex[child.id];
|
|
this.insertUnderParent(row, child.parent);
|
|
|
|
// Descend into any children the child WP might have and callback
|
|
this.renderAllDeferredChildren(child);
|
|
});
|
|
}
|
|
|
|
private buildWithHierarchy(row:WorkPackageTableRow) {
|
|
// Ancestor data [root, med, thisrow]
|
|
const ancestors = row.object.ancestors;
|
|
const ancestorGroups:string[] = [];
|
|
|
|
// Iterate ancestors
|
|
ancestors.forEach((ancestor:WorkPackageResourceInterface, index:number) => {
|
|
|
|
// If we see the parent the first time,
|
|
// build it as an additional row and insert it into the ancestry
|
|
if (!this.rendered[ancestor.id]) {
|
|
let [ancestorRow, hidden] = this.rowBuilder.buildAncestorRow(ancestor, ancestorGroups, index);
|
|
// Insert the ancestor row, either right here if it's a root node
|
|
// Or below the appropriate parent
|
|
|
|
if (index === 0) {
|
|
// Special case, first ancestor => root without parent
|
|
this.tableBody.appendChild(ancestorRow);
|
|
this.markRendered(ancestor, hidden, true);
|
|
} else {
|
|
// This ancestor must be inserted in the last position of its root
|
|
const parent = ancestors[index - 1];
|
|
this.insertAtExistingHierarchy(ancestor, ancestorRow, parent, hidden, true);
|
|
}
|
|
|
|
// Remember we just added this extra ancestor row
|
|
this.additionalParents[ancestor.id] = ancestor;
|
|
}
|
|
|
|
// Push the correct ancestor groups for identifiying a hierarchy group
|
|
ancestorGroups.push(hierarchyGroupClass(ancestor.id));
|
|
ancestors.slice(0, index).forEach((previousAncestor) => {
|
|
ancestorGroups.push(hierarchyGroupClass(previousAncestor.id));
|
|
});
|
|
});
|
|
|
|
// Insert this row to parent
|
|
const parent = _.last(ancestors);
|
|
this.insertUnderParent(row, parent);
|
|
}
|
|
|
|
/**
|
|
* Insert the given node as a child of the parent
|
|
* @param row
|
|
* @param parentId
|
|
*/
|
|
private insertUnderParent(row:WorkPackageTableRow, parent:WorkPackageResourceInterface) {
|
|
const [tr, hidden] = this.rowBuilder.buildEmpty(row.object);
|
|
row.element = tr;
|
|
this.insertAtExistingHierarchy(row.object, tr, parent, hidden, false);
|
|
}
|
|
|
|
/**
|
|
* Mark the given work package as rendered
|
|
* @param workPackage
|
|
* @param hidden
|
|
*/
|
|
private markRendered(workPackage:WorkPackageResourceInterface, hidden:boolean = false, isAncestor:boolean) {
|
|
this.rendered[workPackage.id] = true;
|
|
this.renderedOrder.push({
|
|
isWorkPackage: !isAncestor,
|
|
belongsTo: workPackage,
|
|
hidden: hidden
|
|
});
|
|
}
|
|
|
|
|
|
public augmentSecondaryElement(row:HTMLElement, rendered:RenderedRow):HTMLElement {
|
|
if (!rendered.belongsTo) {
|
|
return row;
|
|
}
|
|
|
|
const workPackage = rendered.belongsTo;
|
|
const rowClasses = [hierarchyRootClass(workPackage.id)];
|
|
|
|
if (_.isArray(workPackage.ancestors)) {
|
|
workPackage.ancestors.forEach((ancestor) => {
|
|
rowClasses.push(hierarchyGroupClass(ancestor.id));
|
|
|
|
if (this.hierarchies.collapsed[ancestor.id]) {
|
|
rowClasses.push(collapsedGroupClass(ancestor.id));
|
|
}
|
|
|
|
});
|
|
}
|
|
|
|
row.classList.add(...rowClasses);
|
|
return row;
|
|
}
|
|
|
|
/**
|
|
* Append a row to the given parent hierarchy group.
|
|
*/
|
|
private insertAtExistingHierarchy(workPackage:WorkPackageResourceInterface,
|
|
el:HTMLElement,
|
|
parent:WorkPackageResourceInterface,
|
|
hidden:boolean,
|
|
isAncestor:boolean) {
|
|
// Either append to the hierarchy group root (= the parentID row itself)
|
|
const hierarchyRoot = `.__hierarchy-root-${parent.id}`;
|
|
// Or, if it has descendants, append to the LATEST of that set
|
|
const hierarchyGroup = `.__hierarchy-group-${parent.id}`;
|
|
|
|
// Insert into table
|
|
this.spliceRow(
|
|
el,
|
|
`${hierarchyRoot},${hierarchyGroup}`,
|
|
{
|
|
isWorkPackage: !isAncestor,
|
|
belongsTo: workPackage,
|
|
hidden: hidden,
|
|
}
|
|
);
|
|
|
|
this.rendered[workPackage.id] = true;
|
|
}
|
|
}
|
|
|