@myrmidon/paged-data-browsers 0.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/README.md ADDED
@@ -0,0 +1,7 @@
1
+ # Paged Data Browsers
2
+
3
+ This library provides simple components to display filtered and paged data from some service.
4
+
5
+ There are currently two components, one for displaying a flat list of data, and another to display hierarchically structured data, combining a tree view with paging.
6
+
7
+ Also, there is a LRU cache service and a compact pager used in the paged tree component.
@@ -0,0 +1,76 @@
1
+ import { Component, EventEmitter, Input, Output } from '@angular/core';
2
+ import * as i0 from "@angular/core";
3
+ import * as i1 from "@angular/common";
4
+ import * as i2 from "@angular/material/badge";
5
+ import * as i3 from "@angular/material/button";
6
+ import * as i4 from "@angular/material/icon";
7
+ import * as i5 from "@angular/material/tooltip";
8
+ import * as i6 from "../compact-pager/compact-pager.component";
9
+ import * as i7 from "../range-view/range-view.component";
10
+ import * as i8 from "@myrmidon/ng-tools";
11
+ /**
12
+ * Browser tree node component view. This wraps some HTML content providing
13
+ * a toggle button to expand/collapse the node, a paging control for the
14
+ * node's children, and a button to edit the node's filter. You should then
15
+ * provide the HTML content to display the node's data inside this component, e.g.
16
+ * <pdb-browser-tree-node [node]="node">
17
+ * <your-node-view [node]="node" />
18
+ * <pdb-browser-tree-node>
19
+ */
20
+ export class BrowserTreeNodeComponent {
21
+ /**
22
+ * The node to display.
23
+ */
24
+ get node() {
25
+ return this._node;
26
+ }
27
+ set node(value) {
28
+ if (this._node === value) {
29
+ return;
30
+ }
31
+ this._node = value;
32
+ }
33
+ constructor() {
34
+ this.toggleExpandedRequest = new EventEmitter();
35
+ this.changePageRequest = new EventEmitter();
36
+ this.editNodeFilterRequest = new EventEmitter();
37
+ }
38
+ onToggleExpanded() {
39
+ if (!this._node) {
40
+ return;
41
+ }
42
+ this.toggleExpandedRequest.emit(this._node);
43
+ }
44
+ onPagingChange(node, paging) {
45
+ this.changePageRequest.emit({
46
+ node,
47
+ paging,
48
+ });
49
+ }
50
+ onEditFilter() {
51
+ if (this._node) {
52
+ this.editNodeFilterRequest.emit(this._node);
53
+ }
54
+ }
55
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.7", ngImport: i0, type: BrowserTreeNodeComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
56
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.2.7", type: BrowserTreeNodeComponent, selector: "pdb-browser-tree-node", inputs: { node: "node", paging: "paging", debug: "debug", hidePaging: "hidePaging" }, outputs: { toggleExpandedRequest: "toggleExpandedRequest", changePageRequest: "changePageRequest", editNodeFilterRequest: "editNodeFilterRequest" }, ngImport: i0, template: "<div id=\"node\" *ngIf=\"node\" [style.margin-left.px]=\"(node.y - 1) * 20\">\r\n <!-- pager -->\r\n <div\r\n *ngIf=\"$any(node).expanded && paging && paging.pageCount > 1\"\r\n id=\"pager\"\r\n [style.display]=\"hidePaging ? 'inherit' : 'block'\"\r\n >\r\n <pdb-compact-pager\r\n [paging]=\"paging\"\r\n (pagingChange)=\"onPagingChange($any(node), $event)\"\r\n />\r\n <pdb-range-view\r\n [width]=\"250\"\r\n [domain]=\"[0, paging.pageCount]\"\r\n [range]=\"[paging.pageNumber - 1, paging.pageNumber]\"\r\n />\r\n </div>\r\n <!-- node -->\r\n <div class=\"form-row\">\r\n <!-- expand/collapse button -->\r\n <button\r\n *ngIf=\"node.y > 0\"\r\n type=\"button\"\r\n mat-icon-button\r\n color=\"primary\"\r\n [matTooltip]=\"$any(node).expanded ? 'Collapse' : 'Expand'\"\r\n [disabled]=\"node.hasChildren === false\"\r\n (click)=\"onToggleExpanded()\"\r\n >\r\n <mat-icon>{{\r\n node.hasChildren === true || node.hasChildren === undefined\r\n ? $any(node).expanded\r\n ? \"expand_less\"\r\n : \"expand_more\"\r\n : \"stop\"\r\n }}</mat-icon>\r\n </button>\r\n <!-- tag -->\r\n <span\r\n class=\"tag\"\r\n [ngStyle]=\"{\r\n 'background-color': (node.tag | stringToColor),\r\n color: node.tag | stringToColor | colorToContrast\r\n }\"\r\n >{{ node.tag }}</span\r\n >\r\n <!-- loc and label -->\r\n <span class=\"loc\">{{ node.y }}.{{ node.x }}</span> - {{ node.label }}\r\n\r\n <!-- PROJECTED NODE -->\r\n <ng-content></ng-content>\r\n\r\n <!-- debug -->\r\n <span *ngIf=\"debug\" class=\"debug\"\r\n >#{{ node.id }}\r\n <span\r\n >| {{ $any(node).paging.pageNumber }}/{{\r\n $any(node).paging.pageCount\r\n }}\r\n ({{ $any(node).paging.total }})</span\r\n ></span\r\n >\r\n\r\n <!-- filter -->\r\n <div class=\"muted\" *ngIf=\"!$any(node).filter && node.y\">\r\n <button\r\n type=\"button\"\r\n mat-icon-button\r\n matTooltip=\"Add filter\"\r\n (click)=\"onEditFilter()\"\r\n >\r\n <mat-icon>filter_list</mat-icon>\r\n </button>\r\n </div>\r\n <div class=\"muted\" *ngIf=\"$any(node).filter && node.y\">\r\n <button type=\"button\" mat-icon-button (click)=\"onEditFilter()\">\r\n <mat-icon [matBadge]=\"$any(node).filter ? 'F' : ''\"\r\n >filter_alt</mat-icon\r\n >\r\n </button>\r\n </div>\r\n </div>\r\n</div>\r\n", styles: [".form-row{display:flex;gap:8px;align-items:center}.form-row *{flex:0 0 auto}.form-row span{flex:0 1 auto;white-space:normal}#node #pager{display:none}#node:hover #pager{display:block}#node{margin-bottom:4px;padding:4px 6px;border:1px solid transparent;border-radius:6px}#node:hover{border:1px solid #98a8d4;border-radius:6px}span.loc{font-size:.85em;color:#666;vertical-align:middle}span.tag{border:1px solid #aaa;border-radius:4px;padding:0 4px}fieldset{border:1px solid silver;border-radius:6px;padding:4px 6px;margin:4px 0}legend{color:silver}.muted{opacity:.3}.muted:hover{opacity:1}.debug{font-size:.85em;color:#9c3d3e}\n"], dependencies: [{ kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { kind: "directive", type: i2.MatBadge, selector: "[matBadge]", inputs: ["matBadgeDisabled", "matBadgeColor", "matBadgeOverlap", "matBadgePosition", "matBadge", "matBadgeDescription", "matBadgeSize", "matBadgeHidden"] }, { kind: "component", type: i3.MatIconButton, selector: "button[mat-icon-button]", inputs: ["disabled", "disableRipple", "color"], exportAs: ["matButton"] }, { kind: "component", type: i4.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "directive", type: i5.MatTooltip, selector: "[matTooltip]", exportAs: ["matTooltip"] }, { kind: "component", type: i6.CompactPagerComponent, selector: "pdb-compact-pager", inputs: ["paging"], outputs: ["pagingChange"] }, { kind: "component", type: i7.RangeViewComponent, selector: "pdb-range-view", inputs: ["domain", "range", "width", "height"] }, { kind: "pipe", type: i8.ColorToContrastPipe, name: "colorToContrast" }, { kind: "pipe", type: i8.StringToColorPipe, name: "stringToColor" }] }); }
57
+ }
58
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.7", ngImport: i0, type: BrowserTreeNodeComponent, decorators: [{
59
+ type: Component,
60
+ args: [{ selector: 'pdb-browser-tree-node', template: "<div id=\"node\" *ngIf=\"node\" [style.margin-left.px]=\"(node.y - 1) * 20\">\r\n <!-- pager -->\r\n <div\r\n *ngIf=\"$any(node).expanded && paging && paging.pageCount > 1\"\r\n id=\"pager\"\r\n [style.display]=\"hidePaging ? 'inherit' : 'block'\"\r\n >\r\n <pdb-compact-pager\r\n [paging]=\"paging\"\r\n (pagingChange)=\"onPagingChange($any(node), $event)\"\r\n />\r\n <pdb-range-view\r\n [width]=\"250\"\r\n [domain]=\"[0, paging.pageCount]\"\r\n [range]=\"[paging.pageNumber - 1, paging.pageNumber]\"\r\n />\r\n </div>\r\n <!-- node -->\r\n <div class=\"form-row\">\r\n <!-- expand/collapse button -->\r\n <button\r\n *ngIf=\"node.y > 0\"\r\n type=\"button\"\r\n mat-icon-button\r\n color=\"primary\"\r\n [matTooltip]=\"$any(node).expanded ? 'Collapse' : 'Expand'\"\r\n [disabled]=\"node.hasChildren === false\"\r\n (click)=\"onToggleExpanded()\"\r\n >\r\n <mat-icon>{{\r\n node.hasChildren === true || node.hasChildren === undefined\r\n ? $any(node).expanded\r\n ? \"expand_less\"\r\n : \"expand_more\"\r\n : \"stop\"\r\n }}</mat-icon>\r\n </button>\r\n <!-- tag -->\r\n <span\r\n class=\"tag\"\r\n [ngStyle]=\"{\r\n 'background-color': (node.tag | stringToColor),\r\n color: node.tag | stringToColor | colorToContrast\r\n }\"\r\n >{{ node.tag }}</span\r\n >\r\n <!-- loc and label -->\r\n <span class=\"loc\">{{ node.y }}.{{ node.x }}</span> - {{ node.label }}\r\n\r\n <!-- PROJECTED NODE -->\r\n <ng-content></ng-content>\r\n\r\n <!-- debug -->\r\n <span *ngIf=\"debug\" class=\"debug\"\r\n >#{{ node.id }}\r\n <span\r\n >| {{ $any(node).paging.pageNumber }}/{{\r\n $any(node).paging.pageCount\r\n }}\r\n ({{ $any(node).paging.total }})</span\r\n ></span\r\n >\r\n\r\n <!-- filter -->\r\n <div class=\"muted\" *ngIf=\"!$any(node).filter && node.y\">\r\n <button\r\n type=\"button\"\r\n mat-icon-button\r\n matTooltip=\"Add filter\"\r\n (click)=\"onEditFilter()\"\r\n >\r\n <mat-icon>filter_list</mat-icon>\r\n </button>\r\n </div>\r\n <div class=\"muted\" *ngIf=\"$any(node).filter && node.y\">\r\n <button type=\"button\" mat-icon-button (click)=\"onEditFilter()\">\r\n <mat-icon [matBadge]=\"$any(node).filter ? 'F' : ''\"\r\n >filter_alt</mat-icon\r\n >\r\n </button>\r\n </div>\r\n </div>\r\n</div>\r\n", styles: [".form-row{display:flex;gap:8px;align-items:center}.form-row *{flex:0 0 auto}.form-row span{flex:0 1 auto;white-space:normal}#node #pager{display:none}#node:hover #pager{display:block}#node{margin-bottom:4px;padding:4px 6px;border:1px solid transparent;border-radius:6px}#node:hover{border:1px solid #98a8d4;border-radius:6px}span.loc{font-size:.85em;color:#666;vertical-align:middle}span.tag{border:1px solid #aaa;border-radius:4px;padding:0 4px}fieldset{border:1px solid silver;border-radius:6px;padding:4px 6px;margin:4px 0}legend{color:silver}.muted{opacity:.3}.muted:hover{opacity:1}.debug{font-size:.85em;color:#9c3d3e}\n"] }]
61
+ }], ctorParameters: function () { return []; }, propDecorators: { node: [{
62
+ type: Input
63
+ }], paging: [{
64
+ type: Input
65
+ }], debug: [{
66
+ type: Input
67
+ }], hidePaging: [{
68
+ type: Input
69
+ }], toggleExpandedRequest: [{
70
+ type: Output
71
+ }], changePageRequest: [{
72
+ type: Output
73
+ }], editNodeFilterRequest: [{
74
+ type: Output
75
+ }] } });
76
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"browser-tree-node.component.js","sourceRoot":"","sources":["../../../../../../../projects/myrmidon/paged-data-browsers/src/lib/components/browser-tree-node/browser-tree-node.component.ts","../../../../../../../projects/myrmidon/paged-data-browsers/src/lib/components/browser-tree-node/browser-tree-node.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,YAAY,EAAE,KAAK,EAAE,MAAM,EAAE,MAAM,eAAe,CAAC;;;;;;;;;;AASvE;;;;;;;;GAQG;AAMH,MAAM,OAAO,wBAAwB;IAGnC;;OAEG;IACH,IACW,IAAI;QACb,OAAO,IAAI,CAAC,KAAK,CAAC;IACpB,CAAC;IAED,IAAW,IAAI,CAAC,KAA4C;QAC1D,IAAI,IAAI,CAAC,KAAK,KAAK,KAAK,EAAE;YACxB,OAAO;SACR;QACD,IAAI,CAAC,KAAK,GAAG,KAAK,CAAC;IACrB,CAAC;IAgCD;QACE,IAAI,CAAC,qBAAqB,GAAG,IAAI,YAAY,EAAsB,CAAC;QACpE,IAAI,CAAC,iBAAiB,GAAG,IAAI,YAAY,EAAqB,CAAC;QAC/D,IAAI,CAAC,qBAAqB,GAAG,IAAI,YAAY,EAAsB,CAAC;IACtE,CAAC;IAEM,gBAAgB;QACrB,IAAI,CAAC,IAAI,CAAC,KAAK,EAAE;YACf,OAAO;SACR;QACD,IAAI,CAAC,qBAAqB,CAAC,IAAI,CAAC,IAAI,CAAC,KAA2B,CAAC,CAAC;IACpE,CAAC;IAEM,cAAc,CAAC,IAAwB,EAAE,MAAkB;QAChE,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC;YAC1B,IAAI;YACJ,MAAM;SACP,CAAC,CAAC;IACL,CAAC;IAEM,YAAY;QACjB,IAAI,IAAI,CAAC,KAAK,EAAE;YACd,IAAI,CAAC,qBAAqB,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;SAC7C;IACH,CAAC;8GAxEU,wBAAwB;kGAAxB,wBAAwB,wSCvBrC,khFAmFA;;2FD5Da,wBAAwB;kBALpC,SAAS;+BACE,uBAAuB;0EAWtB,IAAI;sBADd,KAAK;gBAgBN,MAAM;sBADL,KAAK;gBAOC,KAAK;sBADX,KAAK;gBAIC,UAAU;sBADhB,KAAK;gBAOC,qBAAqB;sBAD3B,MAAM;gBAOA,iBAAiB;sBADvB,MAAM;gBAIA,qBAAqB;sBAD3B,MAAM","sourcesContent":["import { Component, EventEmitter, Input, Output } from '@angular/core';\n\nimport { PagedTreeNode, PagingInfo } from '../../services/paged-tree.store';\n\nexport interface PageChangeRequest {\n  node: PagedTreeNode<any>;\n  paging: PagingInfo;\n}\n\n/**\n * Browser tree node component view. This wraps some HTML content providing\n * a toggle button to expand/collapse the node, a paging control for the\n * node's children, and a button to edit the node's filter. You should then\n * provide the HTML content to display the node's data inside this component, e.g.\n * <pdb-browser-tree-node [node]=\"node\">\n *   <your-node-view [node]=\"node\" />\n * <pdb-browser-tree-node>\n */\n@Component({\n  selector: 'pdb-browser-tree-node',\n  templateUrl: './browser-tree-node.component.html',\n  styleUrls: ['./browser-tree-node.component.css'],\n})\nexport class BrowserTreeNodeComponent {\n  private _node: PagedTreeNode<any> | undefined | null;\n\n  /**\n   * The node to display.\n   */\n  @Input()\n  public get node(): PagedTreeNode<any> | undefined | null {\n    return this._node;\n  }\n\n  public set node(value: PagedTreeNode<any> | undefined | null) {\n    if (this._node === value) {\n      return;\n    }\n    this._node = value;\n  }\n\n  /**\n   * The paging information for the node's children.\n   */\n  @Input()\n  paging?: PagingInfo;\n\n  /**\n   * True to show debug information.\n   */\n  @Input()\n  public debug?: boolean;\n\n  @Input()\n  public hidePaging?: boolean;\n\n  /**\n   * Emits when the user wants to toggle the expanded state of the node.\n   */\n  @Output()\n  public toggleExpandedRequest: EventEmitter<PagedTreeNode<any>>;\n\n  /**\n   * Emits when the user wants to change the page number of the node's children.\n   */\n  @Output()\n  public changePageRequest: EventEmitter<PageChangeRequest>;\n\n  @Output()\n  public editNodeFilterRequest: EventEmitter<PagedTreeNode<any>>;\n\n  constructor() {\n    this.toggleExpandedRequest = new EventEmitter<PagedTreeNode<any>>();\n    this.changePageRequest = new EventEmitter<PageChangeRequest>();\n    this.editNodeFilterRequest = new EventEmitter<PagedTreeNode<any>>();\n  }\n\n  public onToggleExpanded(): void {\n    if (!this._node) {\n      return;\n    }\n    this.toggleExpandedRequest.emit(this._node as PagedTreeNode<any>);\n  }\n\n  public onPagingChange(node: PagedTreeNode<any>, paging: PagingInfo): void {\n    this.changePageRequest.emit({\n      node,\n      paging,\n    });\n  }\n\n  public onEditFilter(): void {\n    if (this._node) {\n      this.editNodeFilterRequest.emit(this._node);\n    }\n  }\n}\n","<div id=\"node\" *ngIf=\"node\" [style.margin-left.px]=\"(node.y - 1) * 20\">\r\n  <!-- pager -->\r\n  <div\r\n    *ngIf=\"$any(node).expanded && paging && paging.pageCount > 1\"\r\n    id=\"pager\"\r\n    [style.display]=\"hidePaging ? 'inherit' : 'block'\"\r\n  >\r\n    <pdb-compact-pager\r\n      [paging]=\"paging\"\r\n      (pagingChange)=\"onPagingChange($any(node), $event)\"\r\n    />\r\n    <pdb-range-view\r\n      [width]=\"250\"\r\n      [domain]=\"[0, paging.pageCount]\"\r\n      [range]=\"[paging.pageNumber - 1, paging.pageNumber]\"\r\n    />\r\n  </div>\r\n  <!-- node -->\r\n  <div class=\"form-row\">\r\n    <!-- expand/collapse button -->\r\n    <button\r\n      *ngIf=\"node.y > 0\"\r\n      type=\"button\"\r\n      mat-icon-button\r\n      color=\"primary\"\r\n      [matTooltip]=\"$any(node).expanded ? 'Collapse' : 'Expand'\"\r\n      [disabled]=\"node.hasChildren === false\"\r\n      (click)=\"onToggleExpanded()\"\r\n    >\r\n      <mat-icon>{{\r\n        node.hasChildren === true || node.hasChildren === undefined\r\n          ? $any(node).expanded\r\n            ? \"expand_less\"\r\n            : \"expand_more\"\r\n          : \"stop\"\r\n      }}</mat-icon>\r\n    </button>\r\n    <!-- tag -->\r\n    <span\r\n      class=\"tag\"\r\n      [ngStyle]=\"{\r\n        'background-color': (node.tag | stringToColor),\r\n        color: node.tag | stringToColor | colorToContrast\r\n      }\"\r\n      >{{ node.tag }}</span\r\n    >\r\n    <!-- loc and label -->\r\n    <span class=\"loc\">{{ node.y }}.{{ node.x }}</span> - {{ node.label }}\r\n\r\n    <!-- PROJECTED NODE -->\r\n    <ng-content></ng-content>\r\n\r\n    <!-- debug -->\r\n    <span *ngIf=\"debug\" class=\"debug\"\r\n      >#{{ node.id }}\r\n      <span\r\n        >| {{ $any(node).paging.pageNumber }}/{{\r\n          $any(node).paging.pageCount\r\n        }}\r\n        ({{ $any(node).paging.total }})</span\r\n      ></span\r\n    >\r\n\r\n    <!-- filter -->\r\n    <div class=\"muted\" *ngIf=\"!$any(node).filter && node.y\">\r\n      <button\r\n        type=\"button\"\r\n        mat-icon-button\r\n        matTooltip=\"Add filter\"\r\n        (click)=\"onEditFilter()\"\r\n      >\r\n        <mat-icon>filter_list</mat-icon>\r\n      </button>\r\n    </div>\r\n    <div class=\"muted\" *ngIf=\"$any(node).filter && node.y\">\r\n      <button type=\"button\" mat-icon-button (click)=\"onEditFilter()\">\r\n        <mat-icon [matBadge]=\"$any(node).filter ? 'F' : ''\"\r\n          >filter_alt</mat-icon\r\n        >\r\n      </button>\r\n    </div>\r\n  </div>\r\n</div>\r\n"]}
@@ -0,0 +1,48 @@
1
+ import { Component, EventEmitter, Input, Output } from '@angular/core';
2
+ import * as i0 from "@angular/core";
3
+ import * as i1 from "@angular/common";
4
+ import * as i2 from "@angular/material/button";
5
+ import * as i3 from "@angular/material/icon";
6
+ export class CompactPagerComponent {
7
+ constructor() {
8
+ this.paging = {
9
+ pageNumber: 0,
10
+ pageCount: 0,
11
+ total: 0,
12
+ };
13
+ this.pagingChange = new EventEmitter();
14
+ }
15
+ onFirst() {
16
+ this.paging = { ...this.paging, pageNumber: 1 };
17
+ this.pagingChange.emit(this.paging);
18
+ }
19
+ onPrevious() {
20
+ if (this.paging.pageNumber > 1) {
21
+ this.paging = { ...this.paging, pageNumber: this.paging.pageNumber - 1 };
22
+ this.pagingChange.emit(this.paging);
23
+ }
24
+ }
25
+ onNext() {
26
+ if (this.paging.pageNumber < this.paging.pageCount) {
27
+ this.paging = { ...this.paging, pageNumber: this.paging.pageNumber + 1 };
28
+ this.pagingChange.emit(this.paging);
29
+ }
30
+ }
31
+ onLast() {
32
+ if (this.paging.pageNumber < this.paging.pageCount) {
33
+ this.paging = { ...this.paging, pageNumber: this.paging.pageCount };
34
+ this.pagingChange.emit(this.paging);
35
+ }
36
+ }
37
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.7", ngImport: i0, type: CompactPagerComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
38
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.2.7", type: CompactPagerComponent, selector: "pdb-compact-pager", inputs: { paging: "paging" }, outputs: { pagingChange: "pagingChange" }, ngImport: i0, template: "<div class=\"form-row\" *ngIf=\"paging.pageCount\">\n <span id=\"pages\">{{ paging.pageNumber }}/{{ paging.pageCount }}</span>\n <button\n type=\"button\"\n mat-icon-button\n (click)=\"onFirst()\"\n [disabled]=\"paging.pageNumber < 2\"\n >\n <mat-icon>first_page</mat-icon>\n </button>\n <button\n type=\"button\"\n mat-icon-button\n (click)=\"onPrevious()\"\n [disabled]=\"paging.pageNumber < 2\"\n >\n <mat-icon>navigate_before</mat-icon>\n </button>\n <button\n type=\"button\"\n mat-icon-button\n (click)=\"onNext()\"\n [disabled]=\"paging.pageNumber === paging.pageCount\"\n >\n <mat-icon>navigate_next</mat-icon>\n </button>\n <button\n type=\"button\"\n mat-icon-button\n (click)=\"onLast()\"\n [disabled]=\"paging.pageNumber === paging.pageCount\"\n >\n <mat-icon>last_page</mat-icon>\n </button>\n <span id=\"total\">{{ paging.total }} </span>\n</div>\n", styles: ["#pages,#total{color:silver}.form-row{display:flex;gap:2px;align-items:center;flex-wrap:wrap}.form-row *{flex:0 0 auto}\n"], dependencies: [{ kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i2.MatIconButton, selector: "button[mat-icon-button]", inputs: ["disabled", "disableRipple", "color"], exportAs: ["matButton"] }, { kind: "component", type: i3.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }] }); }
39
+ }
40
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.7", ngImport: i0, type: CompactPagerComponent, decorators: [{
41
+ type: Component,
42
+ args: [{ selector: 'pdb-compact-pager', template: "<div class=\"form-row\" *ngIf=\"paging.pageCount\">\n <span id=\"pages\">{{ paging.pageNumber }}/{{ paging.pageCount }}</span>\n <button\n type=\"button\"\n mat-icon-button\n (click)=\"onFirst()\"\n [disabled]=\"paging.pageNumber < 2\"\n >\n <mat-icon>first_page</mat-icon>\n </button>\n <button\n type=\"button\"\n mat-icon-button\n (click)=\"onPrevious()\"\n [disabled]=\"paging.pageNumber < 2\"\n >\n <mat-icon>navigate_before</mat-icon>\n </button>\n <button\n type=\"button\"\n mat-icon-button\n (click)=\"onNext()\"\n [disabled]=\"paging.pageNumber === paging.pageCount\"\n >\n <mat-icon>navigate_next</mat-icon>\n </button>\n <button\n type=\"button\"\n mat-icon-button\n (click)=\"onLast()\"\n [disabled]=\"paging.pageNumber === paging.pageCount\"\n >\n <mat-icon>last_page</mat-icon>\n </button>\n <span id=\"total\">{{ paging.total }} </span>\n</div>\n", styles: ["#pages,#total{color:silver}.form-row{display:flex;gap:2px;align-items:center;flex-wrap:wrap}.form-row *{flex:0 0 auto}\n"] }]
43
+ }], ctorParameters: function () { return []; }, propDecorators: { paging: [{
44
+ type: Input
45
+ }], pagingChange: [{
46
+ type: Output
47
+ }] } });
48
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29tcGFjdC1wYWdlci5jb21wb25lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9teXJtaWRvbi9wYWdlZC1kYXRhLWJyb3dzZXJzL3NyYy9saWIvY29tcG9uZW50cy9jb21wYWN0LXBhZ2VyL2NvbXBhY3QtcGFnZXIuY29tcG9uZW50LnRzIiwiLi4vLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvbXlybWlkb24vcGFnZWQtZGF0YS1icm93c2Vycy9zcmMvbGliL2NvbXBvbmVudHMvY29tcGFjdC1wYWdlci9jb21wYWN0LXBhZ2VyLmNvbXBvbmVudC5odG1sIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxTQUFTLEVBQUUsWUFBWSxFQUFFLEtBQUssRUFBRSxNQUFNLEVBQUUsTUFBTSxlQUFlLENBQUM7Ozs7O0FBU3ZFLE1BQU0sT0FBTyxxQkFBcUI7SUFPaEM7UUFDRSxJQUFJLENBQUMsTUFBTSxHQUFHO1lBQ1osVUFBVSxFQUFFLENBQUM7WUFDYixTQUFTLEVBQUUsQ0FBQztZQUNaLEtBQUssRUFBRSxDQUFDO1NBQ1QsQ0FBQztRQUNGLElBQUksQ0FBQyxZQUFZLEdBQUcsSUFBSSxZQUFZLEVBQWMsQ0FBQztJQUNyRCxDQUFDO0lBRU0sT0FBTztRQUNaLElBQUksQ0FBQyxNQUFNLEdBQUcsRUFBRSxHQUFHLElBQUksQ0FBQyxNQUFNLEVBQUUsVUFBVSxFQUFFLENBQUMsRUFBRSxDQUFDO1FBQ2hELElBQUksQ0FBQyxZQUFZLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxNQUFNLENBQUMsQ0FBQztJQUN0QyxDQUFDO0lBRU0sVUFBVTtRQUNmLElBQUksSUFBSSxDQUFDLE1BQU0sQ0FBQyxVQUFVLEdBQUcsQ0FBQyxFQUFFO1lBQzlCLElBQUksQ0FBQyxNQUFNLEdBQUcsRUFBRSxHQUFHLElBQUksQ0FBQyxNQUFNLEVBQUUsVUFBVSxFQUFFLElBQUksQ0FBQyxNQUFNLENBQUMsVUFBVSxHQUFHLENBQUMsRUFBRSxDQUFDO1lBQ3pFLElBQUksQ0FBQyxZQUFZLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxNQUFNLENBQUMsQ0FBQztTQUNyQztJQUNILENBQUM7SUFFTSxNQUFNO1FBQ1gsSUFBSSxJQUFJLENBQUMsTUFBTSxDQUFDLFVBQVUsR0FBRyxJQUFJLENBQUMsTUFBTSxDQUFDLFNBQVMsRUFBRTtZQUNsRCxJQUFJLENBQUMsTUFBTSxHQUFHLEVBQUUsR0FBRyxJQUFJLENBQUMsTUFBTSxFQUFFLFVBQVUsRUFBRSxJQUFJLENBQUMsTUFBTSxDQUFDLFVBQVUsR0FBRyxDQUFDLEVBQUUsQ0FBQztZQUN6RSxJQUFJLENBQUMsWUFBWSxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsTUFBTSxDQUFDLENBQUM7U0FDckM7SUFDSCxDQUFDO0lBRU0sTUFBTTtRQUNYLElBQUksSUFBSSxDQUFDLE1BQU0sQ0FBQyxVQUFVLEdBQUcsSUFBSSxDQUFDLE1BQU0sQ0FBQyxTQUFTLEVBQUU7WUFDbEQsSUFBSSxDQUFDLE1BQU0sR0FBRyxFQUFFLEdBQUcsSUFBSSxDQUFDLE1BQU0sRUFBRSxVQUFVLEVBQUUsSUFBSSxDQUFDLE1BQU0sQ0FBQyxTQUFTLEVBQUUsQ0FBQztZQUNwRSxJQUFJLENBQUMsWUFBWSxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsTUFBTSxDQUFDLENBQUM7U0FDckM7SUFDSCxDQUFDOzhHQXhDVSxxQkFBcUI7a0dBQXJCLHFCQUFxQixrSUNUbEMsMDZCQW9DQTs7MkZEM0JhLHFCQUFxQjtrQkFMakMsU0FBUzsrQkFDRSxtQkFBbUI7MEVBTXRCLE1BQU07c0JBRFosS0FBSztnQkFJQyxZQUFZO3NCQURsQixNQUFNIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ29tcG9uZW50LCBFdmVudEVtaXR0ZXIsIElucHV0LCBPdXRwdXQgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcblxuaW1wb3J0IHsgUGFnaW5nSW5mbyB9IGZyb20gJy4uLy4uL3NlcnZpY2VzL3BhZ2VkLXRyZWUuc3RvcmUnO1xuXG5AQ29tcG9uZW50KHtcbiAgc2VsZWN0b3I6ICdwZGItY29tcGFjdC1wYWdlcicsXG4gIHRlbXBsYXRlVXJsOiAnLi9jb21wYWN0LXBhZ2VyLmNvbXBvbmVudC5odG1sJyxcbiAgc3R5bGVVcmxzOiBbJy4vY29tcGFjdC1wYWdlci5jb21wb25lbnQuc2NzcyddLFxufSlcbmV4cG9ydCBjbGFzcyBDb21wYWN0UGFnZXJDb21wb25lbnQge1xuICBASW5wdXQoKVxuICBwdWJsaWMgcGFnaW5nOiBQYWdpbmdJbmZvO1xuXG4gIEBPdXRwdXQoKVxuICBwdWJsaWMgcGFnaW5nQ2hhbmdlOiBFdmVudEVtaXR0ZXI8UGFnaW5nSW5mbz47XG5cbiAgY29uc3RydWN0b3IoKSB7XG4gICAgdGhpcy5wYWdpbmcgPSB7XG4gICAgICBwYWdlTnVtYmVyOiAwLFxuICAgICAgcGFnZUNvdW50OiAwLFxuICAgICAgdG90YWw6IDAsXG4gICAgfTtcbiAgICB0aGlzLnBhZ2luZ0NoYW5nZSA9IG5ldyBFdmVudEVtaXR0ZXI8UGFnaW5nSW5mbz4oKTtcbiAgfVxuXG4gIHB1YmxpYyBvbkZpcnN0KCk6IHZvaWQge1xuICAgIHRoaXMucGFnaW5nID0geyAuLi50aGlzLnBhZ2luZywgcGFnZU51bWJlcjogMSB9O1xuICAgIHRoaXMucGFnaW5nQ2hhbmdlLmVtaXQodGhpcy5wYWdpbmcpO1xuICB9XG5cbiAgcHVibGljIG9uUHJldmlvdXMoKTogdm9pZCB7XG4gICAgaWYgKHRoaXMucGFnaW5nLnBhZ2VOdW1iZXIgPiAxKSB7XG4gICAgICB0aGlzLnBhZ2luZyA9IHsgLi4udGhpcy5wYWdpbmcsIHBhZ2VOdW1iZXI6IHRoaXMucGFnaW5nLnBhZ2VOdW1iZXIgLSAxIH07XG4gICAgICB0aGlzLnBhZ2luZ0NoYW5nZS5lbWl0KHRoaXMucGFnaW5nKTtcbiAgICB9XG4gIH1cblxuICBwdWJsaWMgb25OZXh0KCk6IHZvaWQge1xuICAgIGlmICh0aGlzLnBhZ2luZy5wYWdlTnVtYmVyIDwgdGhpcy5wYWdpbmcucGFnZUNvdW50KSB7XG4gICAgICB0aGlzLnBhZ2luZyA9IHsgLi4udGhpcy5wYWdpbmcsIHBhZ2VOdW1iZXI6IHRoaXMucGFnaW5nLnBhZ2VOdW1iZXIgKyAxIH07XG4gICAgICB0aGlzLnBhZ2luZ0NoYW5nZS5lbWl0KHRoaXMucGFnaW5nKTtcbiAgICB9XG4gIH1cblxuICBwdWJsaWMgb25MYXN0KCk6IHZvaWQge1xuICAgIGlmICh0aGlzLnBhZ2luZy5wYWdlTnVtYmVyIDwgdGhpcy5wYWdpbmcucGFnZUNvdW50KSB7XG4gICAgICB0aGlzLnBhZ2luZyA9IHsgLi4udGhpcy5wYWdpbmcsIHBhZ2VOdW1iZXI6IHRoaXMucGFnaW5nLnBhZ2VDb3VudCB9O1xuICAgICAgdGhpcy5wYWdpbmdDaGFuZ2UuZW1pdCh0aGlzLnBhZ2luZyk7XG4gICAgfVxuICB9XG59XG4iLCI8ZGl2IGNsYXNzPVwiZm9ybS1yb3dcIiAqbmdJZj1cInBhZ2luZy5wYWdlQ291bnRcIj5cbiAgPHNwYW4gaWQ9XCJwYWdlc1wiPnt7IHBhZ2luZy5wYWdlTnVtYmVyIH19L3t7IHBhZ2luZy5wYWdlQ291bnQgfX08L3NwYW4+XG4gIDxidXR0b25cbiAgICB0eXBlPVwiYnV0dG9uXCJcbiAgICBtYXQtaWNvbi1idXR0b25cbiAgICAoY2xpY2spPVwib25GaXJzdCgpXCJcbiAgICBbZGlzYWJsZWRdPVwicGFnaW5nLnBhZ2VOdW1iZXIgPCAyXCJcbiAgPlxuICAgIDxtYXQtaWNvbj5maXJzdF9wYWdlPC9tYXQtaWNvbj5cbiAgPC9idXR0b24+XG4gIDxidXR0b25cbiAgICB0eXBlPVwiYnV0dG9uXCJcbiAgICBtYXQtaWNvbi1idXR0b25cbiAgICAoY2xpY2spPVwib25QcmV2aW91cygpXCJcbiAgICBbZGlzYWJsZWRdPVwicGFnaW5nLnBhZ2VOdW1iZXIgPCAyXCJcbiAgPlxuICAgIDxtYXQtaWNvbj5uYXZpZ2F0ZV9iZWZvcmU8L21hdC1pY29uPlxuICA8L2J1dHRvbj5cbiAgPGJ1dHRvblxuICAgIHR5cGU9XCJidXR0b25cIlxuICAgIG1hdC1pY29uLWJ1dHRvblxuICAgIChjbGljayk9XCJvbk5leHQoKVwiXG4gICAgW2Rpc2FibGVkXT1cInBhZ2luZy5wYWdlTnVtYmVyID09PSBwYWdpbmcucGFnZUNvdW50XCJcbiAgPlxuICAgIDxtYXQtaWNvbj5uYXZpZ2F0ZV9uZXh0PC9tYXQtaWNvbj5cbiAgPC9idXR0b24+XG4gIDxidXR0b25cbiAgICB0eXBlPVwiYnV0dG9uXCJcbiAgICBtYXQtaWNvbi1idXR0b25cbiAgICAoY2xpY2spPVwib25MYXN0KClcIlxuICAgIFtkaXNhYmxlZF09XCJwYWdpbmcucGFnZU51bWJlciA9PT0gcGFnaW5nLnBhZ2VDb3VudFwiXG4gID5cbiAgICA8bWF0LWljb24+bGFzdF9wYWdlPC9tYXQtaWNvbj5cbiAgPC9idXR0b24+XG4gIDxzcGFuIGlkPVwidG90YWxcIj57eyBwYWdpbmcudG90YWwgfX0gPC9zcGFuPlxuPC9kaXY+XG4iXX0=
@@ -0,0 +1,35 @@
1
+ import { Component, Input } from '@angular/core';
2
+ import * as i0 from "@angular/core";
3
+ export class RangeViewComponent {
4
+ constructor() {
5
+ this.domain = [0, 100];
6
+ this.range = [0, 100];
7
+ this.width = 100;
8
+ this.height = 5;
9
+ this.scaledRange = [];
10
+ }
11
+ ngOnChanges() {
12
+ const domainWidth = this.domain[1] - this.domain[0];
13
+ const rangeWidth = this.range[1] - this.range[0];
14
+ const rangeStart = this.range[0] - this.domain[0];
15
+ this.scaledRange = [
16
+ (rangeStart / domainWidth) * this.width,
17
+ ((rangeStart + rangeWidth) / domainWidth) * this.width,
18
+ ];
19
+ }
20
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.7", ngImport: i0, type: RangeViewComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
21
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.2.7", type: RangeViewComponent, selector: "pdb-range-view", inputs: { domain: "domain", range: "range", width: "width", height: "height" }, usesOnChanges: true, ngImport: i0, template: "<svg [attr.width]=\"width\" [attr.height]=\"height\">\n <rect id=\"rdomain\" [attr.width]=\"width\" [attr.height]=\"height\" />\n <rect\n id=\"rrange\"\n [attr.x]=\"scaledRange[0]\"\n [attr.y]=\"0\"\n [attr.width]=\"scaledRange[1] - scaledRange[0]\"\n [attr.height]=\"height\"\n />\n</svg>\n", styles: ["#rdomain{fill:#d3d3d3;stroke-width:3;stroke:#c1ba9b}#rrange{fill:#91aad3;stroke-width:3}\n"] }); }
22
+ }
23
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.7", ngImport: i0, type: RangeViewComponent, decorators: [{
24
+ type: Component,
25
+ args: [{ selector: 'pdb-range-view', template: "<svg [attr.width]=\"width\" [attr.height]=\"height\">\n <rect id=\"rdomain\" [attr.width]=\"width\" [attr.height]=\"height\" />\n <rect\n id=\"rrange\"\n [attr.x]=\"scaledRange[0]\"\n [attr.y]=\"0\"\n [attr.width]=\"scaledRange[1] - scaledRange[0]\"\n [attr.height]=\"height\"\n />\n</svg>\n", styles: ["#rdomain{fill:#d3d3d3;stroke-width:3;stroke:#c1ba9b}#rrange{fill:#91aad3;stroke-width:3}\n"] }]
26
+ }], ctorParameters: function () { return []; }, propDecorators: { domain: [{
27
+ type: Input
28
+ }], range: [{
29
+ type: Input
30
+ }], width: [{
31
+ type: Input
32
+ }], height: [{
33
+ type: Input
34
+ }] } });
35
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicmFuZ2Utdmlldy5jb21wb25lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9teXJtaWRvbi9wYWdlZC1kYXRhLWJyb3dzZXJzL3NyYy9saWIvY29tcG9uZW50cy9yYW5nZS12aWV3L3JhbmdlLXZpZXcuY29tcG9uZW50LnRzIiwiLi4vLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvbXlybWlkb24vcGFnZWQtZGF0YS1icm93c2Vycy9zcmMvbGliL2NvbXBvbmVudHMvcmFuZ2Utdmlldy9yYW5nZS12aWV3LmNvbXBvbmVudC5odG1sIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxTQUFTLEVBQUUsS0FBSyxFQUFhLE1BQU0sZUFBZSxDQUFDOztBQU81RCxNQUFNLE9BQU8sa0JBQWtCO0lBb0I3QjtRQUNFLElBQUksQ0FBQyxNQUFNLEdBQUcsQ0FBQyxDQUFDLEVBQUUsR0FBRyxDQUFDLENBQUM7UUFDdkIsSUFBSSxDQUFDLEtBQUssR0FBRyxDQUFDLENBQUMsRUFBRSxHQUFHLENBQUMsQ0FBQztRQUN0QixJQUFJLENBQUMsS0FBSyxHQUFHLEdBQUcsQ0FBQztRQUNqQixJQUFJLENBQUMsTUFBTSxHQUFHLENBQUMsQ0FBQztRQUNoQixJQUFJLENBQUMsV0FBVyxHQUFHLEVBQUUsQ0FBQztJQUN4QixDQUFDO0lBRUQsV0FBVztRQUNULE1BQU0sV0FBVyxHQUFHLElBQUksQ0FBQyxNQUFNLENBQUMsQ0FBQyxDQUFDLEdBQUcsSUFBSSxDQUFDLE1BQU0sQ0FBQyxDQUFDLENBQUMsQ0FBQztRQUNwRCxNQUFNLFVBQVUsR0FBRyxJQUFJLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQyxHQUFHLElBQUksQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDLENBQUM7UUFDakQsTUFBTSxVQUFVLEdBQUcsSUFBSSxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsR0FBRyxJQUFJLENBQUMsTUFBTSxDQUFDLENBQUMsQ0FBQyxDQUFDO1FBRWxELElBQUksQ0FBQyxXQUFXLEdBQUc7WUFDakIsQ0FBQyxVQUFVLEdBQUcsV0FBVyxDQUFDLEdBQUcsSUFBSSxDQUFDLEtBQUs7WUFDdkMsQ0FBQyxDQUFDLFVBQVUsR0FBRyxVQUFVLENBQUMsR0FBRyxXQUFXLENBQUMsR0FBRyxJQUFJLENBQUMsS0FBSztTQUN2RCxDQUFDO0lBQ0osQ0FBQzs4R0FyQ1Usa0JBQWtCO2tHQUFsQixrQkFBa0IsMkpDUC9CLHVUQVVBOzsyRkRIYSxrQkFBa0I7a0JBTDlCLFNBQVM7K0JBQ0UsZ0JBQWdCOzBFQVFWLE1BQU07c0JBQXJCLEtBQUs7Z0JBSVUsS0FBSztzQkFBcEIsS0FBSztnQkFJVSxLQUFLO3NCQUFwQixLQUFLO2dCQUlVLE1BQU07c0JBQXJCLEtBQUsiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDb21wb25lbnQsIElucHV0LCBPbkNoYW5nZXMgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcblxuQENvbXBvbmVudCh7XG4gIHNlbGVjdG9yOiAncGRiLXJhbmdlLXZpZXcnLFxuICB0ZW1wbGF0ZVVybDogJy4vcmFuZ2Utdmlldy5jb21wb25lbnQuaHRtbCcsXG4gIHN0eWxlVXJsczogWycuL3JhbmdlLXZpZXcuY29tcG9uZW50LnNjc3MnXSxcbn0pXG5leHBvcnQgY2xhc3MgUmFuZ2VWaWV3Q29tcG9uZW50IGltcGxlbWVudHMgT25DaGFuZ2VzIHtcbiAgLyoqXG4gICAqIFRoZSBkb21haW4gb2YgdGhlIHJhbmdlIHZpZXcgKHN0YXJ0LCBsaW1pdCkuXG4gICAqL1xuICBASW5wdXQoKSBwdWJsaWMgZG9tYWluOiBBcnJheTxudW1iZXI+O1xuICAvKipcbiAgICogVGhlIHJhbmdlIG9mIHRoZSByYW5nZSB2aWV3IChzdGFydCwgbGltaXQpLlxuICAgKi9cbiAgQElucHV0KCkgcHVibGljIHJhbmdlOiBBcnJheTxudW1iZXI+O1xuICAvKipcbiAgICogVGhlIHdpZHRoIG9mIHRoZSBjb21wb25lbnQuXG4gICAqL1xuICBASW5wdXQoKSBwdWJsaWMgd2lkdGg6IG51bWJlcjtcbiAgLyoqXG4gICAqIFRoZSBoZWlnaHQgb2YgdGhlIGNvbXBvbmVudC5cbiAgICovXG4gIEBJbnB1dCgpIHB1YmxpYyBoZWlnaHQ6IG51bWJlcjtcblxuICBwdWJsaWMgc2NhbGVkUmFuZ2U6IEFycmF5PG51bWJlcj47XG5cbiAgY29uc3RydWN0b3IoKSB7XG4gICAgdGhpcy5kb21haW4gPSBbMCwgMTAwXTtcbiAgICB0aGlzLnJhbmdlID0gWzAsIDEwMF07XG4gICAgdGhpcy53aWR0aCA9IDEwMDtcbiAgICB0aGlzLmhlaWdodCA9IDU7XG4gICAgdGhpcy5zY2FsZWRSYW5nZSA9IFtdO1xuICB9XG5cbiAgbmdPbkNoYW5nZXMoKSB7XG4gICAgY29uc3QgZG9tYWluV2lkdGggPSB0aGlzLmRvbWFpblsxXSAtIHRoaXMuZG9tYWluWzBdO1xuICAgIGNvbnN0IHJhbmdlV2lkdGggPSB0aGlzLnJhbmdlWzFdIC0gdGhpcy5yYW5nZVswXTtcbiAgICBjb25zdCByYW5nZVN0YXJ0ID0gdGhpcy5yYW5nZVswXSAtIHRoaXMuZG9tYWluWzBdO1xuXG4gICAgdGhpcy5zY2FsZWRSYW5nZSA9IFtcbiAgICAgIChyYW5nZVN0YXJ0IC8gZG9tYWluV2lkdGgpICogdGhpcy53aWR0aCxcbiAgICAgICgocmFuZ2VTdGFydCArIHJhbmdlV2lkdGgpIC8gZG9tYWluV2lkdGgpICogdGhpcy53aWR0aCxcbiAgICBdO1xuICB9XG59XG4iLCI8c3ZnIFthdHRyLndpZHRoXT1cIndpZHRoXCIgW2F0dHIuaGVpZ2h0XT1cImhlaWdodFwiPlxuICA8cmVjdCBpZD1cInJkb21haW5cIiBbYXR0ci53aWR0aF09XCJ3aWR0aFwiIFthdHRyLmhlaWdodF09XCJoZWlnaHRcIiAvPlxuICA8cmVjdFxuICAgIGlkPVwicnJhbmdlXCJcbiAgICBbYXR0ci54XT1cInNjYWxlZFJhbmdlWzBdXCJcbiAgICBbYXR0ci55XT1cIjBcIlxuICAgIFthdHRyLndpZHRoXT1cInNjYWxlZFJhbmdlWzFdIC0gc2NhbGVkUmFuZ2VbMF1cIlxuICAgIFthdHRyLmhlaWdodF09XCJoZWlnaHRcIlxuICAvPlxuPC9zdmc+XG4iXX0=
@@ -0,0 +1,92 @@
1
+ import { NgModule } from '@angular/core';
2
+ import { CommonModule } from '@angular/common';
3
+ import { ReactiveFormsModule } from '@angular/forms';
4
+ import { MatBadgeModule } from '@angular/material/badge';
5
+ import { MatButtonModule } from '@angular/material/button';
6
+ import { MatChipsModule } from '@angular/material/chips';
7
+ import { MatDialogModule } from '@angular/material/dialog';
8
+ import { MatFormFieldModule } from '@angular/material/form-field';
9
+ import { MatIconModule } from '@angular/material/icon';
10
+ import { MatInputModule } from '@angular/material/input';
11
+ import { MatPaginatorModule } from '@angular/material/paginator';
12
+ import { MatProgressBarModule } from '@angular/material/progress-bar';
13
+ import { MatSelectModule } from '@angular/material/select';
14
+ import { MatTooltipModule } from '@angular/material/tooltip';
15
+ import { NgToolsModule } from '@myrmidon/ng-tools';
16
+ import { CompactPagerComponent } from './components/compact-pager/compact-pager.component';
17
+ import { RangeViewComponent } from './components/range-view/range-view.component';
18
+ import { BrowserTreeNodeComponent } from './components/browser-tree-node/browser-tree-node.component';
19
+ import * as i0 from "@angular/core";
20
+ export class PagedDataBrowsersModule {
21
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.7", ngImport: i0, type: PagedDataBrowsersModule, deps: [], target: i0.ɵɵFactoryTarget.NgModule }); }
22
+ static { this.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "14.0.0", version: "16.2.7", ngImport: i0, type: PagedDataBrowsersModule, declarations: [CompactPagerComponent,
23
+ RangeViewComponent,
24
+ BrowserTreeNodeComponent], imports: [CommonModule,
25
+ ReactiveFormsModule,
26
+ // material
27
+ MatBadgeModule,
28
+ MatButtonModule,
29
+ MatChipsModule,
30
+ MatDialogModule,
31
+ MatFormFieldModule,
32
+ MatIconModule,
33
+ MatInputModule,
34
+ MatPaginatorModule,
35
+ MatProgressBarModule,
36
+ MatSelectModule,
37
+ MatTooltipModule,
38
+ // myrmidon
39
+ NgToolsModule], exports: [CompactPagerComponent,
40
+ RangeViewComponent,
41
+ BrowserTreeNodeComponent] }); }
42
+ static { this.ɵinj = i0.ɵɵngDeclareInjector({ minVersion: "12.0.0", version: "16.2.7", ngImport: i0, type: PagedDataBrowsersModule, imports: [CommonModule,
43
+ ReactiveFormsModule,
44
+ // material
45
+ MatBadgeModule,
46
+ MatButtonModule,
47
+ MatChipsModule,
48
+ MatDialogModule,
49
+ MatFormFieldModule,
50
+ MatIconModule,
51
+ MatInputModule,
52
+ MatPaginatorModule,
53
+ MatProgressBarModule,
54
+ MatSelectModule,
55
+ MatTooltipModule,
56
+ // myrmidon
57
+ NgToolsModule] }); }
58
+ }
59
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.7", ngImport: i0, type: PagedDataBrowsersModule, decorators: [{
60
+ type: NgModule,
61
+ args: [{
62
+ declarations: [
63
+ CompactPagerComponent,
64
+ RangeViewComponent,
65
+ BrowserTreeNodeComponent,
66
+ ],
67
+ imports: [
68
+ CommonModule,
69
+ ReactiveFormsModule,
70
+ // material
71
+ MatBadgeModule,
72
+ MatButtonModule,
73
+ MatChipsModule,
74
+ MatDialogModule,
75
+ MatFormFieldModule,
76
+ MatIconModule,
77
+ MatInputModule,
78
+ MatPaginatorModule,
79
+ MatProgressBarModule,
80
+ MatSelectModule,
81
+ MatTooltipModule,
82
+ // myrmidon
83
+ NgToolsModule
84
+ ],
85
+ exports: [
86
+ CompactPagerComponent,
87
+ RangeViewComponent,
88
+ BrowserTreeNodeComponent,
89
+ ],
90
+ }]
91
+ }] });
92
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicGFnZWQtZGF0YS1icm93c2Vycy5tb2R1bGUuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9teXJtaWRvbi9wYWdlZC1kYXRhLWJyb3dzZXJzL3NyYy9saWIvcGFnZWQtZGF0YS1icm93c2Vycy5tb2R1bGUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFFBQVEsRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUN6QyxPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0saUJBQWlCLENBQUM7QUFDL0MsT0FBTyxFQUFFLG1CQUFtQixFQUFFLE1BQU0sZ0JBQWdCLENBQUM7QUFFckQsT0FBTyxFQUFFLGNBQWMsRUFBRSxNQUFNLHlCQUF5QixDQUFDO0FBQ3pELE9BQU8sRUFBRSxlQUFlLEVBQUUsTUFBTSwwQkFBMEIsQ0FBQztBQUMzRCxPQUFPLEVBQUUsY0FBYyxFQUFFLE1BQU0seUJBQXlCLENBQUM7QUFDekQsT0FBTyxFQUFFLGVBQWUsRUFBRSxNQUFNLDBCQUEwQixDQUFDO0FBQzNELE9BQU8sRUFBRSxrQkFBa0IsRUFBRSxNQUFNLDhCQUE4QixDQUFDO0FBQ2xFLE9BQU8sRUFBRSxhQUFhLEVBQUUsTUFBTSx3QkFBd0IsQ0FBQztBQUN2RCxPQUFPLEVBQUUsY0FBYyxFQUFFLE1BQU0seUJBQXlCLENBQUM7QUFDekQsT0FBTyxFQUFFLGtCQUFrQixFQUFFLE1BQU0sNkJBQTZCLENBQUM7QUFDakUsT0FBTyxFQUFFLG9CQUFvQixFQUFFLE1BQU0sZ0NBQWdDLENBQUM7QUFDdEUsT0FBTyxFQUFFLGVBQWUsRUFBRSxNQUFNLDBCQUEwQixDQUFDO0FBQzNELE9BQU8sRUFBRSxnQkFBZ0IsRUFBRSxNQUFNLDJCQUEyQixDQUFDO0FBRTdELE9BQU8sRUFBRSxhQUFhLEVBQUUsTUFBTSxvQkFBb0IsQ0FBQztBQUVuRCxPQUFPLEVBQUUscUJBQXFCLEVBQUUsTUFBTSxvREFBb0QsQ0FBQztBQUMzRixPQUFPLEVBQUUsa0JBQWtCLEVBQUUsTUFBTSw4Q0FBOEMsQ0FBQztBQUNsRixPQUFPLEVBQUUsd0JBQXdCLEVBQUUsTUFBTSw0REFBNEQsQ0FBQzs7QUFnQ3RHLE1BQU0sT0FBTyx1QkFBdUI7OEdBQXZCLHVCQUF1QjsrR0FBdkIsdUJBQXVCLGlCQTVCaEMscUJBQXFCO1lBQ3JCLGtCQUFrQjtZQUNsQix3QkFBd0IsYUFHeEIsWUFBWTtZQUNaLG1CQUFtQjtZQUNuQixXQUFXO1lBQ1gsY0FBYztZQUNkLGVBQWU7WUFDZixjQUFjO1lBQ2QsZUFBZTtZQUNmLGtCQUFrQjtZQUNsQixhQUFhO1lBQ2IsY0FBYztZQUNkLGtCQUFrQjtZQUNsQixvQkFBb0I7WUFDcEIsZUFBZTtZQUNmLGdCQUFnQjtZQUNoQixXQUFXO1lBQ1gsYUFBYSxhQUdiLHFCQUFxQjtZQUNyQixrQkFBa0I7WUFDbEIsd0JBQXdCOytHQUdmLHVCQUF1QixZQXZCaEMsWUFBWTtZQUNaLG1CQUFtQjtZQUNuQixXQUFXO1lBQ1gsY0FBYztZQUNkLGVBQWU7WUFDZixjQUFjO1lBQ2QsZUFBZTtZQUNmLGtCQUFrQjtZQUNsQixhQUFhO1lBQ2IsY0FBYztZQUNkLGtCQUFrQjtZQUNsQixvQkFBb0I7WUFDcEIsZUFBZTtZQUNmLGdCQUFnQjtZQUNoQixXQUFXO1lBQ1gsYUFBYTs7MkZBUUosdUJBQXVCO2tCQTlCbkMsUUFBUTttQkFBQztvQkFDUixZQUFZLEVBQUU7d0JBQ1oscUJBQXFCO3dCQUNyQixrQkFBa0I7d0JBQ2xCLHdCQUF3QjtxQkFDekI7b0JBQ0QsT0FBTyxFQUFFO3dCQUNQLFlBQVk7d0JBQ1osbUJBQW1CO3dCQUNuQixXQUFXO3dCQUNYLGNBQWM7d0JBQ2QsZUFBZTt3QkFDZixjQUFjO3dCQUNkLGVBQWU7d0JBQ2Ysa0JBQWtCO3dCQUNsQixhQUFhO3dCQUNiLGNBQWM7d0JBQ2Qsa0JBQWtCO3dCQUNsQixvQkFBb0I7d0JBQ3BCLGVBQWU7d0JBQ2YsZ0JBQWdCO3dCQUNoQixXQUFXO3dCQUNYLGFBQWE7cUJBQ2Q7b0JBQ0QsT0FBTyxFQUFFO3dCQUNQLHFCQUFxQjt3QkFDckIsa0JBQWtCO3dCQUNsQix3QkFBd0I7cUJBQ3pCO2lCQUNGIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgTmdNb2R1bGUgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IENvbW1vbk1vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvbW1vbic7XG5pbXBvcnQgeyBSZWFjdGl2ZUZvcm1zTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvZm9ybXMnO1xuXG5pbXBvcnQgeyBNYXRCYWRnZU1vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL21hdGVyaWFsL2JhZGdlJztcbmltcG9ydCB7IE1hdEJ1dHRvbk1vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL21hdGVyaWFsL2J1dHRvbic7XG5pbXBvcnQgeyBNYXRDaGlwc01vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL21hdGVyaWFsL2NoaXBzJztcbmltcG9ydCB7IE1hdERpYWxvZ01vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL21hdGVyaWFsL2RpYWxvZyc7XG5pbXBvcnQgeyBNYXRGb3JtRmllbGRNb2R1bGUgfSBmcm9tICdAYW5ndWxhci9tYXRlcmlhbC9mb3JtLWZpZWxkJztcbmltcG9ydCB7IE1hdEljb25Nb2R1bGUgfSBmcm9tICdAYW5ndWxhci9tYXRlcmlhbC9pY29uJztcbmltcG9ydCB7IE1hdElucHV0TW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvbWF0ZXJpYWwvaW5wdXQnO1xuaW1wb3J0IHsgTWF0UGFnaW5hdG9yTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvbWF0ZXJpYWwvcGFnaW5hdG9yJztcbmltcG9ydCB7IE1hdFByb2dyZXNzQmFyTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvbWF0ZXJpYWwvcHJvZ3Jlc3MtYmFyJztcbmltcG9ydCB7IE1hdFNlbGVjdE1vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL21hdGVyaWFsL3NlbGVjdCc7XG5pbXBvcnQgeyBNYXRUb29sdGlwTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvbWF0ZXJpYWwvdG9vbHRpcCc7XG5cbmltcG9ydCB7IE5nVG9vbHNNb2R1bGUgfSBmcm9tICdAbXlybWlkb24vbmctdG9vbHMnO1xuXG5pbXBvcnQgeyBDb21wYWN0UGFnZXJDb21wb25lbnQgfSBmcm9tICcuL2NvbXBvbmVudHMvY29tcGFjdC1wYWdlci9jb21wYWN0LXBhZ2VyLmNvbXBvbmVudCc7XG5pbXBvcnQgeyBSYW5nZVZpZXdDb21wb25lbnQgfSBmcm9tICcuL2NvbXBvbmVudHMvcmFuZ2Utdmlldy9yYW5nZS12aWV3LmNvbXBvbmVudCc7XG5pbXBvcnQgeyBCcm93c2VyVHJlZU5vZGVDb21wb25lbnQgfSBmcm9tICcuL2NvbXBvbmVudHMvYnJvd3Nlci10cmVlLW5vZGUvYnJvd3Nlci10cmVlLW5vZGUuY29tcG9uZW50JztcblxuQE5nTW9kdWxlKHtcbiAgZGVjbGFyYXRpb25zOiBbXG4gICAgQ29tcGFjdFBhZ2VyQ29tcG9uZW50LFxuICAgIFJhbmdlVmlld0NvbXBvbmVudCxcbiAgICBCcm93c2VyVHJlZU5vZGVDb21wb25lbnQsXG4gIF0sXG4gIGltcG9ydHM6IFtcbiAgICBDb21tb25Nb2R1bGUsXG4gICAgUmVhY3RpdmVGb3Jtc01vZHVsZSxcbiAgICAvLyBtYXRlcmlhbFxuICAgIE1hdEJhZGdlTW9kdWxlLFxuICAgIE1hdEJ1dHRvbk1vZHVsZSxcbiAgICBNYXRDaGlwc01vZHVsZSxcbiAgICBNYXREaWFsb2dNb2R1bGUsXG4gICAgTWF0Rm9ybUZpZWxkTW9kdWxlLFxuICAgIE1hdEljb25Nb2R1bGUsXG4gICAgTWF0SW5wdXRNb2R1bGUsXG4gICAgTWF0UGFnaW5hdG9yTW9kdWxlLFxuICAgIE1hdFByb2dyZXNzQmFyTW9kdWxlLFxuICAgIE1hdFNlbGVjdE1vZHVsZSxcbiAgICBNYXRUb29sdGlwTW9kdWxlLFxuICAgIC8vIG15cm1pZG9uXG4gICAgTmdUb29sc01vZHVsZVxuICBdLFxuICBleHBvcnRzOiBbXG4gICAgQ29tcGFjdFBhZ2VyQ29tcG9uZW50LFxuICAgIFJhbmdlVmlld0NvbXBvbmVudCxcbiAgICBCcm93c2VyVHJlZU5vZGVDb21wb25lbnQsXG4gIF0sXG59KVxuZXhwb3J0IGNsYXNzIFBhZ2VkRGF0YUJyb3dzZXJzTW9kdWxlIHt9XG4iXX0=
@@ -0,0 +1,111 @@
1
+ /**
2
+ * A Least Recently Used cache that can be used to store any type of object.
3
+ * The cache works in two modes: considering the size of the objects or not.
4
+ * If the size is considered, the cache will have a maximum size and will
5
+ * remove the oldest objects when the maximum size is reached. Note that
6
+ * the size is only roughly estimated. This avoids removing too many
7
+ * entries from the cache when the maximum is reached.
8
+ * If the size is not considered, the cache will have a maximum number of
9
+ * objects and will remove the oldest objects when the maximum number is
10
+ * reached.
11
+ */
12
+ export class LRUCache {
13
+ /**
14
+ * Creates a new cache.
15
+ * @param maxSize The maximum size of the cache. This is either
16
+ * the maximum number of items in the cache (when considerSize
17
+ * is false) or the maximum total size of all items in the
18
+ * cache in bytes (when considerSize is true).
19
+ * @param considerSize True if the size of the objects should be
20
+ * considered.
21
+ */
22
+ constructor(maxSize, considerSize = false) {
23
+ this.maxSize = maxSize;
24
+ this.totalSize = 0;
25
+ this.considerSize = considerSize;
26
+ this.cache = new Map();
27
+ this.sizes = new Map();
28
+ }
29
+ /**
30
+ * Get an item from the cache.
31
+ * @param key The key of the item to get.
32
+ * @returns The item or undefined if the item is not in the cache.
33
+ */
34
+ get(key) {
35
+ let item = this.cache.get(key);
36
+ if (item) {
37
+ this.cache.delete(key);
38
+ this.cache.set(key, item);
39
+ }
40
+ return item;
41
+ }
42
+ /**
43
+ * Put an item in the cache.
44
+ * @param key The key of the item to put.
45
+ * @param item The item to put.
46
+ * @param size The estimated size of the item in bytes.
47
+ * This must be calculated by the caller but only when
48
+ * considerSize is true.
49
+ */
50
+ put(key, item, size) {
51
+ this.cache.delete(key);
52
+ this.cache.set(key, item);
53
+ this.sizes.set(key, size);
54
+ if (this.considerSize) {
55
+ this.totalSize += size;
56
+ while (this.totalSize > this.maxSize) {
57
+ const oldestKey = this.cache.keys().next().value;
58
+ let oldestSize = this.sizes.get(oldestKey);
59
+ if (oldestSize) {
60
+ this.totalSize -= oldestSize;
61
+ }
62
+ this.cache.delete(oldestKey);
63
+ this.sizes.delete(oldestKey);
64
+ }
65
+ }
66
+ else {
67
+ while (this.cache.size > this.maxSize) {
68
+ const oldestKey = this.cache.keys().next().value;
69
+ this.cache.delete(oldestKey);
70
+ this.sizes.delete(oldestKey);
71
+ }
72
+ }
73
+ }
74
+ /**
75
+ * Clear the cache.
76
+ */
77
+ clear() {
78
+ this.cache.clear();
79
+ this.sizes.clear();
80
+ this.totalSize = 0;
81
+ }
82
+ /**
83
+ * Estimate the size of an object in bytes.
84
+ * @param obj The object to calculate the size of.
85
+ * @returns The estimated size of the object in bytes.
86
+ */
87
+ static calculateObjectSize(obj) {
88
+ if (!obj) {
89
+ return 0;
90
+ }
91
+ let totalSize = 0;
92
+ let keys = Object.keys(obj);
93
+ for (let key of keys) {
94
+ let value = obj[key];
95
+ if (typeof value === 'string') {
96
+ totalSize += value.length * 2;
97
+ }
98
+ else if (typeof value === 'number') {
99
+ totalSize += 8;
100
+ }
101
+ else if (typeof value === 'boolean') {
102
+ totalSize += 4;
103
+ }
104
+ else if (typeof value === 'object' && value !== null) {
105
+ totalSize += this.calculateObjectSize(value);
106
+ }
107
+ }
108
+ return totalSize;
109
+ }
110
+ }
111
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"lru-cache.js","sourceRoot":"","sources":["../../../../../../projects/myrmidon/paged-data-browsers/src/lib/services/lru-cache.ts"],"names":[],"mappings":"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","sourcesContent":["/**\r\n * A Least Recently Used cache that can be used to store any type of object.\r\n * The cache works in two modes: considering the size of the objects or not.\r\n * If the size is considered, the cache will have a maximum size and will\r\n * remove the oldest objects when the maximum size is reached. Note that\r\n * the size is only roughly estimated. This avoids removing too many\r\n * entries from the cache when the maximum is reached.\r\n * If the size is not considered, the cache will have a maximum number of\r\n * objects and will remove the oldest objects when the maximum number is\r\n * reached.\r\n */\r\nexport class LRUCache<T> {\r\n  private maxSize: number;\r\n  private totalSize: number;\r\n  private considerSize: boolean;\r\n  private cache: Map<string, T>;\r\n  private sizes: Map<string, number>;\r\n\r\n  /**\r\n   * Creates a new cache.\r\n   * @param maxSize The maximum size of the cache. This is either\r\n   * the maximum number of items in the cache (when considerSize\r\n   * is false) or the maximum total size of all items in the\r\n   * cache in bytes (when considerSize is true).\r\n   * @param considerSize True if the size of the objects should be\r\n   * considered.\r\n   */\r\n  constructor(maxSize: number, considerSize: boolean = false) {\r\n    this.maxSize = maxSize;\r\n    this.totalSize = 0;\r\n    this.considerSize = considerSize;\r\n    this.cache = new Map<string, T>();\r\n    this.sizes = new Map<string, number>();\r\n  }\r\n\r\n  /**\r\n   * Get an item from the cache.\r\n   * @param key The key of the item to get.\r\n   * @returns The item or undefined if the item is not in the cache.\r\n   */\r\n  public get(key: string): T | undefined {\r\n    let item: T | undefined = this.cache.get(key);\r\n    if (item) {\r\n      this.cache.delete(key);\r\n      this.cache.set(key, item);\r\n    }\r\n    return item;\r\n  }\r\n\r\n  /**\r\n   * Put an item in the cache.\r\n   * @param key The key of the item to put.\r\n   * @param item The item to put.\r\n   * @param size The estimated size of the item in bytes.\r\n   * This must be calculated by the caller but only when\r\n   * considerSize is true.\r\n   */\r\n  public put(key: string, item: T, size: number): void {\r\n    this.cache.delete(key);\r\n    this.cache.set(key, item);\r\n    this.sizes.set(key, size);\r\n    if (this.considerSize) {\r\n      this.totalSize += size;\r\n      while (this.totalSize > this.maxSize) {\r\n        const oldestKey = this.cache.keys().next().value;\r\n        let oldestSize = this.sizes.get(oldestKey);\r\n        if (oldestSize) {\r\n          this.totalSize -= oldestSize;\r\n        }\r\n        this.cache.delete(oldestKey);\r\n        this.sizes.delete(oldestKey);\r\n      }\r\n    } else {\r\n      while (this.cache.size > this.maxSize) {\r\n        const oldestKey = this.cache.keys().next().value;\r\n        this.cache.delete(oldestKey);\r\n        this.sizes.delete(oldestKey);\r\n      }\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Clear the cache.\r\n   */\r\n  public clear(): void {\r\n    this.cache.clear();\r\n    this.sizes.clear();\r\n    this.totalSize = 0;\r\n  }\r\n\r\n  /**\r\n   * Estimate the size of an object in bytes.\r\n   * @param obj The object to calculate the size of.\r\n   * @returns The estimated size of the object in bytes.\r\n   */\r\n  public static calculateObjectSize(obj: any): number {\r\n    if (!obj) {\r\n      return 0;\r\n    }\r\n    let totalSize = 0;\r\n    let keys = Object.keys(obj);\r\n    for (let key of keys) {\r\n      let value = obj[key];\r\n      if (typeof value === 'string') {\r\n        totalSize += value.length * 2;\r\n      } else if (typeof value === 'number') {\r\n        totalSize += 8;\r\n      } else if (typeof value === 'boolean') {\r\n        totalSize += 4;\r\n      } else if (typeof value === 'object' && value !== null) {\r\n        totalSize += this.calculateObjectSize(value);\r\n      }\r\n    }\r\n    return totalSize;\r\n  }\r\n}\r\n"]}
@@ -0,0 +1,128 @@
1
+ import { BehaviorSubject } from 'rxjs';
2
+ import { LRUCache } from './lru-cache';
3
+ /**
4
+ * Default options for the paged list store.
5
+ */
6
+ export const DEFAULT_PAGED_LIST_STORE_OPTIONS = {
7
+ pageSize: 20,
8
+ cacheSize: 50,
9
+ };
10
+ /**
11
+ * A generic paged list store using a filter object of type F
12
+ * and a list of elements of type E.
13
+ */
14
+ export class PagedListStore {
15
+ /**
16
+ * The size of nodes pages in this store. If you change it, the store
17
+ * is reset. The default value is 20.
18
+ */
19
+ get pageSize() {
20
+ return this._pageSize;
21
+ }
22
+ set pageSize(value) {
23
+ if (this._pageSize === value) {
24
+ return;
25
+ }
26
+ this._pageSize = value;
27
+ this.reset();
28
+ }
29
+ /**
30
+ * Create a new paged list store.
31
+ * @param options Options for the paged list store.
32
+ */
33
+ constructor(_service, options = DEFAULT_PAGED_LIST_STORE_OPTIONS) {
34
+ this._service = _service;
35
+ this._pageSize = options.pageSize;
36
+ this._cache = new LRUCache(options.cacheSize);
37
+ // page
38
+ this._page$ = new BehaviorSubject({
39
+ pageNumber: 0,
40
+ pageCount: 0,
41
+ pageSize: 0,
42
+ total: 0,
43
+ items: [],
44
+ });
45
+ this.page$ = this._page$.asObservable();
46
+ // filter
47
+ this._filter$ = new BehaviorSubject({});
48
+ this.filter$ = this._filter$.asObservable();
49
+ }
50
+ /**
51
+ * Returns true if the store is empty, false otherwise.
52
+ * @returns true if the store is empty, false otherwise.
53
+ */
54
+ isEmpty() {
55
+ return this._page$.value.items.length === 0;
56
+ }
57
+ /**
58
+ * Build the cache key for the given page number and filter.
59
+ * The default implementation just returns a stringified object
60
+ * containing the page number and the filter. You may override
61
+ * this method to provide a custom cache key.
62
+ * @param pageNumber The page number.
63
+ * @param filter The filter.
64
+ * @returns A string to be used as cache key.
65
+ */
66
+ buildCacheKey(pageNumber, filter) {
67
+ if (this._customCacheKeyBuilder) {
68
+ return this._customCacheKeyBuilder(pageNumber, filter);
69
+ }
70
+ return JSON.stringify({ pageNumber, ...filter });
71
+ }
72
+ /**
73
+ * Load the page with the given number.
74
+ * @param pageNumber the page number to load.
75
+ */
76
+ loadPage(pageNumber) {
77
+ return this._service.loadPage(pageNumber, this._pageSize, this._filter$.value);
78
+ }
79
+ /**
80
+ * Set the page with the given number.
81
+ * @param pageNumber The page number to load.
82
+ * @param pageSize The page size.
83
+ * @returns Promise which resolves when the page is loaded.
84
+ */
85
+ setPage(pageNumber, pageSize) {
86
+ if (pageSize && pageSize !== this._pageSize) {
87
+ this._pageSize = pageSize;
88
+ }
89
+ return new Promise((resolve, reject) => {
90
+ // if page is in cache, return it
91
+ const key = this.buildCacheKey(pageNumber, this._filter$.value);
92
+ const cachedPage = this._cache.get(key);
93
+ if (cachedPage) {
94
+ this._page$.next(cachedPage);
95
+ resolve();
96
+ return;
97
+ }
98
+ // else load page
99
+ this.loadPage(pageNumber).subscribe({
100
+ next: (page) => {
101
+ this._page$.next(page);
102
+ resolve();
103
+ },
104
+ error: reject,
105
+ });
106
+ });
107
+ }
108
+ /**
109
+ * Apply the given filter and load the first page.
110
+ * @param filter The filter to apply.
111
+ * @returns Promise which resolves when the page is loaded.
112
+ */
113
+ applyFilter(filter) {
114
+ return new Promise((resolve, reject) => {
115
+ this._filter$.next(filter);
116
+ this.setPage(1).then(resolve, reject);
117
+ });
118
+ }
119
+ /**
120
+ * Reset the filter and load the first page. The cache is cleared.
121
+ * @returns Promise which resolves when the page is loaded.
122
+ */
123
+ reset() {
124
+ this._cache.clear();
125
+ return this.applyFilter({});
126
+ }
127
+ }
128
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"paged-list.store.js","sourceRoot":"","sources":["../../../../../../projects/myrmidon/paged-data-browsers/src/lib/services/paged-list.store.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,eAAe,EAAc,MAAM,MAAM,CAAC;AAInD,OAAO,EAAE,QAAQ,EAAE,MAAM,aAAa,CAAC;AAyBvC;;GAEG;AACH,MAAM,CAAC,MAAM,gCAAgC,GAA0B;IACrE,QAAQ,EAAE,EAAE;IACZ,SAAS,EAAE,EAAE;CACd,CAAC;AAmBF;;;GAGG;AACH,MAAM,OAAO,cAAc;IAoBzB;;;OAGG;IACH,IAAW,QAAQ;QACjB,OAAO,IAAI,CAAC,SAAS,CAAC;IACxB,CAAC;IACD,IAAW,QAAQ,CAAC,KAAa;QAC/B,IAAI,IAAI,CAAC,SAAS,KAAK,KAAK,EAAE;YAC5B,OAAO;SACR;QACD,IAAI,CAAC,SAAS,GAAG,KAAK,CAAC;QACvB,IAAI,CAAC,KAAK,EAAE,CAAC;IACf,CAAC;IAED;;;OAGG;IACH,YACU,QAAqC,EAC7C,UAAiC,gCAAgC;QADzD,aAAQ,GAAR,QAAQ,CAA6B;QAG7C,IAAI,CAAC,SAAS,GAAG,OAAO,CAAC,QAAQ,CAAC;QAClC,IAAI,CAAC,MAAM,GAAG,IAAI,QAAQ,CAAc,OAAO,CAAC,SAAS,CAAC,CAAC;QAC3D,OAAO;QACP,IAAI,CAAC,MAAM,GAAG,IAAI,eAAe,CAAc;YAC7C,UAAU,EAAE,CAAC;YACb,SAAS,EAAE,CAAC;YACZ,QAAQ,EAAE,CAAC;YACX,KAAK,EAAE,CAAC;YACR,KAAK,EAAE,EAAE;SACV,CAAC,CAAC;QACH,IAAI,CAAC,KAAK,GAAG,IAAI,CAAC,MAAM,CAAC,YAAY,EAAE,CAAC;QACxC,SAAS;QACT,IAAI,CAAC,QAAQ,GAAG,IAAI,eAAe,CAAI,EAAO,CAAC,CAAC;QAChD,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC,QAAQ,CAAC,YAAY,EAAE,CAAC;IAC9C,CAAC;IAED;;;OAGG;IACI,OAAO;QACZ,OAAO,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,KAAK,CAAC,MAAM,KAAK,CAAC,CAAC;IAC9C,CAAC;IAED;;;;;;;;OAQG;IACK,aAAa,CAAC,UAAkB,EAAE,MAAS;QACjD,IAAI,IAAI,CAAC,sBAAsB,EAAE;YAC/B,OAAO,IAAI,CAAC,sBAAsB,CAAC,UAAU,EAAE,MAAM,CAAC,CAAC;SACxD;QACD,OAAO,IAAI,CAAC,SAAS,CAAC,EAAE,UAAU,EAAE,GAAG,MAAM,EAAE,CAAC,CAAC;IACnD,CAAC;IAED;;;OAGG;IACK,QAAQ,CAAC,UAAkB;QACjC,OAAO,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAC3B,UAAU,EACV,IAAI,CAAC,SAAS,EACd,IAAI,CAAC,QAAQ,CAAC,KAAK,CACpB,CAAC;IACJ,CAAC;IAED;;;;;OAKG;IACI,OAAO,CAAC,UAAkB,EAAE,QAAiB;QAClD,IAAI,QAAQ,IAAI,QAAQ,KAAK,IAAI,CAAC,SAAS,EAAE;YAC3C,IAAI,CAAC,SAAS,GAAG,QAAQ,CAAC;SAC3B;QACD,OAAO,IAAI,OAAO,CAAC,CAAC,OAAO,EAAE,MAAM,EAAE,EAAE;YACrC,iCAAiC;YACjC,MAAM,GAAG,GAAG,IAAI,CAAC,aAAa,CAAC,UAAU,EAAE,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;YAChE,MAAM,UAAU,GAAG,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;YACxC,IAAI,UAAU,EAAE;gBACd,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;gBAC7B,OAAO,EAAE,CAAC;gBACV,OAAO;aACR;YAED,iBAAiB;YACjB,IAAI,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC,SAAS,CAAC;gBAClC,IAAI,EAAE,CAAC,IAAI,EAAE,EAAE;oBACb,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;oBACvB,OAAO,EAAE,CAAC;gBACZ,CAAC;gBACD,KAAK,EAAE,MAAM;aACd,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;IACL,CAAC;IAED;;;;OAIG;IACI,WAAW,CAAC,MAAS;QAC1B,OAAO,IAAI,OAAO,CAAC,CAAC,OAAO,EAAE,MAAM,EAAE,EAAE;YACrC,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;YAC3B,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,OAAO,EAAE,MAAM,CAAC,CAAC;QACxC,CAAC,CAAC,CAAC;IACL,CAAC;IAED;;;OAGG;IACI,KAAK;QACV,IAAI,CAAC,MAAM,CAAC,KAAK,EAAE,CAAC;QACpB,OAAO,IAAI,CAAC,WAAW,CAAC,EAAO,CAAC,CAAC;IACnC,CAAC;CACF","sourcesContent":["import { BehaviorSubject, Observable } from 'rxjs';\r\n\r\nimport { DataPage } from '@myrmidon/ng-tools';\r\n\r\nimport { LRUCache } from './lru-cache';\r\n\r\n/**\r\n * Options for the paged list store.\r\n */\r\nexport interface PagedListStoreOptions {\r\n  /**\r\n   * The size of pages in the store.\r\n   */\r\n  pageSize: number;\r\n  /**\r\n   * The size of the cache for pages in the store.\r\n   */\r\n  cacheSize: number;\r\n\r\n  /**\r\n   * A custom function for building the cache key for the given page number\r\n   * and filter.\r\n   * @param pageNumber The page number.\r\n   * @param filter The filter.\r\n   * @returns A string to be used as cache key.\r\n   */\r\n  buildCacheKey?: (pageNumber: number, filter: any) => string;\r\n}\r\n\r\n/**\r\n * Default options for the paged list store.\r\n */\r\nexport const DEFAULT_PAGED_LIST_STORE_OPTIONS: PagedListStoreOptions = {\r\n  pageSize: 20,\r\n  cacheSize: 50,\r\n};\r\n\r\n/**\r\n * The interface to be implemented by the service used by PagedListStore.\r\n */\r\nexport interface PagedListStoreService<F, E> {\r\n  /**\r\n   * Load the page with the given number.\r\n   * @param pageNumber The page number to load.\r\n   * @param pageSize The size of the page to load.\r\n   * @param filter The filter to apply.\r\n   */\r\n  loadPage(\r\n    pageNumber: number,\r\n    pageSize: number,\r\n    filter: F\r\n  ): Observable<DataPage<E>>;\r\n}\r\n\r\n/**\r\n * A generic paged list store using a filter object of type F\r\n * and a list of elements of type E.\r\n */\r\nexport class PagedListStore<F, E> {\r\n  private _pageSize: number;\r\n  private readonly _customCacheKeyBuilder?: (\r\n    pageNumber: number,\r\n    filter: any\r\n  ) => string;\r\n  private _page$: BehaviorSubject<DataPage<E>>;\r\n  private _filter$: BehaviorSubject<F>;\r\n  private readonly _cache: LRUCache<DataPage<E>>;\r\n\r\n  /**\r\n   * The page. It is updated when the page is changed or the filter is changed.\r\n   */\r\n  public page$: Observable<Readonly<DataPage<E>>>;\r\n\r\n  /**\r\n   * The filter. It is updated when the filter is changed.\r\n   */\r\n  public filter$: Observable<Readonly<F>>;\r\n\r\n  /**\r\n   * The size of nodes pages in this store. If you change it, the store\r\n   * is reset. The default value is 20.\r\n   */\r\n  public get pageSize(): number {\r\n    return this._pageSize;\r\n  }\r\n  public set pageSize(value: number) {\r\n    if (this._pageSize === value) {\r\n      return;\r\n    }\r\n    this._pageSize = value;\r\n    this.reset();\r\n  }\r\n\r\n  /**\r\n   * Create a new paged list store.\r\n   * @param options Options for the paged list store.\r\n   */\r\n  constructor(\r\n    private _service: PagedListStoreService<F, E>,\r\n    options: PagedListStoreOptions = DEFAULT_PAGED_LIST_STORE_OPTIONS\r\n  ) {\r\n    this._pageSize = options.pageSize;\r\n    this._cache = new LRUCache<DataPage<E>>(options.cacheSize);\r\n    // page\r\n    this._page$ = new BehaviorSubject<DataPage<E>>({\r\n      pageNumber: 0,\r\n      pageCount: 0,\r\n      pageSize: 0,\r\n      total: 0,\r\n      items: [],\r\n    });\r\n    this.page$ = this._page$.asObservable();\r\n    // filter\r\n    this._filter$ = new BehaviorSubject<F>({} as F);\r\n    this.filter$ = this._filter$.asObservable();\r\n  }\r\n\r\n  /**\r\n   * Returns true if the store is empty, false otherwise.\r\n   * @returns true if the store is empty, false otherwise.\r\n   */\r\n  public isEmpty(): boolean {\r\n    return this._page$.value.items.length === 0;\r\n  }\r\n\r\n  /**\r\n   * Build the cache key for the given page number and filter.\r\n   * The default implementation just returns a stringified object\r\n   * containing the page number and the filter. You may override\r\n   * this method to provide a custom cache key.\r\n   * @param pageNumber The page number.\r\n   * @param filter The filter.\r\n   * @returns A string to be used as cache key.\r\n   */\r\n  private buildCacheKey(pageNumber: number, filter: F): string {\r\n    if (this._customCacheKeyBuilder) {\r\n      return this._customCacheKeyBuilder(pageNumber, filter);\r\n    }\r\n    return JSON.stringify({ pageNumber, ...filter });\r\n  }\r\n\r\n  /**\r\n   * Load the page with the given number.\r\n   * @param pageNumber the page number to load.\r\n   */\r\n  private loadPage(pageNumber: number): Observable<DataPage<E>> {\r\n    return this._service.loadPage(\r\n      pageNumber,\r\n      this._pageSize,\r\n      this._filter$.value\r\n    );\r\n  }\r\n\r\n  /**\r\n   * Set the page with the given number.\r\n   * @param pageNumber The page number to load.\r\n   * @param pageSize The page size.\r\n   * @returns Promise which resolves when the page is loaded.\r\n   */\r\n  public setPage(pageNumber: number, pageSize?: number): Promise<void> {\r\n    if (pageSize && pageSize !== this._pageSize) {\r\n      this._pageSize = pageSize;\r\n    }\r\n    return new Promise((resolve, reject) => {\r\n      // if page is in cache, return it\r\n      const key = this.buildCacheKey(pageNumber, this._filter$.value);\r\n      const cachedPage = this._cache.get(key);\r\n      if (cachedPage) {\r\n        this._page$.next(cachedPage);\r\n        resolve();\r\n        return;\r\n      }\r\n\r\n      // else load page\r\n      this.loadPage(pageNumber).subscribe({\r\n        next: (page) => {\r\n          this._page$.next(page);\r\n          resolve();\r\n        },\r\n        error: reject,\r\n      });\r\n    });\r\n  }\r\n\r\n  /**\r\n   * Apply the given filter and load the first page.\r\n   * @param filter The filter to apply.\r\n   * @returns Promise which resolves when the page is loaded.\r\n   */\r\n  public applyFilter(filter: F): Promise<void> {\r\n    return new Promise((resolve, reject) => {\r\n      this._filter$.next(filter);\r\n      this.setPage(1).then(resolve, reject);\r\n    });\r\n  }\r\n\r\n  /**\r\n   * Reset the filter and load the first page. The cache is cleared.\r\n   * @returns Promise which resolves when the page is loaded.\r\n   */\r\n  public reset(): Promise<void> {\r\n    this._cache.clear();\r\n    return this.applyFilter({} as F);\r\n  }\r\n}\r\n"]}