@dereekb/dbx-web 12.5.9 → 12.6.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.
Files changed (53) hide show
  1. package/esm2022/lib/extension/download/blob/download.blob.button.component.mjs +86 -0
  2. package/esm2022/lib/extension/download/blob/index.mjs +2 -0
  3. package/esm2022/lib/extension/download/index.mjs +2 -1
  4. package/esm2022/lib/extension/download/text/download.text.component.mjs +20 -18
  5. package/esm2022/lib/extension/index.mjs +3 -1
  6. package/esm2022/lib/extension/model/model.tracker.mjs +1 -1
  7. package/esm2022/lib/extension/model/model.tracker.view.storage.mjs +3 -3
  8. package/esm2022/lib/extension/preview/index.mjs +4 -0
  9. package/esm2022/lib/extension/preview/webfilepreview.mjs +2 -0
  10. package/esm2022/lib/extension/preview/webfilepreview.service.mjs +60 -0
  11. package/esm2022/lib/extension/preview/webfilepreview.service.preset.mjs +31 -0
  12. package/esm2022/lib/extension/widget/widget.service.mjs +1 -1
  13. package/esm2022/lib/extension/zip/index.mjs +5 -0
  14. package/esm2022/lib/extension/zip/zip.blob.mjs +16 -0
  15. package/esm2022/lib/extension/zip/zip.blob.preview.component.mjs +156 -0
  16. package/esm2022/lib/extension/zip/zip.blob.preview.list.component.mjs +121 -0
  17. package/esm2022/lib/extension/zip/zip.preview.component.mjs +71 -0
  18. package/esm2022/lib/extension/zip/zip.preview.dialog.component.mjs +44 -0
  19. package/esm2022/lib/interaction/iframe/embed.component.mjs +104 -0
  20. package/esm2022/lib/interaction/iframe/embed.dialog.component.mjs +43 -0
  21. package/esm2022/lib/interaction/iframe/iframe.component.mjs +7 -4
  22. package/esm2022/lib/interaction/iframe/iframe.dialog.component.mjs +37 -0
  23. package/esm2022/lib/interaction/iframe/index.mjs +4 -1
  24. package/fesm2022/dereekb-dbx-web.mjs +4596 -3899
  25. package/fesm2022/dereekb-dbx-web.mjs.map +1 -1
  26. package/lib/extension/_extension.scss +19 -0
  27. package/lib/extension/download/blob/download.blob.button.component.d.ts +55 -0
  28. package/lib/extension/download/blob/index.d.ts +1 -0
  29. package/lib/extension/download/index.d.ts +1 -0
  30. package/lib/extension/download/text/download.text.component.d.ts +7 -7
  31. package/lib/extension/index.d.ts +2 -0
  32. package/lib/extension/model/model.tracker.d.ts +3 -3
  33. package/lib/extension/preview/index.d.ts +3 -0
  34. package/lib/extension/preview/webfilepreview.d.ts +13 -0
  35. package/lib/extension/preview/webfilepreview.service.d.ts +31 -0
  36. package/lib/extension/preview/webfilepreview.service.preset.d.ts +13 -0
  37. package/lib/extension/widget/widget.service.d.ts +1 -1
  38. package/lib/extension/zip/_zip.scss +35 -0
  39. package/lib/extension/zip/index.d.ts +4 -0
  40. package/lib/extension/zip/zip.blob.d.ts +9 -0
  41. package/lib/extension/zip/zip.blob.preview.component.d.ts +54 -0
  42. package/lib/extension/zip/zip.blob.preview.list.component.d.ts +25 -0
  43. package/lib/extension/zip/zip.preview.component.d.ts +31 -0
  44. package/lib/extension/zip/zip.preview.dialog.component.d.ts +17 -0
  45. package/lib/interaction/iframe/_iframe.scss +13 -0
  46. package/lib/interaction/iframe/embed.component.d.ts +31 -0
  47. package/lib/interaction/iframe/embed.dialog.component.d.ts +17 -0
  48. package/lib/interaction/iframe/iframe.component.d.ts +0 -1
  49. package/lib/interaction/iframe/iframe.dialog.component.d.ts +13 -0
  50. package/lib/interaction/iframe/index.d.ts +3 -0
  51. package/lib/layout/style/_style.scss +2 -1
  52. package/lib/style/_config.scss +7 -1
  53. package/package.json +3 -2
@@ -0,0 +1,156 @@
1
+ import { ChangeDetectionStrategy, Component, computed, input, signal } from '@angular/core';
2
+ import { DbxInjectionComponent } from '@dereekb/dbx-core';
3
+ import { JsonPipe, NgTemplateOutlet } from '@angular/common';
4
+ import { ZipReader, BlobReader } from '@zip.js/zip.js';
5
+ import { toObservable, toSignal } from '@angular/core/rxjs-interop';
6
+ import { DbxLoadingComponent } from '../../loading';
7
+ import { distinctUntilChanged, map, of, shareReplay, switchMap } from 'rxjs';
8
+ import { loadingStateContext, loadingStateFromObs, valueFromFinishedLoadingState } from '@dereekb/rxjs';
9
+ import { dbxZipBlobPreviewEntryTreeFromEntries } from './zip.blob';
10
+ import { DbxZipPreviewEntryListComponent } from './zip.blob.preview.list.component';
11
+ import { DbxBarHeaderComponent, DbxListEmptyContentComponent, DbxListTitleGroupDirective, DbxSpacerDirective, DbxValueListItemModifierDirective } from '../../layout';
12
+ import { DbxListItemAnchorModifierDirective } from '../../router';
13
+ import { DbxEmbedComponent } from '../../interaction';
14
+ import { MatToolbarModule } from '@angular/material/toolbar';
15
+ import { DbxButtonSpacerDirective, DbxIconButtonComponent } from '../../button';
16
+ import { DbxDownloadBlobButtonComponent } from '../download/blob/download.blob.button.component';
17
+ import * as i0 from "@angular/core";
18
+ /**
19
+ * Used to display a zip preview based on the input blob.
20
+ */
21
+ export class DbxZipBlobPreviewComponent {
22
+ blob = input();
23
+ /**
24
+ * The download file name for the zip file.
25
+ *
26
+ * If not defined, then the file cannot be downloaded directly.
27
+ */
28
+ downloadFileName = input();
29
+ hasBlob = computed(() => !!this.blob());
30
+ zipReader = computed(() => {
31
+ const blob = this.blob();
32
+ return blob ? new ZipReader(new BlobReader(blob)) : undefined;
33
+ });
34
+ downloadZipFileBlobButtonConfigSignal = computed(() => {
35
+ const blob = this.blob();
36
+ const fileName = this.downloadFileName();
37
+ let result = undefined;
38
+ if (blob && fileName) {
39
+ result = {
40
+ blob,
41
+ fileName: fileName ?? 'download.zip',
42
+ buttonDisplay: {
43
+ icon: 'download_for_offline',
44
+ text: 'Download Zip'
45
+ },
46
+ buttonStyle: {
47
+ type: 'stroked'
48
+ }
49
+ };
50
+ }
51
+ return result;
52
+ });
53
+ zipReader$ = toObservable(this.zipReader);
54
+ allEntriesLoadingState$ = loadingStateFromObs(this.zipReader$.pipe(switchMap((x) => (x ? x.getEntries() : of([])))));
55
+ allEntries$ = this.allEntriesLoadingState$.pipe(valueFromFinishedLoadingState(), distinctUntilChanged(), shareReplay(1));
56
+ allEntriesRoot$ = this.allEntries$.pipe(map((x) => dbxZipBlobPreviewEntryTreeFromEntries(x)), shareReplay(1));
57
+ allEntriesRootSignal = toSignal(this.allEntriesRoot$);
58
+ selectedNodeSignal = signal(undefined);
59
+ listTitleGroupDelegate = {
60
+ groupValueForItem: (item) => {
61
+ const group = item.itemValue.value.value.directory ? 'directory' : 'file';
62
+ return group;
63
+ },
64
+ dataForGroupValue: (value, items) => {
65
+ const data = {
66
+ title: (value === 'directory' ? 'Directories' : 'Files') + ` (${items.length})`,
67
+ value,
68
+ sort: value === 'directory' ? 0 : 1,
69
+ cssClasses: ['dbx-zip-blob-preview-list-group']
70
+ };
71
+ return data;
72
+ },
73
+ sortGroupsByData: (a, b) => {
74
+ return a.sort - b.sort;
75
+ }
76
+ };
77
+ mode = computed(() => {
78
+ const selectedNode = this.selectedNodeSignal();
79
+ let mode = 'view_directory';
80
+ if (selectedNode && !selectedNode.value.value.directory) {
81
+ mode = 'view_entry';
82
+ }
83
+ return mode;
84
+ });
85
+ listEntries = computed(() => {
86
+ const allEntries = this.allEntriesRootSignal();
87
+ const selectedNode = this.selectedNodeSignal();
88
+ let entries;
89
+ if (selectedNode) {
90
+ if (selectedNode.value.value.directory) {
91
+ entries = selectedNode.children;
92
+ }
93
+ else {
94
+ entries = selectedNode.parent?.children;
95
+ }
96
+ }
97
+ else {
98
+ entries = allEntries?.children;
99
+ }
100
+ return entries ?? [];
101
+ });
102
+ listEntries$ = toObservable(this.listEntries);
103
+ listEntriesState$ = loadingStateFromObs(this.listEntries$);
104
+ selectedNodeIconSignal = computed(() => {
105
+ const selectedNode = this.selectedNodeSignal();
106
+ let icon;
107
+ if (selectedNode) {
108
+ if (selectedNode.value.value.directory) {
109
+ icon = 'folder';
110
+ }
111
+ else {
112
+ icon = 'note';
113
+ }
114
+ }
115
+ else {
116
+ icon = 'home';
117
+ }
118
+ return icon;
119
+ });
120
+ selectedNodePathSignal = computed(() => {
121
+ const selectedNode = this.selectedNodeSignal();
122
+ return ['Home', ...(selectedNode?.value.slashPathDetails.parts ?? [])].join(' > ');
123
+ });
124
+ selectedFileNodeSignal = computed(() => {
125
+ const selectedNode = this.selectedNodeSignal();
126
+ return selectedNode?.value.value.directory ? undefined : selectedNode;
127
+ });
128
+ selectedFileEntry$ = toObservable(this.selectedFileNodeSignal);
129
+ selectedFileEntryBlob$ = this.selectedFileEntry$.pipe(switchMap((x) => (x && x.value.getBlob ? x.value.getBlob() : of(undefined))), shareReplay(1));
130
+ selectedFileEntryBlobSignal = toSignal(this.selectedFileEntryBlob$);
131
+ context = loadingStateContext({ obs: this.allEntriesLoadingState$ });
132
+ ngOnDestroy() {
133
+ this.context.destroy();
134
+ }
135
+ makeEntryAnchor = (itemValue) => {
136
+ return {
137
+ onClick: () => {
138
+ this.selectedNodeSignal.set(itemValue);
139
+ }
140
+ };
141
+ };
142
+ homeClicked = () => {
143
+ this.selectedNodeSignal.set(undefined);
144
+ };
145
+ backClicked = () => {
146
+ const selectedNode = this.selectedNodeSignal();
147
+ this.selectedNodeSignal.set(selectedNode?.parent);
148
+ };
149
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: DbxZipBlobPreviewComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
150
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: DbxZipBlobPreviewComponent, isStandalone: true, selector: "dbx-zip-blob-preview", inputs: { blob: { classPropertyName: "blob", publicName: "blob", isSignal: true, isRequired: false, transformFunction: null }, downloadFileName: { classPropertyName: "downloadFileName", publicName: "downloadFileName", isSignal: true, isRequired: false, transformFunction: null } }, ngImport: i0, template: "<dbx-loading [context]=\"context\">\n <div class=\"dbx-flex-bar dbx-zip-blob-preview-toolbar\">\n <dbx-icon-button [disabled]=\"!selectedNodeSignal()\" icon=\"arrow_upward\" text=\"Back\" (click)=\"backClicked()\"></dbx-icon-button>\n <dbx-button-spacer></dbx-button-spacer>\n <dbx-icon-button [disabled]=\"!selectedNodeSignal()\" icon=\"home\" text=\"Home\" (click)=\"homeClicked()\"></dbx-icon-button>\n <dbx-spacer></dbx-spacer>\n @if (downloadZipFileBlobButtonConfigSignal()) {\n <dbx-download-blob-button [config]=\"downloadZipFileBlobButtonConfigSignal()\"></dbx-download-blob-button>\n }\n </div>\n <dbx-bar-header class=\"dbx-zip-blob-preview-header\" [icon]=\"selectedNodeIconSignal()\" [text]=\"selectedNodePathSignal()\"></dbx-bar-header>\n <div class=\"dbx-zip-blob-preview-content\">\n @switch (mode()) {\n @case ('view_directory') {\n <dbx-zip-preview-file-entry-list [state$]=\"listEntriesState$\" [dbxListTitleGroup]=\"listTitleGroupDelegate\" [dbxListItemModifier] [dbxListItemAnchorModifier]=\"makeEntryAnchor\"></dbx-zip-preview-file-entry-list>\n }\n @case ('view_entry') {\n <dbx-embed class=\"dbx-zip-blob-preview-content-embed\" [srcBlob]=\"selectedFileEntryBlobSignal()\"></dbx-embed>\n }\n }\n </div>\n</dbx-loading>\n", dependencies: [{ kind: "ngmodule", type: MatToolbarModule }, { kind: "directive", type: DbxButtonSpacerDirective, selector: "dbx-button-spacer,[dbxButtonSpacer]" }, { kind: "component", type: DbxIconButtonComponent, selector: "dbx-icon-button" }, { kind: "component", type: DbxBarHeaderComponent, selector: "dbx-bar-header", inputs: ["text", "icon", "color"] }, { kind: "directive", type: DbxListTitleGroupDirective, selector: "[dbxListTitleGroup]", inputs: ["dbxListTitleGroup"] }, { kind: "component", type: DbxZipPreviewEntryListComponent, selector: "dbx-zip-preview-file-entry-list" }, { kind: "component", type: DbxEmbedComponent, selector: "dbx-embed", inputs: ["sanitizeUrl", "srcUrl", "type", "srcBlob"] }, { kind: "component", type: DbxLoadingComponent, selector: "dbx-loading", inputs: ["padding", "show", "text", "mode", "color", "diameter", "linear", "loading", "error", "context"] }, { kind: "directive", type: DbxValueListItemModifierDirective, selector: "dbxListItemModifier,[dbxListItemModifier]", inputs: ["dbxListItemModifier"] }, { kind: "directive", type: DbxListItemAnchorModifierDirective, selector: "[dbxListItemAnchorModifier]", inputs: ["dbxListItemAnchorModifier"] }, { kind: "component", type: DbxDownloadBlobButtonComponent, selector: "dbx-download-blob-button", inputs: ["config"] }, { kind: "directive", type: DbxSpacerDirective, selector: "dbx-spacer, [dbxSpacer]" }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
151
+ }
152
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: DbxZipBlobPreviewComponent, decorators: [{
153
+ type: Component,
154
+ args: [{ selector: 'dbx-zip-blob-preview', standalone: true, imports: [MatToolbarModule, DbxButtonSpacerDirective, DbxIconButtonComponent, DbxBarHeaderComponent, DbxListTitleGroupDirective, DbxInjectionComponent, DbxZipPreviewEntryListComponent, DbxEmbedComponent, DbxLoadingComponent, JsonPipe, NgTemplateOutlet, DbxValueListItemModifierDirective, DbxListItemAnchorModifierDirective, DbxListTitleGroupDirective, DbxListEmptyContentComponent, DbxDownloadBlobButtonComponent, DbxSpacerDirective], changeDetection: ChangeDetectionStrategy.OnPush, template: "<dbx-loading [context]=\"context\">\n <div class=\"dbx-flex-bar dbx-zip-blob-preview-toolbar\">\n <dbx-icon-button [disabled]=\"!selectedNodeSignal()\" icon=\"arrow_upward\" text=\"Back\" (click)=\"backClicked()\"></dbx-icon-button>\n <dbx-button-spacer></dbx-button-spacer>\n <dbx-icon-button [disabled]=\"!selectedNodeSignal()\" icon=\"home\" text=\"Home\" (click)=\"homeClicked()\"></dbx-icon-button>\n <dbx-spacer></dbx-spacer>\n @if (downloadZipFileBlobButtonConfigSignal()) {\n <dbx-download-blob-button [config]=\"downloadZipFileBlobButtonConfigSignal()\"></dbx-download-blob-button>\n }\n </div>\n <dbx-bar-header class=\"dbx-zip-blob-preview-header\" [icon]=\"selectedNodeIconSignal()\" [text]=\"selectedNodePathSignal()\"></dbx-bar-header>\n <div class=\"dbx-zip-blob-preview-content\">\n @switch (mode()) {\n @case ('view_directory') {\n <dbx-zip-preview-file-entry-list [state$]=\"listEntriesState$\" [dbxListTitleGroup]=\"listTitleGroupDelegate\" [dbxListItemModifier] [dbxListItemAnchorModifier]=\"makeEntryAnchor\"></dbx-zip-preview-file-entry-list>\n }\n @case ('view_entry') {\n <dbx-embed class=\"dbx-zip-blob-preview-content-embed\" [srcBlob]=\"selectedFileEntryBlobSignal()\"></dbx-embed>\n }\n }\n </div>\n</dbx-loading>\n" }]
155
+ }] });
156
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"zip.blob.preview.component.js","sourceRoot":"","sources":["../../../../../../../packages/dbx-web/src/lib/extension/zip/zip.blob.preview.component.ts","../../../../../../../packages/dbx-web/src/lib/extension/zip/zip.blob.preview.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,uBAAuB,EAAE,SAAS,EAAE,QAAQ,EAAE,KAAK,EAAU,MAAM,EAAa,MAAM,eAAe,CAAC;AAC/G,OAAO,EAAE,qBAAqB,EAAE,MAAM,mBAAmB,CAAC;AAC1D,OAAO,EAAE,QAAQ,EAAE,gBAAgB,EAAE,MAAM,iBAAiB,CAAC;AAE7D,OAAO,EAAE,SAAS,EAAE,UAAU,EAAoB,MAAM,gBAAgB,CAAC;AACzE,OAAO,EAAE,YAAY,EAAE,QAAQ,EAAE,MAAM,4BAA4B,CAAC;AACpE,OAAO,EAAE,mBAAmB,EAAE,MAAM,eAAe,CAAC;AACpD,OAAO,EAAE,oBAAoB,EAAE,GAAG,EAAc,EAAE,EAAE,WAAW,EAAE,SAAS,EAAE,MAAM,MAAM,CAAC;AACzF,OAAO,EAAgB,mBAAmB,EAAE,mBAAmB,EAAE,6BAA6B,EAAE,MAAM,eAAe,CAAC;AACtH,OAAO,EAAE,qCAAqC,EAAkC,MAAM,YAAY,CAAC;AACnG,OAAO,EAAE,+BAA+B,EAAE,MAAM,mCAAmC,CAAC;AACpF,OAAO,EAAE,qBAAqB,EAAE,4BAA4B,EAAyB,0BAA0B,EAAkC,kBAAkB,EAAE,iCAAiC,EAAE,MAAM,cAAc,CAAC;AAC7N,OAAO,EAA0B,kCAAkC,EAAE,MAAM,cAAc,CAAC;AAC1F,OAAO,EAAE,iBAAiB,EAAE,MAAM,mBAAmB,CAAC;AACtD,OAAO,EAAE,gBAAgB,EAAE,MAAM,2BAA2B,CAAC;AAC7D,OAAO,EAAE,wBAAwB,EAAE,sBAAsB,EAAE,MAAM,cAAc,CAAC;AAChF,OAAO,EAAE,8BAA8B,EAA+B,MAAM,iDAAiD,CAAC;;AAO9H;;GAEG;AAQH,MAAM,OAAO,0BAA0B;IAC5B,IAAI,GAAG,KAAK,EAAe,CAAC;IAErC;;;;OAIG;IACM,gBAAgB,GAAG,KAAK,EAAiB,CAAC;IAE1C,OAAO,GAAG,QAAQ,CAAC,GAAG,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,EAAE,CAAC,CAAC;IAExC,SAAS,GAAG,QAAQ,CAAC,GAAG,EAAE;QACjC,MAAM,IAAI,GAAG,IAAI,CAAC,IAAI,EAAE,CAAC;QACzB,OAAO,IAAI,CAAC,CAAC,CAAC,IAAI,SAAS,CAAC,IAAI,UAAU,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC;IAChE,CAAC,CAAC,CAAC;IAEM,qCAAqC,GAAG,QAAQ,CAAC,GAAG,EAAE;QAC7D,MAAM,IAAI,GAAG,IAAI,CAAC,IAAI,EAAE,CAAC;QACzB,MAAM,QAAQ,GAAG,IAAI,CAAC,gBAAgB,EAAE,CAAC;QAEzC,IAAI,MAAM,GAAuC,SAAS,CAAC;QAE3D,IAAI,IAAI,IAAI,QAAQ,EAAE,CAAC;YACrB,MAAM,GAAG;gBACP,IAAI;gBACJ,QAAQ,EAAE,QAAQ,IAAI,cAAc;gBACpC,aAAa,EAAE;oBACb,IAAI,EAAE,sBAAsB;oBAC5B,IAAI,EAAE,cAAc;iBACrB;gBACD,WAAW,EAAE;oBACX,IAAI,EAAE,SAAS;iBAChB;aACF,CAAC;QACJ,CAAC;QAED,OAAO,MAAM,CAAC;IAChB,CAAC,CAAC,CAAC;IAEM,UAAU,GAAG,YAAY,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;IAE1C,uBAAuB,GAAsC,mBAAmB,CAAC,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,UAAU,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;IACxJ,WAAW,GAAG,IAAI,CAAC,uBAAuB,CAAC,IAAI,CAAC,6BAA6B,EAAE,EAAE,oBAAoB,EAAE,EAAE,WAAW,CAAC,CAAC,CAAC,CAAC,CAAC;IACzH,eAAe,GAAG,IAAI,CAAC,WAAW,CAAC,IAAI,CAC9C,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,qCAAqC,CAAC,CAAC,CAAC,CAAC,EACpD,WAAW,CAAC,CAAC,CAAC,CACf,CAAC;IAEO,oBAAoB,GAAG,QAAQ,CAAC,IAAI,CAAC,eAAe,CAAC,CAAC;IAEtD,kBAAkB,GAAG,MAAM,CAAwC,SAAS,CAAC,CAAC;IAE9E,sBAAsB,GAA4H;QACzJ,iBAAiB,EAAE,CAAC,IAAI,EAAE,EAAE;YAC1B,MAAM,KAAK,GAAgC,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC,CAAC,WAAW,CAAC,CAAC,CAAC,MAAM,CAAC;YACvG,OAAO,KAAK,CAAC;QACf,CAAC;QACD,iBAAiB,EAAE,CAAC,KAAkC,EAAE,KAAK,EAAE,EAAE;YAC/D,MAAM,IAAI,GAA+B;gBACvC,KAAK,EAAE,CAAC,KAAK,KAAK,WAAW,CAAC,CAAC,CAAC,aAAa,CAAC,CAAC,CAAC,OAAO,CAAC,GAAG,KAAK,KAAK,CAAC,MAAM,GAAG;gBAC/E,KAAK;gBACL,IAAI,EAAE,KAAK,KAAK,WAAW,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;gBACnC,UAAU,EAAE,CAAC,iCAAiC,CAAC;aAChD,CAAC;YAEF,OAAO,IAAI,CAAC;QACd,CAAC;QACD,gBAAgB,EAAE,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE;YACzB,OAAO,CAAC,CAAC,IAAI,GAAG,CAAC,CAAC,IAAI,CAAC;QACzB,CAAC;KACF,CAAC;IAEO,IAAI,GAAG,QAAQ,CAAC,GAAG,EAAE;QAC5B,MAAM,YAAY,GAAG,IAAI,CAAC,kBAAkB,EAAE,CAAC;QAE/C,IAAI,IAAI,GAAG,gBAAgB,CAAC;QAE5B,IAAI,YAAY,IAAI,CAAC,YAAY,CAAC,KAAK,CAAC,KAAK,CAAC,SAAS,EAAE,CAAC;YACxD,IAAI,GAAG,YAAY,CAAC;QACtB,CAAC;QAED,OAAO,IAAI,CAAC;IACd,CAAC,CAAC,CAAC;IAEM,WAAW,GAAG,QAAQ,CAAC,GAAG,EAAE;QACnC,MAAM,UAAU,GAAG,IAAI,CAAC,oBAAoB,EAAE,CAAC;QAC/C,MAAM,YAAY,GAAG,IAAI,CAAC,kBAAkB,EAAE,CAAC;QAE/C,IAAI,OAAgD,CAAC;QAErD,IAAI,YAAY,EAAE,CAAC;YACjB,IAAI,YAAY,CAAC,KAAK,CAAC,KAAK,CAAC,SAAS,EAAE,CAAC;gBACvC,OAAO,GAAG,YAAY,CAAC,QAAQ,CAAC;YAClC,CAAC;iBAAM,CAAC;gBACN,OAAO,GAAG,YAAY,CAAC,MAAM,EAAE,QAAQ,CAAC;YAC1C,CAAC;QACH,CAAC;aAAM,CAAC;YACN,OAAO,GAAG,UAAU,EAAE,QAAQ,CAAC;QACjC,CAAC;QAED,OAAO,OAAO,IAAI,EAAE,CAAC;IACvB,CAAC,CAAC,CAAC;IAEM,YAAY,GAAG,YAAY,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;IAC9C,iBAAiB,GAAG,mBAAmB,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;IAE3D,sBAAsB,GAAG,QAAQ,CAAC,GAAG,EAAE;QAC9C,MAAM,YAAY,GAAG,IAAI,CAAC,kBAAkB,EAAE,CAAC;QAE/C,IAAI,IAAY,CAAC;QAEjB,IAAI,YAAY,EAAE,CAAC;YACjB,IAAI,YAAY,CAAC,KAAK,CAAC,KAAK,CAAC,SAAS,EAAE,CAAC;gBACvC,IAAI,GAAG,QAAQ,CAAC;YAClB,CAAC;iBAAM,CAAC;gBACN,IAAI,GAAG,MAAM,CAAC;YAChB,CAAC;QACH,CAAC;aAAM,CAAC;YACN,IAAI,GAAG,MAAM,CAAC;QAChB,CAAC;QAED,OAAO,IAAI,CAAC;IACd,CAAC,CAAC,CAAC;IAEM,sBAAsB,GAAG,QAAQ,CAAC,GAAG,EAAE;QAC9C,MAAM,YAAY,GAAG,IAAI,CAAC,kBAAkB,EAAE,CAAC;QAC/C,OAAO,CAAC,MAAM,EAAE,GAAG,CAAC,YAAY,EAAE,KAAK,CAAC,gBAAgB,CAAC,KAAK,IAAI,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IACrF,CAAC,CAAC,CAAC;IAEM,sBAAsB,GAA6D,QAAQ,CAAC,GAAG,EAAE;QACxG,MAAM,YAAY,GAAG,IAAI,CAAC,kBAAkB,EAAE,CAAC;QAC/C,OAAO,YAAY,EAAE,KAAK,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAE,YAA0D,CAAC;IACvH,CAAC,CAAC,CAAC;IAEM,kBAAkB,GAAG,YAAY,CAAC,IAAI,CAAC,sBAAsB,CAAC,CAAC;IAC/D,sBAAsB,GAA4B,IAAI,CAAC,kBAAkB,CAAC,IAAI,CACrF,SAAS,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC,SAAS,CAAC,CAAC,CAAC,EAC5E,WAAW,CAAC,CAAC,CAAC,CACf,CAAC;IACO,2BAA2B,GAAwB,QAAQ,CAAC,IAAI,CAAC,sBAAsB,CAAC,CAAC;IAEzF,OAAO,GAAG,mBAAmB,CAAC,EAAE,GAAG,EAAE,IAAI,CAAC,uBAAuB,EAAE,CAAC,CAAC;IAE9E,WAAW;QACT,IAAI,CAAC,OAAO,CAAC,OAAO,EAAE,CAAC;IACzB,CAAC;IAEQ,eAAe,GAA2D,CAAC,SAAS,EAAE,EAAE;QAC/F,OAAO;YACL,OAAO,EAAE,GAAG,EAAE;gBACZ,IAAI,CAAC,kBAAkB,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;YACzC,CAAC;SACF,CAAC;IACJ,CAAC,CAAC;IAEO,WAAW,GAAG,GAAG,EAAE;QAC1B,IAAI,CAAC,kBAAkB,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;IACzC,CAAC,CAAC;IAEO,WAAW,GAAG,GAAG,EAAE;QAC1B,MAAM,YAAY,GAAG,IAAI,CAAC,kBAAkB,EAAE,CAAC;QAC/C,IAAI,CAAC,kBAAkB,CAAC,GAAG,CAAC,YAAY,EAAE,MAAM,CAAC,CAAC;IACpD,CAAC,CAAC;wGAnKS,0BAA0B;4FAA1B,0BAA0B,0WCjCvC,iyCAsBA,2CDQY,gBAAgB,+BAAE,wBAAwB,gFAAE,sBAAsB,4DAAE,qBAAqB,8FAAE,0BAA0B,+FAAyB,+BAA+B,4EAAE,iBAAiB,4GAAE,mBAAmB,mKAA8B,iCAAiC,uHAAE,kCAAkC,+GAA4D,8BAA8B,yFAAE,kBAAkB;;4FAGra,0BAA0B;kBAPtC,SAAS;+BACE,sBAAsB,cAEpB,IAAI,WACP,CAAC,gBAAgB,EAAE,wBAAwB,EAAE,sBAAsB,EAAE,qBAAqB,EAAE,0BAA0B,EAAE,qBAAqB,EAAE,+BAA+B,EAAE,iBAAiB,EAAE,mBAAmB,EAAE,QAAQ,EAAE,gBAAgB,EAAE,iCAAiC,EAAE,kCAAkC,EAAE,0BAA0B,EAAE,4BAA4B,EAAE,8BAA8B,EAAE,kBAAkB,CAAC,mBACha,uBAAuB,CAAC,MAAM","sourcesContent":["import { ChangeDetectionStrategy, Component, computed, input, Signal, signal, OnDestroy } from '@angular/core';\nimport { DbxInjectionComponent } from '@dereekb/dbx-core';\nimport { JsonPipe, NgTemplateOutlet } from '@angular/common';\nimport { Maybe } from '@dereekb/util';\nimport { ZipReader, BlobReader, Entry, FileEntry } from '@zip.js/zip.js';\nimport { toObservable, toSignal } from '@angular/core/rxjs-interop';\nimport { DbxLoadingComponent } from '../../loading';\nimport { distinctUntilChanged, map, Observable, of, shareReplay, switchMap } from 'rxjs';\nimport { LoadingState, loadingStateContext, loadingStateFromObs, valueFromFinishedLoadingState } from '@dereekb/rxjs';\nimport { dbxZipBlobPreviewEntryTreeFromEntries, DbxZipBlobPreviewEntryTreeNode } from './zip.blob';\nimport { DbxZipPreviewEntryListComponent } from './zip.blob.preview.list.component';\nimport { DbxBarHeaderComponent, DbxListEmptyContentComponent, DbxListTitleGroupData, DbxListTitleGroupDirective, DbxListTitleGroupTitleDelegate, DbxSpacerDirective, DbxValueListItemModifierDirective } from '../../layout';\nimport { AnchorForValueFunction, DbxListItemAnchorModifierDirective } from '../../router';\nimport { DbxEmbedComponent } from '../../interaction';\nimport { MatToolbarModule } from '@angular/material/toolbar';\nimport { DbxButtonSpacerDirective, DbxIconButtonComponent } from '../../button';\nimport { DbxDownloadBlobButtonComponent, DbxDownloadBlobButtonConfig } from '../download/blob/download.blob.button.component';\n\nexport type DbxZipBlobPreviewMode = 'view_directory' | 'view_entry';\n\nexport type DbxZipBlobPreviewGroupValue = 'directory' | 'file';\nexport type DbxZipBlobPreviewGroupData = DbxListTitleGroupData<DbxZipBlobPreviewGroupValue> & { sort: number };\n\n/**\n * Used to display a zip preview based on the input blob.\n */\n@Component({\n  selector: 'dbx-zip-blob-preview',\n  templateUrl: './zip.blob.preview.component.html',\n  standalone: true,\n  imports: [MatToolbarModule, DbxButtonSpacerDirective, DbxIconButtonComponent, DbxBarHeaderComponent, DbxListTitleGroupDirective, DbxInjectionComponent, DbxZipPreviewEntryListComponent, DbxEmbedComponent, DbxLoadingComponent, JsonPipe, NgTemplateOutlet, DbxValueListItemModifierDirective, DbxListItemAnchorModifierDirective, DbxListTitleGroupDirective, DbxListEmptyContentComponent, DbxDownloadBlobButtonComponent, DbxSpacerDirective],\n  changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class DbxZipBlobPreviewComponent implements OnDestroy {\n  readonly blob = input<Maybe<Blob>>();\n\n  /**\n   * The download file name for the zip file.\n   *\n   * If not defined, then the file cannot be downloaded directly.\n   */\n  readonly downloadFileName = input<Maybe<string>>();\n\n  readonly hasBlob = computed(() => !!this.blob());\n\n  readonly zipReader = computed(() => {\n    const blob = this.blob();\n    return blob ? new ZipReader(new BlobReader(blob)) : undefined;\n  });\n\n  readonly downloadZipFileBlobButtonConfigSignal = computed(() => {\n    const blob = this.blob();\n    const fileName = this.downloadFileName();\n\n    let result: Maybe<DbxDownloadBlobButtonConfig> = undefined;\n\n    if (blob && fileName) {\n      result = {\n        blob,\n        fileName: fileName ?? 'download.zip',\n        buttonDisplay: {\n          icon: 'download_for_offline',\n          text: 'Download Zip'\n        },\n        buttonStyle: {\n          type: 'stroked'\n        }\n      };\n    }\n\n    return result;\n  });\n\n  readonly zipReader$ = toObservable(this.zipReader);\n\n  readonly allEntriesLoadingState$: Observable<LoadingState<Entry[]>> = loadingStateFromObs(this.zipReader$.pipe(switchMap((x) => (x ? x.getEntries() : of([])))));\n  readonly allEntries$ = this.allEntriesLoadingState$.pipe(valueFromFinishedLoadingState(), distinctUntilChanged(), shareReplay(1));\n  readonly allEntriesRoot$ = this.allEntries$.pipe(\n    map((x) => dbxZipBlobPreviewEntryTreeFromEntries(x)),\n    shareReplay(1)\n  );\n\n  readonly allEntriesRootSignal = toSignal(this.allEntriesRoot$);\n\n  readonly selectedNodeSignal = signal<Maybe<DbxZipBlobPreviewEntryTreeNode>>(undefined);\n\n  readonly listTitleGroupDelegate: DbxListTitleGroupTitleDelegate<DbxZipBlobPreviewEntryTreeNode, DbxZipBlobPreviewGroupValue, DbxZipBlobPreviewGroupData> = {\n    groupValueForItem: (item) => {\n      const group: DbxZipBlobPreviewGroupValue = item.itemValue.value.value.directory ? 'directory' : 'file';\n      return group;\n    },\n    dataForGroupValue: (value: DbxZipBlobPreviewGroupValue, items) => {\n      const data: DbxZipBlobPreviewGroupData = {\n        title: (value === 'directory' ? 'Directories' : 'Files') + ` (${items.length})`,\n        value,\n        sort: value === 'directory' ? 0 : 1,\n        cssClasses: ['dbx-zip-blob-preview-list-group']\n      };\n\n      return data;\n    },\n    sortGroupsByData: (a, b) => {\n      return a.sort - b.sort;\n    }\n  };\n\n  readonly mode = computed(() => {\n    const selectedNode = this.selectedNodeSignal();\n\n    let mode = 'view_directory';\n\n    if (selectedNode && !selectedNode.value.value.directory) {\n      mode = 'view_entry';\n    }\n\n    return mode;\n  });\n\n  readonly listEntries = computed(() => {\n    const allEntries = this.allEntriesRootSignal();\n    const selectedNode = this.selectedNodeSignal();\n\n    let entries: Maybe<DbxZipBlobPreviewEntryTreeNode[]>;\n\n    if (selectedNode) {\n      if (selectedNode.value.value.directory) {\n        entries = selectedNode.children;\n      } else {\n        entries = selectedNode.parent?.children;\n      }\n    } else {\n      entries = allEntries?.children;\n    }\n\n    return entries ?? [];\n  });\n\n  readonly listEntries$ = toObservable(this.listEntries);\n  readonly listEntriesState$ = loadingStateFromObs(this.listEntries$);\n\n  readonly selectedNodeIconSignal = computed(() => {\n    const selectedNode = this.selectedNodeSignal();\n\n    let icon: string;\n\n    if (selectedNode) {\n      if (selectedNode.value.value.directory) {\n        icon = 'folder';\n      } else {\n        icon = 'note';\n      }\n    } else {\n      icon = 'home';\n    }\n\n    return icon;\n  });\n\n  readonly selectedNodePathSignal = computed(() => {\n    const selectedNode = this.selectedNodeSignal();\n    return ['Home', ...(selectedNode?.value.slashPathDetails.parts ?? [])].join(' > ');\n  });\n\n  readonly selectedFileNodeSignal: Signal<Maybe<DbxZipBlobPreviewEntryTreeNode<FileEntry>>> = computed(() => {\n    const selectedNode = this.selectedNodeSignal();\n    return selectedNode?.value.value.directory ? undefined : (selectedNode as DbxZipBlobPreviewEntryTreeNode<FileEntry>);\n  });\n\n  readonly selectedFileEntry$ = toObservable(this.selectedFileNodeSignal);\n  readonly selectedFileEntryBlob$: Observable<Maybe<Blob>> = this.selectedFileEntry$.pipe(\n    switchMap((x) => (x && x.value.getBlob ? x.value.getBlob() : of(undefined))),\n    shareReplay(1)\n  );\n  readonly selectedFileEntryBlobSignal: Signal<Maybe<Blob>> = toSignal(this.selectedFileEntryBlob$);\n\n  readonly context = loadingStateContext({ obs: this.allEntriesLoadingState$ });\n\n  ngOnDestroy(): void {\n    this.context.destroy();\n  }\n\n  readonly makeEntryAnchor: AnchorForValueFunction<DbxZipBlobPreviewEntryTreeNode> = (itemValue) => {\n    return {\n      onClick: () => {\n        this.selectedNodeSignal.set(itemValue);\n      }\n    };\n  };\n\n  readonly homeClicked = () => {\n    this.selectedNodeSignal.set(undefined);\n  };\n\n  readonly backClicked = () => {\n    const selectedNode = this.selectedNodeSignal();\n    this.selectedNodeSignal.set(selectedNode?.parent);\n  };\n}\n","<dbx-loading [context]=\"context\">\n  <div class=\"dbx-flex-bar dbx-zip-blob-preview-toolbar\">\n    <dbx-icon-button [disabled]=\"!selectedNodeSignal()\" icon=\"arrow_upward\" text=\"Back\" (click)=\"backClicked()\"></dbx-icon-button>\n    <dbx-button-spacer></dbx-button-spacer>\n    <dbx-icon-button [disabled]=\"!selectedNodeSignal()\" icon=\"home\" text=\"Home\" (click)=\"homeClicked()\"></dbx-icon-button>\n    <dbx-spacer></dbx-spacer>\n    @if (downloadZipFileBlobButtonConfigSignal()) {\n      <dbx-download-blob-button [config]=\"downloadZipFileBlobButtonConfigSignal()\"></dbx-download-blob-button>\n    }\n  </div>\n  <dbx-bar-header class=\"dbx-zip-blob-preview-header\" [icon]=\"selectedNodeIconSignal()\" [text]=\"selectedNodePathSignal()\"></dbx-bar-header>\n  <div class=\"dbx-zip-blob-preview-content\">\n    @switch (mode()) {\n      @case ('view_directory') {\n        <dbx-zip-preview-file-entry-list [state$]=\"listEntriesState$\" [dbxListTitleGroup]=\"listTitleGroupDelegate\" [dbxListItemModifier] [dbxListItemAnchorModifier]=\"makeEntryAnchor\"></dbx-zip-preview-file-entry-list>\n      }\n      @case ('view_entry') {\n        <dbx-embed class=\"dbx-zip-blob-preview-content-embed\" [srcBlob]=\"selectedFileEntryBlobSignal()\"></dbx-embed>\n      }\n    }\n  </div>\n</dbx-loading>\n"]}
@@ -0,0 +1,121 @@
1
+ import { ChangeDetectionStrategy, Component } from '@angular/core';
2
+ import { DEFAULT_LIST_WRAPPER_COMPONENT_CONFIGURATION_TEMPLATE, DbxListWrapperComponentImportsModule, provideDbxListViewWrapper, AbstractDbxSelectionListWrapperDirective, DEFAULT_DBX_SELECTION_VALUE_LIST_COMPONENT_CONFIGURATION_TEMPLATE, DbxSelectionValueListViewComponentImportsModule, provideDbxListView, AbstractDbxSelectionListViewDirective, AbstractDbxValueListViewItemComponent, DbxSpacerDirective } from '../../layout';
3
+ import { of } from 'rxjs';
4
+ import { DatePipe } from '@angular/common';
5
+ import { DbxIconButtonComponent } from '../../button';
6
+ import { DbxDownloadBlobButtonComponent } from '../download/blob/download.blob.button.component';
7
+ import * as i0 from "@angular/core";
8
+ import * as i1 from "../../layout/list/list.component";
9
+ import * as i2 from "../../layout/list/list.view.value.selection.component";
10
+ export function iconForDbxZipPreviewEntryWithSelection(entry) {
11
+ return entry.value.value.directory ? 'folder' : entry.value.mimeType ? 'note' : 'question_mark';
12
+ }
13
+ export class DbxZipPreviewEntryListComponent extends AbstractDbxSelectionListWrapperDirective {
14
+ constructor() {
15
+ super({
16
+ componentClass: DbxZipPreviewEntryListViewComponent,
17
+ defaultSelectionMode: 'view'
18
+ });
19
+ }
20
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: DbxZipPreviewEntryListComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
21
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: DbxZipPreviewEntryListComponent, isStandalone: true, selector: "dbx-zip-preview-file-entry-list", providers: provideDbxListViewWrapper(DbxZipPreviewEntryListComponent), usesInheritance: true, ngImport: i0, template: "\n <dbx-list [state]=\"currentState$\" [config]=\"configSignal()\" [hasMore]=\"hasMore()\" [disabled]=\"disabled()\" [selectionMode]=\"selectionModeSignal()\">\n <ng-content top select=\"[top]\"></ng-content>\n <ng-content bottom select=\"[bottom]\"></ng-content>\n <ng-content empty select=\"[empty]\"></ng-content>\n <ng-content emptyLoading select=\"[emptyLoading]\"></ng-content>\n <ng-content end select=\"[end]\"></ng-content>\n </dbx-list>", isInline: true, dependencies: [{ kind: "ngmodule", type: DbxListWrapperComponentImportsModule }, { kind: "component", type: i1.DbxListComponent, selector: "dbx-list", inputs: ["padded", "state", "config", "disabled", "selectionMode", "hasMore"], outputs: ["contentScrolled"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
22
+ }
23
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: DbxZipPreviewEntryListComponent, decorators: [{
24
+ type: Component,
25
+ args: [{
26
+ selector: 'dbx-zip-preview-file-entry-list',
27
+ template: DEFAULT_LIST_WRAPPER_COMPONENT_CONFIGURATION_TEMPLATE,
28
+ imports: [DbxListWrapperComponentImportsModule],
29
+ changeDetection: ChangeDetectionStrategy.OnPush,
30
+ providers: provideDbxListViewWrapper(DbxZipPreviewEntryListComponent),
31
+ standalone: true
32
+ }]
33
+ }], ctorParameters: () => [] });
34
+ export class DbxZipPreviewEntryListViewComponent extends AbstractDbxSelectionListViewDirective {
35
+ config = {
36
+ componentClass: DbxZipPreviewEntryListViewItemComponent,
37
+ mapValuesToItemValues: (x) => of(x.map((y) => ({ ...y, icon: iconForDbxZipPreviewEntryWithSelection(y), itemValue: y })))
38
+ };
39
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: DbxZipPreviewEntryListViewComponent, deps: null, target: i0.ɵɵFactoryTarget.Component });
40
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: DbxZipPreviewEntryListViewComponent, isStandalone: true, selector: "dbx-zip-preview-file-entry-list-view", providers: provideDbxListView(DbxZipPreviewEntryListViewComponent), usesInheritance: true, ngImport: i0, template: "<dbx-selection-list-view [config]=\"config\"></dbx-selection-list-view>", isInline: true, dependencies: [{ kind: "ngmodule", type: DbxSelectionValueListViewComponentImportsModule }, { kind: "component", type: i2.DbxSelectionValueListViewComponent, selector: "dbx-selection-list-view" }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
41
+ }
42
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: DbxZipPreviewEntryListViewComponent, decorators: [{
43
+ type: Component,
44
+ args: [{
45
+ selector: 'dbx-zip-preview-file-entry-list-view',
46
+ template: DEFAULT_DBX_SELECTION_VALUE_LIST_COMPONENT_CONFIGURATION_TEMPLATE,
47
+ imports: [DbxSelectionValueListViewComponentImportsModule],
48
+ changeDetection: ChangeDetectionStrategy.OnPush,
49
+ standalone: true,
50
+ providers: provideDbxListView(DbxZipPreviewEntryListViewComponent)
51
+ }]
52
+ }] });
53
+ export class DbxZipPreviewEntryListViewItemComponent extends AbstractDbxValueListViewItemComponent {
54
+ get name() {
55
+ return this.itemValue.value.value.filename ?? 'hello';
56
+ }
57
+ get lastModDate() {
58
+ return this.itemValue.value.value.lastModDate ?? undefined;
59
+ }
60
+ get isDirectory() {
61
+ return this.itemValue.value.value.directory;
62
+ }
63
+ get canDownload() {
64
+ return !this.isDirectory;
65
+ }
66
+ get downloadBlobButtonConfig() {
67
+ return {
68
+ loadBlob: this.itemValue.value.getBlob,
69
+ fileName: this.itemValue.value.slashPathDetails.fileName,
70
+ buttonDisplay: {
71
+ icon: 'download',
72
+ text: 'Download'
73
+ },
74
+ buttonStyle: {
75
+ type: 'stroked'
76
+ }
77
+ };
78
+ }
79
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: DbxZipPreviewEntryListViewItemComponent, deps: null, target: i0.ɵɵFactoryTarget.Component });
80
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: DbxZipPreviewEntryListViewItemComponent, isStandalone: true, selector: "dbx-zip-preview-file-entry-list-view-item", usesInheritance: true, ngImport: i0, template: `
81
+ <div class="dbx-list-item-padded dbx-list-two-line-item">
82
+ <div class="item-left">
83
+ <div class="mat-subtitle-2">{{ name }}</div>
84
+ @if (lastModDate) {
85
+ <div class="item-details">{{ lastModDate | date: 'short' }}</div>
86
+ }
87
+ </div>
88
+ <dbx-spacer></dbx-spacer>
89
+ <div class="item-right">
90
+ @if (canDownload) {
91
+ <dbx-download-blob-button [config]="downloadBlobButtonConfig"></dbx-download-blob-button>
92
+ }
93
+ </div>
94
+ </div>
95
+ `, isInline: true, dependencies: [{ kind: "pipe", type: DatePipe, name: "date" }, { kind: "directive", type: DbxSpacerDirective, selector: "dbx-spacer, [dbxSpacer]" }, { kind: "component", type: DbxDownloadBlobButtonComponent, selector: "dbx-download-blob-button", inputs: ["config"] }] });
96
+ }
97
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: DbxZipPreviewEntryListViewItemComponent, decorators: [{
98
+ type: Component,
99
+ args: [{
100
+ selector: 'dbx-zip-preview-file-entry-list-view-item',
101
+ template: `
102
+ <div class="dbx-list-item-padded dbx-list-two-line-item">
103
+ <div class="item-left">
104
+ <div class="mat-subtitle-2">{{ name }}</div>
105
+ @if (lastModDate) {
106
+ <div class="item-details">{{ lastModDate | date: 'short' }}</div>
107
+ }
108
+ </div>
109
+ <dbx-spacer></dbx-spacer>
110
+ <div class="item-right">
111
+ @if (canDownload) {
112
+ <dbx-download-blob-button [config]="downloadBlobButtonConfig"></dbx-download-blob-button>
113
+ }
114
+ </div>
115
+ </div>
116
+ `,
117
+ imports: [DatePipe, DbxIconButtonComponent, DbxSpacerDirective, DbxDownloadBlobButtonComponent],
118
+ standalone: true
119
+ }]
120
+ }] });
121
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"zip.blob.preview.list.component.js","sourceRoot":"","sources":["../../../../../../../packages/dbx-web/src/lib/extension/zip/zip.blob.preview.list.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component } from '@angular/core';\nimport { DbxValueAsListItem, DEFAULT_LIST_WRAPPER_COMPONENT_CONFIGURATION_TEMPLATE, DbxListWrapperComponentImportsModule, provideDbxListViewWrapper, AbstractDbxSelectionListWrapperDirective, DEFAULT_DBX_SELECTION_VALUE_LIST_COMPONENT_CONFIGURATION_TEMPLATE, DbxSelectionValueListViewComponentImportsModule, provideDbxListView, AbstractDbxSelectionListViewDirective, DbxSelectionValueListViewConfig, AbstractDbxValueListViewItemComponent, DbxSpacerDirective } from '../../layout';\nimport { of } from 'rxjs';\nimport { DbxZipBlobPreviewEntryTreeNode } from './zip.blob';\nimport { DatePipe } from '@angular/common';\nimport { DbxIconButtonComponent } from '../../button';\nimport { DbxDownloadBlobButtonComponent, DbxDownloadBlobButtonConfig } from '../download/blob/download.blob.button.component';\n\nexport type DbxZipPreviewEntryWithSelection = DbxValueAsListItem<DbxZipBlobPreviewEntryTreeNode>;\n\nexport function iconForDbxZipPreviewEntryWithSelection(entry: DbxZipBlobPreviewEntryTreeNode) {\n  return entry.value.value.directory ? 'folder' : entry.value.mimeType ? 'note' : 'question_mark';\n}\n\n@Component({\n  selector: 'dbx-zip-preview-file-entry-list',\n  template: DEFAULT_LIST_WRAPPER_COMPONENT_CONFIGURATION_TEMPLATE,\n  imports: [DbxListWrapperComponentImportsModule],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  providers: provideDbxListViewWrapper(DbxZipPreviewEntryListComponent),\n  standalone: true\n})\nexport class DbxZipPreviewEntryListComponent extends AbstractDbxSelectionListWrapperDirective<DbxZipPreviewEntryWithSelection> {\n  constructor() {\n    super({\n      componentClass: DbxZipPreviewEntryListViewComponent,\n      defaultSelectionMode: 'view'\n    });\n  }\n}\n\n@Component({\n  selector: 'dbx-zip-preview-file-entry-list-view',\n  template: DEFAULT_DBX_SELECTION_VALUE_LIST_COMPONENT_CONFIGURATION_TEMPLATE,\n  imports: [DbxSelectionValueListViewComponentImportsModule],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  standalone: true,\n  providers: provideDbxListView(DbxZipPreviewEntryListViewComponent)\n})\nexport class DbxZipPreviewEntryListViewComponent extends AbstractDbxSelectionListViewDirective<DbxZipPreviewEntryWithSelection> {\n  readonly config: DbxSelectionValueListViewConfig<DbxZipPreviewEntryWithSelection> = {\n    componentClass: DbxZipPreviewEntryListViewItemComponent,\n    mapValuesToItemValues: (x) => of(x.map((y) => ({ ...y, icon: iconForDbxZipPreviewEntryWithSelection(y), itemValue: y })))\n  };\n}\n\n@Component({\n  selector: 'dbx-zip-preview-file-entry-list-view-item',\n  template: `\n    <div class=\"dbx-list-item-padded dbx-list-two-line-item\">\n      <div class=\"item-left\">\n        <div class=\"mat-subtitle-2\">{{ name }}</div>\n        @if (lastModDate) {\n          <div class=\"item-details\">{{ lastModDate | date: 'short' }}</div>\n        }\n      </div>\n      <dbx-spacer></dbx-spacer>\n      <div class=\"item-right\">\n        @if (canDownload) {\n          <dbx-download-blob-button [config]=\"downloadBlobButtonConfig\"></dbx-download-blob-button>\n        }\n      </div>\n    </div>\n  `,\n  imports: [DatePipe, DbxIconButtonComponent, DbxSpacerDirective, DbxDownloadBlobButtonComponent],\n  standalone: true\n})\nexport class DbxZipPreviewEntryListViewItemComponent extends AbstractDbxValueListViewItemComponent<DbxZipPreviewEntryWithSelection> {\n  get name() {\n    return this.itemValue.value.value.filename ?? 'hello';\n  }\n\n  get lastModDate() {\n    return this.itemValue.value.value.lastModDate ?? undefined;\n  }\n\n  get isDirectory() {\n    return this.itemValue.value.value.directory;\n  }\n\n  get canDownload() {\n    return !this.isDirectory;\n  }\n\n  get downloadBlobButtonConfig(): DbxDownloadBlobButtonConfig {\n    return {\n      loadBlob: this.itemValue.value.getBlob,\n      fileName: this.itemValue.value.slashPathDetails.fileName,\n      buttonDisplay: {\n        icon: 'download',\n        text: 'Download'\n      },\n      buttonStyle: {\n        type: 'stroked'\n      }\n    };\n  }\n}\n"]}
@@ -0,0 +1,71 @@
1
+ import { ChangeDetectionStrategy, Component, input } from '@angular/core';
2
+ import { DbxInjectionComponent } from '@dereekb/dbx-core';
3
+ import { NgTemplateOutlet } from '@angular/common';
4
+ import { DbxLoadingComponent } from '../../loading';
5
+ import { toObservable, toSignal } from '@angular/core/rxjs-interop';
6
+ import { combineLatest, distinctUntilChanged, from, of, shareReplay, switchMap } from 'rxjs';
7
+ import { beginLoading, errorResult, loadingStateContext, startWithBeginLoading, successResult, valueFromFinishedLoadingState } from '@dereekb/rxjs';
8
+ import { DbxZipBlobPreviewComponent } from './zip.blob.preview.component';
9
+ import * as i0 from "@angular/core";
10
+ /**
11
+ * Used to display a corresponding widget based on the input data.
12
+ */
13
+ export class DbxZipPreviewComponent {
14
+ /**
15
+ * The URL to download the zip file from, if applicable.
16
+ */
17
+ srcUrl = input();
18
+ /**
19
+ * The blob to use for the zip file, if applicable.
20
+ */
21
+ blob = input();
22
+ /**
23
+ * The file name to use for the zip file.
24
+ */
25
+ downloadFileName = input();
26
+ srcUrl$ = toObservable(this.srcUrl);
27
+ blob$ = toObservable(this.blob);
28
+ zipFileBlobLoadingState$ = combineLatest([this.srcUrl$, this.blob$]).pipe(switchMap(([srcUrl, blob]) => {
29
+ let obs;
30
+ if (blob) {
31
+ obs = of(successResult(blob));
32
+ }
33
+ else if (srcUrl) {
34
+ obs = from(fetch(srcUrl, { method: 'GET' }).then((x) => x
35
+ .blob()
36
+ .then((y) => successResult(y))
37
+ .catch((e) => errorResult(e)))).pipe(startWithBeginLoading());
38
+ }
39
+ else {
40
+ obs = of(beginLoading());
41
+ }
42
+ return obs;
43
+ }));
44
+ zipFileBlob$ = this.zipFileBlobLoadingState$.pipe(valueFromFinishedLoadingState(), distinctUntilChanged(), shareReplay(1));
45
+ zipFileBlobSignal = toSignal(this.zipFileBlob$);
46
+ context = loadingStateContext({ obs: this.zipFileBlobLoadingState$ });
47
+ ngOnDestroy() {
48
+ this.context.destroy();
49
+ }
50
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: DbxZipPreviewComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
51
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.1.0", version: "18.2.13", type: DbxZipPreviewComponent, isStandalone: true, selector: "dbx-zip-preview", inputs: { srcUrl: { classPropertyName: "srcUrl", publicName: "srcUrl", isSignal: true, isRequired: false, transformFunction: null }, blob: { classPropertyName: "blob", publicName: "blob", isSignal: true, isRequired: false, transformFunction: null }, downloadFileName: { classPropertyName: "downloadFileName", publicName: "downloadFileName", isSignal: true, isRequired: false, transformFunction: null } }, ngImport: i0, template: `
52
+ <dbx-loading [context]="context">
53
+ <dbx-zip-blob-preview [downloadFileName]="downloadFileName()" [blob]="zipFileBlobSignal()"></dbx-zip-blob-preview>
54
+ </dbx-loading>
55
+ `, isInline: true, dependencies: [{ kind: "component", type: DbxLoadingComponent, selector: "dbx-loading", inputs: ["padding", "show", "text", "mode", "color", "diameter", "linear", "loading", "error", "context"] }, { kind: "component", type: DbxZipBlobPreviewComponent, selector: "dbx-zip-blob-preview", inputs: ["blob", "downloadFileName"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
56
+ }
57
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: DbxZipPreviewComponent, decorators: [{
58
+ type: Component,
59
+ args: [{
60
+ selector: 'dbx-zip-preview',
61
+ template: `
62
+ <dbx-loading [context]="context">
63
+ <dbx-zip-blob-preview [downloadFileName]="downloadFileName()" [blob]="zipFileBlobSignal()"></dbx-zip-blob-preview>
64
+ </dbx-loading>
65
+ `,
66
+ standalone: true,
67
+ imports: [DbxInjectionComponent, DbxLoadingComponent, DbxZipBlobPreviewComponent, NgTemplateOutlet],
68
+ changeDetection: ChangeDetectionStrategy.OnPush
69
+ }]
70
+ }] });
71
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiemlwLnByZXZpZXcuY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vcGFja2FnZXMvZGJ4LXdlYi9zcmMvbGliL2V4dGVuc2lvbi96aXAvemlwLnByZXZpZXcuY29tcG9uZW50LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSx1QkFBdUIsRUFBRSxTQUFTLEVBQUUsS0FBSyxFQUFhLE1BQU0sZUFBZSxDQUFDO0FBQ3JGLE9BQU8sRUFBRSxxQkFBcUIsRUFBRSxNQUFNLG1CQUFtQixDQUFDO0FBQzFELE9BQU8sRUFBRSxnQkFBZ0IsRUFBRSxNQUFNLGlCQUFpQixDQUFDO0FBQ25ELE9BQU8sRUFBRSxtQkFBbUIsRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUVwRCxPQUFPLEVBQUUsWUFBWSxFQUFFLFFBQVEsRUFBRSxNQUFNLDRCQUE0QixDQUFDO0FBQ3BFLE9BQU8sRUFBRSxhQUFhLEVBQUUsb0JBQW9CLEVBQUUsSUFBSSxFQUFjLEVBQUUsRUFBRSxXQUFXLEVBQUUsU0FBUyxFQUFFLE1BQU0sTUFBTSxDQUFDO0FBQ3pHLE9BQU8sRUFBRSxZQUFZLEVBQUUsV0FBVyxFQUFnQixtQkFBbUIsRUFBRSxxQkFBcUIsRUFBRSxhQUFhLEVBQUUsNkJBQTZCLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDbEssT0FBTyxFQUFFLDBCQUEwQixFQUFFLE1BQU0sOEJBQThCLENBQUM7O0FBRTFFOztHQUVHO0FBWUgsTUFBTSxPQUFPLHNCQUFzQjtJQUNqQzs7T0FFRztJQUNNLE1BQU0sR0FBRyxLQUFLLEVBQStCLENBQUM7SUFFdkQ7O09BRUc7SUFDTSxJQUFJLEdBQUcsS0FBSyxFQUFlLENBQUM7SUFFckM7O09BRUc7SUFDTSxnQkFBZ0IsR0FBRyxLQUFLLEVBQWlCLENBQUM7SUFFMUMsT0FBTyxHQUFHLFlBQVksQ0FBQyxJQUFJLENBQUMsTUFBTSxDQUFDLENBQUM7SUFDcEMsS0FBSyxHQUFHLFlBQVksQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLENBQUM7SUFFaEMsd0JBQXdCLEdBQUcsYUFBYSxDQUFDLENBQUMsSUFBSSxDQUFDLE9BQU8sRUFBRSxJQUFJLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQ2hGLFNBQVMsQ0FBQyxDQUFDLENBQUMsTUFBTSxFQUFFLElBQUksQ0FBQyxFQUFFLEVBQUU7UUFDM0IsSUFBSSxHQUFtQyxDQUFDO1FBRXhDLElBQUksSUFBSSxFQUFFLENBQUM7WUFDVCxHQUFHLEdBQUcsRUFBRSxDQUFDLGFBQWEsQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDO1FBQ2hDLENBQUM7YUFBTSxJQUFJLE1BQU0sRUFBRSxDQUFDO1lBQ2xCLEdBQUcsR0FBRyxJQUFJLENBQ1IsS0FBSyxDQUFDLE1BQU0sRUFBRSxFQUFFLE1BQU0sRUFBRSxLQUFLLEVBQUUsQ0FBQyxDQUFDLElBQUksQ0FBQyxDQUFDLENBQUMsRUFBRSxFQUFFLENBQzFDLENBQUM7aUJBQ0UsSUFBSSxFQUFFO2lCQUNOLElBQUksQ0FBQyxDQUFDLENBQUMsRUFBRSxFQUFFLENBQUMsYUFBYSxDQUFDLENBQUMsQ0FBQyxDQUFDO2lCQUM3QixLQUFLLENBQUMsQ0FBQyxDQUFDLEVBQUUsRUFBRSxDQUFDLFdBQVcsQ0FBTyxDQUFDLENBQUMsQ0FBQyxDQUN0QyxDQUNGLENBQUMsSUFBSSxDQUFDLHFCQUFxQixFQUFFLENBQUMsQ0FBQztRQUNsQyxDQUFDO2FBQU0sQ0FBQztZQUNOLEdBQUcsR0FBRyxFQUFFLENBQUMsWUFBWSxFQUFRLENBQUMsQ0FBQztRQUNqQyxDQUFDO1FBRUQsT0FBTyxHQUFHLENBQUM7SUFDYixDQUFDLENBQUMsQ0FDSCxDQUFDO0lBRU8sWUFBWSxHQUFHLElBQUksQ0FBQyx3QkFBd0IsQ0FBQyxJQUFJLENBQUMsNkJBQTZCLEVBQUUsRUFBRSxvQkFBb0IsRUFBRSxFQUFFLFdBQVcsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDO0lBQzNILGlCQUFpQixHQUFHLFFBQVEsQ0FBQyxJQUFJLENBQUMsWUFBWSxDQUFDLENBQUM7SUFFaEQsT0FBTyxHQUFHLG1CQUFtQixDQUFDLEVBQUUsR0FBRyxFQUFFLElBQUksQ0FBQyx3QkFBd0IsRUFBRSxDQUFDLENBQUM7SUFFL0UsV0FBVztRQUNULElBQUksQ0FBQyxPQUFPLENBQUMsT0FBTyxFQUFFLENBQUM7SUFDekIsQ0FBQzt3R0FqRFUsc0JBQXNCOzRGQUF0QixzQkFBc0IsZ2VBVHZCOzs7O0dBSVQsNERBRWdDLG1CQUFtQixtS0FBRSwwQkFBMEI7OzRGQUdyRSxzQkFBc0I7a0JBWGxDLFNBQVM7bUJBQUM7b0JBQ1QsUUFBUSxFQUFFLGlCQUFpQjtvQkFDM0IsUUFBUSxFQUFFOzs7O0dBSVQ7b0JBQ0QsVUFBVSxFQUFFLElBQUk7b0JBQ2hCLE9BQU8sRUFBRSxDQUFDLHFCQUFxQixFQUFFLG1CQUFtQixFQUFFLDBCQUEwQixFQUFFLGdCQUFnQixDQUFDO29CQUNuRyxlQUFlLEVBQUUsdUJBQXVCLENBQUMsTUFBTTtpQkFDaEQiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDaGFuZ2VEZXRlY3Rpb25TdHJhdGVneSwgQ29tcG9uZW50LCBpbnB1dCwgT25EZXN0cm95IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBEYnhJbmplY3Rpb25Db21wb25lbnQgfSBmcm9tICdAZGVyZWVrYi9kYngtY29yZSc7XG5pbXBvcnQgeyBOZ1RlbXBsYXRlT3V0bGV0IH0gZnJvbSAnQGFuZ3VsYXIvY29tbW9uJztcbmltcG9ydCB7IERieExvYWRpbmdDb21wb25lbnQgfSBmcm9tICcuLi8uLi9sb2FkaW5nJztcbmltcG9ydCB7IE1heWJlLCBXZWJzaXRlVXJsV2l0aFByZWZpeCB9IGZyb20gJ0BkZXJlZWtiL3V0aWwnO1xuaW1wb3J0IHsgdG9PYnNlcnZhYmxlLCB0b1NpZ25hbCB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUvcnhqcy1pbnRlcm9wJztcbmltcG9ydCB7IGNvbWJpbmVMYXRlc3QsIGRpc3RpbmN0VW50aWxDaGFuZ2VkLCBmcm9tLCBPYnNlcnZhYmxlLCBvZiwgc2hhcmVSZXBsYXksIHN3aXRjaE1hcCB9IGZyb20gJ3J4anMnO1xuaW1wb3J0IHsgYmVnaW5Mb2FkaW5nLCBlcnJvclJlc3VsdCwgTG9hZGluZ1N0YXRlLCBsb2FkaW5nU3RhdGVDb250ZXh0LCBzdGFydFdpdGhCZWdpbkxvYWRpbmcsIHN1Y2Nlc3NSZXN1bHQsIHZhbHVlRnJvbUZpbmlzaGVkTG9hZGluZ1N0YXRlIH0gZnJvbSAnQGRlcmVla2Ivcnhqcyc7XG5pbXBvcnQgeyBEYnhaaXBCbG9iUHJldmlld0NvbXBvbmVudCB9IGZyb20gJy4vemlwLmJsb2IucHJldmlldy5jb21wb25lbnQnO1xuXG4vKipcbiAqIFVzZWQgdG8gZGlzcGxheSBhIGNvcnJlc3BvbmRpbmcgd2lkZ2V0IGJhc2VkIG9uIHRoZSBpbnB1dCBkYXRhLlxuICovXG5AQ29tcG9uZW50KHtcbiAgc2VsZWN0b3I6ICdkYngtemlwLXByZXZpZXcnLFxuICB0ZW1wbGF0ZTogYFxuICAgIDxkYngtbG9hZGluZyBbY29udGV4dF09XCJjb250ZXh0XCI+XG4gICAgICA8ZGJ4LXppcC1ibG9iLXByZXZpZXcgW2Rvd25sb2FkRmlsZU5hbWVdPVwiZG93bmxvYWRGaWxlTmFtZSgpXCIgW2Jsb2JdPVwiemlwRmlsZUJsb2JTaWduYWwoKVwiPjwvZGJ4LXppcC1ibG9iLXByZXZpZXc+XG4gICAgPC9kYngtbG9hZGluZz5cbiAgYCxcbiAgc3RhbmRhbG9uZTogdHJ1ZSxcbiAgaW1wb3J0czogW0RieEluamVjdGlvbkNvbXBvbmVudCwgRGJ4TG9hZGluZ0NvbXBvbmVudCwgRGJ4WmlwQmxvYlByZXZpZXdDb21wb25lbnQsIE5nVGVtcGxhdGVPdXRsZXRdLFxuICBjaGFuZ2VEZXRlY3Rpb246IENoYW5nZURldGVjdGlvblN0cmF0ZWd5Lk9uUHVzaFxufSlcbmV4cG9ydCBjbGFzcyBEYnhaaXBQcmV2aWV3Q29tcG9uZW50IGltcGxlbWVudHMgT25EZXN0cm95IHtcbiAgLyoqXG4gICAqIFRoZSBVUkwgdG8gZG93bmxvYWQgdGhlIHppcCBmaWxlIGZyb20sIGlmIGFwcGxpY2FibGUuXG4gICAqL1xuICByZWFkb25seSBzcmNVcmwgPSBpbnB1dDxNYXliZTxXZWJzaXRlVXJsV2l0aFByZWZpeD4+KCk7XG5cbiAgLyoqXG4gICAqIFRoZSBibG9iIHRvIHVzZSBmb3IgdGhlIHppcCBmaWxlLCBpZiBhcHBsaWNhYmxlLlxuICAgKi9cbiAgcmVhZG9ubHkgYmxvYiA9IGlucHV0PE1heWJlPEJsb2I+PigpO1xuXG4gIC8qKlxuICAgKiBUaGUgZmlsZSBuYW1lIHRvIHVzZSBmb3IgdGhlIHppcCBmaWxlLlxuICAgKi9cbiAgcmVhZG9ubHkgZG93bmxvYWRGaWxlTmFtZSA9IGlucHV0PE1heWJlPHN0cmluZz4+KCk7XG5cbiAgcmVhZG9ubHkgc3JjVXJsJCA9IHRvT2JzZXJ2YWJsZSh0aGlzLnNyY1VybCk7XG4gIHJlYWRvbmx5IGJsb2IkID0gdG9PYnNlcnZhYmxlKHRoaXMuYmxvYik7XG5cbiAgcmVhZG9ubHkgemlwRmlsZUJsb2JMb2FkaW5nU3RhdGUkID0gY29tYmluZUxhdGVzdChbdGhpcy5zcmNVcmwkLCB0aGlzLmJsb2IkXSkucGlwZShcbiAgICBzd2l0Y2hNYXAoKFtzcmNVcmwsIGJsb2JdKSA9PiB7XG4gICAgICBsZXQgb2JzOiBPYnNlcnZhYmxlPExvYWRpbmdTdGF0ZTxCbG9iPj47XG5cbiAgICAgIGlmIChibG9iKSB7XG4gICAgICAgIG9icyA9IG9mKHN1Y2Nlc3NSZXN1bHQoYmxvYikpO1xuICAgICAgfSBlbHNlIGlmIChzcmNVcmwpIHtcbiAgICAgICAgb2JzID0gZnJvbShcbiAgICAgICAgICBmZXRjaChzcmNVcmwsIHsgbWV0aG9kOiAnR0VUJyB9KS50aGVuKCh4KSA9PlxuICAgICAgICAgICAgeFxuICAgICAgICAgICAgICAuYmxvYigpXG4gICAgICAgICAgICAgIC50aGVuKCh5KSA9PiBzdWNjZXNzUmVzdWx0KHkpKVxuICAgICAgICAgICAgICAuY2F0Y2goKGUpID0+IGVycm9yUmVzdWx0PEJsb2I+KGUpKVxuICAgICAgICAgIClcbiAgICAgICAgKS5waXBlKHN0YXJ0V2l0aEJlZ2luTG9hZGluZygpKTtcbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIG9icyA9IG9mKGJlZ2luTG9hZGluZzxCbG9iPigpKTtcbiAgICAgIH1cblxuICAgICAgcmV0dXJuIG9icztcbiAgICB9KVxuICApO1xuXG4gIHJlYWRvbmx5IHppcEZpbGVCbG9iJCA9IHRoaXMuemlwRmlsZUJsb2JMb2FkaW5nU3RhdGUkLnBpcGUodmFsdWVGcm9tRmluaXNoZWRMb2FkaW5nU3RhdGUoKSwgZGlzdGluY3RVbnRpbENoYW5nZWQoKSwgc2hhcmVSZXBsYXkoMSkpO1xuICByZWFkb25seSB6aXBGaWxlQmxvYlNpZ25hbCA9IHRvU2lnbmFsKHRoaXMuemlwRmlsZUJsb2IkKTtcblxuICByZWFkb25seSBjb250ZXh0ID0gbG9hZGluZ1N0YXRlQ29udGV4dCh7IG9iczogdGhpcy56aXBGaWxlQmxvYkxvYWRpbmdTdGF0ZSQgfSk7XG5cbiAgbmdPbkRlc3Ryb3koKTogdm9pZCB7XG4gICAgdGhpcy5jb250ZXh0LmRlc3Ryb3koKTtcbiAgfVxufVxuIl19
@@ -0,0 +1,44 @@
1
+ import { ChangeDetectionStrategy, Component } from '@angular/core';
2
+ import { DbxDialogContentDirective, AbstractDialogDirective } from '../../interaction/dialog';
3
+ import { DbxZipPreviewComponent } from './zip.preview.component';
4
+ import * as i0 from "@angular/core";
5
+ export class DbxZipPreviewDialogComponent extends AbstractDialogDirective {
6
+ get srcUrl() {
7
+ return this.data.srcUrl;
8
+ }
9
+ get blob() {
10
+ return this.data.blob;
11
+ }
12
+ get downloadFileName() {
13
+ return this.data.downloadFileName;
14
+ }
15
+ static openDialog(matDialog, config) {
16
+ const dialogRef = matDialog.open(DbxZipPreviewDialogComponent, {
17
+ width: '80vw',
18
+ height: '80vh',
19
+ ...config,
20
+ data: config
21
+ });
22
+ return dialogRef;
23
+ }
24
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: DbxZipPreviewDialogComponent, deps: null, target: i0.ɵɵFactoryTarget.Component });
25
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: DbxZipPreviewDialogComponent, isStandalone: true, selector: "ng-component", usesInheritance: true, ngImport: i0, template: `
26
+ <dbx-dialog-content>
27
+ <dbx-zip-preview [srcUrl]="srcUrl" [blob]="blob" [downloadFileName]="downloadFileName"></dbx-zip-preview>
28
+ </dbx-dialog-content>
29
+ `, isInline: true, dependencies: [{ kind: "component", type: DbxZipPreviewComponent, selector: "dbx-zip-preview", inputs: ["srcUrl", "blob", "downloadFileName"] }, { kind: "directive", type: DbxDialogContentDirective, selector: "dbx-dialog-content,[dbxDialogContent],.dbx-dialog-content", inputs: ["width"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
30
+ }
31
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: DbxZipPreviewDialogComponent, decorators: [{
32
+ type: Component,
33
+ args: [{
34
+ template: `
35
+ <dbx-dialog-content>
36
+ <dbx-zip-preview [srcUrl]="srcUrl" [blob]="blob" [downloadFileName]="downloadFileName"></dbx-zip-preview>
37
+ </dbx-dialog-content>
38
+ `,
39
+ imports: [DbxZipPreviewComponent, DbxDialogContentDirective],
40
+ changeDetection: ChangeDetectionStrategy.OnPush,
41
+ standalone: true
42
+ }]
43
+ }] });
44
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiemlwLnByZXZpZXcuZGlhbG9nLmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uL3BhY2thZ2VzL2RieC13ZWIvc3JjL2xpYi9leHRlbnNpb24vemlwL3ppcC5wcmV2aWV3LmRpYWxvZy5jb21wb25lbnQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLHVCQUF1QixFQUFFLFNBQVMsRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUduRSxPQUFPLEVBQUUseUJBQXlCLEVBQUUsdUJBQXVCLEVBQUUsTUFBTSwwQkFBMEIsQ0FBQztBQUM5RixPQUFPLEVBQUUsc0JBQXNCLEVBQUUsTUFBTSx5QkFBeUIsQ0FBQzs7QUFrQmpFLE1BQU0sT0FBTyw0QkFBNkIsU0FBUSx1QkFBd0Q7SUFDeEcsSUFBSSxNQUFNO1FBQ1IsT0FBTyxJQUFJLENBQUMsSUFBSSxDQUFDLE1BQU0sQ0FBQztJQUMxQixDQUFDO0lBRUQsSUFBSSxJQUFJO1FBQ04sT0FBTyxJQUFJLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQztJQUN4QixDQUFDO0lBRUQsSUFBSSxnQkFBZ0I7UUFDbEIsT0FBTyxJQUFJLENBQUMsSUFBSSxDQUFDLGdCQUFnQixDQUFDO0lBQ3BDLENBQUM7SUFFRCxNQUFNLENBQUMsVUFBVSxDQUFDLFNBQW9CLEVBQUUsTUFBaUM7UUFDdkUsTUFBTSxTQUFTLEdBQUcsU0FBUyxDQUFDLElBQUksQ0FBQyw0QkFBNEIsRUFBRTtZQUM3RCxLQUFLLEVBQUUsTUFBTTtZQUNiLE1BQU0sRUFBRSxNQUFNO1lBQ2QsR0FBRyxNQUFNO1lBQ1QsSUFBSSxFQUFFLE1BQU07U0FDYixDQUFDLENBQUM7UUFFSCxPQUFPLFNBQVMsQ0FBQztJQUNuQixDQUFDO3dHQXRCVSw0QkFBNEI7NEZBQTVCLDRCQUE0QiwrRkFUN0I7Ozs7R0FJVCw0REFDUyxzQkFBc0IsNEdBQUUseUJBQXlCOzs0RkFJaEQsNEJBQTRCO2tCQVZ4QyxTQUFTO21CQUFDO29CQUNULFFBQVEsRUFBRTs7OztHQUlUO29CQUNELE9BQU8sRUFBRSxDQUFDLHNCQUFzQixFQUFFLHlCQUF5QixDQUFDO29CQUM1RCxlQUFlLEVBQUUsdUJBQXVCLENBQUMsTUFBTTtvQkFDL0MsVUFBVSxFQUFFLElBQUk7aUJBQ2pCIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ2hhbmdlRGV0ZWN0aW9uU3RyYXRlZ3ksIENvbXBvbmVudCB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgTWF0RGlhbG9nLCBNYXREaWFsb2dSZWYsIE1hdERpYWxvZ0NvbmZpZyB9IGZyb20gJ0Bhbmd1bGFyL21hdGVyaWFsL2RpYWxvZyc7XG5pbXBvcnQgeyBXZWJzaXRlVXJsV2l0aFByZWZpeCwgTWF5YmUgfSBmcm9tICdAZGVyZWVrYi91dGlsJztcbmltcG9ydCB7IERieERpYWxvZ0NvbnRlbnREaXJlY3RpdmUsIEFic3RyYWN0RGlhbG9nRGlyZWN0aXZlIH0gZnJvbSAnLi4vLi4vaW50ZXJhY3Rpb24vZGlhbG9nJztcbmltcG9ydCB7IERieFppcFByZXZpZXdDb21wb25lbnQgfSBmcm9tICcuL3ppcC5wcmV2aWV3LmNvbXBvbmVudCc7XG5cbmV4cG9ydCBpbnRlcmZhY2UgRGJ4WmlwUHJldmlld0RpYWxvZ0NvbmZpZyBleHRlbmRzIE9taXQ8TWF0RGlhbG9nQ29uZmlnLCAnZGF0YSc+IHtcbiAgcmVhZG9ubHkgc3JjVXJsPzogTWF5YmU8V2Vic2l0ZVVybFdpdGhQcmVmaXg+O1xuICByZWFkb25seSBibG9iPzogTWF5YmU8QmxvYj47XG4gIHJlYWRvbmx5IGRvd25sb2FkRmlsZU5hbWU/OiBNYXliZTxzdHJpbmc+O1xufVxuXG5AQ29tcG9uZW50KHtcbiAgdGVtcGxhdGU6IGBcbiAgICA8ZGJ4LWRpYWxvZy1jb250ZW50PlxuICAgICAgPGRieC16aXAtcHJldmlldyBbc3JjVXJsXT1cInNyY1VybFwiIFtibG9iXT1cImJsb2JcIiBbZG93bmxvYWRGaWxlTmFtZV09XCJkb3dubG9hZEZpbGVOYW1lXCI+PC9kYngtemlwLXByZXZpZXc+XG4gICAgPC9kYngtZGlhbG9nLWNvbnRlbnQ+XG4gIGAsXG4gIGltcG9ydHM6IFtEYnhaaXBQcmV2aWV3Q29tcG9uZW50LCBEYnhEaWFsb2dDb250ZW50RGlyZWN0aXZlXSxcbiAgY2hhbmdlRGV0ZWN0aW9uOiBDaGFuZ2VEZXRlY3Rpb25TdHJhdGVneS5PblB1c2gsXG4gIHN0YW5kYWxvbmU6IHRydWVcbn0pXG5leHBvcnQgY2xhc3MgRGJ4WmlwUHJldmlld0RpYWxvZ0NvbXBvbmVudCBleHRlbmRzIEFic3RyYWN0RGlhbG9nRGlyZWN0aXZlPHZvaWQsIERieFppcFByZXZpZXdEaWFsb2dDb25maWc+IHtcbiAgZ2V0IHNyY1VybCgpIHtcbiAgICByZXR1cm4gdGhpcy5kYXRhLnNyY1VybDtcbiAgfVxuXG4gIGdldCBibG9iKCkge1xuICAgIHJldHVybiB0aGlzLmRhdGEuYmxvYjtcbiAgfVxuXG4gIGdldCBkb3dubG9hZEZpbGVOYW1lKCkge1xuICAgIHJldHVybiB0aGlzLmRhdGEuZG93bmxvYWRGaWxlTmFtZTtcbiAgfVxuXG4gIHN0YXRpYyBvcGVuRGlhbG9nKG1hdERpYWxvZzogTWF0RGlhbG9nLCBjb25maWc6IERieFppcFByZXZpZXdEaWFsb2dDb25maWcpOiBNYXREaWFsb2dSZWY8RGJ4WmlwUHJldmlld0RpYWxvZ0NvbXBvbmVudCwgdm9pZD4ge1xuICAgIGNvbnN0IGRpYWxvZ1JlZiA9IG1hdERpYWxvZy5vcGVuKERieFppcFByZXZpZXdEaWFsb2dDb21wb25lbnQsIHtcbiAgICAgIHdpZHRoOiAnODB2dycsXG4gICAgICBoZWlnaHQ6ICc4MHZoJyxcbiAgICAgIC4uLmNvbmZpZyxcbiAgICAgIGRhdGE6IGNvbmZpZ1xuICAgIH0pO1xuXG4gICAgcmV0dXJuIGRpYWxvZ1JlZjtcbiAgfVxufVxuIl19
@@ -0,0 +1,104 @@
1
+ import { Component, viewChild, ElementRef, ChangeDetectionStrategy, computed, inject, SecurityContext, input, effect, signal } from '@angular/core';
2
+ import { toObservable } from '@angular/core/rxjs-interop';
3
+ import { combineLatest } from 'rxjs';
4
+ import { AbstractSubscriptionDirective } from '@dereekb/dbx-core';
5
+ import { DomSanitizer } from '@angular/platform-browser';
6
+ import { browserObjectUrlRef } from '@dereekb/browser';
7
+ import * as i0 from "@angular/core";
8
+ export class DbxEmbedComponent extends AbstractSubscriptionDirective {
9
+ _browserObjectUrlRef = browserObjectUrlRef();
10
+ sanitizer = inject(DomSanitizer);
11
+ root = viewChild('root', { read: ElementRef });
12
+ sanitizeUrl = input(false);
13
+ srcUrl = input();
14
+ type = input();
15
+ /**
16
+ * The input src blob or media source to use.
17
+ *
18
+ * If set, the srcUrl will be updated with the browser object URL.
19
+ */
20
+ srcBlob = input();
21
+ srcUrlFromBlob = signal(undefined);
22
+ typeFromBlob = signal(undefined);
23
+ srcBlobEffect = effect(() => {
24
+ const blob = this.srcBlob();
25
+ this.srcUrlFromBlob.set(this._browserObjectUrlRef.createBrowserUrl(blob));
26
+ this.typeFromBlob.set(blob?.type);
27
+ }, {
28
+ allowSignalWrites: true
29
+ });
30
+ srcUrlSignal = computed(() => {
31
+ const srcUrl = this.srcUrl();
32
+ const srcUrlFromBlob = this.srcUrlFromBlob();
33
+ const baseUrl = srcUrl ?? srcUrlFromBlob;
34
+ let url = baseUrl;
35
+ const sanitizeUrl = this.sanitizeUrl();
36
+ if (url && typeof url === 'string' && sanitizeUrl) {
37
+ url = this.sanitizer.bypassSecurityTrustResourceUrl(url);
38
+ }
39
+ return url;
40
+ });
41
+ typeSignal = computed(() => {
42
+ const type = this.type();
43
+ const typeFromBlob = this.typeFromBlob();
44
+ return type ?? typeFromBlob;
45
+ });
46
+ root$ = toObservable(this.root);
47
+ srcUrl$ = toObservable(this.srcUrlSignal);
48
+ type$ = toObservable(this.typeSignal);
49
+ constructor() {
50
+ super();
51
+ this.sub = combineLatest([this.srcUrl$, this.root$, this.type$]).subscribe(([srcUrl, root, type]) => {
52
+ const element = root?.nativeElement;
53
+ if (element) {
54
+ // remove all embeds from the element
55
+ element?.childNodes.forEach((x) => element.removeChild(x));
56
+ if (srcUrl) {
57
+ // NOTE: We do this because of the following chromium behavior:
58
+ // https://issues.chromium.org/issues/40508296
59
+ //
60
+ // Embed cannot have src change dynamically, so we create a new embed element each time the src changes.
61
+ //
62
+ const embed = document.createElement('embed');
63
+ let url = undefined;
64
+ if (srcUrl != null && typeof srcUrl !== 'string') {
65
+ url = this.sanitizer.sanitize(SecurityContext.URL, srcUrl);
66
+ }
67
+ else {
68
+ url = srcUrl;
69
+ }
70
+ embed.setAttribute('src', url ?? '');
71
+ // only set the type if it is presented
72
+ if (type) {
73
+ embed.setAttribute('type', type);
74
+ // if the type is an image, also add the embed-image class
75
+ if (type.startsWith('image/')) {
76
+ embed.setAttribute('class', 'embed-image');
77
+ }
78
+ }
79
+ element.appendChild(embed);
80
+ }
81
+ }
82
+ });
83
+ }
84
+ ngOnDestroy() {
85
+ super.ngOnDestroy();
86
+ this._browserObjectUrlRef.destroy();
87
+ }
88
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: DbxEmbedComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
89
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.2.0", version: "18.2.13", type: DbxEmbedComponent, isStandalone: true, selector: "dbx-embed", inputs: { sanitizeUrl: { classPropertyName: "sanitizeUrl", publicName: "sanitizeUrl", isSignal: true, isRequired: false, transformFunction: null }, srcUrl: { classPropertyName: "srcUrl", publicName: "srcUrl", isSignal: true, isRequired: false, transformFunction: null }, type: { classPropertyName: "type", publicName: "type", isSignal: true, isRequired: false, transformFunction: null }, srcBlob: { classPropertyName: "srcBlob", publicName: "srcBlob", isSignal: true, isRequired: false, transformFunction: null } }, viewQueries: [{ propertyName: "root", first: true, predicate: ["root"], descendants: true, read: ElementRef, isSignal: true }], usesInheritance: true, ngImport: i0, template: `
90
+ <span #root></span>
91
+ `, isInline: true, changeDetection: i0.ChangeDetectionStrategy.OnPush });
92
+ }
93
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: DbxEmbedComponent, decorators: [{
94
+ type: Component,
95
+ args: [{
96
+ selector: 'dbx-embed',
97
+ template: `
98
+ <span #root></span>
99
+ `,
100
+ changeDetection: ChangeDetectionStrategy.OnPush,
101
+ standalone: true
102
+ }]
103
+ }], ctorParameters: () => [] });
104
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"embed.component.js","sourceRoot":"","sources":["../../../../../../../packages/dbx-web/src/lib/interaction/iframe/embed.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { Component, viewChild, ElementRef, ChangeDetectionStrategy, computed, inject, SecurityContext, input, effect, signal, OnDestroy } from '@angular/core';\nimport { toObservable } from '@angular/core/rxjs-interop';\nimport { combineLatest } from 'rxjs';\nimport { ContentTypeMimeType, Maybe } from '@dereekb/util';\nimport { AbstractSubscriptionDirective } from '@dereekb/dbx-core';\nimport { DomSanitizer, SafeResourceUrl } from '@angular/platform-browser';\nimport { browserObjectUrlRef } from '@dereekb/browser';\n\n@Component({\n  selector: 'dbx-embed',\n  template: `\n    <span #root></span>\n  `,\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  standalone: true\n})\nexport class DbxEmbedComponent extends AbstractSubscriptionDirective implements OnDestroy {\n  private readonly _browserObjectUrlRef = browserObjectUrlRef();\n\n  readonly sanitizer = inject(DomSanitizer);\n\n  readonly root = viewChild<string, ElementRef<HTMLSpanElement>>('root', { read: ElementRef });\n\n  readonly sanitizeUrl = input<Maybe<boolean>>(false);\n\n  readonly srcUrl = input<Maybe<string | SafeResourceUrl>>();\n  readonly type = input<Maybe<ContentTypeMimeType | string>>();\n\n  /**\n   * The input src blob or media source to use.\n   *\n   * If set, the srcUrl will be updated with the browser object URL.\n   */\n  readonly srcBlob = input<Maybe<Blob>>();\n  readonly srcUrlFromBlob = signal<Maybe<string>>(undefined);\n  readonly typeFromBlob = signal<Maybe<string>>(undefined);\n\n  readonly srcBlobEffect = effect(\n    () => {\n      const blob = this.srcBlob();\n\n      this.srcUrlFromBlob.set(this._browserObjectUrlRef.createBrowserUrl(blob));\n      this.typeFromBlob.set(blob?.type);\n    },\n    {\n      allowSignalWrites: true\n    }\n  );\n\n  readonly srcUrlSignal = computed(() => {\n    const srcUrl = this.srcUrl();\n    const srcUrlFromBlob = this.srcUrlFromBlob();\n\n    const baseUrl = srcUrl ?? srcUrlFromBlob;\n\n    let url: Maybe<string | SafeResourceUrl> = baseUrl;\n    const sanitizeUrl = this.sanitizeUrl();\n\n    if (url && typeof url === 'string' && sanitizeUrl) {\n      url = this.sanitizer.bypassSecurityTrustResourceUrl(url);\n    }\n\n    return url;\n  });\n\n  readonly typeSignal = computed(() => {\n    const type = this.type();\n    const typeFromBlob = this.typeFromBlob();\n    return type ?? typeFromBlob;\n  });\n\n  readonly root$ = toObservable(this.root);\n  readonly srcUrl$ = toObservable(this.srcUrlSignal);\n  readonly type$ = toObservable(this.typeSignal);\n\n  constructor() {\n    super();\n\n    this.sub = combineLatest([this.srcUrl$, this.root$, this.type$]).subscribe(([srcUrl, root, type]) => {\n      const element = root?.nativeElement;\n\n      if (element) {\n        // remove all embeds from the element\n        element?.childNodes.forEach((x) => element.removeChild(x));\n\n        if (srcUrl) {\n          // NOTE: We do this because of the following chromium behavior:\n          // https://issues.chromium.org/issues/40508296\n          //\n          // Embed cannot have src change dynamically, so we create a new embed element each time the src changes.\n          //\n          const embed = document.createElement('embed');\n\n          let url: Maybe<string> = undefined;\n\n          if (srcUrl != null && typeof srcUrl !== 'string') {\n            url = this.sanitizer.sanitize(SecurityContext.URL, srcUrl);\n          } else {\n            url = srcUrl;\n          }\n\n          embed.setAttribute('src', url ?? '');\n\n          // only set the type if it is presented\n          if (type) {\n            embed.setAttribute('type', type);\n\n            // if the type is an image, also add the embed-image class\n            if (type.startsWith('image/')) {\n              embed.setAttribute('class', 'embed-image');\n            }\n          }\n\n          element.appendChild(embed);\n        }\n      }\n    });\n  }\n\n  override ngOnDestroy(): void {\n    super.ngOnDestroy();\n    this._browserObjectUrlRef.destroy();\n  }\n}\n"]}