@rivet-health/design-system 23.0.0 → 23.1.0
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/esm2020/lib/content-toggle/toggle/toggle.component.mjs +32 -0
- package/esm2020/lib/modal/menu/menu.component.mjs +8 -5
- package/esm2020/lib/navigation/views/all-views-modal/all-views-modal.component.mjs +43 -0
- package/esm2020/lib/navigation/views/state.mjs +119 -20
- package/esm2020/lib/navigation/views/view-menu/view-menu.component.mjs +30 -0
- package/esm2020/lib/navigation/views/views/views.component.mjs +17 -17
- package/esm2020/lib/riv.module.mjs +18 -3
- package/esm2020/lib/table/table-column-settings/table-column-settings-side-sheet.component.mjs +1 -1
- package/esm2020/public-api.mjs +4 -1
- package/fesm2015/rivet-health-design-system.mjs +191 -31
- package/fesm2015/rivet-health-design-system.mjs.map +1 -1
- package/fesm2020/rivet-health-design-system.mjs +228 -30
- package/fesm2020/rivet-health-design-system.mjs.map +1 -1
- package/lib/content-toggle/toggle/toggle.component.d.ts +13 -0
- package/lib/modal/menu/menu.component.d.ts +3 -1
- package/lib/navigation/views/all-views-modal/all-views-modal.component.d.ts +14 -0
- package/lib/navigation/views/state.d.ts +34 -5
- package/lib/navigation/views/view-menu/view-menu.component.d.ts +13 -0
- package/lib/navigation/views/views/views.component.d.ts +3 -3
- package/lib/riv.module.d.ts +86 -83
- package/package.json +1 -1
- package/public-api.d.ts +3 -0
|
@@ -0,0 +1,32 @@
|
|
|
1
|
+
import { ChangeDetectionStrategy, Component, ContentChild, 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 "../../icon/icon.component";
|
|
5
|
+
export class ToggleComponent {
|
|
6
|
+
constructor() {
|
|
7
|
+
this.open = false;
|
|
8
|
+
this.disabled = false;
|
|
9
|
+
this.openChange = new EventEmitter();
|
|
10
|
+
}
|
|
11
|
+
toggleOpen() {
|
|
12
|
+
const newOpenState = !this.open;
|
|
13
|
+
this.open = newOpenState;
|
|
14
|
+
this.openChange.emit(newOpenState);
|
|
15
|
+
}
|
|
16
|
+
}
|
|
17
|
+
ToggleComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: ToggleComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
18
|
+
ToggleComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: ToggleComponent, selector: "riv-toggle", inputs: { open: "open", disabled: "disabled" }, outputs: { openChange: "openChange" }, queries: [{ propertyName: "triggerTemplate", first: true, predicate: ["trigger"], descendants: true }], ngImport: i0, template: "<button\n type=\"button\"\n class=\"toggle-button\"\n [attr.disabled]=\"disabled ? true : null\"\n (click)=\"toggleOpen()\"\n>\n <div *ngIf=\"triggerTemplate; let tpl\" class=\"content\">\n <ng-container\n *ngTemplateOutlet=\"tpl; context: { open: open }\"\n ></ng-container>\n </div>\n <riv-icon [class.open]=\"open\" [name]=\"'ChevronDown'\" [size]=\"24\"></riv-icon>\n</button>\n\n<ng-container *ngIf=\"open\">\n <ng-content></ng-content>\n</ng-container>\n", styles: [".toggle-button{display:flex;align-items:center;justify-content:space-between;width:100%;border:none;background:none;padding:var(--size-xsmall);font:var(--input-medium);text-align:left;border-radius:var(--border-radius-medium);color:inherit;gap:var(--size-small);cursor:pointer;transition:background-color var(--short-transition)}.toggle-button:hover:not(:disabled){background-color:var(--surface-light-1)}.toggle-button:active:not(:disabled){background-color:var(--surface-light-2)}.toggle-button:disabled{cursor:default;color:var(--type-light-disabled)}.toggle-button .content{flex-grow:1}riv-icon{transition:transform var(--medium-transition)}riv-icon.open{transform:rotate(-180deg)}\n"], dependencies: [{ kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "component", type: i2.IconComponent, selector: "riv-icon", inputs: ["name", "size", "customSize", "strokeWidth"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
19
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: ToggleComponent, decorators: [{
|
|
20
|
+
type: Component,
|
|
21
|
+
args: [{ selector: 'riv-toggle', changeDetection: ChangeDetectionStrategy.OnPush, template: "<button\n type=\"button\"\n class=\"toggle-button\"\n [attr.disabled]=\"disabled ? true : null\"\n (click)=\"toggleOpen()\"\n>\n <div *ngIf=\"triggerTemplate; let tpl\" class=\"content\">\n <ng-container\n *ngTemplateOutlet=\"tpl; context: { open: open }\"\n ></ng-container>\n </div>\n <riv-icon [class.open]=\"open\" [name]=\"'ChevronDown'\" [size]=\"24\"></riv-icon>\n</button>\n\n<ng-container *ngIf=\"open\">\n <ng-content></ng-content>\n</ng-container>\n", styles: [".toggle-button{display:flex;align-items:center;justify-content:space-between;width:100%;border:none;background:none;padding:var(--size-xsmall);font:var(--input-medium);text-align:left;border-radius:var(--border-radius-medium);color:inherit;gap:var(--size-small);cursor:pointer;transition:background-color var(--short-transition)}.toggle-button:hover:not(:disabled){background-color:var(--surface-light-1)}.toggle-button:active:not(:disabled){background-color:var(--surface-light-2)}.toggle-button:disabled{cursor:default;color:var(--type-light-disabled)}.toggle-button .content{flex-grow:1}riv-icon{transition:transform var(--medium-transition)}riv-icon.open{transform:rotate(-180deg)}\n"] }]
|
|
22
|
+
}], propDecorators: { open: [{
|
|
23
|
+
type: Input
|
|
24
|
+
}], disabled: [{
|
|
25
|
+
type: Input
|
|
26
|
+
}], openChange: [{
|
|
27
|
+
type: Output
|
|
28
|
+
}], triggerTemplate: [{
|
|
29
|
+
type: ContentChild,
|
|
30
|
+
args: ['trigger']
|
|
31
|
+
}] } });
|
|
32
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidG9nZ2xlLmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL3Jpdi9zcmMvbGliL2NvbnRlbnQtdG9nZ2xlL3RvZ2dsZS90b2dnbGUuY29tcG9uZW50LnRzIiwiLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvcml2L3NyYy9saWIvY29udGVudC10b2dnbGUvdG9nZ2xlL3RvZ2dsZS5jb21wb25lbnQuaHRtbCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQ0wsdUJBQXVCLEVBQ3ZCLFNBQVMsRUFDVCxZQUFZLEVBQ1osWUFBWSxFQUNaLEtBQUssRUFDTCxNQUFNLEdBRVAsTUFBTSxlQUFlLENBQUM7Ozs7QUFRdkIsTUFBTSxPQUFPLGVBQWU7SUFONUI7UUFRRSxTQUFJLEdBQVksS0FBSyxDQUFDO1FBR3RCLGFBQVEsR0FBWSxLQUFLLENBQUM7UUFHMUIsZUFBVSxHQUEwQixJQUFJLFlBQVksRUFBVyxDQUFDO0tBVWpFO0lBTEMsVUFBVTtRQUNSLE1BQU0sWUFBWSxHQUFHLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQztRQUNoQyxJQUFJLENBQUMsSUFBSSxHQUFHLFlBQVksQ0FBQztRQUN6QixJQUFJLENBQUMsVUFBVSxDQUFDLElBQUksQ0FBQyxZQUFZLENBQUMsQ0FBQztJQUNyQyxDQUFDOzs0R0FqQlUsZUFBZTtnR0FBZixlQUFlLGlQQ2hCNUIsaWVBaUJBOzJGRERhLGVBQWU7a0JBTjNCLFNBQVM7K0JBQ0UsWUFBWSxtQkFHTCx1QkFBdUIsQ0FBQyxNQUFNOzhCQUkvQyxJQUFJO3NCQURILEtBQUs7Z0JBSU4sUUFBUTtzQkFEUCxLQUFLO2dCQUlOLFVBQVU7c0JBRFQsTUFBTTtnQkFJUCxlQUFlO3NCQURkLFlBQVk7dUJBQUMsU0FBUyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7XG4gIENoYW5nZURldGVjdGlvblN0cmF0ZWd5LFxuICBDb21wb25lbnQsXG4gIENvbnRlbnRDaGlsZCxcbiAgRXZlbnRFbWl0dGVyLFxuICBJbnB1dCxcbiAgT3V0cHV0LFxuICBUZW1wbGF0ZVJlZixcbn0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5cbkBDb21wb25lbnQoe1xuICBzZWxlY3RvcjogJ3Jpdi10b2dnbGUnLFxuICB0ZW1wbGF0ZVVybDogJy4vdG9nZ2xlLmNvbXBvbmVudC5odG1sJyxcbiAgc3R5bGVVcmxzOiBbJy4vdG9nZ2xlLmNvbXBvbmVudC5jc3MnXSxcbiAgY2hhbmdlRGV0ZWN0aW9uOiBDaGFuZ2VEZXRlY3Rpb25TdHJhdGVneS5PblB1c2gsXG59KVxuZXhwb3J0IGNsYXNzIFRvZ2dsZUNvbXBvbmVudCB7XG4gIEBJbnB1dCgpXG4gIG9wZW46IGJvb2xlYW4gPSBmYWxzZTtcblxuICBASW5wdXQoKVxuICBkaXNhYmxlZDogYm9vbGVhbiA9IGZhbHNlO1xuXG4gIEBPdXRwdXQoKVxuICBvcGVuQ2hhbmdlOiBFdmVudEVtaXR0ZXI8Ym9vbGVhbj4gPSBuZXcgRXZlbnRFbWl0dGVyPGJvb2xlYW4+KCk7XG5cbiAgQENvbnRlbnRDaGlsZCgndHJpZ2dlcicpXG4gIHRyaWdnZXJUZW1wbGF0ZT86IFRlbXBsYXRlUmVmPHsgb3BlbjogYm9vbGVhbiB9PjtcblxuICB0b2dnbGVPcGVuKCk6IHZvaWQge1xuICAgIGNvbnN0IG5ld09wZW5TdGF0ZSA9ICF0aGlzLm9wZW47XG4gICAgdGhpcy5vcGVuID0gbmV3T3BlblN0YXRlO1xuICAgIHRoaXMub3BlbkNoYW5nZS5lbWl0KG5ld09wZW5TdGF0ZSk7XG4gIH1cbn1cbiIsIjxidXR0b25cbiAgdHlwZT1cImJ1dHRvblwiXG4gIGNsYXNzPVwidG9nZ2xlLWJ1dHRvblwiXG4gIFthdHRyLmRpc2FibGVkXT1cImRpc2FibGVkID8gdHJ1ZSA6IG51bGxcIlxuICAoY2xpY2spPVwidG9nZ2xlT3BlbigpXCJcbj5cbiAgPGRpdiAqbmdJZj1cInRyaWdnZXJUZW1wbGF0ZTsgbGV0IHRwbFwiIGNsYXNzPVwiY29udGVudFwiPlxuICAgIDxuZy1jb250YWluZXJcbiAgICAgICpuZ1RlbXBsYXRlT3V0bGV0PVwidHBsOyBjb250ZXh0OiB7IG9wZW46IG9wZW4gfVwiXG4gICAgPjwvbmctY29udGFpbmVyPlxuICA8L2Rpdj5cbiAgPHJpdi1pY29uIFtjbGFzcy5vcGVuXT1cIm9wZW5cIiBbbmFtZV09XCInQ2hldnJvbkRvd24nXCIgW3NpemVdPVwiMjRcIj48L3Jpdi1pY29uPlxuPC9idXR0b24+XG5cbjxuZy1jb250YWluZXIgKm5nSWY9XCJvcGVuXCI+XG4gIDxuZy1jb250ZW50PjwvbmctY29udGVudD5cbjwvbmctY29udGFpbmVyPlxuIl19
|
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import { ChangeDetectionStrategy, Component, ContentChild, ElementRef, ViewChild, } from '@angular/core';
|
|
1
|
+
import { ChangeDetectionStrategy, Component, ContentChild, ElementRef, Input, ViewChild, } from '@angular/core';
|
|
2
2
|
import { BehaviorSubject } from 'rxjs';
|
|
3
3
|
import * as i0 from "@angular/core";
|
|
4
4
|
import * as i1 from "@angular/common";
|
|
@@ -8,6 +8,7 @@ import * as i4 from "../overlay.directive";
|
|
|
8
8
|
export class MenuComponent {
|
|
9
9
|
constructor() {
|
|
10
10
|
this.open = new BehaviorSubject(false);
|
|
11
|
+
this.preferredPosition = 'bottom-left';
|
|
11
12
|
}
|
|
12
13
|
getTrigger() {
|
|
13
14
|
return (this.customTriggerButton?.nativeElement ??
|
|
@@ -15,11 +16,13 @@ export class MenuComponent {
|
|
|
15
16
|
}
|
|
16
17
|
}
|
|
17
18
|
MenuComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: MenuComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
18
|
-
MenuComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: MenuComponent, selector: "riv-menu", queries: [{ propertyName: "triggerTemplate", first: true, predicate: ["trigger"], descendants: true }], viewQueries: [{ propertyName: "customTriggerButton", first: true, predicate: ["customTriggerButton"], descendants: true, read: ElementRef }, { propertyName: "standardTriggerButton", first: true, predicate: ["standardTriggerButton"], descendants: true, read: ElementRef }], ngImport: i0, template: "<ng-container *ngIf=\"triggerTemplate; else standardTrigger\">\n <button #customTriggerButton class=\"custom-trigger\" (click)=\"open.next(true)\">\n <ng-container *ngTemplateOutlet=\"triggerTemplate\"></ng-container>\n </button>\n</ng-container>\n\n<ng-template #standardTrigger>\n <button\n #standardTriggerButton\n rivButton\n [variant]=\"'ghost'\"\n [size]=\"'xsmall'\"\n [icon]=\"'MoreVertical'\"\n (click)=\"open.next(true)\"\n ></button>\n</ng-template>\n\n<ng-container *ngIf=\"open | async\">\n <riv-callout\n *riv-overlay\n [anchor]=\"getTrigger()\"\n [showCaret]=\"false\"\n [preferredPosition]=\"
|
|
19
|
+
MenuComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: MenuComponent, selector: "riv-menu", inputs: { preferredPosition: "preferredPosition" }, queries: [{ propertyName: "triggerTemplate", first: true, predicate: ["trigger"], descendants: true }], viewQueries: [{ propertyName: "customTriggerButton", first: true, predicate: ["customTriggerButton"], descendants: true, read: ElementRef }, { propertyName: "standardTriggerButton", first: true, predicate: ["standardTriggerButton"], descendants: true, read: ElementRef }], ngImport: i0, template: "<ng-container *ngIf=\"triggerTemplate; else standardTrigger\">\n <button #customTriggerButton class=\"custom-trigger\" (click)=\"open.next(true)\">\n <ng-container *ngTemplateOutlet=\"triggerTemplate\"></ng-container>\n </button>\n</ng-container>\n\n<ng-template #standardTrigger>\n <button\n #standardTriggerButton\n rivButton\n [variant]=\"'ghost'\"\n [size]=\"'xsmall'\"\n [icon]=\"'MoreVertical'\"\n (click)=\"open.next(true)\"\n ></button>\n</ng-template>\n\n<ng-container *ngIf=\"open | async\">\n <riv-callout\n *riv-overlay\n [anchor]=\"getTrigger()\"\n [showCaret]=\"false\"\n [preferredPosition]=\"preferredPosition\"\n [theme]=\"'light'\"\n (close)=\"open.next(false)\"\n >\n <div class=\"options\">\n <ng-content></ng-content>\n </div>\n </riv-callout>\n</ng-container>\n", styles: [":host{display:inline-flex;justify-content:center;align-items:center}.custom-trigger{max-width:100%}.options{display:flex;flex-direction:column;align-items:stretch;gap:var(--size-small);padding:var(--size-large);min-width:calc(var(--base-grid-size) * 58)}\n"], dependencies: [{ kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "component", type: i2.ButtonComponent, selector: "[rivButton]", inputs: ["locked", "disabled", "loading", "full", "size", "variant", "icon", "iconPosition", "active"] }, { kind: "component", type: i3.CalloutComponent, selector: "riv-callout", inputs: ["anchor", "isModal", "preferredPosition", "allowedPositions", "fallbackDirection", "showCaret", "theme"], outputs: ["close"] }, { kind: "directive", type: i4.OverlayDirective, selector: "[riv-overlay]" }, { kind: "pipe", type: i1.AsyncPipe, name: "async" }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
19
20
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: MenuComponent, decorators: [{
|
|
20
21
|
type: Component,
|
|
21
|
-
args: [{ selector: 'riv-menu', changeDetection: ChangeDetectionStrategy.OnPush, template: "<ng-container *ngIf=\"triggerTemplate; else standardTrigger\">\n <button #customTriggerButton class=\"custom-trigger\" (click)=\"open.next(true)\">\n <ng-container *ngTemplateOutlet=\"triggerTemplate\"></ng-container>\n </button>\n</ng-container>\n\n<ng-template #standardTrigger>\n <button\n #standardTriggerButton\n rivButton\n [variant]=\"'ghost'\"\n [size]=\"'xsmall'\"\n [icon]=\"'MoreVertical'\"\n (click)=\"open.next(true)\"\n ></button>\n</ng-template>\n\n<ng-container *ngIf=\"open | async\">\n <riv-callout\n *riv-overlay\n [anchor]=\"getTrigger()\"\n [showCaret]=\"false\"\n [preferredPosition]=\"
|
|
22
|
-
}], propDecorators: {
|
|
22
|
+
args: [{ selector: 'riv-menu', changeDetection: ChangeDetectionStrategy.OnPush, template: "<ng-container *ngIf=\"triggerTemplate; else standardTrigger\">\n <button #customTriggerButton class=\"custom-trigger\" (click)=\"open.next(true)\">\n <ng-container *ngTemplateOutlet=\"triggerTemplate\"></ng-container>\n </button>\n</ng-container>\n\n<ng-template #standardTrigger>\n <button\n #standardTriggerButton\n rivButton\n [variant]=\"'ghost'\"\n [size]=\"'xsmall'\"\n [icon]=\"'MoreVertical'\"\n (click)=\"open.next(true)\"\n ></button>\n</ng-template>\n\n<ng-container *ngIf=\"open | async\">\n <riv-callout\n *riv-overlay\n [anchor]=\"getTrigger()\"\n [showCaret]=\"false\"\n [preferredPosition]=\"preferredPosition\"\n [theme]=\"'light'\"\n (close)=\"open.next(false)\"\n >\n <div class=\"options\">\n <ng-content></ng-content>\n </div>\n </riv-callout>\n</ng-container>\n", styles: [":host{display:inline-flex;justify-content:center;align-items:center}.custom-trigger{max-width:100%}.options{display:flex;flex-direction:column;align-items:stretch;gap:var(--size-small);padding:var(--size-large);min-width:calc(var(--base-grid-size) * 58)}\n"] }]
|
|
23
|
+
}], propDecorators: { preferredPosition: [{
|
|
24
|
+
type: Input
|
|
25
|
+
}], triggerTemplate: [{
|
|
23
26
|
type: ContentChild,
|
|
24
27
|
args: ['trigger']
|
|
25
28
|
}], customTriggerButton: [{
|
|
@@ -29,4 +32,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImpor
|
|
|
29
32
|
type: ViewChild,
|
|
30
33
|
args: ['standardTriggerButton', { read: ElementRef }]
|
|
31
34
|
}] } });
|
|
32
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
35
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibWVudS5jb21wb25lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9yaXYvc3JjL2xpYi9tb2RhbC9tZW51L21lbnUuY29tcG9uZW50LnRzIiwiLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvcml2L3NyYy9saWIvbW9kYWwvbWVudS9tZW51LmNvbXBvbmVudC5odG1sIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFDTCx1QkFBdUIsRUFDdkIsU0FBUyxFQUNULFlBQVksRUFDWixVQUFVLEVBQ1YsS0FBSyxFQUVMLFNBQVMsR0FDVixNQUFNLGVBQWUsQ0FBQztBQUN2QixPQUFPLEVBQUUsZUFBZSxFQUFFLE1BQU0sTUFBTSxDQUFDOzs7Ozs7QUFTdkMsTUFBTSxPQUFPLGFBQWE7SUFOMUI7UUFPa0IsU0FBSSxHQUFHLElBQUksZUFBZSxDQUFVLEtBQUssQ0FBQyxDQUFDO1FBRzNELHNCQUFpQixHQUFzQyxhQUFhLENBQUM7S0FpQnRFO0lBTkMsVUFBVTtRQUNSLE9BQU8sQ0FDTCxJQUFJLENBQUMsbUJBQW1CLEVBQUUsYUFBYTtZQUN2QyxJQUFJLENBQUMscUJBQXFCLEVBQUUsYUFBYSxDQUMxQyxDQUFDO0lBQ0osQ0FBQzs7MEdBcEJVLGFBQWE7OEZBQWIsYUFBYSxtVEFTa0IsVUFBVSx5SEFHUixVQUFVLDZCQzlCeEQsMDBCQStCQTsyRkRiYSxhQUFhO2tCQU56QixTQUFTOytCQUNFLFVBQVUsbUJBR0gsdUJBQXVCLENBQUMsTUFBTTs4QkFNL0MsaUJBQWlCO3NCQURoQixLQUFLO2dCQUlOLGVBQWU7c0JBRGQsWUFBWTt1QkFBQyxTQUFTO2dCQUl2QixtQkFBbUI7c0JBRGxCLFNBQVM7dUJBQUMscUJBQXFCLEVBQUUsRUFBRSxJQUFJLEVBQUUsVUFBVSxFQUFFO2dCQUl0RCxxQkFBcUI7c0JBRHBCLFNBQVM7dUJBQUMsdUJBQXVCLEVBQUUsRUFBRSxJQUFJLEVBQUUsVUFBVSxFQUFFIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHtcbiAgQ2hhbmdlRGV0ZWN0aW9uU3RyYXRlZ3ksXG4gIENvbXBvbmVudCxcbiAgQ29udGVudENoaWxkLFxuICBFbGVtZW50UmVmLFxuICBJbnB1dCxcbiAgVGVtcGxhdGVSZWYsXG4gIFZpZXdDaGlsZCxcbn0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBCZWhhdmlvclN1YmplY3QgfSBmcm9tICdyeGpzJztcbmltcG9ydCB7IENhbGxvdXRDb21wb25lbnQgfSBmcm9tICcuLi9jYWxsb3V0L2NhbGxvdXQuY29tcG9uZW50JztcblxuQENvbXBvbmVudCh7XG4gIHNlbGVjdG9yOiAncml2LW1lbnUnLFxuICB0ZW1wbGF0ZVVybDogJy4vbWVudS5jb21wb25lbnQuaHRtbCcsXG4gIHN0eWxlVXJsczogWycuL21lbnUuY29tcG9uZW50LmNzcyddLFxuICBjaGFuZ2VEZXRlY3Rpb246IENoYW5nZURldGVjdGlvblN0cmF0ZWd5Lk9uUHVzaCxcbn0pXG5leHBvcnQgY2xhc3MgTWVudUNvbXBvbmVudCB7XG4gIHB1YmxpYyByZWFkb25seSBvcGVuID0gbmV3IEJlaGF2aW9yU3ViamVjdDxib29sZWFuPihmYWxzZSk7XG5cbiAgQElucHV0KClcbiAgcHJlZmVycmVkUG9zaXRpb246IENhbGxvdXRDb21wb25lbnQuQW5jaG9yZWRQb3NpdGlvbiA9ICdib3R0b20tbGVmdCc7XG5cbiAgQENvbnRlbnRDaGlsZCgndHJpZ2dlcicpXG4gIHRyaWdnZXJUZW1wbGF0ZT86IFRlbXBsYXRlUmVmPHZvaWQ+O1xuXG4gIEBWaWV3Q2hpbGQoJ2N1c3RvbVRyaWdnZXJCdXR0b24nLCB7IHJlYWQ6IEVsZW1lbnRSZWYgfSlcbiAgY3VzdG9tVHJpZ2dlckJ1dHRvbj86IEVsZW1lbnRSZWY7XG5cbiAgQFZpZXdDaGlsZCgnc3RhbmRhcmRUcmlnZ2VyQnV0dG9uJywgeyByZWFkOiBFbGVtZW50UmVmIH0pXG4gIHN0YW5kYXJkVHJpZ2dlckJ1dHRvbj86IEVsZW1lbnRSZWY7XG5cbiAgZ2V0VHJpZ2dlcigpOiBFbGVtZW50IHwgbnVsbCB7XG4gICAgcmV0dXJuIChcbiAgICAgIHRoaXMuY3VzdG9tVHJpZ2dlckJ1dHRvbj8ubmF0aXZlRWxlbWVudCA/P1xuICAgICAgdGhpcy5zdGFuZGFyZFRyaWdnZXJCdXR0b24/Lm5hdGl2ZUVsZW1lbnRcbiAgICApO1xuICB9XG59XG4iLCI8bmctY29udGFpbmVyICpuZ0lmPVwidHJpZ2dlclRlbXBsYXRlOyBlbHNlIHN0YW5kYXJkVHJpZ2dlclwiPlxuICA8YnV0dG9uICNjdXN0b21UcmlnZ2VyQnV0dG9uIGNsYXNzPVwiY3VzdG9tLXRyaWdnZXJcIiAoY2xpY2spPVwib3Blbi5uZXh0KHRydWUpXCI+XG4gICAgPG5nLWNvbnRhaW5lciAqbmdUZW1wbGF0ZU91dGxldD1cInRyaWdnZXJUZW1wbGF0ZVwiPjwvbmctY29udGFpbmVyPlxuICA8L2J1dHRvbj5cbjwvbmctY29udGFpbmVyPlxuXG48bmctdGVtcGxhdGUgI3N0YW5kYXJkVHJpZ2dlcj5cbiAgPGJ1dHRvblxuICAgICNzdGFuZGFyZFRyaWdnZXJCdXR0b25cbiAgICByaXZCdXR0b25cbiAgICBbdmFyaWFudF09XCInZ2hvc3QnXCJcbiAgICBbc2l6ZV09XCIneHNtYWxsJ1wiXG4gICAgW2ljb25dPVwiJ01vcmVWZXJ0aWNhbCdcIlxuICAgIChjbGljayk9XCJvcGVuLm5leHQodHJ1ZSlcIlxuICA+PC9idXR0b24+XG48L25nLXRlbXBsYXRlPlxuXG48bmctY29udGFpbmVyICpuZ0lmPVwib3BlbiB8IGFzeW5jXCI+XG4gIDxyaXYtY2FsbG91dFxuICAgICpyaXYtb3ZlcmxheVxuICAgIFthbmNob3JdPVwiZ2V0VHJpZ2dlcigpXCJcbiAgICBbc2hvd0NhcmV0XT1cImZhbHNlXCJcbiAgICBbcHJlZmVycmVkUG9zaXRpb25dPVwicHJlZmVycmVkUG9zaXRpb25cIlxuICAgIFt0aGVtZV09XCInbGlnaHQnXCJcbiAgICAoY2xvc2UpPVwib3Blbi5uZXh0KGZhbHNlKVwiXG4gID5cbiAgICA8ZGl2IGNsYXNzPVwib3B0aW9uc1wiPlxuICAgICAgPG5nLWNvbnRlbnQ+PC9uZy1jb250ZW50PlxuICAgIDwvZGl2PlxuICA8L3Jpdi1jYWxsb3V0PlxuPC9uZy1jb250YWluZXI+XG4iXX0=
|
|
@@ -0,0 +1,43 @@
|
|
|
1
|
+
import { ChangeDetectionStrategy, Component, EventEmitter, Input, Output, } from '@angular/core';
|
|
2
|
+
import { RivViews } from '../state';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
import * as i1 from "@angular/common";
|
|
5
|
+
import * as i2 from "../../../input/button/button.component";
|
|
6
|
+
import * as i3 from "../../../visualization/highlight/highlight.component";
|
|
7
|
+
import * as i4 from "../../../icon/icon.component";
|
|
8
|
+
import * as i5 from "../../../input/search/search.component";
|
|
9
|
+
import * as i6 from "../../../modal/side-sheet/side-sheet.component";
|
|
10
|
+
import * as i7 from "../../../content-toggle/toggle/toggle.component";
|
|
11
|
+
import * as i8 from "../view-menu/view-menu.component";
|
|
12
|
+
import * as i9 from "../../../visualization/zero-state/zero-state.component";
|
|
13
|
+
export class AllViewsModalComponent {
|
|
14
|
+
constructor() {
|
|
15
|
+
this.topOffset = 0;
|
|
16
|
+
this.close = new EventEmitter();
|
|
17
|
+
this.getPermissionTitle = RivViews.getPermissionTitle;
|
|
18
|
+
this.getPermissionDescription = RivViews.getPermissionDescription;
|
|
19
|
+
}
|
|
20
|
+
getOpenChange(permission) {
|
|
21
|
+
return (open) => this.manager?.actions?.next({
|
|
22
|
+
type: 'setAllViewsToggle',
|
|
23
|
+
permission,
|
|
24
|
+
open,
|
|
25
|
+
});
|
|
26
|
+
}
|
|
27
|
+
trackById(_, view) {
|
|
28
|
+
return view.id;
|
|
29
|
+
}
|
|
30
|
+
}
|
|
31
|
+
AllViewsModalComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: AllViewsModalComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
32
|
+
AllViewsModalComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: AllViewsModalComponent, selector: "riv-all-views-modal", inputs: { manager: "manager", topOffset: "topOffset" }, outputs: { close: "close" }, ngImport: i0, template: "<ng-container *ngIf=\"manager?.state | async; let s\">\n <riv-side-sheet [topOffset]=\"topOffset\" (close)=\"close.emit()\">\n <div class=\"content\">\n <header>\n <span class=\"title\">All views</span>\n <button\n rivButton\n [variant]=\"'ghost'\"\n [icon]=\"'X'\"\n (click)=\"close.emit()\"\n ></button>\n </header>\n\n <riv-search\n (valueChange)=\"\n manager?.actions?.next({\n type: 'setAllViewsSearchQuery',\n query: $event\n })\n \"\n [value]=\"s.all.search\"\n placeholder=\"Search\"\n ></riv-search>\n\n <div class=\"views\">\n <!-- View sections by permission type -->\n <ng-container\n [ngTemplateOutlet]=\"permissionSection\"\n [ngTemplateOutletContext]=\"{\n permission: 'private',\n open: s.all.privateOpen,\n openChange: getOpenChange('private'),\n views: s.all.privateViews,\n title: getPermissionTitle('private'),\n description: getPermissionDescription('private'),\n emptyStateMessage: s.all.privateEmptyStateMessage\n }\"\n ></ng-container>\n <ng-container\n [ngTemplateOutlet]=\"permissionSection\"\n [ngTemplateOutletContext]=\"{\n permission: 'shared',\n open: s.all.sharedOpen,\n openChange: getOpenChange('shared'),\n views: s.all.sharedViews,\n title: getPermissionTitle('shared'),\n description: getPermissionDescription('shared'),\n emptyStateMessage: s.all.sharedEmptyStateMessage\n }\"\n ></ng-container>\n <ng-container\n [ngTemplateOutlet]=\"permissionSection\"\n [ngTemplateOutletContext]=\"{\n permission: 'public',\n open: s.all.publicOpen,\n openChange: getOpenChange('public'),\n views: s.all.publicViews,\n title: getPermissionTitle('public'),\n description: getPermissionDescription('public'),\n emptyStateMessage: s.all.publicEmptyStateMessage\n }\"\n ></ng-container>\n\n <!-- Permission section -->\n <ng-template\n #permissionSection\n let-permission=\"permission\"\n let-open=\"open\"\n let-openChange=\"openChange\"\n let-views=\"views\"\n let-title=\"title\"\n let-description=\"description\"\n let-emptyStateMessage=\"emptyStateMessage\"\n >\n <riv-toggle\n [open]=\"open\"\n (openChange)=\"openChange($event)\"\n [disabled]=\"!!s.all.search\"\n >\n <ng-template #trigger>\n <div>\n <h3 class=\"permission-title\">{{ title }}</h3>\n <span class=\"permission-description\">{{ description }}</span>\n </div>\n </ng-template>\n\n <div class=\"views-list\" *ngIf=\"views.length > 0; else noViews\">\n <button\n *ngFor=\"let view of views; trackBy: trackById\"\n class=\"view-item\"\n (click)=\"\n manager?.actions?.next({\n type: 'setActiveView',\n id: view.id\n });\n close.emit()\n \"\n >\n <riv-highlight\n [text]=\"view.title\"\n [indices]=\"view.titleHighlightIndices ?? []\"\n >\n </riv-highlight>\n <riv-view-menu\n [manager]=\"manager\"\n [view]=\"view\"\n ></riv-view-menu>\n <button\n class=\"enable-view\"\n [class.enabled]=\"view.enabled\"\n (click)=\"\n $event.stopPropagation();\n manager?.actions?.next({\n type: 'setEnabled',\n id: view.id,\n enabled: !view.enabled\n })\n \"\n >\n <riv-icon\n [name]=\"view.enabled ? 'Eye' : 'EyeOff'\"\n [size]=\"20\"\n ></riv-icon>\n </button>\n </button>\n </div>\n <ng-template #noViews>\n <riv-zero-state [message]=\"emptyStateMessage\"></riv-zero-state>\n </ng-template>\n </riv-toggle>\n </ng-template>\n </div>\n </div>\n </riv-side-sheet>\n</ng-container>\n", styles: [".content{padding:var(--size-large);display:flex;flex-direction:column;gap:var(--size-medium);height:100%}header{display:flex;justify-content:space-between;align-items:center}.title{font:var(--title-02)}.views{flex-grow:1;overflow-y:auto;display:flex;flex-direction:column;gap:var(--size-xlarge)}.permission-title{font:var(--title-02)}.permission-description{font:var(--body-medium);color:var(--type-light-low-contrast)}.views-list{display:flex;flex-direction:column;gap:var(--size-small)}.view-item{display:flex;align-items:center;padding-left:var(--size-small);border-radius:var(--border-radius-small);cursor:pointer;background-color:var(--surface-light-0);transition:background-color var(--short-transition)}.view-item:hover{background-color:var(--surface-light-1)}.view-item:active{background-color:var(--surface-light-2)}.view-item riv-highlight{text-align:left;flex-grow:1}.view-item riv-view-menu{align-self:flex-end;opacity:0;transition:opacity var(--short-transition)}.view-item:hover riv-view-menu{opacity:1}.enable-view{cursor:pointer;display:inline-flex;justify-content:center;align-items:center;padding:0 var(--size-xsmall);transition:color var(--medium-transition);color:var(--type-light-low-contrast)}.enable-view.enabled{color:var(--type-light-link)}.enable-view.enabled:hover{color:var(--type-light-link-hover)}.enable-view.enabled:active{color:var(--type-light-link-active)}\n"], dependencies: [{ kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "component", type: i2.ButtonComponent, selector: "[rivButton]", inputs: ["locked", "disabled", "loading", "full", "size", "variant", "icon", "iconPosition", "active"] }, { kind: "component", type: i3.HighlightComponent, selector: "riv-highlight", inputs: ["text", "indices"] }, { kind: "component", type: i4.IconComponent, selector: "riv-icon", inputs: ["name", "size", "customSize", "strokeWidth"] }, { kind: "component", type: i5.SearchComponent, selector: "riv-search", inputs: ["placeholder", "name"] }, { kind: "component", type: i6.SideSheetComponent, selector: "riv-side-sheet", inputs: ["topOffset"], outputs: ["close"] }, { kind: "component", type: i7.ToggleComponent, selector: "riv-toggle", inputs: ["open", "disabled"], outputs: ["openChange"] }, { kind: "component", type: i8.ViewMenuComponent, selector: "riv-view-menu", inputs: ["manager", "view"] }, { kind: "component", type: i9.ZeroStateComponent, selector: "riv-zero-state", inputs: ["message"] }, { kind: "pipe", type: i1.AsyncPipe, name: "async" }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
33
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: AllViewsModalComponent, decorators: [{
|
|
34
|
+
type: Component,
|
|
35
|
+
args: [{ selector: 'riv-all-views-modal', changeDetection: ChangeDetectionStrategy.OnPush, template: "<ng-container *ngIf=\"manager?.state | async; let s\">\n <riv-side-sheet [topOffset]=\"topOffset\" (close)=\"close.emit()\">\n <div class=\"content\">\n <header>\n <span class=\"title\">All views</span>\n <button\n rivButton\n [variant]=\"'ghost'\"\n [icon]=\"'X'\"\n (click)=\"close.emit()\"\n ></button>\n </header>\n\n <riv-search\n (valueChange)=\"\n manager?.actions?.next({\n type: 'setAllViewsSearchQuery',\n query: $event\n })\n \"\n [value]=\"s.all.search\"\n placeholder=\"Search\"\n ></riv-search>\n\n <div class=\"views\">\n <!-- View sections by permission type -->\n <ng-container\n [ngTemplateOutlet]=\"permissionSection\"\n [ngTemplateOutletContext]=\"{\n permission: 'private',\n open: s.all.privateOpen,\n openChange: getOpenChange('private'),\n views: s.all.privateViews,\n title: getPermissionTitle('private'),\n description: getPermissionDescription('private'),\n emptyStateMessage: s.all.privateEmptyStateMessage\n }\"\n ></ng-container>\n <ng-container\n [ngTemplateOutlet]=\"permissionSection\"\n [ngTemplateOutletContext]=\"{\n permission: 'shared',\n open: s.all.sharedOpen,\n openChange: getOpenChange('shared'),\n views: s.all.sharedViews,\n title: getPermissionTitle('shared'),\n description: getPermissionDescription('shared'),\n emptyStateMessage: s.all.sharedEmptyStateMessage\n }\"\n ></ng-container>\n <ng-container\n [ngTemplateOutlet]=\"permissionSection\"\n [ngTemplateOutletContext]=\"{\n permission: 'public',\n open: s.all.publicOpen,\n openChange: getOpenChange('public'),\n views: s.all.publicViews,\n title: getPermissionTitle('public'),\n description: getPermissionDescription('public'),\n emptyStateMessage: s.all.publicEmptyStateMessage\n }\"\n ></ng-container>\n\n <!-- Permission section -->\n <ng-template\n #permissionSection\n let-permission=\"permission\"\n let-open=\"open\"\n let-openChange=\"openChange\"\n let-views=\"views\"\n let-title=\"title\"\n let-description=\"description\"\n let-emptyStateMessage=\"emptyStateMessage\"\n >\n <riv-toggle\n [open]=\"open\"\n (openChange)=\"openChange($event)\"\n [disabled]=\"!!s.all.search\"\n >\n <ng-template #trigger>\n <div>\n <h3 class=\"permission-title\">{{ title }}</h3>\n <span class=\"permission-description\">{{ description }}</span>\n </div>\n </ng-template>\n\n <div class=\"views-list\" *ngIf=\"views.length > 0; else noViews\">\n <button\n *ngFor=\"let view of views; trackBy: trackById\"\n class=\"view-item\"\n (click)=\"\n manager?.actions?.next({\n type: 'setActiveView',\n id: view.id\n });\n close.emit()\n \"\n >\n <riv-highlight\n [text]=\"view.title\"\n [indices]=\"view.titleHighlightIndices ?? []\"\n >\n </riv-highlight>\n <riv-view-menu\n [manager]=\"manager\"\n [view]=\"view\"\n ></riv-view-menu>\n <button\n class=\"enable-view\"\n [class.enabled]=\"view.enabled\"\n (click)=\"\n $event.stopPropagation();\n manager?.actions?.next({\n type: 'setEnabled',\n id: view.id,\n enabled: !view.enabled\n })\n \"\n >\n <riv-icon\n [name]=\"view.enabled ? 'Eye' : 'EyeOff'\"\n [size]=\"20\"\n ></riv-icon>\n </button>\n </button>\n </div>\n <ng-template #noViews>\n <riv-zero-state [message]=\"emptyStateMessage\"></riv-zero-state>\n </ng-template>\n </riv-toggle>\n </ng-template>\n </div>\n </div>\n </riv-side-sheet>\n</ng-container>\n", styles: [".content{padding:var(--size-large);display:flex;flex-direction:column;gap:var(--size-medium);height:100%}header{display:flex;justify-content:space-between;align-items:center}.title{font:var(--title-02)}.views{flex-grow:1;overflow-y:auto;display:flex;flex-direction:column;gap:var(--size-xlarge)}.permission-title{font:var(--title-02)}.permission-description{font:var(--body-medium);color:var(--type-light-low-contrast)}.views-list{display:flex;flex-direction:column;gap:var(--size-small)}.view-item{display:flex;align-items:center;padding-left:var(--size-small);border-radius:var(--border-radius-small);cursor:pointer;background-color:var(--surface-light-0);transition:background-color var(--short-transition)}.view-item:hover{background-color:var(--surface-light-1)}.view-item:active{background-color:var(--surface-light-2)}.view-item riv-highlight{text-align:left;flex-grow:1}.view-item riv-view-menu{align-self:flex-end;opacity:0;transition:opacity var(--short-transition)}.view-item:hover riv-view-menu{opacity:1}.enable-view{cursor:pointer;display:inline-flex;justify-content:center;align-items:center;padding:0 var(--size-xsmall);transition:color var(--medium-transition);color:var(--type-light-low-contrast)}.enable-view.enabled{color:var(--type-light-link)}.enable-view.enabled:hover{color:var(--type-light-link-hover)}.enable-view.enabled:active{color:var(--type-light-link-active)}\n"] }]
|
|
36
|
+
}], propDecorators: { manager: [{
|
|
37
|
+
type: Input
|
|
38
|
+
}], topOffset: [{
|
|
39
|
+
type: Input
|
|
40
|
+
}], close: [{
|
|
41
|
+
type: Output
|
|
42
|
+
}] } });
|
|
43
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"all-views-modal.component.js","sourceRoot":"","sources":["../../../../../../../projects/riv/src/lib/navigation/views/all-views-modal/all-views-modal.component.ts","../../../../../../../projects/riv/src/lib/navigation/views/all-views-modal/all-views-modal.component.html"],"names":[],"mappings":"AAAA,OAAO,EACL,uBAAuB,EACvB,SAAS,EACT,YAAY,EACZ,KAAK,EACL,MAAM,GACP,MAAM,eAAe,CAAC;AACvB,OAAO,EAAE,QAAQ,EAAE,MAAM,UAAU,CAAC;;;;;;;;;;;AAQpC,MAAM,OAAO,sBAAsB;IANnC;QAWE,cAAS,GAAW,CAAC,CAAC;QAGtB,UAAK,GAAG,IAAI,YAAY,EAAE,CAAC;QAW3B,uBAAkB,GAAG,QAAQ,CAAC,kBAAkB,CAAC;QACjD,6BAAwB,GAAG,QAAQ,CAAC,wBAAwB,CAAC;KAK9D;IAfC,aAAa,CAAC,UAAmC;QAC/C,OAAO,CAAC,IAAa,EAAE,EAAE,CACvB,IAAI,CAAC,OAAO,EAAE,OAAO,EAAE,IAAI,CAAC;YAC1B,IAAI,EAAE,mBAAmB;YACzB,UAAU;YACV,IAAI;SACL,CAAC,CAAC;IACP,CAAC;IAKD,SAAS,CAAC,CAAS,EAAE,IAA0B;QAC7C,OAAO,IAAI,CAAC,EAAE,CAAC;IACjB,CAAC;;mHAxBU,sBAAsB;uGAAtB,sBAAsB,gJCfnC,2mJAuIA;2FDxHa,sBAAsB;kBANlC,SAAS;+BACE,qBAAqB,mBAGd,uBAAuB,CAAC,MAAM;8BAI/C,OAAO;sBADN,KAAK;gBAIN,SAAS;sBADR,KAAK;gBAIN,KAAK;sBADJ,MAAM","sourcesContent":["import {\n  ChangeDetectionStrategy,\n  Component,\n  EventEmitter,\n  Input,\n  Output,\n} from '@angular/core';\nimport { RivViews } from '../state';\n\n@Component({\n  selector: 'riv-all-views-modal',\n  templateUrl: './all-views-modal.component.html',\n  styleUrls: ['./all-views-modal.component.css'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class AllViewsModalComponent<V extends RivViews.View> {\n  @Input()\n  manager?: RivViews.Manager<V>;\n\n  @Input()\n  topOffset: number = 0;\n\n  @Output()\n  close = new EventEmitter();\n\n  getOpenChange(permission: RivViews.ViewPermission): (open: boolean) => void {\n    return (open: boolean) =>\n      this.manager?.actions?.next({\n        type: 'setAllViewsToggle',\n        permission,\n        open,\n      });\n  }\n\n  getPermissionTitle = RivViews.getPermissionTitle;\n  getPermissionDescription = RivViews.getPermissionDescription;\n\n  trackById(_: number, view: RivViews.FullView<V>): V['id'] {\n    return view.id;\n  }\n}\n","<ng-container *ngIf=\"manager?.state | async; let s\">\n  <riv-side-sheet [topOffset]=\"topOffset\" (close)=\"close.emit()\">\n    <div class=\"content\">\n      <header>\n        <span class=\"title\">All views</span>\n        <button\n          rivButton\n          [variant]=\"'ghost'\"\n          [icon]=\"'X'\"\n          (click)=\"close.emit()\"\n        ></button>\n      </header>\n\n      <riv-search\n        (valueChange)=\"\n          manager?.actions?.next({\n            type: 'setAllViewsSearchQuery',\n            query: $event\n          })\n        \"\n        [value]=\"s.all.search\"\n        placeholder=\"Search\"\n      ></riv-search>\n\n      <div class=\"views\">\n        <!-- View sections by permission type -->\n        <ng-container\n          [ngTemplateOutlet]=\"permissionSection\"\n          [ngTemplateOutletContext]=\"{\n            permission: 'private',\n            open: s.all.privateOpen,\n            openChange: getOpenChange('private'),\n            views: s.all.privateViews,\n            title: getPermissionTitle('private'),\n            description: getPermissionDescription('private'),\n            emptyStateMessage: s.all.privateEmptyStateMessage\n          }\"\n        ></ng-container>\n        <ng-container\n          [ngTemplateOutlet]=\"permissionSection\"\n          [ngTemplateOutletContext]=\"{\n            permission: 'shared',\n            open: s.all.sharedOpen,\n            openChange: getOpenChange('shared'),\n            views: s.all.sharedViews,\n            title: getPermissionTitle('shared'),\n            description: getPermissionDescription('shared'),\n            emptyStateMessage: s.all.sharedEmptyStateMessage\n          }\"\n        ></ng-container>\n        <ng-container\n          [ngTemplateOutlet]=\"permissionSection\"\n          [ngTemplateOutletContext]=\"{\n            permission: 'public',\n            open: s.all.publicOpen,\n            openChange: getOpenChange('public'),\n            views: s.all.publicViews,\n            title: getPermissionTitle('public'),\n            description: getPermissionDescription('public'),\n            emptyStateMessage: s.all.publicEmptyStateMessage\n          }\"\n        ></ng-container>\n\n        <!-- Permission section -->\n        <ng-template\n          #permissionSection\n          let-permission=\"permission\"\n          let-open=\"open\"\n          let-openChange=\"openChange\"\n          let-views=\"views\"\n          let-title=\"title\"\n          let-description=\"description\"\n          let-emptyStateMessage=\"emptyStateMessage\"\n        >\n          <riv-toggle\n            [open]=\"open\"\n            (openChange)=\"openChange($event)\"\n            [disabled]=\"!!s.all.search\"\n          >\n            <ng-template #trigger>\n              <div>\n                <h3 class=\"permission-title\">{{ title }}</h3>\n                <span class=\"permission-description\">{{ description }}</span>\n              </div>\n            </ng-template>\n\n            <div class=\"views-list\" *ngIf=\"views.length > 0; else noViews\">\n              <button\n                *ngFor=\"let view of views; trackBy: trackById\"\n                class=\"view-item\"\n                (click)=\"\n                  manager?.actions?.next({\n                    type: 'setActiveView',\n                    id: view.id\n                  });\n                  close.emit()\n                \"\n              >\n                <riv-highlight\n                  [text]=\"view.title\"\n                  [indices]=\"view.titleHighlightIndices ?? []\"\n                >\n                </riv-highlight>\n                <riv-view-menu\n                  [manager]=\"manager\"\n                  [view]=\"view\"\n                ></riv-view-menu>\n                <button\n                  class=\"enable-view\"\n                  [class.enabled]=\"view.enabled\"\n                  (click)=\"\n                    $event.stopPropagation();\n                    manager?.actions?.next({\n                      type: 'setEnabled',\n                      id: view.id,\n                      enabled: !view.enabled\n                    })\n                  \"\n                >\n                  <riv-icon\n                    [name]=\"view.enabled ? 'Eye' : 'EyeOff'\"\n                    [size]=\"20\"\n                  ></riv-icon>\n                </button>\n              </button>\n            </div>\n            <ng-template #noViews>\n              <riv-zero-state [message]=\"emptyStateMessage\"></riv-zero-state>\n            </ng-template>\n          </riv-toggle>\n        </ng-template>\n      </div>\n    </div>\n  </riv-side-sheet>\n</ng-container>\n"]}
|
|
@@ -1,41 +1,64 @@
|
|
|
1
1
|
import Fuse from 'fuse.js';
|
|
2
|
-
import { defaultsDeep } from 'lodash';
|
|
2
|
+
import { defaultsDeep, sortBy } from 'lodash';
|
|
3
3
|
import { connectable, from, map, mergeScan, ReplaySubject, startWith, Subject, } from 'rxjs';
|
|
4
4
|
export var RivViews;
|
|
5
5
|
(function (RivViews) {
|
|
6
6
|
RivViews.ViewPermissions = ['private', 'public', 'shared'];
|
|
7
|
-
function createManager(source, duplicate, options) {
|
|
7
|
+
function createManager(source, duplicate, copyLink, options) {
|
|
8
8
|
const defaultState = {
|
|
9
9
|
views: [],
|
|
10
10
|
activeViewId: null,
|
|
11
11
|
defaultViewId: null,
|
|
12
12
|
containerWidth: 0,
|
|
13
|
-
|
|
14
|
-
|
|
13
|
+
overflow: {
|
|
14
|
+
search: '',
|
|
15
|
+
pickerOpen: false,
|
|
16
|
+
},
|
|
17
|
+
all: {
|
|
18
|
+
search: '',
|
|
19
|
+
privateOpen: true,
|
|
20
|
+
sharedOpen: true,
|
|
21
|
+
publicOpen: true,
|
|
22
|
+
},
|
|
15
23
|
};
|
|
16
24
|
const defaultedInitialState = defaultsDeep({}, options?.initialState, defaultState);
|
|
17
25
|
const actions = new Subject();
|
|
18
|
-
const coreState = actions.pipe(mergeScan((state, action) => from(reduce(state, action, source, duplicate)), defaultedInitialState, 1), startWith(defaultedInitialState));
|
|
26
|
+
const coreState = actions.pipe(mergeScan((state, action) => from(reduce(state, action, source, duplicate, copyLink)), defaultedInitialState, 1), startWith(defaultedInitialState));
|
|
19
27
|
const state = connectable(coreState.pipe(map(s => {
|
|
20
28
|
// If for some reason the active view is not currently enabled, we'll
|
|
21
29
|
// default to making the first view active instead.
|
|
22
30
|
const activeView = s.views.find(view => view.enabled && view.id === s.activeViewId);
|
|
23
|
-
const
|
|
24
|
-
.filter(view => view.enabled)
|
|
25
|
-
.map((view, i) => ({
|
|
31
|
+
const allViews = s.views.map((view, i) => ({
|
|
26
32
|
...view,
|
|
27
33
|
active: activeView ? activeView.id === view.id : i === 0,
|
|
28
34
|
default: view.id === s.defaultViewId,
|
|
29
35
|
}));
|
|
30
|
-
const
|
|
31
|
-
const
|
|
32
|
-
|
|
33
|
-
|
|
36
|
+
const allEnabledViews = allViews.filter(view => view.enabled);
|
|
37
|
+
const displayedViews = calculateDisplayedViews(allEnabledViews, calculateOverflowViewIndex(s), s.activeViewId);
|
|
38
|
+
const enabledViews = s.overflow.search
|
|
39
|
+
? searchViews(allEnabledViews, s.overflow.search, options?.fuseOptions)
|
|
40
|
+
: allEnabledViews;
|
|
41
|
+
const allViewsFiltered = s.all.search
|
|
42
|
+
? searchViews(allViews, s.all.search, options?.fuseOptions)
|
|
43
|
+
: allViews;
|
|
44
|
+
const searchingAllViews = !!s.all.search;
|
|
34
45
|
return {
|
|
35
46
|
...s,
|
|
36
|
-
|
|
47
|
+
allViews,
|
|
37
48
|
displayedViews,
|
|
38
|
-
|
|
49
|
+
enabledViews,
|
|
50
|
+
all: {
|
|
51
|
+
...s.all,
|
|
52
|
+
privateOpen: searchingAllViews || s.all.privateOpen,
|
|
53
|
+
privateViews: viewsByPermission(allViewsFiltered, 'private'),
|
|
54
|
+
privateEmptyStateMessage: getEmptyStateMessage(s, 'private'),
|
|
55
|
+
sharedOpen: searchingAllViews || s.all.sharedOpen,
|
|
56
|
+
sharedViews: viewsByPermission(allViewsFiltered, 'shared'),
|
|
57
|
+
sharedEmptyStateMessage: getEmptyStateMessage(s, 'shared'),
|
|
58
|
+
publicOpen: searchingAllViews || s.all.publicOpen,
|
|
59
|
+
publicViews: viewsByPermission(allViewsFiltered, 'public'),
|
|
60
|
+
publicEmptyStateMessage: getEmptyStateMessage(s, 'public'),
|
|
61
|
+
},
|
|
39
62
|
};
|
|
40
63
|
})), {
|
|
41
64
|
connector: () => new ReplaySubject(1),
|
|
@@ -74,8 +97,11 @@ export var RivViews;
|
|
|
74
97
|
}
|
|
75
98
|
return displayedViews;
|
|
76
99
|
}
|
|
77
|
-
function
|
|
78
|
-
|
|
100
|
+
function viewsByPermission(views, permission) {
|
|
101
|
+
return sortBy(views.filter(view => view.permission === permission), 'title');
|
|
102
|
+
}
|
|
103
|
+
function searchViews(views, query, options) {
|
|
104
|
+
const fuse = new Fuse(views, {
|
|
79
105
|
threshold: 0.2,
|
|
80
106
|
keys: ['title'],
|
|
81
107
|
includeMatches: true,
|
|
@@ -133,7 +159,13 @@ export var RivViews;
|
|
|
133
159
|
];
|
|
134
160
|
}
|
|
135
161
|
}
|
|
136
|
-
|
|
162
|
+
function getEmptyStateMessage(state, permission) {
|
|
163
|
+
if (state.all.search)
|
|
164
|
+
return `No ${permission} views matching "${state.all.search}"`;
|
|
165
|
+
else
|
|
166
|
+
return `No ${permission} views`;
|
|
167
|
+
}
|
|
168
|
+
async function reduce(state, action, source, duplicate, copyLink) {
|
|
137
169
|
switch (action.type) {
|
|
138
170
|
case 'load':
|
|
139
171
|
return {
|
|
@@ -144,6 +176,15 @@ export var RivViews;
|
|
|
144
176
|
return {
|
|
145
177
|
...state,
|
|
146
178
|
activeViewId: action.id,
|
|
179
|
+
views: state.views.map(view => view.id === action.id ? { ...view, enabled: true } : view),
|
|
180
|
+
all: {
|
|
181
|
+
...state.all,
|
|
182
|
+
search: '',
|
|
183
|
+
},
|
|
184
|
+
overflow: {
|
|
185
|
+
...state.overflow,
|
|
186
|
+
search: '',
|
|
187
|
+
},
|
|
147
188
|
};
|
|
148
189
|
case 'setDefaultView':
|
|
149
190
|
return {
|
|
@@ -160,12 +201,26 @@ export var RivViews;
|
|
|
160
201
|
case 'setOverflowPickerOpen':
|
|
161
202
|
return {
|
|
162
203
|
...state,
|
|
163
|
-
|
|
204
|
+
overflow: {
|
|
205
|
+
...state.overflow,
|
|
206
|
+
pickerOpen: action.open,
|
|
207
|
+
},
|
|
164
208
|
};
|
|
165
209
|
case 'setOverflowSearchQuery':
|
|
166
210
|
return {
|
|
167
211
|
...state,
|
|
168
|
-
|
|
212
|
+
overflow: {
|
|
213
|
+
...state.overflow,
|
|
214
|
+
search: action.query,
|
|
215
|
+
},
|
|
216
|
+
};
|
|
217
|
+
case 'setAllViewsSearchQuery':
|
|
218
|
+
return {
|
|
219
|
+
...state,
|
|
220
|
+
all: {
|
|
221
|
+
...state.all,
|
|
222
|
+
search: action.query,
|
|
223
|
+
},
|
|
169
224
|
};
|
|
170
225
|
case 'updateView':
|
|
171
226
|
return {
|
|
@@ -179,7 +234,51 @@ export var RivViews;
|
|
|
179
234
|
...state,
|
|
180
235
|
views: await withDuplicateView(state.views, action.view, duplicate),
|
|
181
236
|
};
|
|
237
|
+
case 'copyLinkToView':
|
|
238
|
+
await copyLink(action.view);
|
|
239
|
+
return state;
|
|
240
|
+
case 'setAllViewsToggle':
|
|
241
|
+
return {
|
|
242
|
+
...state,
|
|
243
|
+
all: {
|
|
244
|
+
...state.all,
|
|
245
|
+
privateOpen: action.permission === 'private'
|
|
246
|
+
? action.open
|
|
247
|
+
: state.all.privateOpen,
|
|
248
|
+
sharedOpen: action.permission === 'shared'
|
|
249
|
+
? action.open
|
|
250
|
+
: state.all.sharedOpen,
|
|
251
|
+
publicOpen: action.permission === 'public'
|
|
252
|
+
? action.open
|
|
253
|
+
: state.all.publicOpen,
|
|
254
|
+
},
|
|
255
|
+
};
|
|
256
|
+
}
|
|
257
|
+
}
|
|
258
|
+
///////////////
|
|
259
|
+
// Utilities //
|
|
260
|
+
///////////////
|
|
261
|
+
function getPermissionTitle(permission) {
|
|
262
|
+
switch (permission) {
|
|
263
|
+
case 'private':
|
|
264
|
+
return 'Private';
|
|
265
|
+
case 'shared':
|
|
266
|
+
return 'Shared';
|
|
267
|
+
case 'public':
|
|
268
|
+
return 'Public';
|
|
269
|
+
}
|
|
270
|
+
}
|
|
271
|
+
RivViews.getPermissionTitle = getPermissionTitle;
|
|
272
|
+
function getPermissionDescription(permission) {
|
|
273
|
+
switch (permission) {
|
|
274
|
+
case 'private':
|
|
275
|
+
return 'Only visible to you';
|
|
276
|
+
case 'shared':
|
|
277
|
+
return 'Only visible to those with access';
|
|
278
|
+
case 'public':
|
|
279
|
+
return 'Visible to everyone';
|
|
182
280
|
}
|
|
183
281
|
}
|
|
282
|
+
RivViews.getPermissionDescription = getPermissionDescription;
|
|
184
283
|
})(RivViews || (RivViews = {}));
|
|
185
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"state.js","sourceRoot":"","sources":["../../../../../../projects/riv/src/lib/navigation/views/state.ts"],"names":[],"mappings":"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","sourcesContent":["import Fuse, { FuseResult, IFuseOptions } from 'fuse.js';\nimport { defaultsDeep } from 'lodash';\nimport {\n  connectable,\n  Connectable,\n  from,\n  map,\n  mergeScan,\n  Observable,\n  ReplaySubject,\n  startWith,\n  Subject,\n} from 'rxjs';\nimport { DeepPartial } from 'utility-types';\nimport { HighlightComponent } from '../../visualization/highlight/highlight.component';\n\nexport namespace RivViews {\n  export const ViewPermissions = ['private', 'public', 'shared'] as const;\n  export type ViewPermission = (typeof ViewPermissions)[number];\n\n  export type View = {\n    id: string | number;\n    title: string;\n    enabled: boolean;\n    permission: ViewPermission;\n  };\n\n  export type FullView<V extends View = View> = V & {\n    active: boolean;\n    default: boolean;\n    titleHighlightIndices?: HighlightComponent.HighlightIndices[];\n  };\n\n  ///////////\n  // State //\n  ///////////\n\n  export type CoreState<V extends View> = {\n    views: V[];\n    activeViewId: V['id'] | null;\n    defaultViewId: V['id'] | null;\n    containerWidth: number;\n    overflowPickerOpen: boolean;\n    overflowSearchQuery: string;\n  };\n\n  export type FullState<V extends View> = CoreState<V> & {\n    fullViews: FullView<V>[];\n    displayedViews: FullView<V>[];\n    overflowViews: FullView<V>[];\n  };\n\n  /////////////\n  // Actions //\n  /////////////\n\n  export type Action<V extends View> =\n    | { type: 'load' }\n    | { type: 'setActiveView'; id: V['id'] }\n    | { type: 'setDefaultView'; id: V['id'] }\n    | { type: 'setEnabled'; id: V['id']; enabled: boolean }\n    | { type: 'setContainerWidth'; width: number }\n    | { type: 'setOverflowPickerOpen'; open: boolean }\n    | { type: 'setOverflowSearchQuery'; query: string }\n    | { type: 'updateView'; view: V }\n    | { type: 'deleteView'; view: V }\n    | { type: 'duplicateView'; view: V };\n\n  ////////////\n  // Source //\n  ////////////\n\n  export type Source<V extends View> = () => V[] | Promise<V[]>;\n  export type Duplicate<V extends View> = (\n    viewToDuplicate: V,\n  ) => V | Promise<V>;\n\n  /////////////\n  // Manager //\n  /////////////\n\n  export type Manager<V extends View> = {\n    actions: Subject<Action<V>>;\n    state: Connectable<FullState<V>>;\n  };\n\n  export type ManagerOptions<V extends View> = {\n    initialState?: DeepPartial<CoreState<V>>;\n    fuseOptions?: IFuseOptions<V>;\n  };\n\n  export function createManager<V extends View>(\n    source: Source<V>,\n    duplicate: Duplicate<V>,\n    options?: ManagerOptions<V>,\n  ): Manager<V> {\n    const defaultState: CoreState<V> = {\n      views: [],\n      activeViewId: null,\n      defaultViewId: null,\n      containerWidth: 0,\n      overflowPickerOpen: false,\n      overflowSearchQuery: '',\n    };\n\n    const defaultedInitialState: CoreState<V> = defaultsDeep(\n      {},\n      options?.initialState,\n      defaultState,\n    );\n\n    const actions: Subject<Action<V>> = new Subject();\n    const coreState: Observable<CoreState<V>> = actions.pipe(\n      mergeScan(\n        (state, action) => from(reduce(state, action, source, duplicate)),\n        defaultedInitialState,\n        1,\n      ),\n      startWith(defaultedInitialState),\n    );\n\n    const state: Connectable<FullState<V>> = connectable(\n      coreState.pipe(\n        map(s => {\n          // If for some reason the active view is not currently enabled, we'll\n          // default to making the first view active instead.\n          const activeView: V | undefined = s.views.find(\n            view => view.enabled && view.id === s.activeViewId,\n          );\n          const fullViews = s.views\n            .filter(view => view.enabled)\n            .map((view, i) => ({\n              ...view,\n              active: activeView ? activeView.id === view.id : i === 0,\n              default: view.id === s.defaultViewId,\n            }));\n          const displayedViews = calculateDisplayedViews(\n            fullViews,\n            calculateOverflowViewIndex(s),\n            s.activeViewId,\n          );\n\n          const overflowViews = s.overflowSearchQuery\n            ? searchViews(\n                fullViews,\n                s.overflowSearchQuery,\n                options?.fuseOptions,\n              )\n            : fullViews;\n\n          return {\n            ...s,\n            fullViews,\n            displayedViews,\n            overflowViews,\n          };\n        }),\n      ),\n      {\n        connector: () => new ReplaySubject(1),\n        resetOnDisconnect: false,\n      },\n    );\n    state.connect();\n\n    // Load the views immediately\n    actions.next({ type: 'load' });\n\n    return { actions, state };\n  }\n\n  //////////////////////////////////\n  // State reducer (with helpers) //\n  //////////////////////////////////\n\n  function calculateOverflowViewIndex<V extends View>(\n    state: CoreState<V>,\n  ): number {\n    const enabledViews = state.views.filter(view => view.enabled);\n\n    // Tabs have a minimum width of 92px.\n    const possibleSlotCount = Math.floor(state.containerWidth / 92);\n\n    // The \"all views\" sidebar trigger takes up one slot.\n    const availableSlotCount = possibleSlotCount - 1;\n\n    // If there are too many tabs, the \"N more\" trigger takes up one slot as well.\n    // But we want to avoid the \"1 more\" case, so we only use a slot for \"N more\" if N != 1.\n    if (availableSlotCount >= enabledViews.length) return enabledViews.length;\n    else return availableSlotCount - 1; // the \"N more\" uses up a slot.\n  }\n\n  function calculateDisplayedViews<V extends View>(\n    fullViews: FullView<V>[],\n    overflowViewIndex: number,\n    activeViewId: CoreState<V>['activeViewId'],\n  ): FullView<V>[] {\n    const displayedViews = fullViews.slice(0, overflowViewIndex);\n\n    const activeViewInDisplay = displayedViews.some(\n      view => view.id === activeViewId,\n    );\n\n    if (!activeViewInDisplay && activeViewId !== null) {\n      const activeView = fullViews.find(view => view.id === activeViewId);\n      if (activeView && displayedViews.length > 0) {\n        return [...displayedViews.slice(0, -1), activeView];\n      }\n    }\n\n    return displayedViews;\n  }\n\n  function searchViews<V extends View>(\n    fullViews: FullView<V>[],\n    query: string,\n    options?: IFuseOptions<V>,\n  ): FullView<V>[] {\n    const fuse = new Fuse(fullViews, {\n      threshold: 0.2,\n      keys: ['title'],\n      includeMatches: true,\n      ...options,\n    });\n\n    const matchedOptions: FuseResult<FullView<V>>[] = fuse.search(query);\n\n    return matchedOptions.map(result => ({\n      ...result.item,\n      titleHighlightIndices: [\n        ...(result.matches?.find(match => match.key === 'title')?.indices ??\n          []),\n      ],\n    }));\n  }\n\n  function getNextBestActiveView<V extends View>(\n    state: CoreState<V>,\n  ): V['id'] | null {\n    const enabledViews = state.views.filter(view => view.enabled);\n    const currentActiveViewIndex = enabledViews.findIndex(\n      view => view.id === state.activeViewId,\n    );\n\n    if (currentActiveViewIndex === -1) return enabledViews[0]?.id ?? null;\n    return (\n      enabledViews[currentActiveViewIndex + 1]?.id ??\n      enabledViews[currentActiveViewIndex - 1]?.id ??\n      null\n    );\n  }\n\n  function deleteView<V extends View>(\n    state: CoreState<V>,\n    id: V['id'],\n  ): CoreState<V> {\n    const activeViewId =\n      state.activeViewId === id\n        ? getNextBestActiveView(state)\n        : state.activeViewId;\n    const views = state.views.filter(view => view.id !== id);\n    return {\n      ...state,\n      views,\n      activeViewId,\n    };\n  }\n\n  function setEnabled<V extends View>(\n    state: CoreState<V>,\n    id: V['id'],\n    enabled: boolean,\n  ): CoreState<V> {\n    return {\n      ...state,\n      views: state.views.map(view =>\n        view.id === id ? { ...view, enabled } : view,\n      ),\n      activeViewId:\n        !enabled && state.activeViewId === id\n          ? getNextBestActiveView(state)\n          : state.activeViewId,\n    };\n  }\n\n  async function withDuplicateView<V extends View>(\n    views: V[],\n    viewToDuplicate: V,\n    duplicate: Duplicate<V>,\n  ): Promise<V[]> {\n    const newView = await duplicate(viewToDuplicate);\n    const sourceIndex = views.findIndex(view => view.id === viewToDuplicate.id);\n    if (sourceIndex === -1) {\n      return [newView, ...views];\n    } else {\n      return [\n        ...views.slice(0, sourceIndex + 1),\n        newView,\n        ...views.slice(sourceIndex + 1),\n      ];\n    }\n  }\n\n  async function reduce<V extends View>(\n    state: CoreState<V>,\n    action: Action<V>,\n    source: Source<V>,\n    duplicate: Duplicate<V>,\n  ): Promise<CoreState<V>> {\n    switch (action.type) {\n      case 'load':\n        return {\n          ...state,\n          views: await source(),\n        };\n\n      case 'setActiveView':\n        return {\n          ...state,\n          activeViewId: action.id,\n        };\n\n      case 'setDefaultView':\n        return {\n          ...state,\n          defaultViewId: action.id,\n        };\n\n      case 'setEnabled':\n        return setEnabled(state, action.id, action.enabled);\n\n      case 'setContainerWidth':\n        return {\n          ...state,\n          containerWidth: action.width,\n        };\n\n      case 'setOverflowPickerOpen':\n        return {\n          ...state,\n          overflowPickerOpen: action.open,\n        };\n\n      case 'setOverflowSearchQuery':\n        return {\n          ...state,\n          overflowSearchQuery: action.query,\n        };\n\n      case 'updateView':\n        return {\n          ...state,\n          views: state.views.map(view =>\n            view.id === action.view.id ? action.view : view,\n          ),\n        };\n\n      case 'deleteView':\n        return deleteView(state, action.view.id);\n\n      case 'duplicateView':\n        return {\n          ...state,\n          views: await withDuplicateView(state.views, action.view, duplicate),\n        };\n    }\n  }\n}\n"]}
|
|
284
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"state.js","sourceRoot":"","sources":["../../../../../../projects/riv/src/lib/navigation/views/state.ts"],"names":[],"mappings":"AAAA,OAAO,IAAkC,MAAM,SAAS,CAAC;AACzD,OAAO,EAAE,YAAY,EAAE,MAAM,EAAE,MAAM,QAAQ,CAAC;AAC9C,OAAO,EACL,WAAW,EAEX,IAAI,EACJ,GAAG,EACH,SAAS,EAET,aAAa,EACb,SAAS,EACT,OAAO,GACR,MAAM,MAAM,CAAC;AAId,MAAM,KAAW,QAAQ,CAmfxB;AAnfD,WAAiB,QAAQ;IACV,wBAAe,GAAG,CAAC,SAAS,EAAE,QAAQ,EAAE,QAAQ,CAAU,CAAC;IA8FxE,SAAgB,aAAa,CAC3B,MAAiB,EACjB,SAAuB,EACvB,QAAqB,EACrB,OAA2B;QAE3B,MAAM,YAAY,GAAiB;YACjC,KAAK,EAAE,EAAE;YACT,YAAY,EAAE,IAAI;YAClB,aAAa,EAAE,IAAI;YACnB,cAAc,EAAE,CAAC;YACjB,QAAQ,EAAE;gBACR,MAAM,EAAE,EAAE;gBACV,UAAU,EAAE,KAAK;aAClB;YACD,GAAG,EAAE;gBACH,MAAM,EAAE,EAAE;gBACV,WAAW,EAAE,IAAI;gBACjB,UAAU,EAAE,IAAI;gBAChB,UAAU,EAAE,IAAI;aACjB;SACF,CAAC;QAEF,MAAM,qBAAqB,GAAiB,YAAY,CACtD,EAAE,EACF,OAAO,EAAE,YAAY,EACrB,YAAY,CACb,CAAC;QAEF,MAAM,OAAO,GAAuB,IAAI,OAAO,EAAE,CAAC;QAClD,MAAM,SAAS,GAA6B,OAAO,CAAC,IAAI,CACtD,SAAS,CACP,CAAC,KAAK,EAAE,MAAM,EAAE,EAAE,CAChB,IAAI,CAAC,MAAM,CAAC,KAAK,EAAE,MAAM,EAAE,MAAM,EAAE,SAAS,EAAE,QAAQ,CAAC,CAAC,EAC1D,qBAAqB,EACrB,CAAC,CACF,EACD,SAAS,CAAC,qBAAqB,CAAC,CACjC,CAAC;QAEF,MAAM,KAAK,GAA8B,WAAW,CAClD,SAAS,CAAC,IAAI,CACZ,GAAG,CAAC,CAAC,CAAC,EAAE;YACN,qEAAqE;YACrE,mDAAmD;YACnD,MAAM,UAAU,GAAkB,CAAC,CAAC,KAAK,CAAC,IAAI,CAC5C,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,OAAO,IAAI,IAAI,CAAC,EAAE,KAAK,CAAC,CAAC,YAAY,CACnD,CAAC;YAEF,MAAM,QAAQ,GAAkB,CAAC,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,IAAI,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC;gBACxD,GAAG,IAAI;gBACP,MAAM,EAAE,UAAU,CAAC,CAAC,CAAC,UAAU,CAAC,EAAE,KAAK,IAAI,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC;gBACxD,OAAO,EAAE,IAAI,CAAC,EAAE,KAAK,CAAC,CAAC,aAAa;aACrC,CAAC,CAAC,CAAC;YAEJ,MAAM,eAAe,GAAkB,QAAQ,CAAC,MAAM,CACpD,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,OAAO,CACrB,CAAC;YAEF,MAAM,cAAc,GAAG,uBAAuB,CAC5C,eAAe,EACf,0BAA0B,CAAC,CAAC,CAAC,EAC7B,CAAC,CAAC,YAAY,CACf,CAAC;YAEF,MAAM,YAAY,GAAG,CAAC,CAAC,QAAQ,CAAC,MAAM;gBACpC,CAAC,CAAC,WAAW,CACT,eAAe,EACf,CAAC,CAAC,QAAQ,CAAC,MAAM,EACjB,OAAO,EAAE,WAAW,CACrB;gBACH,CAAC,CAAC,eAAe,CAAC;YAEpB,MAAM,gBAAgB,GAAG,CAAC,CAAC,GAAG,CAAC,MAAM;gBACnC,CAAC,CAAC,WAAW,CAAC,QAAQ,EAAE,CAAC,CAAC,GAAG,CAAC,MAAM,EAAE,OAAO,EAAE,WAAW,CAAC;gBAC3D,CAAC,CAAC,QAAQ,CAAC;YAEb,MAAM,iBAAiB,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,MAAM,CAAC;YAEzC,OAAO;gBACL,GAAG,CAAC;gBACJ,QAAQ;gBACR,cAAc;gBACd,YAAY;gBACZ,GAAG,EAAE;oBACH,GAAG,CAAC,CAAC,GAAG;oBACR,WAAW,EAAE,iBAAiB,IAAI,CAAC,CAAC,GAAG,CAAC,WAAW;oBACnD,YAAY,EAAE,iBAAiB,CAAC,gBAAgB,EAAE,SAAS,CAAC;oBAC5D,wBAAwB,EAAE,oBAAoB,CAAC,CAAC,EAAE,SAAS,CAAC;oBAC5D,UAAU,EAAE,iBAAiB,IAAI,CAAC,CAAC,GAAG,CAAC,UAAU;oBACjD,WAAW,EAAE,iBAAiB,CAAC,gBAAgB,EAAE,QAAQ,CAAC;oBAC1D,uBAAuB,EAAE,oBAAoB,CAAC,CAAC,EAAE,QAAQ,CAAC;oBAC1D,UAAU,EAAE,iBAAiB,IAAI,CAAC,CAAC,GAAG,CAAC,UAAU;oBACjD,WAAW,EAAE,iBAAiB,CAAC,gBAAgB,EAAE,QAAQ,CAAC;oBAC1D,uBAAuB,EAAE,oBAAoB,CAAC,CAAC,EAAE,QAAQ,CAAC;iBAC3D;aACF,CAAC;QACJ,CAAC,CAAC,CACH,EACD;YACE,SAAS,EAAE,GAAG,EAAE,CAAC,IAAI,aAAa,CAAC,CAAC,CAAC;YACrC,iBAAiB,EAAE,KAAK;SACzB,CACF,CAAC;QACF,KAAK,CAAC,OAAO,EAAE,CAAC;QAEhB,6BAA6B;QAC7B,OAAO,CAAC,IAAI,CAAC,EAAE,IAAI,EAAE,MAAM,EAAE,CAAC,CAAC;QAE/B,OAAO,EAAE,OAAO,EAAE,KAAK,EAAE,CAAC;IAC5B,CAAC;IA9Ge,sBAAa,gBA8G5B,CAAA;IAED,kCAAkC;IAClC,kCAAkC;IAClC,kCAAkC;IAElC,SAAS,0BAA0B,CACjC,KAAmB;QAEnB,MAAM,YAAY,GAAG,KAAK,CAAC,KAAK,CAAC,MAAM,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;QAE9D,qCAAqC;QACrC,MAAM,iBAAiB,GAAG,IAAI,CAAC,KAAK,CAAC,KAAK,CAAC,cAAc,GAAG,EAAE,CAAC,CAAC;QAEhE,qDAAqD;QACrD,MAAM,kBAAkB,GAAG,iBAAiB,GAAG,CAAC,CAAC;QAEjD,8EAA8E;QAC9E,wFAAwF;QACxF,IAAI,kBAAkB,IAAI,YAAY,CAAC,MAAM;YAAE,OAAO,YAAY,CAAC,MAAM,CAAC;;YACrE,OAAO,kBAAkB,GAAG,CAAC,CAAC,CAAC,+BAA+B;IACrE,CAAC;IAED,SAAS,uBAAuB,CAC9B,SAAwB,EACxB,iBAAyB,EACzB,YAA0C;QAE1C,MAAM,cAAc,GAAG,SAAS,CAAC,KAAK,CAAC,CAAC,EAAE,iBAAiB,CAAC,CAAC;QAE7D,MAAM,mBAAmB,GAAG,cAAc,CAAC,IAAI,CAC7C,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,EAAE,KAAK,YAAY,CACjC,CAAC;QAEF,IAAI,CAAC,mBAAmB,IAAI,YAAY,KAAK,IAAI,EAAE;YACjD,MAAM,UAAU,GAAG,SAAS,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,EAAE,KAAK,YAAY,CAAC,CAAC;YACpE,IAAI,UAAU,IAAI,cAAc,CAAC,MAAM,GAAG,CAAC,EAAE;gBAC3C,OAAO,CAAC,GAAG,cAAc,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,EAAE,UAAU,CAAC,CAAC;aACrD;SACF;QAED,OAAO,cAAc,CAAC;IACxB,CAAC;IAED,SAAS,iBAAiB,CACxB,KAAoB,EACpB,UAA0B;QAE1B,OAAO,MAAM,CACX,KAAK,CAAC,MAAM,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,UAAU,KAAK,UAAU,CAAC,EACpD,OAAO,CACR,CAAC;IACJ,CAAC;IAED,SAAS,WAAW,CAClB,KAAoB,EACpB,KAAa,EACb,OAAyB;QAEzB,MAAM,IAAI,GAAG,IAAI,IAAI,CAAC,KAAK,EAAE;YAC3B,SAAS,EAAE,GAAG;YACd,IAAI,EAAE,CAAC,OAAO,CAAC;YACf,cAAc,EAAE,IAAI;YACpB,GAAG,OAAO;SACX,CAAC,CAAC;QAEH,MAAM,cAAc,GAA8B,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;QAErE,OAAO,cAAc,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC;YACnC,GAAG,MAAM,CAAC,IAAI;YACd,qBAAqB,EAAE;gBACrB,GAAG,CAAC,MAAM,CAAC,OAAO,EAAE,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC,KAAK,CAAC,GAAG,KAAK,OAAO,CAAC,EAAE,OAAO;oBAC/D,EAAE,CAAC;aACN;SACF,CAAC,CAAC,CAAC;IACN,CAAC;IAED,SAAS,qBAAqB,CAC5B,KAAmB;QAEnB,MAAM,YAAY,GAAG,KAAK,CAAC,KAAK,CAAC,MAAM,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;QAC9D,MAAM,sBAAsB,GAAG,YAAY,CAAC,SAAS,CACnD,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,EAAE,KAAK,KAAK,CAAC,YAAY,CACvC,CAAC;QAEF,IAAI,sBAAsB,KAAK,CAAC,CAAC;YAAE,OAAO,YAAY,CAAC,CAAC,CAAC,EAAE,EAAE,IAAI,IAAI,CAAC;QACtE,OAAO,CACL,YAAY,CAAC,sBAAsB,GAAG,CAAC,CAAC,EAAE,EAAE;YAC5C,YAAY,CAAC,sBAAsB,GAAG,CAAC,CAAC,EAAE,EAAE;YAC5C,IAAI,CACL,CAAC;IACJ,CAAC;IAED,SAAS,UAAU,CACjB,KAAmB,EACnB,EAAW;QAEX,MAAM,YAAY,GAChB,KAAK,CAAC,YAAY,KAAK,EAAE;YACvB,CAAC,CAAC,qBAAqB,CAAC,KAAK,CAAC;YAC9B,CAAC,CAAC,KAAK,CAAC,YAAY,CAAC;QACzB,MAAM,KAAK,GAAG,KAAK,CAAC,KAAK,CAAC,MAAM,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,CAAC,CAAC;QACzD,OAAO;YACL,GAAG,KAAK;YACR,KAAK;YACL,YAAY;SACb,CAAC;IACJ,CAAC;IAED,SAAS,UAAU,CACjB,KAAmB,EACnB,EAAW,EACX,OAAgB;QAEhB,OAAO;YACL,GAAG,KAAK;YACR,KAAK,EAAE,KAAK,CAAC,KAAK,CAAC,GAAG,CAAC,IAAI,CAAC,EAAE,CAC5B,IAAI,CAAC,EAAE,KAAK,EAAE,CAAC,CAAC,CAAC,EAAE,GAAG,IAAI,EAAE,OAAO,EAAE,CAAC,CAAC,CAAC,IAAI,CAC7C;YACD,YAAY,EACV,CAAC,OAAO,IAAI,KAAK,CAAC,YAAY,KAAK,EAAE;gBACnC,CAAC,CAAC,qBAAqB,CAAC,KAAK,CAAC;gBAC9B,CAAC,CAAC,KAAK,CAAC,YAAY;SACzB,CAAC;IACJ,CAAC;IAED,KAAK,UAAU,iBAAiB,CAC9B,KAAU,EACV,eAAkB,EAClB,SAAuB;QAEvB,MAAM,OAAO,GAAG,MAAM,SAAS,CAAC,eAAe,CAAC,CAAC;QACjD,MAAM,WAAW,GAAG,KAAK,CAAC,SAAS,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,EAAE,KAAK,eAAe,CAAC,EAAE,CAAC,CAAC;QAC5E,IAAI,WAAW,KAAK,CAAC,CAAC,EAAE;YACtB,OAAO,CAAC,OAAO,EAAE,GAAG,KAAK,CAAC,CAAC;SAC5B;aAAM;YACL,OAAO;gBACL,GAAG,KAAK,CAAC,KAAK,CAAC,CAAC,EAAE,WAAW,GAAG,CAAC,CAAC;gBAClC,OAAO;gBACP,GAAG,KAAK,CAAC,KAAK,CAAC,WAAW,GAAG,CAAC,CAAC;aAChC,CAAC;SACH;IACH,CAAC;IAED,SAAS,oBAAoB,CAC3B,KAAmB,EACnB,UAA0B;QAE1B,IAAI,KAAK,CAAC,GAAG,CAAC,MAAM;YAClB,OAAO,MAAM,UAAU,oBAAoB,KAAK,CAAC,GAAG,CAAC,MAAM,GAAG,CAAC;;YAC5D,OAAO,MAAM,UAAU,QAAQ,CAAC;IACvC,CAAC;IAED,KAAK,UAAU,MAAM,CACnB,KAAmB,EACnB,MAAiB,EACjB,MAAiB,EACjB,SAAuB,EACvB,QAAqB;QAErB,QAAQ,MAAM,CAAC,IAAI,EAAE;YACnB,KAAK,MAAM;gBACT,OAAO;oBACL,GAAG,KAAK;oBACR,KAAK,EAAE,MAAM,MAAM,EAAE;iBACtB,CAAC;YAEJ,KAAK,eAAe;gBAClB,OAAO;oBACL,GAAG,KAAK;oBACR,YAAY,EAAE,MAAM,CAAC,EAAE;oBACvB,KAAK,EAAE,KAAK,CAAC,KAAK,CAAC,GAAG,CAAC,IAAI,CAAC,EAAE,CAC5B,IAAI,CAAC,EAAE,KAAK,MAAM,CAAC,EAAE,CAAC,CAAC,CAAC,EAAE,GAAG,IAAI,EAAE,OAAO,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,IAAI,CAC1D;oBACD,GAAG,EAAE;wBACH,GAAG,KAAK,CAAC,GAAG;wBACZ,MAAM,EAAE,EAAE;qBACX;oBACD,QAAQ,EAAE;wBACR,GAAG,KAAK,CAAC,QAAQ;wBACjB,MAAM,EAAE,EAAE;qBACX;iBACF,CAAC;YAEJ,KAAK,gBAAgB;gBACnB,OAAO;oBACL,GAAG,KAAK;oBACR,aAAa,EAAE,MAAM,CAAC,EAAE;iBACzB,CAAC;YAEJ,KAAK,YAAY;gBACf,OAAO,UAAU,CAAC,KAAK,EAAE,MAAM,CAAC,EAAE,EAAE,MAAM,CAAC,OAAO,CAAC,CAAC;YAEtD,KAAK,mBAAmB;gBACtB,OAAO;oBACL,GAAG,KAAK;oBACR,cAAc,EAAE,MAAM,CAAC,KAAK;iBAC7B,CAAC;YAEJ,KAAK,uBAAuB;gBAC1B,OAAO;oBACL,GAAG,KAAK;oBACR,QAAQ,EAAE;wBACR,GAAG,KAAK,CAAC,QAAQ;wBACjB,UAAU,EAAE,MAAM,CAAC,IAAI;qBACxB;iBACF,CAAC;YAEJ,KAAK,wBAAwB;gBAC3B,OAAO;oBACL,GAAG,KAAK;oBACR,QAAQ,EAAE;wBACR,GAAG,KAAK,CAAC,QAAQ;wBACjB,MAAM,EAAE,MAAM,CAAC,KAAK;qBACrB;iBACF,CAAC;YAEJ,KAAK,wBAAwB;gBAC3B,OAAO;oBACL,GAAG,KAAK;oBACR,GAAG,EAAE;wBACH,GAAG,KAAK,CAAC,GAAG;wBACZ,MAAM,EAAE,MAAM,CAAC,KAAK;qBACrB;iBACF,CAAC;YAEJ,KAAK,YAAY;gBACf,OAAO;oBACL,GAAG,KAAK;oBACR,KAAK,EAAE,KAAK,CAAC,KAAK,CAAC,GAAG,CAAC,IAAI,CAAC,EAAE,CAC5B,IAAI,CAAC,EAAE,KAAK,MAAM,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,IAAI,CAChD;iBACF,CAAC;YAEJ,KAAK,YAAY;gBACf,OAAO,UAAU,CAAC,KAAK,EAAE,MAAM,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;YAE3C,KAAK,eAAe;gBAClB,OAAO;oBACL,GAAG,KAAK;oBACR,KAAK,EAAE,MAAM,iBAAiB,CAAC,KAAK,CAAC,KAAK,EAAE,MAAM,CAAC,IAAI,EAAE,SAAS,CAAC;iBACpE,CAAC;YAEJ,KAAK,gBAAgB;gBACnB,MAAM,QAAQ,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;gBAC5B,OAAO,KAAK,CAAC;YAEf,KAAK,mBAAmB;gBACtB,OAAO;oBACL,GAAG,KAAK;oBACR,GAAG,EAAE;wBACH,GAAG,KAAK,CAAC,GAAG;wBACZ,WAAW,EACT,MAAM,CAAC,UAAU,KAAK,SAAS;4BAC7B,CAAC,CAAC,MAAM,CAAC,IAAI;4BACb,CAAC,CAAC,KAAK,CAAC,GAAG,CAAC,WAAW;wBAC3B,UAAU,EACR,MAAM,CAAC,UAAU,KAAK,QAAQ;4BAC5B,CAAC,CAAC,MAAM,CAAC,IAAI;4BACb,CAAC,CAAC,KAAK,CAAC,GAAG,CAAC,UAAU;wBAC1B,UAAU,EACR,MAAM,CAAC,UAAU,KAAK,QAAQ;4BAC5B,CAAC,CAAC,MAAM,CAAC,IAAI;4BACb,CAAC,CAAC,KAAK,CAAC,GAAG,CAAC,UAAU;qBAC3B;iBACF,CAAC;SACL;IACH,CAAC;IAED,eAAe;IACf,eAAe;IACf,eAAe;IAEf,SAAgB,kBAAkB,CAAC,UAA0B;QAC3D,QAAQ,UAAU,EAAE;YAClB,KAAK,SAAS;gBACZ,OAAO,SAAS,CAAC;YACnB,KAAK,QAAQ;gBACX,OAAO,QAAQ,CAAC;YAClB,KAAK,QAAQ;gBACX,OAAO,QAAQ,CAAC;SACnB;IACH,CAAC;IATe,2BAAkB,qBASjC,CAAA;IAED,SAAgB,wBAAwB,CAAC,UAA0B;QACjE,QAAQ,UAAU,EAAE;YAClB,KAAK,SAAS;gBACZ,OAAO,qBAAqB,CAAC;YAC/B,KAAK,QAAQ;gBACX,OAAO,mCAAmC,CAAC;YAC7C,KAAK,QAAQ;gBACX,OAAO,qBAAqB,CAAC;SAChC;IACH,CAAC;IATe,iCAAwB,2BASvC,CAAA;AACH,CAAC,EAnfgB,QAAQ,KAAR,QAAQ,QAmfxB","sourcesContent":["import Fuse, { FuseResult, IFuseOptions } from 'fuse.js';\nimport { defaultsDeep, sortBy } from 'lodash';\nimport {\n  connectable,\n  Connectable,\n  from,\n  map,\n  mergeScan,\n  Observable,\n  ReplaySubject,\n  startWith,\n  Subject,\n} from 'rxjs';\nimport { DeepPartial } from 'utility-types';\nimport { HighlightComponent } from '../../visualization/highlight/highlight.component';\n\nexport namespace RivViews {\n  export const ViewPermissions = ['private', 'public', 'shared'] as const;\n  export type ViewPermission = (typeof ViewPermissions)[number];\n\n  export type View = {\n    id: string | number;\n    title: string;\n    enabled: boolean;\n    permission: ViewPermission;\n  };\n\n  export type FullView<V extends View = View> = V & {\n    active: boolean;\n    default: boolean;\n    titleHighlightIndices?: HighlightComponent.HighlightIndices[];\n  };\n\n  ///////////\n  // State //\n  ///////////\n\n  export type CoreState<V extends View> = {\n    views: V[];\n    activeViewId: V['id'] | null;\n    defaultViewId: V['id'] | null;\n    containerWidth: number;\n    overflow: {\n      search: string;\n      pickerOpen: boolean;\n    };\n    all: {\n      search: string;\n      privateOpen: boolean;\n      sharedOpen: boolean;\n      publicOpen: boolean;\n    };\n  };\n\n  export type FullState<V extends View> = CoreState<V> & {\n    allViews: FullView<V>[];\n    displayedViews: FullView<V>[];\n    enabledViews: FullView<V>[];\n    all: {\n      privateViews: FullView<V>[];\n      privateEmptyStateMessage: string;\n      sharedViews: FullView<V>[];\n      sharedEmptyStateMessage: string;\n      publicViews: FullView<V>[];\n      publicEmptyStateMessage: string;\n    };\n  };\n\n  /////////////\n  // Actions //\n  /////////////\n\n  export type Action<V extends View> =\n    | { type: 'load' }\n    | { type: 'setActiveView'; id: V['id'] }\n    | { type: 'setDefaultView'; id: V['id'] }\n    | { type: 'setEnabled'; id: V['id']; enabled: boolean }\n    | { type: 'setContainerWidth'; width: number }\n    | { type: 'setOverflowPickerOpen'; open: boolean }\n    | { type: 'setOverflowSearchQuery'; query: string }\n    | { type: 'setAllViewsSearchQuery'; query: string }\n    | { type: 'updateView'; view: V }\n    | { type: 'deleteView'; view: V }\n    | { type: 'duplicateView'; view: V }\n    | { type: 'copyLinkToView'; view: V }\n    | { type: 'setAllViewsToggle'; permission: ViewPermission; open: boolean };\n\n  ////////////\n  // Source //\n  ////////////\n\n  export type Source<V extends View> = () => V[] | Promise<V[]>;\n  export type Duplicate<V extends View> = (\n    viewToDuplicate: V,\n  ) => V | Promise<V>;\n  export type CopyLink<V extends View> = (view: V) => void | Promise<void>;\n\n  /////////////\n  // Manager //\n  /////////////\n\n  export type Manager<V extends View> = {\n    actions: Subject<Action<V>>;\n    state: Connectable<FullState<V>>;\n  };\n\n  export type ManagerOptions<V extends View> = {\n    initialState?: DeepPartial<CoreState<V>>;\n    fuseOptions?: IFuseOptions<V>;\n  };\n\n  export function createManager<V extends View>(\n    source: Source<V>,\n    duplicate: Duplicate<V>,\n    copyLink: CopyLink<V>,\n    options?: ManagerOptions<V>,\n  ): Manager<V> {\n    const defaultState: CoreState<V> = {\n      views: [],\n      activeViewId: null,\n      defaultViewId: null,\n      containerWidth: 0,\n      overflow: {\n        search: '',\n        pickerOpen: false,\n      },\n      all: {\n        search: '',\n        privateOpen: true,\n        sharedOpen: true,\n        publicOpen: true,\n      },\n    };\n\n    const defaultedInitialState: CoreState<V> = defaultsDeep(\n      {},\n      options?.initialState,\n      defaultState,\n    );\n\n    const actions: Subject<Action<V>> = new Subject();\n    const coreState: Observable<CoreState<V>> = actions.pipe(\n      mergeScan(\n        (state, action) =>\n          from(reduce(state, action, source, duplicate, copyLink)),\n        defaultedInitialState,\n        1,\n      ),\n      startWith(defaultedInitialState),\n    );\n\n    const state: Connectable<FullState<V>> = connectable(\n      coreState.pipe(\n        map(s => {\n          // If for some reason the active view is not currently enabled, we'll\n          // default to making the first view active instead.\n          const activeView: V | undefined = s.views.find(\n            view => view.enabled && view.id === s.activeViewId,\n          );\n\n          const allViews: FullView<V>[] = s.views.map((view, i) => ({\n            ...view,\n            active: activeView ? activeView.id === view.id : i === 0,\n            default: view.id === s.defaultViewId,\n          }));\n\n          const allEnabledViews: FullView<V>[] = allViews.filter(\n            view => view.enabled,\n          );\n\n          const displayedViews = calculateDisplayedViews(\n            allEnabledViews,\n            calculateOverflowViewIndex(s),\n            s.activeViewId,\n          );\n\n          const enabledViews = s.overflow.search\n            ? searchViews(\n                allEnabledViews,\n                s.overflow.search,\n                options?.fuseOptions,\n              )\n            : allEnabledViews;\n\n          const allViewsFiltered = s.all.search\n            ? searchViews(allViews, s.all.search, options?.fuseOptions)\n            : allViews;\n\n          const searchingAllViews = !!s.all.search;\n\n          return {\n            ...s,\n            allViews,\n            displayedViews,\n            enabledViews,\n            all: {\n              ...s.all,\n              privateOpen: searchingAllViews || s.all.privateOpen,\n              privateViews: viewsByPermission(allViewsFiltered, 'private'),\n              privateEmptyStateMessage: getEmptyStateMessage(s, 'private'),\n              sharedOpen: searchingAllViews || s.all.sharedOpen,\n              sharedViews: viewsByPermission(allViewsFiltered, 'shared'),\n              sharedEmptyStateMessage: getEmptyStateMessage(s, 'shared'),\n              publicOpen: searchingAllViews || s.all.publicOpen,\n              publicViews: viewsByPermission(allViewsFiltered, 'public'),\n              publicEmptyStateMessage: getEmptyStateMessage(s, 'public'),\n            },\n          };\n        }),\n      ),\n      {\n        connector: () => new ReplaySubject(1),\n        resetOnDisconnect: false,\n      },\n    );\n    state.connect();\n\n    // Load the views immediately\n    actions.next({ type: 'load' });\n\n    return { actions, state };\n  }\n\n  //////////////////////////////////\n  // State reducer (with helpers) //\n  //////////////////////////////////\n\n  function calculateOverflowViewIndex<V extends View>(\n    state: CoreState<V>,\n  ): number {\n    const enabledViews = state.views.filter(view => view.enabled);\n\n    // Tabs have a minimum width of 92px.\n    const possibleSlotCount = Math.floor(state.containerWidth / 92);\n\n    // The \"all views\" sidebar trigger takes up one slot.\n    const availableSlotCount = possibleSlotCount - 1;\n\n    // If there are too many tabs, the \"N more\" trigger takes up one slot as well.\n    // But we want to avoid the \"1 more\" case, so we only use a slot for \"N more\" if N != 1.\n    if (availableSlotCount >= enabledViews.length) return enabledViews.length;\n    else return availableSlotCount - 1; // the \"N more\" uses up a slot.\n  }\n\n  function calculateDisplayedViews<V extends View>(\n    fullViews: FullView<V>[],\n    overflowViewIndex: number,\n    activeViewId: CoreState<V>['activeViewId'],\n  ): FullView<V>[] {\n    const displayedViews = fullViews.slice(0, overflowViewIndex);\n\n    const activeViewInDisplay = displayedViews.some(\n      view => view.id === activeViewId,\n    );\n\n    if (!activeViewInDisplay && activeViewId !== null) {\n      const activeView = fullViews.find(view => view.id === activeViewId);\n      if (activeView && displayedViews.length > 0) {\n        return [...displayedViews.slice(0, -1), activeView];\n      }\n    }\n\n    return displayedViews;\n  }\n\n  function viewsByPermission<V extends View>(\n    views: FullView<V>[],\n    permission: ViewPermission,\n  ): FullView<V>[] {\n    return sortBy(\n      views.filter(view => view.permission === permission),\n      'title',\n    );\n  }\n\n  function searchViews<V extends View>(\n    views: FullView<V>[],\n    query: string,\n    options?: IFuseOptions<V>,\n  ): FullView<V>[] {\n    const fuse = new Fuse(views, {\n      threshold: 0.2,\n      keys: ['title'],\n      includeMatches: true,\n      ...options,\n    });\n\n    const matchedOptions: FuseResult<FullView<V>>[] = fuse.search(query);\n\n    return matchedOptions.map(result => ({\n      ...result.item,\n      titleHighlightIndices: [\n        ...(result.matches?.find(match => match.key === 'title')?.indices ??\n          []),\n      ],\n    }));\n  }\n\n  function getNextBestActiveView<V extends View>(\n    state: CoreState<V>,\n  ): V['id'] | null {\n    const enabledViews = state.views.filter(view => view.enabled);\n    const currentActiveViewIndex = enabledViews.findIndex(\n      view => view.id === state.activeViewId,\n    );\n\n    if (currentActiveViewIndex === -1) return enabledViews[0]?.id ?? null;\n    return (\n      enabledViews[currentActiveViewIndex + 1]?.id ??\n      enabledViews[currentActiveViewIndex - 1]?.id ??\n      null\n    );\n  }\n\n  function deleteView<V extends View>(\n    state: CoreState<V>,\n    id: V['id'],\n  ): CoreState<V> {\n    const activeViewId =\n      state.activeViewId === id\n        ? getNextBestActiveView(state)\n        : state.activeViewId;\n    const views = state.views.filter(view => view.id !== id);\n    return {\n      ...state,\n      views,\n      activeViewId,\n    };\n  }\n\n  function setEnabled<V extends View>(\n    state: CoreState<V>,\n    id: V['id'],\n    enabled: boolean,\n  ): CoreState<V> {\n    return {\n      ...state,\n      views: state.views.map(view =>\n        view.id === id ? { ...view, enabled } : view,\n      ),\n      activeViewId:\n        !enabled && state.activeViewId === id\n          ? getNextBestActiveView(state)\n          : state.activeViewId,\n    };\n  }\n\n  async function withDuplicateView<V extends View>(\n    views: V[],\n    viewToDuplicate: V,\n    duplicate: Duplicate<V>,\n  ): Promise<V[]> {\n    const newView = await duplicate(viewToDuplicate);\n    const sourceIndex = views.findIndex(view => view.id === viewToDuplicate.id);\n    if (sourceIndex === -1) {\n      return [newView, ...views];\n    } else {\n      return [\n        ...views.slice(0, sourceIndex + 1),\n        newView,\n        ...views.slice(sourceIndex + 1),\n      ];\n    }\n  }\n\n  function getEmptyStateMessage<V extends View>(\n    state: CoreState<V>,\n    permission: ViewPermission,\n  ): string {\n    if (state.all.search)\n      return `No ${permission} views matching \"${state.all.search}\"`;\n    else return `No ${permission} views`;\n  }\n\n  async function reduce<V extends View>(\n    state: CoreState<V>,\n    action: Action<V>,\n    source: Source<V>,\n    duplicate: Duplicate<V>,\n    copyLink: CopyLink<V>,\n  ): Promise<CoreState<V>> {\n    switch (action.type) {\n      case 'load':\n        return {\n          ...state,\n          views: await source(),\n        };\n\n      case 'setActiveView':\n        return {\n          ...state,\n          activeViewId: action.id,\n          views: state.views.map(view =>\n            view.id === action.id ? { ...view, enabled: true } : view,\n          ),\n          all: {\n            ...state.all,\n            search: '',\n          },\n          overflow: {\n            ...state.overflow,\n            search: '',\n          },\n        };\n\n      case 'setDefaultView':\n        return {\n          ...state,\n          defaultViewId: action.id,\n        };\n\n      case 'setEnabled':\n        return setEnabled(state, action.id, action.enabled);\n\n      case 'setContainerWidth':\n        return {\n          ...state,\n          containerWidth: action.width,\n        };\n\n      case 'setOverflowPickerOpen':\n        return {\n          ...state,\n          overflow: {\n            ...state.overflow,\n            pickerOpen: action.open,\n          },\n        };\n\n      case 'setOverflowSearchQuery':\n        return {\n          ...state,\n          overflow: {\n            ...state.overflow,\n            search: action.query,\n          },\n        };\n\n      case 'setAllViewsSearchQuery':\n        return {\n          ...state,\n          all: {\n            ...state.all,\n            search: action.query,\n          },\n        };\n\n      case 'updateView':\n        return {\n          ...state,\n          views: state.views.map(view =>\n            view.id === action.view.id ? action.view : view,\n          ),\n        };\n\n      case 'deleteView':\n        return deleteView(state, action.view.id);\n\n      case 'duplicateView':\n        return {\n          ...state,\n          views: await withDuplicateView(state.views, action.view, duplicate),\n        };\n\n      case 'copyLinkToView':\n        await copyLink(action.view);\n        return state;\n\n      case 'setAllViewsToggle':\n        return {\n          ...state,\n          all: {\n            ...state.all,\n            privateOpen:\n              action.permission === 'private'\n                ? action.open\n                : state.all.privateOpen,\n            sharedOpen:\n              action.permission === 'shared'\n                ? action.open\n                : state.all.sharedOpen,\n            publicOpen:\n              action.permission === 'public'\n                ? action.open\n                : state.all.publicOpen,\n          },\n        };\n    }\n  }\n\n  ///////////////\n  // Utilities //\n  ///////////////\n\n  export function getPermissionTitle(permission: ViewPermission): string {\n    switch (permission) {\n      case 'private':\n        return 'Private';\n      case 'shared':\n        return 'Shared';\n      case 'public':\n        return 'Public';\n    }\n  }\n\n  export function getPermissionDescription(permission: ViewPermission): string {\n    switch (permission) {\n      case 'private':\n        return 'Only visible to you';\n      case 'shared':\n        return 'Only visible to those with access';\n      case 'public':\n        return 'Visible to everyone';\n    }\n  }\n}\n"]}
|
|
@@ -0,0 +1,30 @@
|
|
|
1
|
+
import { ChangeDetectionStrategy, Component, ContentChild, ElementRef, Input, ViewChild, } from '@angular/core';
|
|
2
|
+
import * as i0 from "@angular/core";
|
|
3
|
+
import * as i1 from "@angular/common";
|
|
4
|
+
import * as i2 from "../edit-view/edit-view.component";
|
|
5
|
+
import * as i3 from "../../../modal/menu/menu.component";
|
|
6
|
+
import * as i4 from "../../../modal/menu/menu-divider/menu-divider.component";
|
|
7
|
+
import * as i5 from "../../../modal/menu/menu-item/menu-item.component";
|
|
8
|
+
export class ViewMenuComponent {
|
|
9
|
+
constructor() {
|
|
10
|
+
this.editingView = null;
|
|
11
|
+
this.renamingView = false;
|
|
12
|
+
}
|
|
13
|
+
}
|
|
14
|
+
ViewMenuComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: ViewMenuComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
15
|
+
ViewMenuComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: ViewMenuComponent, selector: "riv-view-menu", inputs: { manager: "manager", view: "view" }, queries: [{ propertyName: "triggerTemplate", first: true, predicate: ["menuTrigger"], descendants: true }], viewQueries: [{ propertyName: "menuElement", first: true, predicate: ["menu"], descendants: true, read: ElementRef }], ngImport: i0, template: "<ng-container *ngIf=\"view\">\n <riv-menu\n #menu\n [preferredPosition]=\"'bottom-right'\"\n (click)=\"$event.stopPropagation()\"\n >\n <ng-container *ngIf=\"triggerTemplate\">\n <ng-template #trigger>\n <ng-container [ngTemplateOutlet]=\"triggerTemplate\"></ng-container>\n </ng-template>\n </ng-container>\n <button\n riv-menu-item\n (click)=\"\n manager?.actions?.next({\n type: 'setEnabled',\n id: view.id,\n enabled: !view.enabled\n })\n \"\n >\n {{ view.enabled ? 'Hide tab' : 'Show tab' }}\n </button>\n <riv-menu-divider></riv-menu-divider>\n <button riv-menu-item (click)=\"editingView = view\">Edit view</button>\n <button riv-menu-item (click)=\"editingView = view; renamingView = true\">\n Rename view\n </button>\n <button\n riv-menu-item\n (click)=\"manager?.actions?.next({ type: 'duplicateView', view })\"\n >\n Duplicate view\n </button>\n <button\n riv-menu-item\n [variant]=\"'danger'\"\n (click)=\"manager?.actions?.next({ type: 'deleteView', view })\"\n >\n Delete view\n </button>\n <riv-menu-divider></riv-menu-divider>\n <button\n riv-menu-item\n (click)=\"manager?.actions?.next({ type: 'copyLinkToView', view })\"\n >\n Copy link to view\n </button>\n </riv-menu>\n\n <riv-edit-view\n *ngIf=\"editingView\"\n [view]=\"editingView\"\n [autoSelectName]=\"renamingView\"\n [anchor]=\"menuElement?.nativeElement\"\n (save)=\"manager?.actions?.next({ type: 'updateView', view: $event })\"\n (delete)=\"manager?.actions?.next({ type: 'deleteView', view: $event })\"\n (close)=\"editingView = null; renamingView = false\"\n ></riv-edit-view>\n</ng-container>\n", styles: [":host{display:inline-flex;justify-content:center;align-items:center}riv-menu{max-width:100%}\n"], dependencies: [{ kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "component", type: i2.EditViewComponent, selector: "riv-edit-view", inputs: ["anchor", "autoSelectName", "view"], outputs: ["save", "delete", "close"] }, { kind: "component", type: i3.MenuComponent, selector: "riv-menu", inputs: ["preferredPosition"] }, { kind: "component", type: i4.MenuDividerComponent, selector: "riv-menu-divider" }, { kind: "component", type: i5.MenuItemComponent, selector: "[riv-menu-item]", inputs: ["locked", "disabled", "variant"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
16
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: ViewMenuComponent, decorators: [{
|
|
17
|
+
type: Component,
|
|
18
|
+
args: [{ selector: 'riv-view-menu', changeDetection: ChangeDetectionStrategy.OnPush, template: "<ng-container *ngIf=\"view\">\n <riv-menu\n #menu\n [preferredPosition]=\"'bottom-right'\"\n (click)=\"$event.stopPropagation()\"\n >\n <ng-container *ngIf=\"triggerTemplate\">\n <ng-template #trigger>\n <ng-container [ngTemplateOutlet]=\"triggerTemplate\"></ng-container>\n </ng-template>\n </ng-container>\n <button\n riv-menu-item\n (click)=\"\n manager?.actions?.next({\n type: 'setEnabled',\n id: view.id,\n enabled: !view.enabled\n })\n \"\n >\n {{ view.enabled ? 'Hide tab' : 'Show tab' }}\n </button>\n <riv-menu-divider></riv-menu-divider>\n <button riv-menu-item (click)=\"editingView = view\">Edit view</button>\n <button riv-menu-item (click)=\"editingView = view; renamingView = true\">\n Rename view\n </button>\n <button\n riv-menu-item\n (click)=\"manager?.actions?.next({ type: 'duplicateView', view })\"\n >\n Duplicate view\n </button>\n <button\n riv-menu-item\n [variant]=\"'danger'\"\n (click)=\"manager?.actions?.next({ type: 'deleteView', view })\"\n >\n Delete view\n </button>\n <riv-menu-divider></riv-menu-divider>\n <button\n riv-menu-item\n (click)=\"manager?.actions?.next({ type: 'copyLinkToView', view })\"\n >\n Copy link to view\n </button>\n </riv-menu>\n\n <riv-edit-view\n *ngIf=\"editingView\"\n [view]=\"editingView\"\n [autoSelectName]=\"renamingView\"\n [anchor]=\"menuElement?.nativeElement\"\n (save)=\"manager?.actions?.next({ type: 'updateView', view: $event })\"\n (delete)=\"manager?.actions?.next({ type: 'deleteView', view: $event })\"\n (close)=\"editingView = null; renamingView = false\"\n ></riv-edit-view>\n</ng-container>\n", styles: [":host{display:inline-flex;justify-content:center;align-items:center}riv-menu{max-width:100%}\n"] }]
|
|
19
|
+
}], propDecorators: { manager: [{
|
|
20
|
+
type: Input
|
|
21
|
+
}], view: [{
|
|
22
|
+
type: Input
|
|
23
|
+
}], menuElement: [{
|
|
24
|
+
type: ViewChild,
|
|
25
|
+
args: ['menu', { read: ElementRef }]
|
|
26
|
+
}], triggerTemplate: [{
|
|
27
|
+
type: ContentChild,
|
|
28
|
+
args: ['menuTrigger']
|
|
29
|
+
}] } });
|
|
30
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidmlldy1tZW51LmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL3Jpdi9zcmMvbGliL25hdmlnYXRpb24vdmlld3Mvdmlldy1tZW51L3ZpZXctbWVudS5jb21wb25lbnQudHMiLCIuLi8uLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9yaXYvc3JjL2xpYi9uYXZpZ2F0aW9uL3ZpZXdzL3ZpZXctbWVudS92aWV3LW1lbnUuY29tcG9uZW50Lmh0bWwiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUNMLHVCQUF1QixFQUN2QixTQUFTLEVBQ1QsWUFBWSxFQUNaLFVBQVUsRUFDVixLQUFLLEVBRUwsU0FBUyxHQUNWLE1BQU0sZUFBZSxDQUFDOzs7Ozs7O0FBU3ZCLE1BQU0sT0FBTyxpQkFBaUI7SUFOOUI7UUFtQlMsZ0JBQVcsR0FBZ0MsSUFBSSxDQUFDO1FBQ2hELGlCQUFZLEdBQVksS0FBSyxDQUFDO0tBQ3RDOzs4R0FmWSxpQkFBaUI7a0dBQWpCLGlCQUFpQiwrUkFPRCxVQUFVLDZCQ3hCdkMsOHdEQTREQTsyRkQzQ2EsaUJBQWlCO2tCQU43QixTQUFTOytCQUNFLGVBQWUsbUJBR1IsdUJBQXVCLENBQUMsTUFBTTs4QkFJL0MsT0FBTztzQkFETixLQUFLO2dCQUlOLElBQUk7c0JBREgsS0FBSztnQkFJTixXQUFXO3NCQURWLFNBQVM7dUJBQUMsTUFBTSxFQUFFLEVBQUUsSUFBSSxFQUFFLFVBQVUsRUFBRTtnQkFJdkMsZUFBZTtzQkFEZCxZQUFZO3VCQUFDLGFBQWEiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQge1xuICBDaGFuZ2VEZXRlY3Rpb25TdHJhdGVneSxcbiAgQ29tcG9uZW50LFxuICBDb250ZW50Q2hpbGQsXG4gIEVsZW1lbnRSZWYsXG4gIElucHV0LFxuICBUZW1wbGF0ZVJlZixcbiAgVmlld0NoaWxkLFxufSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IFJpdlZpZXdzIH0gZnJvbSAnLi4vc3RhdGUnO1xuXG5AQ29tcG9uZW50KHtcbiAgc2VsZWN0b3I6ICdyaXYtdmlldy1tZW51JyxcbiAgdGVtcGxhdGVVcmw6ICcuL3ZpZXctbWVudS5jb21wb25lbnQuaHRtbCcsXG4gIHN0eWxlVXJsczogWycuL3ZpZXctbWVudS5jb21wb25lbnQuY3NzJ10sXG4gIGNoYW5nZURldGVjdGlvbjogQ2hhbmdlRGV0ZWN0aW9uU3RyYXRlZ3kuT25QdXNoLFxufSlcbmV4cG9ydCBjbGFzcyBWaWV3TWVudUNvbXBvbmVudDxWIGV4dGVuZHMgUml2Vmlld3MuVmlldz4ge1xuICBASW5wdXQoKVxuICBtYW5hZ2VyPzogUml2Vmlld3MuTWFuYWdlcjxWPjtcblxuICBASW5wdXQoKVxuICB2aWV3PzogUml2Vmlld3MuRnVsbFZpZXc8Vj47XG5cbiAgQFZpZXdDaGlsZCgnbWVudScsIHsgcmVhZDogRWxlbWVudFJlZiB9KVxuICBtZW51RWxlbWVudD86IEVsZW1lbnRSZWY7XG5cbiAgQENvbnRlbnRDaGlsZCgnbWVudVRyaWdnZXInKVxuICB0cmlnZ2VyVGVtcGxhdGU/OiBUZW1wbGF0ZVJlZjx2b2lkPjtcblxuICBwdWJsaWMgZWRpdGluZ1ZpZXc6IFJpdlZpZXdzLkZ1bGxWaWV3PFY+IHwgbnVsbCA9IG51bGw7XG4gIHB1YmxpYyByZW5hbWluZ1ZpZXc6IGJvb2xlYW4gPSBmYWxzZTtcbn1cbiIsIjxuZy1jb250YWluZXIgKm5nSWY9XCJ2aWV3XCI+XG4gIDxyaXYtbWVudVxuICAgICNtZW51XG4gICAgW3ByZWZlcnJlZFBvc2l0aW9uXT1cIidib3R0b20tcmlnaHQnXCJcbiAgICAoY2xpY2spPVwiJGV2ZW50LnN0b3BQcm9wYWdhdGlvbigpXCJcbiAgPlxuICAgIDxuZy1jb250YWluZXIgKm5nSWY9XCJ0cmlnZ2VyVGVtcGxhdGVcIj5cbiAgICAgIDxuZy10ZW1wbGF0ZSAjdHJpZ2dlcj5cbiAgICAgICAgPG5nLWNvbnRhaW5lciBbbmdUZW1wbGF0ZU91dGxldF09XCJ0cmlnZ2VyVGVtcGxhdGVcIj48L25nLWNvbnRhaW5lcj5cbiAgICAgIDwvbmctdGVtcGxhdGU+XG4gICAgPC9uZy1jb250YWluZXI+XG4gICAgPGJ1dHRvblxuICAgICAgcml2LW1lbnUtaXRlbVxuICAgICAgKGNsaWNrKT1cIlxuICAgICAgICBtYW5hZ2VyPy5hY3Rpb25zPy5uZXh0KHtcbiAgICAgICAgICB0eXBlOiAnc2V0RW5hYmxlZCcsXG4gICAgICAgICAgaWQ6IHZpZXcuaWQsXG4gICAgICAgICAgZW5hYmxlZDogIXZpZXcuZW5hYmxlZFxuICAgICAgICB9KVxuICAgICAgXCJcbiAgICA+XG4gICAgICB7eyB2aWV3LmVuYWJsZWQgPyAnSGlkZSB0YWInIDogJ1Nob3cgdGFiJyB9fVxuICAgIDwvYnV0dG9uPlxuICAgIDxyaXYtbWVudS1kaXZpZGVyPjwvcml2LW1lbnUtZGl2aWRlcj5cbiAgICA8YnV0dG9uIHJpdi1tZW51LWl0ZW0gKGNsaWNrKT1cImVkaXRpbmdWaWV3ID0gdmlld1wiPkVkaXQgdmlldzwvYnV0dG9uPlxuICAgIDxidXR0b24gcml2LW1lbnUtaXRlbSAoY2xpY2spPVwiZWRpdGluZ1ZpZXcgPSB2aWV3OyByZW5hbWluZ1ZpZXcgPSB0cnVlXCI+XG4gICAgICBSZW5hbWUgdmlld1xuICAgIDwvYnV0dG9uPlxuICAgIDxidXR0b25cbiAgICAgIHJpdi1tZW51LWl0ZW1cbiAgICAgIChjbGljayk9XCJtYW5hZ2VyPy5hY3Rpb25zPy5uZXh0KHsgdHlwZTogJ2R1cGxpY2F0ZVZpZXcnLCB2aWV3IH0pXCJcbiAgICA+XG4gICAgICBEdXBsaWNhdGUgdmlld1xuICAgIDwvYnV0dG9uPlxuICAgIDxidXR0b25cbiAgICAgIHJpdi1tZW51LWl0ZW1cbiAgICAgIFt2YXJpYW50XT1cIidkYW5nZXInXCJcbiAgICAgIChjbGljayk9XCJtYW5hZ2VyPy5hY3Rpb25zPy5uZXh0KHsgdHlwZTogJ2RlbGV0ZVZpZXcnLCB2aWV3IH0pXCJcbiAgICA+XG4gICAgICBEZWxldGUgdmlld1xuICAgIDwvYnV0dG9uPlxuICAgIDxyaXYtbWVudS1kaXZpZGVyPjwvcml2LW1lbnUtZGl2aWRlcj5cbiAgICA8YnV0dG9uXG4gICAgICByaXYtbWVudS1pdGVtXG4gICAgICAoY2xpY2spPVwibWFuYWdlcj8uYWN0aW9ucz8ubmV4dCh7IHR5cGU6ICdjb3B5TGlua1RvVmlldycsIHZpZXcgfSlcIlxuICAgID5cbiAgICAgIENvcHkgbGluayB0byB2aWV3XG4gICAgPC9idXR0b24+XG4gIDwvcml2LW1lbnU+XG5cbiAgPHJpdi1lZGl0LXZpZXdcbiAgICAqbmdJZj1cImVkaXRpbmdWaWV3XCJcbiAgICBbdmlld109XCJlZGl0aW5nVmlld1wiXG4gICAgW2F1dG9TZWxlY3ROYW1lXT1cInJlbmFtaW5nVmlld1wiXG4gICAgW2FuY2hvcl09XCJtZW51RWxlbWVudD8ubmF0aXZlRWxlbWVudFwiXG4gICAgKHNhdmUpPVwibWFuYWdlcj8uYWN0aW9ucz8ubmV4dCh7IHR5cGU6ICd1cGRhdGVWaWV3JywgdmlldzogJGV2ZW50IH0pXCJcbiAgICAoZGVsZXRlKT1cIm1hbmFnZXI/LmFjdGlvbnM/Lm5leHQoeyB0eXBlOiAnZGVsZXRlVmlldycsIHZpZXc6ICRldmVudCB9KVwiXG4gICAgKGNsb3NlKT1cImVkaXRpbmdWaWV3ID0gbnVsbDsgcmVuYW1pbmdWaWV3ID0gZmFsc2VcIlxuICA+PC9yaXYtZWRpdC12aWV3PlxuPC9uZy1jb250YWluZXI+XG4iXX0=
|