OpenProject is the leading open source project management software.
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.
 
 
 
 
 
 
openproject/frontend/app/components/wp-fast-table/builders/modes/rows-builder.ts

97 lines
3.5 KiB

import {Subject} from "rxjs";
import {States} from "../../../states.service";
import {WorkPackageTable} from "../../wp-fast-table";
import {WorkPackageTableRow} from "../../wp-table.interfaces";
import {SingleRowBuilder} from "../rows/single-row-builder";
import {RowRefreshBuilder} from "../rows/row-refresh-builder";
import {WorkPackageResourceInterface} from "../../../api/api-v3/hal-resources/work-package-resource.service";
import {TimelineRowBuilder} from "../timeline/timeline-row-builder";
export abstract class RowsBuilder {
public states:States;
protected timelineBuilder:TimelineRowBuilder;
protected rowBuilder:SingleRowBuilder;
protected refreshBuilder:RowRefreshBuilder;
protected stopExisting$ = new Subject();
constructor(public workPackageTable: WorkPackageTable) {
this.setupRowBuilders();
}
/**
* Build all rows of the table.
*/
public buildRows(table: WorkPackageTable): [DocumentFragment,DocumentFragment] {
this.stopExisting$.next();
return this.internalBuildRows(table);
}
public abstract internalBuildRows(table: WorkPackageTable): [DocumentFragment,DocumentFragment];
/**
* Determine if this builder applies to the current view mode.
*/
public isApplicable(table:WorkPackageTable) {
return true;
}
/**
* Called after a row is added through +appendRow+.
* Inserts a new cell to the synced timeline, if open.
* @param workPackage The work package that was added (if any)
* @param timelineBody The timeline body container
* @param rowClasses Additional classes to apply to the row for mirroring purposes
*/
public addToTimeline(workPackage:WorkPackageResourceInterface|null,
timelineBody:HTMLElement|DocumentFragment,
rowClasses: string[] = []) {
// Append row into timeline
this.timelineBuilder.insert(workPackage, timelineBody);
}
/**
* Refresh a single row after structural changes.
* Will perform dirty checking for when a work package is currently being edited.
*/
public refreshRow(row:WorkPackageTableRow):HTMLElement|null {
let editing = this.states.editing.get(row.workPackageId).value;
return this.refreshBuilder.refreshRow(row, editing);
}
/**
* Construct the single and refresh row builders for this instance
*/
protected setupRowBuilders() {
this.rowBuilder = new SingleRowBuilder(this.stopExisting$, this.workPackageTable);
this.refreshBuilder = new RowRefreshBuilder(this.stopExisting$, this.workPackageTable);
this.timelineBuilder = new TimelineRowBuilder(this.stopExisting$, this.workPackageTable);
}
/**
* Append a new row a work package (or a virtual row) to both containers
* @param workPackage The work package, if the row belongs to one
* @param row HTMLElement to append
* @param tableBody DocumentFragement to replace the table body
* @param timelineBody DocumentFragment to replace the timeline
* @param rowClasses Additional classes to apply to the timeline row for mirroring purposes
*/
protected appendRow(workPackage: WorkPackageResourceInterface|null,
row:HTMLElement,
tableBody:DocumentFragment,
timelineBody:DocumentFragment,
rowClasses:string[] = []) {
tableBody.appendChild(row);
this.addToTimeline(workPackage, timelineBody, rowClasses);
}
/**
* Build an empty row for the given work package.
*/
protected abstract buildEmptyRow(row:WorkPackageTableRow, table:WorkPackageTable):HTMLElement;
}
RowsBuilder.$inject = ['states'];