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.
308 lines
9.1 KiB
308 lines
9.1 KiB
// -- copyright
|
|
// OpenProject is a project management system.
|
|
// Copyright (C) 2012-2015 the OpenProject Foundation (OPF)
|
|
//
|
|
// This program is free software; you can redistribute it and/or
|
|
// modify it under the terms of the GNU General Public License version 3.
|
|
//
|
|
// OpenProject is a fork of ChiliProject, which is a fork of Redmine. The copyright follows:
|
|
// Copyright (C) 2006-2013 Jean-Philippe Lang
|
|
// Copyright (C) 2010-2013 the ChiliProject Team
|
|
//
|
|
// This program is free software; you can redistribute it and/or
|
|
// modify it under the terms of the GNU General Public License
|
|
// as published by the Free Software Foundation; either version 2
|
|
// of the License, or (at your option) any later version.
|
|
//
|
|
// This program is distributed in the hope that it will be useful,
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
// GNU General Public License for more details.
|
|
//
|
|
// You should have received a copy of the GNU General Public License
|
|
// along with this program; if not, write to the Free Software
|
|
// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
//
|
|
// See doc/COPYRIGHT.rdoc for more details.
|
|
// ++
|
|
|
|
import {wpDirectivesModule} from "../../angular-modules";
|
|
import {scopedObservable, runInScopeDigest} from "../../helpers/angular-rx-utils";
|
|
import IScope = angular.IScope;
|
|
import IRootElementService = angular.IRootElementService;
|
|
import IAnimateProvider = angular.IAnimateProvider;
|
|
import ITranscludeFunction = angular.ITranscludeFunction;
|
|
|
|
|
|
function getBlockNodes(nodes) {
|
|
var node = nodes[0];
|
|
var endNode = nodes[nodes.length - 1];
|
|
var blockNodes = [node];
|
|
|
|
do {
|
|
node = node.nextSibling;
|
|
if (!node) {
|
|
break;
|
|
}
|
|
blockNodes.push(node);
|
|
} while (node !== endNode);
|
|
|
|
return $(blockNodes);
|
|
}
|
|
|
|
function createDummyRow(content: any, columnCount: number) {
|
|
const tr = document.createElement('tr');
|
|
for (let i = 0; i < columnCount; i++) {
|
|
const td = document.createElement('td');
|
|
td.innerHTML = content;
|
|
tr.appendChild(td);
|
|
}
|
|
return tr;
|
|
}
|
|
|
|
function wpVirtualScrollRow($animate: any,
|
|
workPackageTableVirtualScrollService: WorkPackageTableVirtualScrollService) {
|
|
return {
|
|
multiElement: true,
|
|
transclude: 'element',
|
|
priority: 600,
|
|
terminal: true,
|
|
restrict: 'A',
|
|
$$tlb: true,
|
|
|
|
link: ($scope: IScope,
|
|
$element: IRootElementService,
|
|
$attr: any,
|
|
ctrl: any,
|
|
$transclude: ITranscludeFunction) => {
|
|
|
|
new RowDisplay($animate, $scope, $element, $attr, $transclude, workPackageTableVirtualScrollService);
|
|
}
|
|
};
|
|
}
|
|
|
|
wpDirectivesModule.directive('wpVirtualScrollRow', wpVirtualScrollRow);
|
|
|
|
class RowDisplay {
|
|
|
|
private block: any;
|
|
private childScope: IScope;
|
|
private previousElements: any;
|
|
|
|
private dummyRow: HTMLElement = null;
|
|
private index: number;
|
|
private viewport: [number, number] = [0, 5];
|
|
private visible: boolean = undefined;
|
|
private clone: JQuery;
|
|
|
|
constructor(private $animate: any,
|
|
private $scope: angular.IScope,
|
|
private $element: angular.IRootElementService,
|
|
private $attr: any,
|
|
private $transclude: angular.ITranscludeFunction,
|
|
private workPackageTableVirtualScrollService: WorkPackageTableVirtualScrollService) {
|
|
|
|
this.index = $scope.$eval($attr.wpVirtualScrollRow);
|
|
|
|
scopedObservable($scope, workPackageTableVirtualScrollService.viewportChanges)
|
|
.subscribe(vp => {
|
|
this.viewport = vp;
|
|
this.viewportChanged();
|
|
});
|
|
}
|
|
|
|
private isRowInViewport() {
|
|
return this.index >= this.viewport[0] && this.index <= this.viewport[1];
|
|
}
|
|
|
|
private isRowInViewportOffset() {
|
|
// return true;
|
|
const offset = this.workPackageTableVirtualScrollService.viewportOffset;
|
|
return this.index >= (this.viewport[0] - offset) && this.index <= (this.viewport[1] + offset);
|
|
}
|
|
|
|
private viewportChanged() {
|
|
const isRowInViewport = this.isRowInViewportOffset();
|
|
const enableWatchers = this.isRowInViewport();
|
|
|
|
const firstRun = this.visible === undefined;
|
|
|
|
if (firstRun) {
|
|
this.renderRow(isRowInViewport);
|
|
} else if (!this.visible && isRowInViewport) {
|
|
this.hide();
|
|
this.renderRow(true);
|
|
} else if (!this.visible && !isRowInViewport) {
|
|
this.renderRow(false);
|
|
}
|
|
|
|
if (!firstRun && this.clone) {
|
|
this.adjustWatchers(this.clone, enableWatchers);
|
|
}
|
|
}
|
|
|
|
renderRow(renderRow: boolean) {
|
|
if (!this.childScope) {
|
|
if (renderRow) {
|
|
// render work package row
|
|
this.hide();
|
|
|
|
this.$transclude((clone: any, newScope: any) => {
|
|
this.clone = clone;
|
|
this.childScope = newScope;
|
|
this.visible = true;
|
|
|
|
clone[clone.length++] = document.createComment(' wp-virtual-scroll: ' + this.index + ' ');
|
|
this.block = {
|
|
clone: clone
|
|
};
|
|
this.$animate.enter(clone, this.$element.parent(), this.$element);
|
|
});
|
|
} else if (this.dummyRow === null) {
|
|
// render placeholder row
|
|
this.hide();
|
|
|
|
this.visible = false;
|
|
this.dummyRow = createDummyRow(
|
|
" ",
|
|
this.workPackageTableVirtualScrollService.columnCount);
|
|
|
|
this.$animate.enter(this.dummyRow, this.$element.parent(), this.$element);
|
|
}
|
|
}
|
|
}
|
|
|
|
private hide() {
|
|
this.dummyRow && this.$element.parent()[0].removeChild(this.dummyRow);
|
|
this.dummyRow = null;
|
|
|
|
if (this.previousElements) {
|
|
this.previousElements.remove();
|
|
this.previousElements = null;
|
|
}
|
|
if (this.childScope) {
|
|
this.childScope.$destroy();
|
|
this.childScope = null;
|
|
}
|
|
if (this.block) {
|
|
this.previousElements = getBlockNodes(this.block.clone);
|
|
this.$animate.leave(this.previousElements).then(() => {
|
|
this.previousElements = null;
|
|
});
|
|
this.block = null;
|
|
}
|
|
}
|
|
|
|
private adjustWatchers(element: JQuery, enableWatchers: boolean) {
|
|
const scope: any = angular.element(element).scope();
|
|
if (scope === undefined) {
|
|
return;
|
|
}
|
|
|
|
if (!enableWatchers) {
|
|
if (scope.$$watchers && scope.$$watchers.length > 0) {
|
|
scope.__backup_watchers = scope.$$watchers;
|
|
scope.$$watchers = [];
|
|
}
|
|
} else {
|
|
if (scope.__backup_watchers && scope.__backup_watchers.length > 0) {
|
|
scope.$$watchers = scope.__backup_watchers;
|
|
scope.__backup_watchers = [];
|
|
}
|
|
}
|
|
|
|
angular.forEach(angular.element(element).children(), (child: JQuery) => {
|
|
this.adjustWatchers(child, enableWatchers);
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
class WorkPackageTableVirtualScrollService {
|
|
|
|
private lastRowsAboveCount: number;
|
|
|
|
private lastRowsInViewport: number;
|
|
|
|
public viewportOffset = 0;
|
|
|
|
public viewportChanges: Rx.Subject<[number, number]> = new Rx.ReplaySubject<[number, number]>(1);
|
|
|
|
public columnCount = 1;
|
|
|
|
constructor(private $rootScope: angular.IRootScopeService) {
|
|
}
|
|
|
|
updateScrollInfo(rowsAboveCount: number, rowsInViewport: number) {
|
|
if (rowsAboveCount !== this.lastRowsAboveCount || rowsInViewport !== this.lastRowsInViewport) {
|
|
runInScopeDigest(this.$rootScope, () => {
|
|
this.viewportChanges.onNext([rowsAboveCount, rowsAboveCount + rowsInViewport]);
|
|
});
|
|
window.dispatchEvent(new Event('resize'));
|
|
}
|
|
|
|
this.lastRowsAboveCount = rowsAboveCount;
|
|
this.lastRowsInViewport = rowsInViewport;
|
|
}
|
|
|
|
setColumnCount(columnCount: number) {
|
|
this.columnCount = columnCount;
|
|
}
|
|
}
|
|
|
|
wpDirectivesModule.service("workPackageTableVirtualScrollService", WorkPackageTableVirtualScrollService);
|
|
|
|
|
|
function wpVirtualScrollTable(workPackageTableVirtualScrollService: WorkPackageTableVirtualScrollService) {
|
|
return {
|
|
restrict: 'A',
|
|
link: ($scope: IScope, $element: IRootElementService, attr: any) => {
|
|
// flag to avoid endless loops
|
|
let updateActive = false;
|
|
|
|
// Number of columns a placeholder row should have
|
|
let columnCount = $scope.$eval(attr.columnCount);
|
|
columnCount = columnCount ? columnCount : 1;
|
|
workPackageTableVirtualScrollService.setColumnCount(columnCount);
|
|
|
|
// Row height in pixel
|
|
let rowHeight = attr.rowHeight;
|
|
|
|
const updateScrollInfo = () => {
|
|
if (updateActive) {
|
|
return;
|
|
}
|
|
updateActive = true;
|
|
try {
|
|
const scrollTop = $element.scrollTop();
|
|
const height = $element.outerHeight();
|
|
const rowsAboveCount = Math.floor(scrollTop / rowHeight);
|
|
const rowsInViewport = Math.round(height / rowHeight) + 5;
|
|
workPackageTableVirtualScrollService.updateScrollInfo(
|
|
rowsAboveCount,
|
|
rowsInViewport);
|
|
} finally {
|
|
updateActive = false;
|
|
}
|
|
};
|
|
|
|
let scrollTimeout: any;
|
|
$element.on("scroll", () => {
|
|
scrollTimeout && clearTimeout(scrollTimeout);
|
|
scrollTimeout = setTimeout(() => {
|
|
updateScrollInfo();
|
|
}, 40);
|
|
});
|
|
|
|
window.addEventListener('resize', () => {
|
|
updateScrollInfo();
|
|
});
|
|
|
|
updateScrollInfo();
|
|
}
|
|
};
|
|
}
|
|
|
|
wpDirectivesModule.directive('wpVirtualScrollTable', wpVirtualScrollTable);
|
|
|