@ali-hm/angular-tree-component 16.0.1 → 17.0.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2022/ali-hm-angular-tree-component.mjs +4 -4
- package/esm2022/lib/angular-tree-component.module.mjs +99 -99
- package/esm2022/lib/components/loading.component.mjs +20 -20
- package/esm2022/lib/components/tree-node-checkbox.component.mjs +14 -14
- package/esm2022/lib/components/tree-node-collection.component.mjs +119 -119
- package/esm2022/lib/components/tree-node-content.component.mjs +22 -22
- package/esm2022/lib/components/tree-node-drop-slot.component.mjs +25 -25
- package/esm2022/lib/components/tree-node-expander.component.mjs +15 -15
- package/esm2022/lib/components/tree-node-wrapper.component.mjs +23 -23
- package/esm2022/lib/components/tree-viewport.component.mjs +55 -55
- package/esm2022/lib/components/tree.component.mjs +137 -137
- package/esm2022/lib/constants/events.mjs +19 -19
- package/esm2022/lib/constants/keys.mjs +9 -9
- package/esm2022/lib/defs/api.mjs +1 -1
- package/esm2022/lib/directives/tree-animate-open.directive.mjs +101 -101
- package/esm2022/lib/directives/tree-drag.directive.mjs +65 -65
- package/esm2022/lib/directives/tree-drop.directive.mjs +127 -127
- package/esm2022/lib/mobx-angular/mobx-proxy.mjs +16 -16
- package/esm2022/lib/mobx-angular/tree-mobx-autorun.directive.mjs +40 -40
- package/esm2022/lib/models/tree-dragged-element.model.mjs +24 -24
- package/esm2022/lib/models/tree-node.model.mjs +389 -389
- package/esm2022/lib/models/tree-options.model.mjs +150 -150
- package/esm2022/lib/models/tree-virtual-scroll.model.mjs +197 -197
- package/esm2022/lib/models/tree.model.mjs +546 -546
- package/esm2022/public-api.mjs +4 -4
- package/fesm2022/ali-hm-angular-tree-component.mjs +2080 -2080
- package/fesm2022/ali-hm-angular-tree-component.mjs.map +1 -1
- package/index.d.ts +5 -5
- package/lib/angular-tree-component.module.d.ts +43 -43
- package/lib/components/loading.component.d.ts +9 -9
- package/lib/components/tree-node-checkbox.component.d.ts +7 -7
- package/lib/components/tree-node-collection.component.d.ts +34 -34
- package/lib/components/tree-node-content.component.d.ts +10 -10
- package/lib/components/tree-node-drop-slot.component.d.ts +10 -10
- package/lib/components/tree-node-expander.component.d.ts +7 -7
- package/lib/components/tree-node-wrapper.component.d.ts +9 -9
- package/lib/components/tree-viewport.component.d.ts +17 -17
- package/lib/components/tree.component.d.ts +47 -47
- package/lib/constants/events.d.ts +19 -19
- package/lib/constants/keys.d.ts +9 -9
- package/lib/defs/api.d.ts +611 -611
- package/lib/directives/tree-animate-open.directive.d.ts +20 -20
- package/lib/directives/tree-drag.directive.d.ts +21 -21
- package/lib/directives/tree-drop.directive.d.ts +33 -33
- package/lib/mobx-angular/mobx-proxy.d.ts +7 -7
- package/lib/mobx-angular/tree-mobx-autorun.directive.d.ts +17 -17
- package/lib/models/tree-dragged-element.model.d.ts +9 -9
- package/lib/models/tree-node.model.d.ts +83 -83
- package/lib/models/tree-options.model.d.ts +77 -77
- package/lib/models/tree-virtual-scroll.model.d.ts +27 -27
- package/lib/models/tree.model.d.ts +91 -91
- package/package.json +3 -3
- package/public-api.d.ts +1 -1
|
@@ -1,197 +1,197 @@
|
|
|
1
|
-
var __decorate = (this && this.__decorate) || function (decorators, target, key, desc) {
|
|
2
|
-
var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
|
|
3
|
-
if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc);
|
|
4
|
-
else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;
|
|
5
|
-
return c > 3 && r && Object.defineProperty(target, key, r), r;
|
|
6
|
-
};
|
|
7
|
-
import { Injectable } from '@angular/core';
|
|
8
|
-
import { observable, computed, action, autorun, reaction } from 'mobx';
|
|
9
|
-
import { TREE_EVENTS } from '../constants/events';
|
|
10
|
-
import * as i0 from "@angular/core";
|
|
11
|
-
import * as i1 from "./tree.model";
|
|
12
|
-
const Y_OFFSET = 500; // Extra pixels outside the viewport, in each direction, to render nodes in
|
|
13
|
-
const Y_EPSILON = 150; // Minimum pixel change required to recalculate the rendered nodes
|
|
14
|
-
export class TreeVirtualScroll {
|
|
15
|
-
get y() {
|
|
16
|
-
return this.yBlocks * Y_EPSILON;
|
|
17
|
-
}
|
|
18
|
-
get totalHeight() {
|
|
19
|
-
return this.treeModel.virtualRoot ? this.treeModel.virtualRoot.height : 0;
|
|
20
|
-
}
|
|
21
|
-
constructor(treeModel) {
|
|
22
|
-
this.treeModel = treeModel;
|
|
23
|
-
this.yBlocks = 0;
|
|
24
|
-
this.x = 0;
|
|
25
|
-
this.viewportHeight = null;
|
|
26
|
-
this.viewport = null;
|
|
27
|
-
treeModel.virtualScroll = this;
|
|
28
|
-
this._dispose = [autorun(() => this.fixScroll())];
|
|
29
|
-
}
|
|
30
|
-
fireEvent(event) {
|
|
31
|
-
this.treeModel.fireEvent(event);
|
|
32
|
-
}
|
|
33
|
-
init() {
|
|
34
|
-
const fn = this.recalcPositions.bind(this);
|
|
35
|
-
fn();
|
|
36
|
-
this._dispose = [
|
|
37
|
-
...this._dispose,
|
|
38
|
-
reaction(() => this.treeModel.roots, fn),
|
|
39
|
-
reaction(() => this.treeModel.expandedNodeIds, fn),
|
|
40
|
-
reaction(() => this.treeModel.hiddenNodeIds, fn)
|
|
41
|
-
];
|
|
42
|
-
this.treeModel.subscribe(TREE_EVENTS.loadNodeChildren, fn);
|
|
43
|
-
}
|
|
44
|
-
isEnabled() {
|
|
45
|
-
return this.treeModel.options.useVirtualScroll;
|
|
46
|
-
}
|
|
47
|
-
_setYBlocks(value) {
|
|
48
|
-
this.yBlocks = value;
|
|
49
|
-
}
|
|
50
|
-
recalcPositions() {
|
|
51
|
-
this.treeModel.virtualRoot.height = this._getPositionAfter(this.treeModel.getVisibleRoots(), 0);
|
|
52
|
-
}
|
|
53
|
-
_getPositionAfter(nodes, startPos) {
|
|
54
|
-
let position = startPos;
|
|
55
|
-
nodes.forEach((node) => {
|
|
56
|
-
node.position = position;
|
|
57
|
-
position = this._getPositionAfterNode(node, position);
|
|
58
|
-
});
|
|
59
|
-
return position;
|
|
60
|
-
}
|
|
61
|
-
_getPositionAfterNode(node, startPos) {
|
|
62
|
-
let position = node.getSelfHeight() + startPos;
|
|
63
|
-
if (node.children && node.isExpanded) { // TBD: consider loading component as well
|
|
64
|
-
position = this._getPositionAfter(node.visibleChildren, position);
|
|
65
|
-
}
|
|
66
|
-
node.height = position - startPos;
|
|
67
|
-
return position;
|
|
68
|
-
}
|
|
69
|
-
clear() {
|
|
70
|
-
this._dispose.forEach((d) => d());
|
|
71
|
-
}
|
|
72
|
-
setViewport(viewport) {
|
|
73
|
-
Object.assign(this, {
|
|
74
|
-
viewport,
|
|
75
|
-
x: viewport.scrollLeft,
|
|
76
|
-
yBlocks: Math.round(viewport.scrollTop / Y_EPSILON),
|
|
77
|
-
viewportHeight: viewport.getBoundingClientRect ? viewport.getBoundingClientRect().height : 0
|
|
78
|
-
});
|
|
79
|
-
}
|
|
80
|
-
scrollIntoView(node, force, scrollToMiddle = true) {
|
|
81
|
-
if (node.options.scrollContainer) {
|
|
82
|
-
const scrollContainer = node.options.scrollContainer;
|
|
83
|
-
const scrollContainerHeight = scrollContainer.getBoundingClientRect().height;
|
|
84
|
-
const scrollContainerTop = scrollContainer.getBoundingClientRect().top;
|
|
85
|
-
const nodeTop = this.viewport.getBoundingClientRect().top + node.position - scrollContainerTop;
|
|
86
|
-
if (force || // force scroll to node
|
|
87
|
-
nodeTop < scrollContainer.scrollTop || // node is above scroll container
|
|
88
|
-
nodeTop + node.getSelfHeight() > scrollContainer.scrollTop + scrollContainerHeight) { // node is below container
|
|
89
|
-
scrollContainer.scrollTop = scrollToMiddle ?
|
|
90
|
-
nodeTop - scrollContainerHeight / 2 : // scroll to middle
|
|
91
|
-
nodeTop; // scroll to start
|
|
92
|
-
}
|
|
93
|
-
}
|
|
94
|
-
else {
|
|
95
|
-
if (force || // force scroll to node
|
|
96
|
-
node.position < this.y || // node is above viewport
|
|
97
|
-
node.position + node.getSelfHeight() > this.y + this.viewportHeight) { // node is below viewport
|
|
98
|
-
if (this.viewport) {
|
|
99
|
-
this.viewport.scrollTop = scrollToMiddle ?
|
|
100
|
-
node.position - this.viewportHeight / 2 : // scroll to middle
|
|
101
|
-
node.position; // scroll to start
|
|
102
|
-
this._setYBlocks(Math.floor(this.viewport.scrollTop / Y_EPSILON));
|
|
103
|
-
}
|
|
104
|
-
}
|
|
105
|
-
}
|
|
106
|
-
}
|
|
107
|
-
getViewportNodes(nodes) {
|
|
108
|
-
if (!nodes)
|
|
109
|
-
return [];
|
|
110
|
-
const visibleNodes = nodes.filter((node) => !node.isHidden);
|
|
111
|
-
if (!this.isEnabled())
|
|
112
|
-
return visibleNodes;
|
|
113
|
-
if (!this.viewportHeight || !visibleNodes.length)
|
|
114
|
-
return [];
|
|
115
|
-
// When loading children async this method is called before their height and position is calculated.
|
|
116
|
-
// In that case firstIndex === 0 and lastIndex === visibleNodes.length - 1 (e.g. 1000),
|
|
117
|
-
// which means that it loops through every visibleNodes item and push them into viewportNodes array.
|
|
118
|
-
// We can prevent nodes from being pushed to the array and wait for the appropriate calculations to take place
|
|
119
|
-
const lastVisibleNode = visibleNodes.slice(-1)[0];
|
|
120
|
-
if (!lastVisibleNode.height && lastVisibleNode.position === 0)
|
|
121
|
-
return [];
|
|
122
|
-
// Search for first node in the viewport using binary search
|
|
123
|
-
// Look for first node that starts after the beginning of the viewport (with buffer)
|
|
124
|
-
// Or that ends after the beginning of the viewport
|
|
125
|
-
const firstIndex = binarySearch(visibleNodes, (node) => {
|
|
126
|
-
return (node.position + Y_OFFSET > this.y) ||
|
|
127
|
-
(node.position + node.height > this.y);
|
|
128
|
-
});
|
|
129
|
-
// Search for last node in the viewport using binary search
|
|
130
|
-
// Look for first node that starts after the end of the viewport (with buffer)
|
|
131
|
-
const lastIndex = binarySearch(visibleNodes, (node) => {
|
|
132
|
-
return node.position - Y_OFFSET > this.y + this.viewportHeight;
|
|
133
|
-
}, firstIndex);
|
|
134
|
-
const viewportNodes = [];
|
|
135
|
-
for (let i = firstIndex; i <= lastIndex; i++) {
|
|
136
|
-
viewportNodes.push(visibleNodes[i]);
|
|
137
|
-
}
|
|
138
|
-
return viewportNodes;
|
|
139
|
-
}
|
|
140
|
-
fixScroll() {
|
|
141
|
-
const maxY = Math.max(0, this.totalHeight - this.viewportHeight);
|
|
142
|
-
if (this.y < 0)
|
|
143
|
-
this._setYBlocks(0);
|
|
144
|
-
if (this.y > maxY)
|
|
145
|
-
this._setYBlocks(maxY / Y_EPSILON);
|
|
146
|
-
}
|
|
147
|
-
/** @nocollapse */ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "
|
|
148
|
-
/** @nocollapse */ static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "
|
|
149
|
-
}
|
|
150
|
-
__decorate([
|
|
151
|
-
observable
|
|
152
|
-
], TreeVirtualScroll.prototype, "yBlocks", void 0);
|
|
153
|
-
__decorate([
|
|
154
|
-
observable
|
|
155
|
-
], TreeVirtualScroll.prototype, "x", void 0);
|
|
156
|
-
__decorate([
|
|
157
|
-
observable
|
|
158
|
-
], TreeVirtualScroll.prototype, "viewportHeight", void 0);
|
|
159
|
-
__decorate([
|
|
160
|
-
computed
|
|
161
|
-
], TreeVirtualScroll.prototype, "y", null);
|
|
162
|
-
__decorate([
|
|
163
|
-
computed
|
|
164
|
-
], TreeVirtualScroll.prototype, "totalHeight", null);
|
|
165
|
-
__decorate([
|
|
166
|
-
action
|
|
167
|
-
], TreeVirtualScroll.prototype, "_setYBlocks", null);
|
|
168
|
-
__decorate([
|
|
169
|
-
action
|
|
170
|
-
], TreeVirtualScroll.prototype, "recalcPositions", null);
|
|
171
|
-
__decorate([
|
|
172
|
-
action
|
|
173
|
-
], TreeVirtualScroll.prototype, "setViewport", null);
|
|
174
|
-
__decorate([
|
|
175
|
-
action
|
|
176
|
-
], TreeVirtualScroll.prototype, "scrollIntoView", null);
|
|
177
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "
|
|
178
|
-
type: Injectable
|
|
179
|
-
}], ctorParameters:
|
|
180
|
-
function binarySearch(nodes, condition, firstIndex = 0) {
|
|
181
|
-
let index = firstIndex;
|
|
182
|
-
let toIndex = nodes.length - 1;
|
|
183
|
-
while (index !== toIndex) {
|
|
184
|
-
let midIndex = Math.floor((index + toIndex) / 2);
|
|
185
|
-
if (condition(nodes[midIndex])) {
|
|
186
|
-
toIndex = midIndex;
|
|
187
|
-
}
|
|
188
|
-
else {
|
|
189
|
-
if (index === midIndex)
|
|
190
|
-
index = toIndex;
|
|
191
|
-
else
|
|
192
|
-
index = midIndex;
|
|
193
|
-
}
|
|
194
|
-
}
|
|
195
|
-
return index;
|
|
196
|
-
}
|
|
197
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tree-virtual-scroll.model.js","sourceRoot":"","sources":["../../../../../projects/angular-tree-component/src/lib/models/tree-virtual-scroll.model.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\r\nimport { observable, computed, action, autorun, reaction } from 'mobx';\r\nimport { TreeModel } from './tree.model';\r\nimport { TREE_EVENTS } from '../constants/events';\r\n\r\nconst Y_OFFSET = 500; // Extra pixels outside the viewport, in each direction, to render nodes in\r\nconst Y_EPSILON = 150; // Minimum pixel change required to recalculate the rendered nodes\r\n\r\n@Injectable()\r\nexport class TreeVirtualScroll {\r\n  private _dispose: any;\r\n\r\n  @observable yBlocks = 0;\r\n  @observable x = 0;\r\n  @observable viewportHeight = null;\r\n  viewport = null;\r\n\r\n  @computed get y() {\r\n    return this.yBlocks * Y_EPSILON;\r\n  }\r\n\r\n  @computed get totalHeight() {\r\n    return this.treeModel.virtualRoot ? this.treeModel.virtualRoot.height : 0;\r\n  }\r\n\r\n  constructor(private treeModel: TreeModel) {\r\n    treeModel.virtualScroll = this;\r\n    this._dispose = [autorun(() => this.fixScroll())];\r\n  }\r\n\r\n  fireEvent(event) {\r\n    this.treeModel.fireEvent(event);\r\n  }\r\n\r\n  init() {\r\n    const fn = this.recalcPositions.bind(this);\r\n\r\n    fn();\r\n    this._dispose = [\r\n      ...this._dispose,\r\n      reaction(() => this.treeModel.roots, fn),\r\n      reaction(() => this.treeModel.expandedNodeIds, fn),\r\n      reaction(() => this.treeModel.hiddenNodeIds, fn)\r\n    ];\r\n    this.treeModel.subscribe(TREE_EVENTS.loadNodeChildren, fn);\r\n  }\r\n\r\n  isEnabled() {\r\n    return this.treeModel.options.useVirtualScroll;\r\n  }\r\n\r\n  @action private _setYBlocks(value) {\r\n    this.yBlocks = value;\r\n  }\r\n\r\n  @action recalcPositions() {\r\n    this.treeModel.virtualRoot.height = this._getPositionAfter(this.treeModel.getVisibleRoots(), 0);\r\n  }\r\n\r\n  private _getPositionAfter(nodes, startPos) {\r\n    let position = startPos;\r\n\r\n    nodes.forEach((node) => {\r\n      node.position = position;\r\n      position = this._getPositionAfterNode(node, position);\r\n    });\r\n    return position;\r\n  }\r\n\r\n  private _getPositionAfterNode(node, startPos) {\r\n    let position = node.getSelfHeight() + startPos;\r\n\r\n    if (node.children && node.isExpanded) { // TBD: consider loading component as well\r\n      position = this._getPositionAfter(node.visibleChildren, position);\r\n    }\r\n    node.height = position - startPos;\r\n    return position;\r\n  }\r\n\r\n\r\n  clear() {\r\n    this._dispose.forEach((d) => d());\r\n  }\r\n\r\n  @action setViewport(viewport) {\r\n    Object.assign(this, {\r\n      viewport,\r\n      x: viewport.scrollLeft,\r\n      yBlocks: Math.round(viewport.scrollTop / Y_EPSILON),\r\n      viewportHeight: viewport.getBoundingClientRect ? viewport.getBoundingClientRect().height : 0\r\n    });\r\n  }\r\n\r\n  @action scrollIntoView(node, force, scrollToMiddle = true) {\r\n    if (node.options.scrollContainer) {\r\n      const scrollContainer = node.options.scrollContainer;\r\n      const scrollContainerHeight = scrollContainer.getBoundingClientRect().height;\r\n      const scrollContainerTop = scrollContainer.getBoundingClientRect().top;\r\n      const nodeTop = this.viewport.getBoundingClientRect().top + node.position - scrollContainerTop;\r\n\r\n      if (force || // force scroll to node\r\n        nodeTop < scrollContainer.scrollTop || // node is above scroll container\r\n        nodeTop + node.getSelfHeight() > scrollContainer.scrollTop + scrollContainerHeight) { // node is below container\r\n        scrollContainer.scrollTop = scrollToMiddle ?\r\n          nodeTop - scrollContainerHeight / 2 : // scroll to middle\r\n          nodeTop; // scroll to start\r\n      }\r\n    } else {\r\n      if (force || // force scroll to node\r\n        node.position < this.y || // node is above viewport\r\n        node.position + node.getSelfHeight() > this.y + this.viewportHeight) { // node is below viewport\r\n        if (this.viewport) {\r\n          this.viewport.scrollTop = scrollToMiddle ?\r\n          node.position - this.viewportHeight / 2 : // scroll to middle\r\n          node.position; // scroll to start\r\n\r\n          this._setYBlocks(Math.floor(this.viewport.scrollTop / Y_EPSILON));\r\n        }\r\n      }\r\n    }\r\n  }\r\n\r\n  getViewportNodes(nodes) {\r\n    if (!nodes) return [];\r\n\r\n    const visibleNodes = nodes.filter((node) => !node.isHidden);\r\n\r\n    if (!this.isEnabled()) return visibleNodes;\r\n\r\n    if (!this.viewportHeight || !visibleNodes.length) return [];\r\n\r\n    // When loading children async this method is called before their height and position is calculated.\r\n    // In that case firstIndex === 0 and lastIndex === visibleNodes.length - 1 (e.g. 1000),\r\n    // which means that it loops through every visibleNodes item and push them into viewportNodes array.\r\n    // We can prevent nodes from being pushed to the array and wait for the appropriate calculations to take place\r\n    const lastVisibleNode = visibleNodes.slice(-1)[0]\r\n    if (!lastVisibleNode.height && lastVisibleNode.position === 0) return [];\r\n\r\n    // Search for first node in the viewport using binary search\r\n    // Look for first node that starts after the beginning of the viewport (with buffer)\r\n    // Or that ends after the beginning of the viewport\r\n    const firstIndex = binarySearch(visibleNodes, (node) => {\r\n      return (node.position + Y_OFFSET > this.y) ||\r\n             (node.position + node.height > this.y);\r\n    });\r\n\r\n    // Search for last node in the viewport using binary search\r\n    // Look for first node that starts after the end of the viewport (with buffer)\r\n    const lastIndex = binarySearch(visibleNodes, (node) => {\r\n      return node.position - Y_OFFSET > this.y + this.viewportHeight;\r\n    }, firstIndex);\r\n\r\n    const viewportNodes = [];\r\n\r\n    for (let i = firstIndex; i <= lastIndex; i++) {\r\n      viewportNodes.push(visibleNodes[i]);\r\n    }\r\n\r\n    return viewportNodes;\r\n  }\r\n\r\n  fixScroll() {\r\n    const maxY = Math.max(0, this.totalHeight - this.viewportHeight);\r\n\r\n    if (this.y < 0) this._setYBlocks(0);\r\n    if (this.y > maxY) this._setYBlocks(maxY / Y_EPSILON);\r\n  }\r\n}\r\n\r\nfunction binarySearch(nodes, condition, firstIndex = 0) {\r\n  let index = firstIndex;\r\n  let toIndex = nodes.length - 1;\r\n\r\n  while (index !== toIndex) {\r\n    let midIndex = Math.floor((index + toIndex) / 2);\r\n\r\n    if (condition(nodes[midIndex])) {\r\n      toIndex = midIndex;\r\n    }\r\n    else {\r\n      if (index === midIndex) index = toIndex;\r\n      else index = midIndex;\r\n    }\r\n  }\r\n  return index;\r\n}\r\n"]}
|
|
1
|
+
var __decorate = (this && this.__decorate) || function (decorators, target, key, desc) {
|
|
2
|
+
var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
|
|
3
|
+
if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc);
|
|
4
|
+
else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;
|
|
5
|
+
return c > 3 && r && Object.defineProperty(target, key, r), r;
|
|
6
|
+
};
|
|
7
|
+
import { Injectable } from '@angular/core';
|
|
8
|
+
import { observable, computed, action, autorun, reaction } from 'mobx';
|
|
9
|
+
import { TREE_EVENTS } from '../constants/events';
|
|
10
|
+
import * as i0 from "@angular/core";
|
|
11
|
+
import * as i1 from "./tree.model";
|
|
12
|
+
const Y_OFFSET = 500; // Extra pixels outside the viewport, in each direction, to render nodes in
|
|
13
|
+
const Y_EPSILON = 150; // Minimum pixel change required to recalculate the rendered nodes
|
|
14
|
+
export class TreeVirtualScroll {
|
|
15
|
+
get y() {
|
|
16
|
+
return this.yBlocks * Y_EPSILON;
|
|
17
|
+
}
|
|
18
|
+
get totalHeight() {
|
|
19
|
+
return this.treeModel.virtualRoot ? this.treeModel.virtualRoot.height : 0;
|
|
20
|
+
}
|
|
21
|
+
constructor(treeModel) {
|
|
22
|
+
this.treeModel = treeModel;
|
|
23
|
+
this.yBlocks = 0;
|
|
24
|
+
this.x = 0;
|
|
25
|
+
this.viewportHeight = null;
|
|
26
|
+
this.viewport = null;
|
|
27
|
+
treeModel.virtualScroll = this;
|
|
28
|
+
this._dispose = [autorun(() => this.fixScroll())];
|
|
29
|
+
}
|
|
30
|
+
fireEvent(event) {
|
|
31
|
+
this.treeModel.fireEvent(event);
|
|
32
|
+
}
|
|
33
|
+
init() {
|
|
34
|
+
const fn = this.recalcPositions.bind(this);
|
|
35
|
+
fn();
|
|
36
|
+
this._dispose = [
|
|
37
|
+
...this._dispose,
|
|
38
|
+
reaction(() => this.treeModel.roots, fn),
|
|
39
|
+
reaction(() => this.treeModel.expandedNodeIds, fn),
|
|
40
|
+
reaction(() => this.treeModel.hiddenNodeIds, fn)
|
|
41
|
+
];
|
|
42
|
+
this.treeModel.subscribe(TREE_EVENTS.loadNodeChildren, fn);
|
|
43
|
+
}
|
|
44
|
+
isEnabled() {
|
|
45
|
+
return this.treeModel.options.useVirtualScroll;
|
|
46
|
+
}
|
|
47
|
+
_setYBlocks(value) {
|
|
48
|
+
this.yBlocks = value;
|
|
49
|
+
}
|
|
50
|
+
recalcPositions() {
|
|
51
|
+
this.treeModel.virtualRoot.height = this._getPositionAfter(this.treeModel.getVisibleRoots(), 0);
|
|
52
|
+
}
|
|
53
|
+
_getPositionAfter(nodes, startPos) {
|
|
54
|
+
let position = startPos;
|
|
55
|
+
nodes.forEach((node) => {
|
|
56
|
+
node.position = position;
|
|
57
|
+
position = this._getPositionAfterNode(node, position);
|
|
58
|
+
});
|
|
59
|
+
return position;
|
|
60
|
+
}
|
|
61
|
+
_getPositionAfterNode(node, startPos) {
|
|
62
|
+
let position = node.getSelfHeight() + startPos;
|
|
63
|
+
if (node.children && node.isExpanded) { // TBD: consider loading component as well
|
|
64
|
+
position = this._getPositionAfter(node.visibleChildren, position);
|
|
65
|
+
}
|
|
66
|
+
node.height = position - startPos;
|
|
67
|
+
return position;
|
|
68
|
+
}
|
|
69
|
+
clear() {
|
|
70
|
+
this._dispose.forEach((d) => d());
|
|
71
|
+
}
|
|
72
|
+
setViewport(viewport) {
|
|
73
|
+
Object.assign(this, {
|
|
74
|
+
viewport,
|
|
75
|
+
x: viewport.scrollLeft,
|
|
76
|
+
yBlocks: Math.round(viewport.scrollTop / Y_EPSILON),
|
|
77
|
+
viewportHeight: viewport.getBoundingClientRect ? viewport.getBoundingClientRect().height : 0
|
|
78
|
+
});
|
|
79
|
+
}
|
|
80
|
+
scrollIntoView(node, force, scrollToMiddle = true) {
|
|
81
|
+
if (node.options.scrollContainer) {
|
|
82
|
+
const scrollContainer = node.options.scrollContainer;
|
|
83
|
+
const scrollContainerHeight = scrollContainer.getBoundingClientRect().height;
|
|
84
|
+
const scrollContainerTop = scrollContainer.getBoundingClientRect().top;
|
|
85
|
+
const nodeTop = this.viewport.getBoundingClientRect().top + node.position - scrollContainerTop;
|
|
86
|
+
if (force || // force scroll to node
|
|
87
|
+
nodeTop < scrollContainer.scrollTop || // node is above scroll container
|
|
88
|
+
nodeTop + node.getSelfHeight() > scrollContainer.scrollTop + scrollContainerHeight) { // node is below container
|
|
89
|
+
scrollContainer.scrollTop = scrollToMiddle ?
|
|
90
|
+
nodeTop - scrollContainerHeight / 2 : // scroll to middle
|
|
91
|
+
nodeTop; // scroll to start
|
|
92
|
+
}
|
|
93
|
+
}
|
|
94
|
+
else {
|
|
95
|
+
if (force || // force scroll to node
|
|
96
|
+
node.position < this.y || // node is above viewport
|
|
97
|
+
node.position + node.getSelfHeight() > this.y + this.viewportHeight) { // node is below viewport
|
|
98
|
+
if (this.viewport) {
|
|
99
|
+
this.viewport.scrollTop = scrollToMiddle ?
|
|
100
|
+
node.position - this.viewportHeight / 2 : // scroll to middle
|
|
101
|
+
node.position; // scroll to start
|
|
102
|
+
this._setYBlocks(Math.floor(this.viewport.scrollTop / Y_EPSILON));
|
|
103
|
+
}
|
|
104
|
+
}
|
|
105
|
+
}
|
|
106
|
+
}
|
|
107
|
+
getViewportNodes(nodes) {
|
|
108
|
+
if (!nodes)
|
|
109
|
+
return [];
|
|
110
|
+
const visibleNodes = nodes.filter((node) => !node.isHidden);
|
|
111
|
+
if (!this.isEnabled())
|
|
112
|
+
return visibleNodes;
|
|
113
|
+
if (!this.viewportHeight || !visibleNodes.length)
|
|
114
|
+
return [];
|
|
115
|
+
// When loading children async this method is called before their height and position is calculated.
|
|
116
|
+
// In that case firstIndex === 0 and lastIndex === visibleNodes.length - 1 (e.g. 1000),
|
|
117
|
+
// which means that it loops through every visibleNodes item and push them into viewportNodes array.
|
|
118
|
+
// We can prevent nodes from being pushed to the array and wait for the appropriate calculations to take place
|
|
119
|
+
const lastVisibleNode = visibleNodes.slice(-1)[0];
|
|
120
|
+
if (!lastVisibleNode.height && lastVisibleNode.position === 0)
|
|
121
|
+
return [];
|
|
122
|
+
// Search for first node in the viewport using binary search
|
|
123
|
+
// Look for first node that starts after the beginning of the viewport (with buffer)
|
|
124
|
+
// Or that ends after the beginning of the viewport
|
|
125
|
+
const firstIndex = binarySearch(visibleNodes, (node) => {
|
|
126
|
+
return (node.position + Y_OFFSET > this.y) ||
|
|
127
|
+
(node.position + node.height > this.y);
|
|
128
|
+
});
|
|
129
|
+
// Search for last node in the viewport using binary search
|
|
130
|
+
// Look for first node that starts after the end of the viewport (with buffer)
|
|
131
|
+
const lastIndex = binarySearch(visibleNodes, (node) => {
|
|
132
|
+
return node.position - Y_OFFSET > this.y + this.viewportHeight;
|
|
133
|
+
}, firstIndex);
|
|
134
|
+
const viewportNodes = [];
|
|
135
|
+
for (let i = firstIndex; i <= lastIndex; i++) {
|
|
136
|
+
viewportNodes.push(visibleNodes[i]);
|
|
137
|
+
}
|
|
138
|
+
return viewportNodes;
|
|
139
|
+
}
|
|
140
|
+
fixScroll() {
|
|
141
|
+
const maxY = Math.max(0, this.totalHeight - this.viewportHeight);
|
|
142
|
+
if (this.y < 0)
|
|
143
|
+
this._setYBlocks(0);
|
|
144
|
+
if (this.y > maxY)
|
|
145
|
+
this._setYBlocks(maxY / Y_EPSILON);
|
|
146
|
+
}
|
|
147
|
+
/** @nocollapse */ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: TreeVirtualScroll, deps: [{ token: i1.TreeModel }], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
148
|
+
/** @nocollapse */ static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: TreeVirtualScroll }); }
|
|
149
|
+
}
|
|
150
|
+
__decorate([
|
|
151
|
+
observable
|
|
152
|
+
], TreeVirtualScroll.prototype, "yBlocks", void 0);
|
|
153
|
+
__decorate([
|
|
154
|
+
observable
|
|
155
|
+
], TreeVirtualScroll.prototype, "x", void 0);
|
|
156
|
+
__decorate([
|
|
157
|
+
observable
|
|
158
|
+
], TreeVirtualScroll.prototype, "viewportHeight", void 0);
|
|
159
|
+
__decorate([
|
|
160
|
+
computed
|
|
161
|
+
], TreeVirtualScroll.prototype, "y", null);
|
|
162
|
+
__decorate([
|
|
163
|
+
computed
|
|
164
|
+
], TreeVirtualScroll.prototype, "totalHeight", null);
|
|
165
|
+
__decorate([
|
|
166
|
+
action
|
|
167
|
+
], TreeVirtualScroll.prototype, "_setYBlocks", null);
|
|
168
|
+
__decorate([
|
|
169
|
+
action
|
|
170
|
+
], TreeVirtualScroll.prototype, "recalcPositions", null);
|
|
171
|
+
__decorate([
|
|
172
|
+
action
|
|
173
|
+
], TreeVirtualScroll.prototype, "setViewport", null);
|
|
174
|
+
__decorate([
|
|
175
|
+
action
|
|
176
|
+
], TreeVirtualScroll.prototype, "scrollIntoView", null);
|
|
177
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: TreeVirtualScroll, decorators: [{
|
|
178
|
+
type: Injectable
|
|
179
|
+
}], ctorParameters: () => [{ type: i1.TreeModel }], propDecorators: { yBlocks: [], x: [], viewportHeight: [], y: [], totalHeight: [], _setYBlocks: [], recalcPositions: [], setViewport: [], scrollIntoView: [] } });
|
|
180
|
+
function binarySearch(nodes, condition, firstIndex = 0) {
|
|
181
|
+
let index = firstIndex;
|
|
182
|
+
let toIndex = nodes.length - 1;
|
|
183
|
+
while (index !== toIndex) {
|
|
184
|
+
let midIndex = Math.floor((index + toIndex) / 2);
|
|
185
|
+
if (condition(nodes[midIndex])) {
|
|
186
|
+
toIndex = midIndex;
|
|
187
|
+
}
|
|
188
|
+
else {
|
|
189
|
+
if (index === midIndex)
|
|
190
|
+
index = toIndex;
|
|
191
|
+
else
|
|
192
|
+
index = midIndex;
|
|
193
|
+
}
|
|
194
|
+
}
|
|
195
|
+
return index;
|
|
196
|
+
}
|
|
197
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tree-virtual-scroll.model.js","sourceRoot":"","sources":["../../../../../projects/angular-tree-component/src/lib/models/tree-virtual-scroll.model.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\r\nimport { observable, computed, action, autorun, reaction } from 'mobx';\r\nimport { TreeModel } from './tree.model';\r\nimport { TREE_EVENTS } from '../constants/events';\r\n\r\nconst Y_OFFSET = 500; // Extra pixels outside the viewport, in each direction, to render nodes in\r\nconst Y_EPSILON = 150; // Minimum pixel change required to recalculate the rendered nodes\r\n\r\n@Injectable()\r\nexport class TreeVirtualScroll {\r\n  private _dispose: any;\r\n\r\n  @observable yBlocks = 0;\r\n  @observable x = 0;\r\n  @observable viewportHeight = null;\r\n  viewport = null;\r\n\r\n  @computed get y() {\r\n    return this.yBlocks * Y_EPSILON;\r\n  }\r\n\r\n  @computed get totalHeight() {\r\n    return this.treeModel.virtualRoot ? this.treeModel.virtualRoot.height : 0;\r\n  }\r\n\r\n  constructor(private treeModel: TreeModel) {\r\n    treeModel.virtualScroll = this;\r\n    this._dispose = [autorun(() => this.fixScroll())];\r\n  }\r\n\r\n  fireEvent(event) {\r\n    this.treeModel.fireEvent(event);\r\n  }\r\n\r\n  init() {\r\n    const fn = this.recalcPositions.bind(this);\r\n\r\n    fn();\r\n    this._dispose = [\r\n      ...this._dispose,\r\n      reaction(() => this.treeModel.roots, fn),\r\n      reaction(() => this.treeModel.expandedNodeIds, fn),\r\n      reaction(() => this.treeModel.hiddenNodeIds, fn)\r\n    ];\r\n    this.treeModel.subscribe(TREE_EVENTS.loadNodeChildren, fn);\r\n  }\r\n\r\n  isEnabled() {\r\n    return this.treeModel.options.useVirtualScroll;\r\n  }\r\n\r\n  @action private _setYBlocks(value) {\r\n    this.yBlocks = value;\r\n  }\r\n\r\n  @action recalcPositions() {\r\n    this.treeModel.virtualRoot.height = this._getPositionAfter(this.treeModel.getVisibleRoots(), 0);\r\n  }\r\n\r\n  private _getPositionAfter(nodes, startPos) {\r\n    let position = startPos;\r\n\r\n    nodes.forEach((node) => {\r\n      node.position = position;\r\n      position = this._getPositionAfterNode(node, position);\r\n    });\r\n    return position;\r\n  }\r\n\r\n  private _getPositionAfterNode(node, startPos) {\r\n    let position = node.getSelfHeight() + startPos;\r\n\r\n    if (node.children && node.isExpanded) { // TBD: consider loading component as well\r\n      position = this._getPositionAfter(node.visibleChildren, position);\r\n    }\r\n    node.height = position - startPos;\r\n    return position;\r\n  }\r\n\r\n\r\n  clear() {\r\n    this._dispose.forEach((d) => d());\r\n  }\r\n\r\n  @action setViewport(viewport) {\r\n    Object.assign(this, {\r\n      viewport,\r\n      x: viewport.scrollLeft,\r\n      yBlocks: Math.round(viewport.scrollTop / Y_EPSILON),\r\n      viewportHeight: viewport.getBoundingClientRect ? viewport.getBoundingClientRect().height : 0\r\n    });\r\n  }\r\n\r\n  @action scrollIntoView(node, force, scrollToMiddle = true) {\r\n    if (node.options.scrollContainer) {\r\n      const scrollContainer = node.options.scrollContainer;\r\n      const scrollContainerHeight = scrollContainer.getBoundingClientRect().height;\r\n      const scrollContainerTop = scrollContainer.getBoundingClientRect().top;\r\n      const nodeTop = this.viewport.getBoundingClientRect().top + node.position - scrollContainerTop;\r\n\r\n      if (force || // force scroll to node\r\n        nodeTop < scrollContainer.scrollTop || // node is above scroll container\r\n        nodeTop + node.getSelfHeight() > scrollContainer.scrollTop + scrollContainerHeight) { // node is below container\r\n        scrollContainer.scrollTop = scrollToMiddle ?\r\n          nodeTop - scrollContainerHeight / 2 : // scroll to middle\r\n          nodeTop; // scroll to start\r\n      }\r\n    } else {\r\n      if (force || // force scroll to node\r\n        node.position < this.y || // node is above viewport\r\n        node.position + node.getSelfHeight() > this.y + this.viewportHeight) { // node is below viewport\r\n        if (this.viewport) {\r\n          this.viewport.scrollTop = scrollToMiddle ?\r\n          node.position - this.viewportHeight / 2 : // scroll to middle\r\n          node.position; // scroll to start\r\n\r\n          this._setYBlocks(Math.floor(this.viewport.scrollTop / Y_EPSILON));\r\n        }\r\n      }\r\n    }\r\n  }\r\n\r\n  getViewportNodes(nodes) {\r\n    if (!nodes) return [];\r\n\r\n    const visibleNodes = nodes.filter((node) => !node.isHidden);\r\n\r\n    if (!this.isEnabled()) return visibleNodes;\r\n\r\n    if (!this.viewportHeight || !visibleNodes.length) return [];\r\n\r\n    // When loading children async this method is called before their height and position is calculated.\r\n    // In that case firstIndex === 0 and lastIndex === visibleNodes.length - 1 (e.g. 1000),\r\n    // which means that it loops through every visibleNodes item and push them into viewportNodes array.\r\n    // We can prevent nodes from being pushed to the array and wait for the appropriate calculations to take place\r\n    const lastVisibleNode = visibleNodes.slice(-1)[0]\r\n    if (!lastVisibleNode.height && lastVisibleNode.position === 0) return [];\r\n\r\n    // Search for first node in the viewport using binary search\r\n    // Look for first node that starts after the beginning of the viewport (with buffer)\r\n    // Or that ends after the beginning of the viewport\r\n    const firstIndex = binarySearch(visibleNodes, (node) => {\r\n      return (node.position + Y_OFFSET > this.y) ||\r\n             (node.position + node.height > this.y);\r\n    });\r\n\r\n    // Search for last node in the viewport using binary search\r\n    // Look for first node that starts after the end of the viewport (with buffer)\r\n    const lastIndex = binarySearch(visibleNodes, (node) => {\r\n      return node.position - Y_OFFSET > this.y + this.viewportHeight;\r\n    }, firstIndex);\r\n\r\n    const viewportNodes = [];\r\n\r\n    for (let i = firstIndex; i <= lastIndex; i++) {\r\n      viewportNodes.push(visibleNodes[i]);\r\n    }\r\n\r\n    return viewportNodes;\r\n  }\r\n\r\n  fixScroll() {\r\n    const maxY = Math.max(0, this.totalHeight - this.viewportHeight);\r\n\r\n    if (this.y < 0) this._setYBlocks(0);\r\n    if (this.y > maxY) this._setYBlocks(maxY / Y_EPSILON);\r\n  }\r\n}\r\n\r\nfunction binarySearch(nodes, condition, firstIndex = 0) {\r\n  let index = firstIndex;\r\n  let toIndex = nodes.length - 1;\r\n\r\n  while (index !== toIndex) {\r\n    let midIndex = Math.floor((index + toIndex) / 2);\r\n\r\n    if (condition(nodes[midIndex])) {\r\n      toIndex = midIndex;\r\n    }\r\n    else {\r\n      if (index === midIndex) index = toIndex;\r\n      else index = midIndex;\r\n    }\r\n  }\r\n  return index;\r\n}\r\n"]}
|