@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.
- package/esm2022/lib/extension/download/blob/download.blob.button.component.mjs +86 -0
- package/esm2022/lib/extension/download/blob/index.mjs +2 -0
- package/esm2022/lib/extension/download/index.mjs +2 -1
- package/esm2022/lib/extension/download/text/download.text.component.mjs +20 -18
- package/esm2022/lib/extension/index.mjs +3 -1
- package/esm2022/lib/extension/model/model.tracker.mjs +1 -1
- package/esm2022/lib/extension/model/model.tracker.view.storage.mjs +3 -3
- package/esm2022/lib/extension/preview/index.mjs +4 -0
- package/esm2022/lib/extension/preview/webfilepreview.mjs +2 -0
- package/esm2022/lib/extension/preview/webfilepreview.service.mjs +60 -0
- package/esm2022/lib/extension/preview/webfilepreview.service.preset.mjs +31 -0
- package/esm2022/lib/extension/widget/widget.service.mjs +1 -1
- package/esm2022/lib/extension/zip/index.mjs +5 -0
- package/esm2022/lib/extension/zip/zip.blob.mjs +16 -0
- package/esm2022/lib/extension/zip/zip.blob.preview.component.mjs +156 -0
- package/esm2022/lib/extension/zip/zip.blob.preview.list.component.mjs +121 -0
- package/esm2022/lib/extension/zip/zip.preview.component.mjs +71 -0
- package/esm2022/lib/extension/zip/zip.preview.dialog.component.mjs +44 -0
- package/esm2022/lib/interaction/iframe/embed.component.mjs +104 -0
- package/esm2022/lib/interaction/iframe/embed.dialog.component.mjs +43 -0
- package/esm2022/lib/interaction/iframe/iframe.component.mjs +7 -4
- package/esm2022/lib/interaction/iframe/iframe.dialog.component.mjs +37 -0
- package/esm2022/lib/interaction/iframe/index.mjs +4 -1
- package/fesm2022/dereekb-dbx-web.mjs +4596 -3899
- package/fesm2022/dereekb-dbx-web.mjs.map +1 -1
- package/lib/extension/_extension.scss +19 -0
- package/lib/extension/download/blob/download.blob.button.component.d.ts +55 -0
- package/lib/extension/download/blob/index.d.ts +1 -0
- package/lib/extension/download/index.d.ts +1 -0
- package/lib/extension/download/text/download.text.component.d.ts +7 -7
- package/lib/extension/index.d.ts +2 -0
- package/lib/extension/model/model.tracker.d.ts +3 -3
- package/lib/extension/preview/index.d.ts +3 -0
- package/lib/extension/preview/webfilepreview.d.ts +13 -0
- package/lib/extension/preview/webfilepreview.service.d.ts +31 -0
- package/lib/extension/preview/webfilepreview.service.preset.d.ts +13 -0
- package/lib/extension/widget/widget.service.d.ts +1 -1
- package/lib/extension/zip/_zip.scss +35 -0
- package/lib/extension/zip/index.d.ts +4 -0
- package/lib/extension/zip/zip.blob.d.ts +9 -0
- package/lib/extension/zip/zip.blob.preview.component.d.ts +54 -0
- package/lib/extension/zip/zip.blob.preview.list.component.d.ts +25 -0
- package/lib/extension/zip/zip.preview.component.d.ts +31 -0
- package/lib/extension/zip/zip.preview.dialog.component.d.ts +17 -0
- package/lib/interaction/iframe/_iframe.scss +13 -0
- package/lib/interaction/iframe/embed.component.d.ts +31 -0
- package/lib/interaction/iframe/embed.dialog.component.d.ts +17 -0
- package/lib/interaction/iframe/iframe.component.d.ts +0 -1
- package/lib/interaction/iframe/iframe.dialog.component.d.ts +13 -0
- package/lib/interaction/iframe/index.d.ts +3 -0
- package/lib/layout/style/_style.scss +2 -1
- package/lib/style/_config.scss +7 -1
- 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":"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","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"]}
|