@dereekb/dbx-firebase 12.4.5 → 12.5.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (52) hide show
  1. package/esm2022/lib/auth/login/login.button.component.mjs +1 -1
  2. package/esm2022/lib/auth/login/login.email.content.component.mjs +2 -2
  3. package/esm2022/lib/development/development.scheduler.list.component.mjs +1 -1
  4. package/esm2022/lib/firestore/firebase.firestore.providers.mjs +21 -2
  5. package/esm2022/lib/model/loader/collection.loader.instance.mjs +65 -5
  6. package/esm2022/lib/model/modules/store/index.mjs +2 -1
  7. package/esm2022/lib/model/modules/store/store.collection.crud.mjs +28 -0
  8. package/esm2022/lib/model/modules/store/store.collection.directive.mjs +20 -2
  9. package/esm2022/lib/model/modules/store/store.collection.mjs +14 -1
  10. package/esm2022/lib/modules/index.mjs +2 -1
  11. package/esm2022/lib/modules/storagefile/container/index.mjs +7 -0
  12. package/esm2022/lib/modules/storagefile/container/storagefile.upload.action.handler.directive.mjs +118 -0
  13. package/esm2022/lib/modules/storagefile/container/storagefile.upload.error.mjs +12 -0
  14. package/esm2022/lib/modules/storagefile/container/storagefile.upload.handler.mjs +260 -0
  15. package/esm2022/lib/modules/storagefile/container/storagefile.upload.initialize.document.directive.mjs +39 -0
  16. package/esm2022/lib/modules/storagefile/container/storagefile.upload.store.directive.mjs +38 -0
  17. package/esm2022/lib/modules/storagefile/container/storagefile.upload.sync.directive.mjs +36 -0
  18. package/esm2022/lib/modules/storagefile/index.mjs +4 -0
  19. package/esm2022/lib/modules/storagefile/storagefile.upload.module.mjs +83 -0
  20. package/esm2022/lib/modules/storagefile/store/index.mjs +6 -0
  21. package/esm2022/lib/modules/storagefile/store/storagefile.collection.store.directive.mjs +22 -0
  22. package/esm2022/lib/modules/storagefile/store/storagefile.collection.store.mjs +18 -0
  23. package/esm2022/lib/modules/storagefile/store/storagefile.document.store.directive.mjs +21 -0
  24. package/esm2022/lib/modules/storagefile/store/storagefile.document.store.mjs +22 -0
  25. package/esm2022/lib/modules/storagefile/store/storagefile.upload.store.mjs +63 -0
  26. package/esm2022/lib/storage/firebase.storage.module.mjs +2 -2
  27. package/fesm2022/dereekb-dbx-firebase.mjs +839 -30
  28. package/fesm2022/dereekb-dbx-firebase.mjs.map +1 -1
  29. package/lib/firestore/firebase.firestore.providers.d.ts +14 -1
  30. package/lib/model/loader/collection.loader.instance.d.ts +26 -1
  31. package/lib/model/modules/store/index.d.ts +1 -0
  32. package/lib/model/modules/store/store.collection.crud.d.ts +21 -0
  33. package/lib/model/modules/store/store.collection.d.ts +19 -2
  34. package/lib/model/modules/store/store.collection.directive.d.ts +12 -2
  35. package/lib/modules/index.d.ts +1 -0
  36. package/lib/modules/storagefile/container/index.d.ts +6 -0
  37. package/lib/modules/storagefile/container/storagefile.upload.action.handler.directive.d.ts +47 -0
  38. package/lib/modules/storagefile/container/storagefile.upload.error.d.ts +9 -0
  39. package/lib/modules/storagefile/container/storagefile.upload.handler.d.ts +183 -0
  40. package/lib/modules/storagefile/container/storagefile.upload.initialize.document.directive.d.ts +14 -0
  41. package/lib/modules/storagefile/container/storagefile.upload.store.directive.d.ts +20 -0
  42. package/lib/modules/storagefile/container/storagefile.upload.sync.directive.d.ts +18 -0
  43. package/lib/modules/storagefile/index.d.ts +3 -0
  44. package/lib/modules/storagefile/storagefile.upload.module.d.ts +29 -0
  45. package/lib/modules/storagefile/store/index.d.ts +5 -0
  46. package/lib/modules/storagefile/store/storagefile.collection.store.d.ts +10 -0
  47. package/lib/modules/storagefile/store/storagefile.collection.store.directive.d.ts +9 -0
  48. package/lib/modules/storagefile/store/storagefile.document.store.d.ts +15 -0
  49. package/lib/modules/storagefile/store/storagefile.document.store.directive.d.ts +9 -0
  50. package/lib/modules/storagefile/store/storagefile.upload.store.d.ts +166 -0
  51. package/lib/storage/firebase.storage.module.d.ts +1 -1
  52. package/package.json +1 -1
@@ -0,0 +1,260 @@
1
+ import { catchError, map, Observable, of, shareReplay } from 'rxjs';
2
+ import { runAsyncTasksForValues, separateValues } from '@dereekb/util';
3
+ import { MultiSubscriptionObject } from '@dereekb/rxjs';
4
+ /**
5
+ * Default implementation of StorageFileUploadHandler.
6
+ */
7
+ export function storageFileUploadHandler(config) {
8
+ const { storageService, storageFileUploadConfigFactory } = config;
9
+ let resumable;
10
+ return {
11
+ uploadFile: async (file) => {
12
+ const storageFileUploadConfig = await storageFileUploadConfigFactory(file);
13
+ const { storagePath } = storageFileUploadConfig;
14
+ const storageAccessorFile = storageService.file(storagePath);
15
+ const upload = new Observable((x) => {
16
+ if (storageAccessorFile.uploadResumable) {
17
+ resumable = storageAccessorFile.uploadResumable(file);
18
+ // subscribe to the event by piping this observable to it
19
+ resumable
20
+ .streamSnapshotEvents()
21
+ .pipe(map((x) => {
22
+ const { bytesTransferred, totalBytes } = x;
23
+ const progress = {
24
+ file,
25
+ fileRef: storageAccessorFile,
26
+ uploadRef: resumable,
27
+ bytesTransferred,
28
+ totalBytes,
29
+ progress: bytesTransferred / totalBytes
30
+ };
31
+ return progress;
32
+ }))
33
+ .pipe(catchError((error) => {
34
+ // if an error occurs, catch it and emit it as a progress
35
+ const progress = {
36
+ file,
37
+ fileRef: storageAccessorFile,
38
+ uploadRef: resumable,
39
+ error,
40
+ failed: true
41
+ };
42
+ return of(progress);
43
+ }))
44
+ .subscribe(x);
45
+ }
46
+ else {
47
+ throw new Error('uploadResumable() function was unavailable.');
48
+ }
49
+ }).pipe(shareReplay(1));
50
+ const instance = {
51
+ upload,
52
+ taskRef: storageAccessorFile,
53
+ pause: () => resumable?.pause() ?? false,
54
+ resume: () => resumable?.resume() ?? false,
55
+ cancel: () => resumable?.cancel() ?? false
56
+ };
57
+ return instance;
58
+ }
59
+ };
60
+ }
61
+ /**
62
+ * Uploads files using the provided upload handler and files.
63
+ *
64
+ * An observable is returned that emits the latest file events from any file that is being uploaded.
65
+ *
66
+ * @param input
67
+ * @returns
68
+ */
69
+ export function storageFileUploadFiles(input) {
70
+ const { uploadHandler, files, maxParallelUploads: inputMaxParallelUploads } = input;
71
+ const maxParallelTasks = inputMaxParallelUploads ?? 3;
72
+ const multiUploadsSubscriptionObject = new MultiSubscriptionObject();
73
+ // begin the upload for each file
74
+ const allFiles = Array.from(files);
75
+ // unsubscribe from all previous uploads
76
+ multiUploadsSubscriptionObject.unsub();
77
+ const allFilesAndLatestProgress = new Array(allFiles.length);
78
+ const allFilesAndDetails = allFiles.map((file) => ({ file }));
79
+ const overallProgressPerCompletedFile = (1 / allFilesAndLatestProgress.length);
80
+ /**
81
+ * Once set, any new file upload task that hits this will return an cancel failure.
82
+ */
83
+ let flaggedCancel = false;
84
+ const cancel = () => {
85
+ flaggedCancel = true;
86
+ };
87
+ const upload = new Observable((subscriber) => {
88
+ const overallStartTime = new Date();
89
+ const incompleteFileFileIndexes = new Set(allFiles.map((_, index) => index));
90
+ const activeFileIndexes = new Set();
91
+ const doneFileIndexes = new Set();
92
+ let latestOverallProgress = 0;
93
+ function onStartFileUpload(index, uploadInstance) {
94
+ activeFileIndexes.add(index);
95
+ allFilesAndDetails[index].startTime = new Date();
96
+ allFilesAndDetails[index].uploadInstance = uploadInstance;
97
+ allFilesAndDetails[index].fileRef = uploadInstance.taskRef;
98
+ }
99
+ function onStartFileUploadFlaggedCancelled(index) {
100
+ allFilesAndDetails[index].startTime = new Date();
101
+ // immediately mark it done
102
+ _markFileUploadDone(index, true);
103
+ // emit new progress event
104
+ _emitEvent();
105
+ }
106
+ function _markFileUploadDone(fileIndex, finalError) {
107
+ doneFileIndexes.add(fileIndex); // add to done file indexes
108
+ activeFileIndexes.delete(fileIndex); // remove from active file indexes if it exists
109
+ incompleteFileFileIndexes.delete(fileIndex); // remove from incomplete file indexes
110
+ // update details
111
+ allFilesAndDetails[fileIndex].endTime = new Date();
112
+ const error = finalError ?? allFilesAndDetails[fileIndex].error;
113
+ allFilesAndDetails[fileIndex].success = !error;
114
+ allFilesAndDetails[fileIndex].error = error;
115
+ }
116
+ function updateUploadProgress(input) {
117
+ const { index: fileIndex, nextProgress, fileUploadTaskDone: inputFileUploadTaskDone, nonProgressError } = input;
118
+ const error = nonProgressError ?? nextProgress?.error;
119
+ const fileUploadTaskDone = inputFileUploadTaskDone ?? Boolean(error);
120
+ // the task may already be done, as after a progress-related error the complete task can get called.
121
+ const isTaskAlreadyDone = allFilesAndDetails[fileIndex].endTime != null;
122
+ if (!isTaskAlreadyDone) {
123
+ let nextOverallProgress = latestOverallProgress;
124
+ const nextProgressPercent = fileUploadTaskDone ? 100 : (nextProgress?.progress ?? 0) * 100;
125
+ // update the overall progress percentage
126
+ if (nextProgressPercent) {
127
+ // update the overall percentage
128
+ const previousProgress = allFilesAndLatestProgress[fileIndex];
129
+ const previousProgressPercent = previousProgress?.progress != null ? previousProgress.progress * 100 : 0;
130
+ const progressPercentChange = nextProgressPercent - previousProgressPercent;
131
+ // increase overall progress by the change
132
+ nextOverallProgress += progressPercentChange * overallProgressPerCompletedFile;
133
+ }
134
+ // update the file progress
135
+ if (nextProgress) {
136
+ // update the latest FileProgress
137
+ allFilesAndLatestProgress[fileIndex] = nextProgress;
138
+ // only set fileRef once
139
+ if (!allFilesAndDetails[fileIndex].fileRef) {
140
+ allFilesAndDetails[fileIndex].fileRef = nextProgress.fileRef;
141
+ }
142
+ }
143
+ // if complete, update the indexes and details
144
+ if (fileUploadTaskDone) {
145
+ _markFileUploadDone(fileIndex, error);
146
+ }
147
+ // update the overall progress
148
+ latestOverallProgress = nextOverallProgress;
149
+ // emit the event to send it
150
+ _emitEvent(nextProgress);
151
+ }
152
+ }
153
+ function _emitEvent(nextProgress) {
154
+ const isComplete = incompleteFileFileIndexes.size === 0;
155
+ let overallProgress = latestOverallProgress;
156
+ let result = undefined;
157
+ if (isComplete) {
158
+ overallProgress = 100; // set to 100%
159
+ const overallEndTime = new Date();
160
+ const fileResults = allFiles.map((file, index) => {
161
+ const result = {
162
+ startTime: allFilesAndDetails[index].startTime,
163
+ endTime: allFilesAndDetails[index].endTime,
164
+ file,
165
+ fileRef: allFilesAndDetails[index].fileRef,
166
+ success: !allFilesAndDetails[index].error,
167
+ error: allFilesAndDetails[index].error
168
+ };
169
+ return result;
170
+ });
171
+ const { included: successFileResults, excluded: errorFileResults } = separateValues(fileResults, (x) => x.success);
172
+ // all are done, set the result on the next event
173
+ result = {
174
+ startTime: overallStartTime,
175
+ endTime: overallEndTime,
176
+ successFileResults,
177
+ errorFileResults,
178
+ fileResults
179
+ };
180
+ }
181
+ const nextEvent = {
182
+ allFiles,
183
+ isComplete,
184
+ overallProgress,
185
+ uploadProgress: nextProgress,
186
+ incompleteFileCount: incompleteFileFileIndexes.size,
187
+ activeFileCount: activeFileIndexes.size,
188
+ doneFileCount: doneFileIndexes.size,
189
+ result
190
+ };
191
+ subscriber.next(nextEvent);
192
+ }
193
+ async function runUploadTaskForFile([file, index]) {
194
+ if (flaggedCancel) {
195
+ onStartFileUploadFlaggedCancelled(index);
196
+ return Promise.resolve();
197
+ }
198
+ return new Promise((resolve) => {
199
+ const updateFileUploadProgress = (nextProgress) => {
200
+ // update the progress
201
+ updateUploadProgress({
202
+ index,
203
+ nextProgress
204
+ });
205
+ };
206
+ const updateFileUploadProgressWithUncaughtError = (error) => {
207
+ // error occurred, update the progress with the error
208
+ updateUploadProgress({
209
+ index,
210
+ nonProgressError: error,
211
+ fileUploadTaskDone: true
212
+ });
213
+ // always resolve, never reject
214
+ resolve();
215
+ };
216
+ const completeFileUploadProgress = () => {
217
+ updateUploadProgress({
218
+ index,
219
+ fileUploadTaskDone: true
220
+ });
221
+ resolve();
222
+ };
223
+ // upload the file, subscribe to the progress
224
+ try {
225
+ uploadHandler
226
+ .uploadFile(file)
227
+ .then((uploadInstance) => {
228
+ // add to active file indexes
229
+ onStartFileUpload(index, uploadInstance);
230
+ const uploadSubscription = uploadInstance.upload.subscribe({
231
+ next: updateFileUploadProgress,
232
+ error: updateFileUploadProgressWithUncaughtError,
233
+ complete: completeFileUploadProgress
234
+ });
235
+ multiUploadsSubscriptionObject.addSubs(uploadSubscription);
236
+ })
237
+ .catch(updateFileUploadProgressWithUncaughtError);
238
+ }
239
+ catch (error) {
240
+ updateFileUploadProgressWithUncaughtError(error);
241
+ }
242
+ });
243
+ }
244
+ // run upload task for each file
245
+ const fileTuples = allFiles.map((file, index) => [file, index]);
246
+ runAsyncTasksForValues(fileTuples, runUploadTaskForFile, {
247
+ maxParallelTasks,
248
+ retriesAllowed: 0 // no retries allowed
249
+ }).then(() => {
250
+ // all tasks are finished. Complete the subscriber.
251
+ subscriber.complete();
252
+ });
253
+ }).pipe(shareReplay(1));
254
+ const instance = {
255
+ cancel,
256
+ upload
257
+ };
258
+ return instance;
259
+ }
260
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"storagefile.upload.handler.js","sourceRoot":"","sources":["../../../../../../../../packages/dbx-firebase/src/lib/modules/storagefile/container/storagefile.upload.handler.ts"],"names":[],"mappings":"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","sourcesContent":["import { catchError, map, Observable, of, shareReplay } from 'rxjs';\nimport { DbxFirebaseStorageFileUploadStoreFileProgress } from '../store';\nimport { DbxFirebaseStorageService } from '../../../storage/firebase.storage.service';\nimport { IndexNumber, Maybe, PercentDecimal, PercentNumber, PromiseOrValue, runAsyncTasksForValues, separateValues } from '@dereekb/util';\nimport { MultiSubscriptionObject } from '@dereekb/rxjs';\nimport { FirebaseStorageAccessorFile, StorageCustomMetadata, StoragePathInput, StorageUploadOptions, StorageUploadTask } from '@dereekb/firebase';\n\n/**\n * Creates a new observable for uploading a file.\n */\nexport type StorageFileUploadHandlerFunction = (file: File) => Promise<StorageFileUploadHandlerInstance>;\n\nexport interface StorageFileUploadHandlerInstance extends Pick<StorageUploadTask<FirebaseStorageAccessorFile>, 'taskRef' | 'pause' | 'resume' | 'cancel'> {\n  /**\n   * The upload observable.\n   *\n   * Must be subscribed to in order for the upload to begin.\n   */\n  readonly upload: Observable<DbxFirebaseStorageFileUploadStoreFileProgress>;\n}\n\n/**\n * Handles uploading files.\n */\nexport interface StorageFileUploadHandler {\n  /**\n   * Uploads a file, and returns the file progress as it is uploading.\n   */\n  readonly uploadFile: StorageFileUploadHandlerFunction;\n}\n\n/**\n * Configuration for a single file upload.\n */\nexport interface StorageFileUploadConfig {\n  /**\n   * Path for where to upload the file to\n   */\n  readonly storagePath: StoragePathInput;\n  /**\n   * Upload options for the file.\n   *\n   * Resumable is not supported.\n   */\n  readonly uploadOptions?: StorageFileUploadConfigOptions;\n  /**\n   * Custom metadata for the file.\n   *\n   * Is merged with uploadOptions's metadata.\n   */\n  readonly customMetadata?: StorageCustomMetadata;\n}\n\n/**\n * StorageFileUploadConfig upload options.\n */\nexport type StorageFileUploadConfigOptions = Omit<StorageUploadOptions, 'resumable'>;\n\n/**\n * Function used to generate file names for the uploaded files.\n *\n * If not set, the file name will be used as is.\n */\nexport type StorageFileUploadConfigFactory = (file: File) => PromiseOrValue<StorageFileUploadConfig>;\n\n/**\n * Configuration for StorageFileUploadHandler().\n */\nexport interface StorageFileUploadHandlerConfig {\n  readonly storageService: DbxFirebaseStorageService;\n  readonly storageFileUploadConfigFactory: StorageFileUploadConfigFactory;\n}\n\n/**\n * Default implementation of StorageFileUploadHandler.\n */\nexport function storageFileUploadHandler(config: StorageFileUploadHandlerConfig): StorageFileUploadHandler {\n  const { storageService, storageFileUploadConfigFactory } = config;\n\n  let resumable: Maybe<StorageUploadTask>;\n\n  return {\n    uploadFile: async (file) => {\n      const storageFileUploadConfig = await storageFileUploadConfigFactory(file);\n\n      const { storagePath } = storageFileUploadConfig;\n      const storageAccessorFile = storageService.file(storagePath);\n\n      const upload = new Observable<DbxFirebaseStorageFileUploadStoreFileProgress>((x) => {\n        if (storageAccessorFile.uploadResumable) {\n          resumable = storageAccessorFile.uploadResumable(file);\n\n          // subscribe to the event by piping this observable to it\n          resumable\n            .streamSnapshotEvents()\n            .pipe(\n              map((x) => {\n                const { bytesTransferred, totalBytes } = x;\n\n                const progress: DbxFirebaseStorageFileUploadStoreFileProgress = {\n                  file,\n                  fileRef: storageAccessorFile,\n                  uploadRef: resumable,\n                  bytesTransferred,\n                  totalBytes,\n                  progress: bytesTransferred / totalBytes\n                };\n\n                return progress;\n              })\n            )\n            .pipe(\n              catchError((error) => {\n                // if an error occurs, catch it and emit it as a progress\n                const progress: DbxFirebaseStorageFileUploadStoreFileProgress = {\n                  file,\n                  fileRef: storageAccessorFile,\n                  uploadRef: resumable,\n                  error,\n                  failed: true\n                };\n\n                return of(progress);\n              })\n            )\n            .subscribe(x);\n        } else {\n          throw new Error('uploadResumable() function was unavailable.');\n        }\n      }).pipe(shareReplay(1));\n\n      const instance: StorageFileUploadHandlerInstance = {\n        upload,\n        taskRef: storageAccessorFile,\n        pause: () => resumable?.pause() ?? false,\n        resume: () => resumable?.resume() ?? false,\n        cancel: () => resumable?.cancel() ?? false\n      };\n\n      return instance;\n    }\n  };\n}\n\n// MARK: Upload Files\nexport interface StorageFileUploadFilesInput {\n  readonly uploadHandler: StorageFileUploadHandler;\n  /**\n   * Files to upload\n   */\n  readonly files: File[];\n  /**\n   * The number of max parallel uploads to perform at a time.\n   *\n   * Defaults to 3\n   */\n  readonly maxParallelUploads?: number;\n}\n\nexport interface StorageFileUploadFilesInstance {\n  /**\n   * Cancels the upload of the remaining files.\n   */\n  cancel(): void;\n  /**\n   * The upload observable.\n   *\n   * Must be subscribed to in order for the upload to begin.\n   */\n  readonly upload: Observable<StorageFileUploadFilesEvent>;\n}\n\nexport interface StorageFileUploadFilesEvent {\n  /**\n   * All files being uploaded\n   */\n  readonly allFiles: File[];\n  /**\n   * Returns true if all files have been uploaded.\n   *\n   * The result value should be available.\n   */\n  readonly isComplete: boolean;\n  /**\n   * Returns true if the upload was canceled.\n   */\n  readonly isCanceled?: Maybe<boolean>;\n  /**\n   * The overall progress of all files being uploaded.\n   */\n  readonly overallProgress: PercentNumber;\n  /**\n   * The upload progress that triggered this event.\n   */\n  readonly uploadProgress?: Maybe<DbxFirebaseStorageFileUploadStoreFileProgress>;\n  /**\n   * The final result.\n   *\n   * Set when the final file has been uploaded or failed.\n   */\n  readonly result?: StorageFileUploadFilesFinalResult;\n  /**\n   * The number of files that are still uploading or queued for upload.\n   */\n  readonly incompleteFileCount: number;\n  /**\n   * The number of files that are active.\n   */\n  readonly activeFileCount: number;\n  /**\n   * The number of files that are done.\n   */\n  readonly doneFileCount: number;\n}\n\nexport interface StorageFileUploadFilesFinalResult {\n  readonly startTime: Date;\n  readonly endTime: Date;\n  readonly fileResults: StorageFileUploadFilesFinalFileResult[];\n  readonly successFileResults: StorageFileUploadFilesFinalFileResult[];\n  readonly errorFileResults: StorageFileUploadFilesFinalFileResult[];\n}\n\nexport interface StorageFileUploadFilesFinalFileResult {\n  /**\n   * The start time of the file upload.\n   */\n  readonly startTime: Date;\n  /**\n   * The end time of the file upload, or when it failed or was canceled.\n   */\n  readonly endTime: Date;\n  /**\n   * The file that was uploaded.\n   */\n  readonly file: File;\n  /**\n   * The accessor file for the file, if available.\n   *\n   * Is generally available if success is true.\n   */\n  readonly fileRef?: Maybe<FirebaseStorageAccessorFile>;\n  /**\n   * True if the file was uploaded successfully.\n   */\n  readonly success: boolean;\n  /**\n   * Error if the file failed to upload.\n   */\n  readonly error?: Maybe<unknown>;\n  /**\n   * True if the file upload was cancelled.\n   */\n  readonly canceled?: Maybe<boolean>;\n}\n\n/**\n * Uploads files using the provided upload handler and files.\n *\n * An observable is returned that emits the latest file events from any file that is being uploaded.\n *\n * @param input\n * @returns\n */\nexport function storageFileUploadFiles(input: StorageFileUploadFilesInput): StorageFileUploadFilesInstance {\n  const { uploadHandler, files, maxParallelUploads: inputMaxParallelUploads } = input;\n  const maxParallelTasks = inputMaxParallelUploads ?? 3;\n\n  const multiUploadsSubscriptionObject = new MultiSubscriptionObject();\n\n  // begin the upload for each file\n  const allFiles = Array.from(files);\n\n  // unsubscribe from all previous uploads\n  multiUploadsSubscriptionObject.unsub();\n\n  interface UpdateUploadProgressInput {\n    /**\n     * The file index number.\n     */\n    readonly index: IndexNumber;\n    /**\n     * The next progress event, if applicable.\n     */\n    readonly nextProgress?: Maybe<DbxFirebaseStorageFileUploadStoreFileProgress>;\n    /**\n     * An error that occured outside of the nextProgress, if applicable.\n     *\n     * These are typically \"uncaught\" errors.\n     */\n    readonly nonProgressError?: Maybe<unknown>;\n    /**\n     * Passed as true when the upload task is done.\n     *\n     * This is also inferred as true when a progress error occurs for an upload.\n     *\n     * Does not specify whether or not success was achieved or not.\n     */\n    readonly fileUploadTaskDone?: boolean;\n    /**\n     * True if the upload was canceled.\n     */\n    readonly canceled?: boolean;\n  }\n\n  interface FileUploadDetails {\n    readonly file: File;\n    /**\n     * The current upload instance for the file.\n     *\n     * Set if the file is currently uploading.\n     */\n    uploadInstance?: StorageFileUploadHandlerInstance;\n    fileRef?: Maybe<FirebaseStorageAccessorFile>;\n    startTime?: Date;\n    endTime?: Date;\n    success?: boolean;\n    canceled?: Maybe<boolean>;\n    error?: Maybe<unknown>;\n  }\n\n  const allFilesAndLatestProgress: Maybe<DbxFirebaseStorageFileUploadStoreFileProgress>[] = new Array(allFiles.length);\n  const allFilesAndDetails: FileUploadDetails[] = allFiles.map((file) => ({ file }));\n  const overallProgressPerCompletedFile: PercentDecimal = (1 / allFilesAndLatestProgress.length) as PercentDecimal;\n\n  /**\n   * Once set, any new file upload task that hits this will return an cancel failure.\n   */\n  let flaggedCancel = false;\n\n  const cancel = () => {\n    flaggedCancel = true;\n  };\n\n  const upload = new Observable<StorageFileUploadFilesEvent>((subscriber) => {\n    const overallStartTime = new Date();\n\n    const incompleteFileFileIndexes = new Set<IndexNumber>(allFiles.map((_, index) => index));\n    const activeFileIndexes = new Set<IndexNumber>();\n    const doneFileIndexes = new Set<IndexNumber>();\n    let latestOverallProgress: PercentNumber = 0;\n\n    function onStartFileUpload(index: IndexNumber, uploadInstance: StorageFileUploadHandlerInstance) {\n      activeFileIndexes.add(index);\n      allFilesAndDetails[index].startTime = new Date();\n      allFilesAndDetails[index].uploadInstance = uploadInstance;\n      allFilesAndDetails[index].fileRef = uploadInstance.taskRef;\n    }\n\n    function onStartFileUploadFlaggedCancelled(index: IndexNumber) {\n      allFilesAndDetails[index].startTime = new Date();\n\n      // immediately mark it done\n      _markFileUploadDone(index, true);\n\n      // emit new progress event\n      _emitEvent();\n    }\n\n    function _markFileUploadDone(fileIndex: IndexNumber, finalError?: Maybe<unknown>) {\n      doneFileIndexes.add(fileIndex); // add to done file indexes\n      activeFileIndexes.delete(fileIndex); // remove from active file indexes if it exists\n      incompleteFileFileIndexes.delete(fileIndex); // remove from incomplete file indexes\n\n      // update details\n      allFilesAndDetails[fileIndex].endTime = new Date();\n\n      const error = finalError ?? allFilesAndDetails[fileIndex].error;\n      allFilesAndDetails[fileIndex].success = !error;\n      allFilesAndDetails[fileIndex].error = error;\n    }\n\n    function updateUploadProgress(input: UpdateUploadProgressInput) {\n      const { index: fileIndex, nextProgress, fileUploadTaskDone: inputFileUploadTaskDone, nonProgressError } = input;\n      const error = nonProgressError ?? nextProgress?.error;\n      const fileUploadTaskDone = inputFileUploadTaskDone ?? Boolean(error);\n\n      // the task may already be done, as after a progress-related error the complete task can get called.\n      const isTaskAlreadyDone = allFilesAndDetails[fileIndex].endTime != null;\n\n      if (!isTaskAlreadyDone) {\n        let nextOverallProgress = latestOverallProgress;\n        const nextProgressPercent = fileUploadTaskDone ? 100 : (nextProgress?.progress ?? 0) * 100;\n\n        // update the overall progress percentage\n        if (nextProgressPercent) {\n          // update the overall percentage\n          const previousProgress = allFilesAndLatestProgress[fileIndex];\n          const previousProgressPercent = previousProgress?.progress != null ? previousProgress.progress * 100 : 0;\n          const progressPercentChange = nextProgressPercent - previousProgressPercent;\n\n          // increase overall progress by the change\n          nextOverallProgress += progressPercentChange * overallProgressPerCompletedFile;\n        }\n\n        // update the file progress\n        if (nextProgress) {\n          // update the latest FileProgress\n          allFilesAndLatestProgress[fileIndex] = nextProgress;\n\n          // only set fileRef once\n          if (!allFilesAndDetails[fileIndex].fileRef) {\n            allFilesAndDetails[fileIndex].fileRef = nextProgress.fileRef;\n          }\n        }\n\n        // if complete, update the indexes and details\n        if (fileUploadTaskDone) {\n          _markFileUploadDone(fileIndex, error);\n        }\n\n        // update the overall progress\n        latestOverallProgress = nextOverallProgress;\n\n        // emit the event to send it\n        _emitEvent(nextProgress);\n      }\n    }\n\n    function _emitEvent(nextProgress?: Maybe<DbxFirebaseStorageFileUploadStoreFileProgress>) {\n      const isComplete = incompleteFileFileIndexes.size === 0;\n\n      let overallProgress = latestOverallProgress;\n      let result: Maybe<StorageFileUploadFilesFinalResult> = undefined;\n\n      if (isComplete) {\n        overallProgress = 100; // set to 100%\n        const overallEndTime = new Date();\n\n        const fileResults = allFiles.map((file, index) => {\n          const result: StorageFileUploadFilesFinalFileResult = {\n            startTime: allFilesAndDetails[index].startTime as Date,\n            endTime: allFilesAndDetails[index].endTime as Date,\n            file,\n            fileRef: allFilesAndDetails[index].fileRef,\n            success: !allFilesAndDetails[index].error,\n            error: allFilesAndDetails[index].error\n          };\n\n          return result;\n        });\n\n        const { included: successFileResults, excluded: errorFileResults } = separateValues(fileResults, (x) => x.success);\n\n        // all are done, set the result on the next event\n        result = {\n          startTime: overallStartTime,\n          endTime: overallEndTime,\n          successFileResults,\n          errorFileResults,\n          fileResults\n        };\n      }\n\n      const nextEvent: StorageFileUploadFilesEvent = {\n        allFiles,\n        isComplete,\n        overallProgress,\n        uploadProgress: nextProgress,\n        incompleteFileCount: incompleteFileFileIndexes.size,\n        activeFileCount: activeFileIndexes.size,\n        doneFileCount: doneFileIndexes.size,\n        result\n      };\n\n      subscriber.next(nextEvent);\n    }\n\n    async function runUploadTaskForFile([file, index]: readonly [File, IndexNumber]) {\n      if (flaggedCancel) {\n        onStartFileUploadFlaggedCancelled(index);\n        return Promise.resolve();\n      }\n\n      return new Promise<void>((resolve) => {\n        const updateFileUploadProgress = (nextProgress: DbxFirebaseStorageFileUploadStoreFileProgress) => {\n          // update the progress\n          updateUploadProgress({\n            index,\n            nextProgress\n          });\n        };\n\n        const updateFileUploadProgressWithUncaughtError = (error: unknown) => {\n          // error occurred, update the progress with the error\n          updateUploadProgress({\n            index,\n            nonProgressError: error,\n            fileUploadTaskDone: true\n          });\n\n          // always resolve, never reject\n          resolve();\n        };\n\n        const completeFileUploadProgress = () => {\n          updateUploadProgress({\n            index,\n            fileUploadTaskDone: true\n          });\n\n          resolve();\n        };\n\n        // upload the file, subscribe to the progress\n        try {\n          uploadHandler\n            .uploadFile(file)\n            .then((uploadInstance) => {\n              // add to active file indexes\n              onStartFileUpload(index, uploadInstance);\n\n              const uploadSubscription = uploadInstance.upload.subscribe({\n                next: updateFileUploadProgress,\n                error: updateFileUploadProgressWithUncaughtError,\n                complete: completeFileUploadProgress\n              });\n\n              multiUploadsSubscriptionObject.addSubs(uploadSubscription);\n            })\n            .catch(updateFileUploadProgressWithUncaughtError);\n        } catch (error) {\n          updateFileUploadProgressWithUncaughtError(error);\n        }\n      });\n    }\n\n    // run upload task for each file\n    const fileTuples = allFiles.map((file, index) => [file, index] as const);\n\n    runAsyncTasksForValues(fileTuples, runUploadTaskForFile, {\n      maxParallelTasks,\n      retriesAllowed: 0 // no retries allowed\n    }).then(() => {\n      // all tasks are finished. Complete the subscriber.\n      subscriber.complete();\n    });\n  }).pipe(shareReplay(1));\n\n  const instance: StorageFileUploadFilesInstance = {\n    cancel,\n    upload\n  };\n\n  return instance;\n}\n"]}
@@ -0,0 +1,39 @@
1
+ import { Directive, inject } from '@angular/core';
2
+ import { DbxFirebaseStorageFileUploadStore, StorageFileDocumentStore } from '../store';
3
+ import { AbstractSubscriptionDirective } from '@dereekb/dbx-core';
4
+ import * as i0 from "@angular/core";
5
+ /**
6
+ * Directive that passes the upload result to a StorageFileDocumentStore to initialize the document immediately.
7
+ */
8
+ export class DbxFirebaseStorageFileUploadInitializeDocumentDirective extends AbstractSubscriptionDirective {
9
+ uploadStore = inject(DbxFirebaseStorageFileUploadStore);
10
+ storageFileDocumentStore = inject(StorageFileDocumentStore);
11
+ constructor() {
12
+ super();
13
+ this.sub = this.uploadStore.uploadResult$.subscribe(async (result) => {
14
+ const successFileResult = result.successFileResults.find((x) => x.fileRef != null);
15
+ if (successFileResult) {
16
+ const fileRef = successFileResult.fileRef;
17
+ if (fileRef) {
18
+ this.storageFileDocumentStore
19
+ .initializeStorageFileFromUpload({
20
+ pathString: fileRef.storagePath.pathString,
21
+ bucketId: fileRef.storagePath.bucketId
22
+ })
23
+ .subscribe();
24
+ }
25
+ }
26
+ });
27
+ }
28
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: DbxFirebaseStorageFileUploadInitializeDocumentDirective, deps: [], target: i0.ɵɵFactoryTarget.Directive });
29
+ static ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "18.2.13", type: DbxFirebaseStorageFileUploadInitializeDocumentDirective, isStandalone: true, selector: "[dbxFirebaseStorageFileUploadInitializeDocument]", exportAs: ["dbxFirebaseStorageFileUploadInitializeDocument"], usesInheritance: true, ngImport: i0 });
30
+ }
31
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: DbxFirebaseStorageFileUploadInitializeDocumentDirective, decorators: [{
32
+ type: Directive,
33
+ args: [{
34
+ selector: '[dbxFirebaseStorageFileUploadInitializeDocument]',
35
+ exportAs: 'dbxFirebaseStorageFileUploadInitializeDocument',
36
+ standalone: true
37
+ }]
38
+ }], ctorParameters: () => [] });
39
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3RvcmFnZWZpbGUudXBsb2FkLmluaXRpYWxpemUuZG9jdW1lbnQuZGlyZWN0aXZlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vLi4vcGFja2FnZXMvZGJ4LWZpcmViYXNlL3NyYy9saWIvbW9kdWxlcy9zdG9yYWdlZmlsZS9jb250YWluZXIvc3RvcmFnZWZpbGUudXBsb2FkLmluaXRpYWxpemUuZG9jdW1lbnQuZGlyZWN0aXZlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxTQUFTLEVBQUUsTUFBTSxFQUFhLE1BQU0sZUFBZSxDQUFDO0FBQzdELE9BQU8sRUFBRSxpQ0FBaUMsRUFBRSx3QkFBd0IsRUFBRSxNQUFNLFVBQVUsQ0FBQztBQUN2RixPQUFPLEVBQUUsNkJBQTZCLEVBQUUsTUFBTSxtQkFBbUIsQ0FBQzs7QUFHbEU7O0dBRUc7QUFNSCxNQUFNLE9BQU8sdURBQXdELFNBQVEsNkJBQTZCO0lBQy9GLFdBQVcsR0FBRyxNQUFNLENBQUMsaUNBQWlDLENBQUMsQ0FBQztJQUN4RCx3QkFBd0IsR0FBRyxNQUFNLENBQUMsd0JBQXdCLENBQUMsQ0FBQztJQUVyRTtRQUNFLEtBQUssRUFBRSxDQUFDO1FBRVIsSUFBSSxDQUFDLEdBQUcsR0FBRyxJQUFJLENBQUMsV0FBVyxDQUFDLGFBQWEsQ0FBQyxTQUFTLENBQUMsS0FBSyxFQUFFLE1BQU0sRUFBRSxFQUFFO1lBQ25FLE1BQU0saUJBQWlCLEdBQUcsTUFBTSxDQUFDLGtCQUFrQixDQUFDLElBQUksQ0FBQyxDQUFDLENBQUMsRUFBRSxFQUFFLENBQUMsQ0FBQyxDQUFDLE9BQU8sSUFBSSxJQUFJLENBQUMsQ0FBQztZQUVuRixJQUFJLGlCQUFpQixFQUFFLENBQUM7Z0JBQ3RCLE1BQU0sT0FBTyxHQUFHLGlCQUFpQixDQUFDLE9BQXNDLENBQUM7Z0JBRXpFLElBQUksT0FBTyxFQUFFLENBQUM7b0JBQ1osSUFBSSxDQUFDLHdCQUF3Qjt5QkFDMUIsK0JBQStCLENBQUM7d0JBQy9CLFVBQVUsRUFBRSxPQUFPLENBQUMsV0FBVyxDQUFDLFVBQVU7d0JBQzFDLFFBQVEsRUFBRSxPQUFPLENBQUMsV0FBVyxDQUFDLFFBQVE7cUJBQ3ZDLENBQUM7eUJBQ0QsU0FBUyxFQUFFLENBQUM7Z0JBQ2pCLENBQUM7WUFDSCxDQUFDO1FBQ0gsQ0FBQyxDQUFDLENBQUM7SUFDTCxDQUFDO3dHQXZCVSx1REFBdUQ7NEZBQXZELHVEQUF1RDs7NEZBQXZELHVEQUF1RDtrQkFMbkUsU0FBUzttQkFBQztvQkFDVCxRQUFRLEVBQUUsa0RBQWtEO29CQUM1RCxRQUFRLEVBQUUsZ0RBQWdEO29CQUMxRCxVQUFVLEVBQUUsSUFBSTtpQkFDakIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBEaXJlY3RpdmUsIGluamVjdCwgT25EZXN0cm95IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBEYnhGaXJlYmFzZVN0b3JhZ2VGaWxlVXBsb2FkU3RvcmUsIFN0b3JhZ2VGaWxlRG9jdW1lbnRTdG9yZSB9IGZyb20gJy4uL3N0b3JlJztcbmltcG9ydCB7IEFic3RyYWN0U3Vic2NyaXB0aW9uRGlyZWN0aXZlIH0gZnJvbSAnQGRlcmVla2IvZGJ4LWNvcmUnO1xuaW1wb3J0IHsgRmlyZWJhc2VTdG9yYWdlQWNjZXNzb3JGaWxlIH0gZnJvbSAnQGRlcmVla2IvZmlyZWJhc2UnO1xuXG4vKipcbiAqIERpcmVjdGl2ZSB0aGF0IHBhc3NlcyB0aGUgdXBsb2FkIHJlc3VsdCB0byBhIFN0b3JhZ2VGaWxlRG9jdW1lbnRTdG9yZSB0byBpbml0aWFsaXplIHRoZSBkb2N1bWVudCBpbW1lZGlhdGVseS5cbiAqL1xuQERpcmVjdGl2ZSh7XG4gIHNlbGVjdG9yOiAnW2RieEZpcmViYXNlU3RvcmFnZUZpbGVVcGxvYWRJbml0aWFsaXplRG9jdW1lbnRdJyxcbiAgZXhwb3J0QXM6ICdkYnhGaXJlYmFzZVN0b3JhZ2VGaWxlVXBsb2FkSW5pdGlhbGl6ZURvY3VtZW50JyxcbiAgc3RhbmRhbG9uZTogdHJ1ZVxufSlcbmV4cG9ydCBjbGFzcyBEYnhGaXJlYmFzZVN0b3JhZ2VGaWxlVXBsb2FkSW5pdGlhbGl6ZURvY3VtZW50RGlyZWN0aXZlIGV4dGVuZHMgQWJzdHJhY3RTdWJzY3JpcHRpb25EaXJlY3RpdmUgaW1wbGVtZW50cyBPbkRlc3Ryb3kge1xuICByZWFkb25seSB1cGxvYWRTdG9yZSA9IGluamVjdChEYnhGaXJlYmFzZVN0b3JhZ2VGaWxlVXBsb2FkU3RvcmUpO1xuICByZWFkb25seSBzdG9yYWdlRmlsZURvY3VtZW50U3RvcmUgPSBpbmplY3QoU3RvcmFnZUZpbGVEb2N1bWVudFN0b3JlKTtcblxuICBjb25zdHJ1Y3RvcigpIHtcbiAgICBzdXBlcigpO1xuXG4gICAgdGhpcy5zdWIgPSB0aGlzLnVwbG9hZFN0b3JlLnVwbG9hZFJlc3VsdCQuc3Vic2NyaWJlKGFzeW5jIChyZXN1bHQpID0+IHtcbiAgICAgIGNvbnN0IHN1Y2Nlc3NGaWxlUmVzdWx0ID0gcmVzdWx0LnN1Y2Nlc3NGaWxlUmVzdWx0cy5maW5kKCh4KSA9PiB4LmZpbGVSZWYgIT0gbnVsbCk7XG5cbiAgICAgIGlmIChzdWNjZXNzRmlsZVJlc3VsdCkge1xuICAgICAgICBjb25zdCBmaWxlUmVmID0gc3VjY2Vzc0ZpbGVSZXN1bHQuZmlsZVJlZiBhcyBGaXJlYmFzZVN0b3JhZ2VBY2Nlc3NvckZpbGU7XG5cbiAgICAgICAgaWYgKGZpbGVSZWYpIHtcbiAgICAgICAgICB0aGlzLnN0b3JhZ2VGaWxlRG9jdW1lbnRTdG9yZVxuICAgICAgICAgICAgLmluaXRpYWxpemVTdG9yYWdlRmlsZUZyb21VcGxvYWQoe1xuICAgICAgICAgICAgICBwYXRoU3RyaW5nOiBmaWxlUmVmLnN0b3JhZ2VQYXRoLnBhdGhTdHJpbmcsXG4gICAgICAgICAgICAgIGJ1Y2tldElkOiBmaWxlUmVmLnN0b3JhZ2VQYXRoLmJ1Y2tldElkXG4gICAgICAgICAgICB9KVxuICAgICAgICAgICAgLnN1YnNjcmliZSgpO1xuICAgICAgICB9XG4gICAgICB9XG4gICAgfSk7XG4gIH1cbn1cbiJdfQ==
@@ -0,0 +1,38 @@
1
+ import { Directive, inject, input } from '@angular/core';
2
+ import { skipAllInitialMaybe, SubscriptionObject } from '@dereekb/rxjs';
3
+ import { DbxFirebaseStorageFileUploadStore } from '../store';
4
+ import { toObservable } from '@angular/core/rxjs-interop';
5
+ import { shareReplay } from 'rxjs';
6
+ import * as i0 from "@angular/core";
7
+ /**
8
+ * Directive that provides a DbxFirebaseStorageFileUploadStore, and sync's the inputs to the store.
9
+ */
10
+ export class DbxFirebaseStorageFileUploadStoreDirective {
11
+ _allowedSub = new SubscriptionObject();
12
+ _multiSub = new SubscriptionObject();
13
+ uploadStore = inject(DbxFirebaseStorageFileUploadStore);
14
+ multipleUpload = input();
15
+ fileTypesAccepted = input();
16
+ fileTypesAccepted$ = toObservable(this.fileTypesAccepted).pipe(skipAllInitialMaybe(), shareReplay(1));
17
+ isMultiUploadAllowed$ = toObservable(this.multipleUpload).pipe(skipAllInitialMaybe(), shareReplay(1));
18
+ constructor() {
19
+ this._allowedSub.subscription = this.fileTypesAccepted$.subscribe((x) => this.uploadStore.setFileTypesAccepted(x));
20
+ this._multiSub.subscription = this.isMultiUploadAllowed$.subscribe((x) => this.uploadStore.setIsMultiUploadAllowed(x));
21
+ }
22
+ ngOnDestroy() {
23
+ this._allowedSub.destroy();
24
+ this._multiSub.destroy();
25
+ }
26
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: DbxFirebaseStorageFileUploadStoreDirective, deps: [], target: i0.ɵɵFactoryTarget.Directive });
27
+ static ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "17.1.0", version: "18.2.13", type: DbxFirebaseStorageFileUploadStoreDirective, isStandalone: true, selector: "[dbxFirebaseStorageFileUploadStore]", inputs: { multipleUpload: { classPropertyName: "multipleUpload", publicName: "multipleUpload", isSignal: true, isRequired: false, transformFunction: null }, fileTypesAccepted: { classPropertyName: "fileTypesAccepted", publicName: "fileTypesAccepted", isSignal: true, isRequired: false, transformFunction: null } }, providers: [DbxFirebaseStorageFileUploadStore], exportAs: ["dbxFirebaseStorageFileUploadStore"], ngImport: i0 });
28
+ }
29
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: DbxFirebaseStorageFileUploadStoreDirective, decorators: [{
30
+ type: Directive,
31
+ args: [{
32
+ selector: '[dbxFirebaseStorageFileUploadStore]',
33
+ exportAs: 'dbxFirebaseStorageFileUploadStore',
34
+ providers: [DbxFirebaseStorageFileUploadStore],
35
+ standalone: true
36
+ }]
37
+ }], ctorParameters: () => [] });
38
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3RvcmFnZWZpbGUudXBsb2FkLnN0b3JlLmRpcmVjdGl2ZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uLy4uL3BhY2thZ2VzL2RieC1maXJlYmFzZS9zcmMvbGliL21vZHVsZXMvc3RvcmFnZWZpbGUvY29udGFpbmVyL3N0b3JhZ2VmaWxlLnVwbG9hZC5zdG9yZS5kaXJlY3RpdmUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFNBQVMsRUFBRSxNQUFNLEVBQUUsS0FBSyxFQUFhLE1BQU0sZUFBZSxDQUFDO0FBQ3BFLE9BQU8sRUFBRSxtQkFBbUIsRUFBRSxrQkFBa0IsRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUN4RSxPQUFPLEVBQUUsaUNBQWlDLEVBQUUsTUFBTSxVQUFVLENBQUM7QUFFN0QsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLDRCQUE0QixDQUFDO0FBQzFELE9BQU8sRUFBRSxXQUFXLEVBQUUsTUFBTSxNQUFNLENBQUM7O0FBR25DOztHQUVHO0FBT0gsTUFBTSxPQUFPLDBDQUEwQztJQUNwQyxXQUFXLEdBQUcsSUFBSSxrQkFBa0IsRUFBRSxDQUFDO0lBQ3ZDLFNBQVMsR0FBRyxJQUFJLGtCQUFrQixFQUFFLENBQUM7SUFFN0MsV0FBVyxHQUFHLE1BQU0sQ0FBQyxpQ0FBaUMsQ0FBQyxDQUFDO0lBRXhELGNBQWMsR0FBRyxLQUFLLEVBQWtCLENBQUM7SUFDekMsaUJBQWlCLEdBQUcsS0FBSyxFQUFtRCxDQUFDO0lBRTdFLGtCQUFrQixHQUFHLFlBQVksQ0FBQyxJQUFJLENBQUMsaUJBQWlCLENBQUMsQ0FBQyxJQUFJLENBQUMsbUJBQW1CLEVBQUUsRUFBRSxXQUFXLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQztJQUN0RyxxQkFBcUIsR0FBRyxZQUFZLENBQUMsSUFBSSxDQUFDLGNBQWMsQ0FBQyxDQUFDLElBQUksQ0FBQyxtQkFBbUIsRUFBRSxFQUFFLFdBQVcsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDO0lBRS9HO1FBQ0UsSUFBSSxDQUFDLFdBQVcsQ0FBQyxZQUFZLEdBQUcsSUFBSSxDQUFDLGtCQUFrQixDQUFDLFNBQVMsQ0FBQyxDQUFDLENBQUMsRUFBRSxFQUFFLENBQUMsSUFBSSxDQUFDLFdBQVcsQ0FBQyxvQkFBb0IsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDO1FBQ25ILElBQUksQ0FBQyxTQUFTLENBQUMsWUFBWSxHQUFHLElBQUksQ0FBQyxxQkFBcUIsQ0FBQyxTQUFTLENBQUMsQ0FBQyxDQUFDLEVBQUUsRUFBRSxDQUFDLElBQUksQ0FBQyxXQUFXLENBQUMsdUJBQXVCLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQztJQUN6SCxDQUFDO0lBRUQsV0FBVztRQUNULElBQUksQ0FBQyxXQUFXLENBQUMsT0FBTyxFQUFFLENBQUM7UUFDM0IsSUFBSSxDQUFDLFNBQVMsQ0FBQyxPQUFPLEVBQUUsQ0FBQztJQUMzQixDQUFDO3dHQXBCVSwwQ0FBMEM7NEZBQTFDLDBDQUEwQyw2WUFIMUMsQ0FBQyxpQ0FBaUMsQ0FBQzs7NEZBR25DLDBDQUEwQztrQkFOdEQsU0FBUzttQkFBQztvQkFDVCxRQUFRLEVBQUUscUNBQXFDO29CQUMvQyxRQUFRLEVBQUUsbUNBQW1DO29CQUM3QyxTQUFTLEVBQUUsQ0FBQyxpQ0FBaUMsQ0FBQztvQkFDOUMsVUFBVSxFQUFFLElBQUk7aUJBQ2pCIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgRGlyZWN0aXZlLCBpbmplY3QsIGlucHV0LCBPbkRlc3Ryb3kgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IHNraXBBbGxJbml0aWFsTWF5YmUsIFN1YnNjcmlwdGlvbk9iamVjdCB9IGZyb20gJ0BkZXJlZWtiL3J4anMnO1xuaW1wb3J0IHsgRGJ4RmlyZWJhc2VTdG9yYWdlRmlsZVVwbG9hZFN0b3JlIH0gZnJvbSAnLi4vc3RvcmUnO1xuaW1wb3J0IHsgQXJyYXlPclZhbHVlLCBNYXliZSB9IGZyb20gJ0BkZXJlZWtiL3V0aWwnO1xuaW1wb3J0IHsgdG9PYnNlcnZhYmxlIH0gZnJvbSAnQGFuZ3VsYXIvY29yZS9yeGpzLWludGVyb3AnO1xuaW1wb3J0IHsgc2hhcmVSZXBsYXkgfSBmcm9tICdyeGpzJztcbmltcG9ydCB7IEZpbGVBY2NlcHRGaWx0ZXJUeXBlU3RyaW5nIH0gZnJvbSAnQGRlcmVla2IvZGJ4LXdlYic7XG5cbi8qKlxuICogRGlyZWN0aXZlIHRoYXQgcHJvdmlkZXMgYSBEYnhGaXJlYmFzZVN0b3JhZ2VGaWxlVXBsb2FkU3RvcmUsIGFuZCBzeW5jJ3MgdGhlIGlucHV0cyB0byB0aGUgc3RvcmUuXG4gKi9cbkBEaXJlY3RpdmUoe1xuICBzZWxlY3RvcjogJ1tkYnhGaXJlYmFzZVN0b3JhZ2VGaWxlVXBsb2FkU3RvcmVdJyxcbiAgZXhwb3J0QXM6ICdkYnhGaXJlYmFzZVN0b3JhZ2VGaWxlVXBsb2FkU3RvcmUnLFxuICBwcm92aWRlcnM6IFtEYnhGaXJlYmFzZVN0b3JhZ2VGaWxlVXBsb2FkU3RvcmVdLFxuICBzdGFuZGFsb25lOiB0cnVlXG59KVxuZXhwb3J0IGNsYXNzIERieEZpcmViYXNlU3RvcmFnZUZpbGVVcGxvYWRTdG9yZURpcmVjdGl2ZSBpbXBsZW1lbnRzIE9uRGVzdHJveSB7XG4gIHByaXZhdGUgcmVhZG9ubHkgX2FsbG93ZWRTdWIgPSBuZXcgU3Vic2NyaXB0aW9uT2JqZWN0KCk7XG4gIHByaXZhdGUgcmVhZG9ubHkgX211bHRpU3ViID0gbmV3IFN1YnNjcmlwdGlvbk9iamVjdCgpO1xuXG4gIHJlYWRvbmx5IHVwbG9hZFN0b3JlID0gaW5qZWN0KERieEZpcmViYXNlU3RvcmFnZUZpbGVVcGxvYWRTdG9yZSk7XG5cbiAgcmVhZG9ubHkgbXVsdGlwbGVVcGxvYWQgPSBpbnB1dDxNYXliZTxib29sZWFuPj4oKTtcbiAgcmVhZG9ubHkgZmlsZVR5cGVzQWNjZXB0ZWQgPSBpbnB1dDxNYXliZTxBcnJheU9yVmFsdWU8RmlsZUFjY2VwdEZpbHRlclR5cGVTdHJpbmc+Pj4oKTtcblxuICByZWFkb25seSBmaWxlVHlwZXNBY2NlcHRlZCQgPSB0b09ic2VydmFibGUodGhpcy5maWxlVHlwZXNBY2NlcHRlZCkucGlwZShza2lwQWxsSW5pdGlhbE1heWJlKCksIHNoYXJlUmVwbGF5KDEpKTtcbiAgcmVhZG9ubHkgaXNNdWx0aVVwbG9hZEFsbG93ZWQkID0gdG9PYnNlcnZhYmxlKHRoaXMubXVsdGlwbGVVcGxvYWQpLnBpcGUoc2tpcEFsbEluaXRpYWxNYXliZSgpLCBzaGFyZVJlcGxheSgxKSk7XG5cbiAgY29uc3RydWN0b3IoKSB7XG4gICAgdGhpcy5fYWxsb3dlZFN1Yi5zdWJzY3JpcHRpb24gPSB0aGlzLmZpbGVUeXBlc0FjY2VwdGVkJC5zdWJzY3JpYmUoKHgpID0+IHRoaXMudXBsb2FkU3RvcmUuc2V0RmlsZVR5cGVzQWNjZXB0ZWQoeCkpO1xuICAgIHRoaXMuX211bHRpU3ViLnN1YnNjcmlwdGlvbiA9IHRoaXMuaXNNdWx0aVVwbG9hZEFsbG93ZWQkLnN1YnNjcmliZSgoeCkgPT4gdGhpcy51cGxvYWRTdG9yZS5zZXRJc011bHRpVXBsb2FkQWxsb3dlZCh4KSk7XG4gIH1cblxuICBuZ09uRGVzdHJveSgpOiB2b2lkIHtcbiAgICB0aGlzLl9hbGxvd2VkU3ViLmRlc3Ryb3koKTtcbiAgICB0aGlzLl9tdWx0aVN1Yi5kZXN0cm95KCk7XG4gIH1cbn1cbiJdfQ==
@@ -0,0 +1,36 @@
1
+ import { Directive, inject } from '@angular/core';
2
+ import { SubscriptionObject } from '@dereekb/rxjs';
3
+ import { DbxFirebaseStorageFileUploadStore } from '../store';
4
+ import { DbxFileUploadComponent } from '@dereekb/dbx-web';
5
+ import * as i0 from "@angular/core";
6
+ /**
7
+ * Directive that syncs a DbxFirebaseStorageFileUploadStore's configuration to a DbxFileUploadComponent.
8
+ */
9
+ export class DbxFirebaseStorageFileUploadSyncDirective {
10
+ _allowedSub = new SubscriptionObject();
11
+ _multiSub = new SubscriptionObject();
12
+ _filesSub = new SubscriptionObject();
13
+ uploadStore = inject(DbxFirebaseStorageFileUploadStore);
14
+ uploadComponent = inject(DbxFileUploadComponent);
15
+ constructor() {
16
+ this._allowedSub.subscription = this.uploadStore.fileTypesAllowed$.subscribe((x) => this.uploadComponent.setAccept(x));
17
+ this._multiSub.subscription = this.uploadStore.isMultiUploadAllowed$.subscribe((x) => this.uploadComponent.setMultiple(x));
18
+ this._filesSub.subscription = this.uploadComponent.filesChanged.subscribe((files) => this.uploadStore.setFiles(files.matchResult.accepted));
19
+ }
20
+ ngOnDestroy() {
21
+ this._allowedSub.destroy();
22
+ this._multiSub.destroy();
23
+ this._filesSub.destroy();
24
+ }
25
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: DbxFirebaseStorageFileUploadSyncDirective, deps: [], target: i0.ɵɵFactoryTarget.Directive });
26
+ static ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "18.2.13", type: DbxFirebaseStorageFileUploadSyncDirective, isStandalone: true, selector: "[dbxFirebaseStorageFileUploadSync]", exportAs: ["dbxFirebaseStorageFileUploadSync"], ngImport: i0 });
27
+ }
28
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: DbxFirebaseStorageFileUploadSyncDirective, decorators: [{
29
+ type: Directive,
30
+ args: [{
31
+ selector: '[dbxFirebaseStorageFileUploadSync]',
32
+ exportAs: 'dbxFirebaseStorageFileUploadSync',
33
+ standalone: true
34
+ }]
35
+ }], ctorParameters: () => [] });
36
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3RvcmFnZWZpbGUudXBsb2FkLnN5bmMuZGlyZWN0aXZlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vLi4vcGFja2FnZXMvZGJ4LWZpcmViYXNlL3NyYy9saWIvbW9kdWxlcy9zdG9yYWdlZmlsZS9jb250YWluZXIvc3RvcmFnZWZpbGUudXBsb2FkLnN5bmMuZGlyZWN0aXZlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxTQUFTLEVBQUUsTUFBTSxFQUFhLE1BQU0sZUFBZSxDQUFDO0FBQzdELE9BQU8sRUFBRSxrQkFBa0IsRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUNuRCxPQUFPLEVBQUUsaUNBQWlDLEVBQUUsTUFBTSxVQUFVLENBQUM7QUFDN0QsT0FBTyxFQUFFLHNCQUFzQixFQUFFLE1BQU0sa0JBQWtCLENBQUM7O0FBRTFEOztHQUVHO0FBTUgsTUFBTSxPQUFPLHlDQUF5QztJQUNuQyxXQUFXLEdBQUcsSUFBSSxrQkFBa0IsRUFBRSxDQUFDO0lBQ3ZDLFNBQVMsR0FBRyxJQUFJLGtCQUFrQixFQUFFLENBQUM7SUFDckMsU0FBUyxHQUFHLElBQUksa0JBQWtCLEVBQUUsQ0FBQztJQUU3QyxXQUFXLEdBQUcsTUFBTSxDQUFDLGlDQUFpQyxDQUFDLENBQUM7SUFDeEQsZUFBZSxHQUFHLE1BQU0sQ0FBQyxzQkFBc0IsQ0FBQyxDQUFDO0lBRTFEO1FBQ0UsSUFBSSxDQUFDLFdBQVcsQ0FBQyxZQUFZLEdBQUcsSUFBSSxDQUFDLFdBQVcsQ0FBQyxpQkFBaUIsQ0FBQyxTQUFTLENBQUMsQ0FBQyxDQUFDLEVBQUUsRUFBRSxDQUFDLElBQUksQ0FBQyxlQUFlLENBQUMsU0FBUyxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUM7UUFDdkgsSUFBSSxDQUFDLFNBQVMsQ0FBQyxZQUFZLEdBQUcsSUFBSSxDQUFDLFdBQVcsQ0FBQyxxQkFBcUIsQ0FBQyxTQUFTLENBQUMsQ0FBQyxDQUFDLEVBQUUsRUFBRSxDQUFDLElBQUksQ0FBQyxlQUFlLENBQUMsV0FBVyxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUM7UUFDM0gsSUFBSSxDQUFDLFNBQVMsQ0FBQyxZQUFZLEdBQUcsSUFBSSxDQUFDLGVBQWUsQ0FBQyxZQUFZLENBQUMsU0FBUyxDQUFDLENBQUMsS0FBSyxFQUFFLEVBQUUsQ0FBQyxJQUFJLENBQUMsV0FBVyxDQUFDLFFBQVEsQ0FBQyxLQUFLLENBQUMsV0FBVyxDQUFDLFFBQVEsQ0FBQyxDQUFDLENBQUM7SUFDOUksQ0FBQztJQUVELFdBQVc7UUFDVCxJQUFJLENBQUMsV0FBVyxDQUFDLE9BQU8sRUFBRSxDQUFDO1FBQzNCLElBQUksQ0FBQyxTQUFTLENBQUMsT0FBTyxFQUFFLENBQUM7UUFDekIsSUFBSSxDQUFDLFNBQVMsQ0FBQyxPQUFPLEVBQUUsQ0FBQztJQUMzQixDQUFDO3dHQWxCVSx5Q0FBeUM7NEZBQXpDLHlDQUF5Qzs7NEZBQXpDLHlDQUF5QztrQkFMckQsU0FBUzttQkFBQztvQkFDVCxRQUFRLEVBQUUsb0NBQW9DO29CQUM5QyxRQUFRLEVBQUUsa0NBQWtDO29CQUM1QyxVQUFVLEVBQUUsSUFBSTtpQkFDakIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBEaXJlY3RpdmUsIGluamVjdCwgT25EZXN0cm95IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBTdWJzY3JpcHRpb25PYmplY3QgfSBmcm9tICdAZGVyZWVrYi9yeGpzJztcbmltcG9ydCB7IERieEZpcmViYXNlU3RvcmFnZUZpbGVVcGxvYWRTdG9yZSB9IGZyb20gJy4uL3N0b3JlJztcbmltcG9ydCB7IERieEZpbGVVcGxvYWRDb21wb25lbnQgfSBmcm9tICdAZGVyZWVrYi9kYngtd2ViJztcblxuLyoqXG4gKiBEaXJlY3RpdmUgdGhhdCBzeW5jcyBhIERieEZpcmViYXNlU3RvcmFnZUZpbGVVcGxvYWRTdG9yZSdzIGNvbmZpZ3VyYXRpb24gdG8gYSBEYnhGaWxlVXBsb2FkQ29tcG9uZW50LlxuICovXG5ARGlyZWN0aXZlKHtcbiAgc2VsZWN0b3I6ICdbZGJ4RmlyZWJhc2VTdG9yYWdlRmlsZVVwbG9hZFN5bmNdJyxcbiAgZXhwb3J0QXM6ICdkYnhGaXJlYmFzZVN0b3JhZ2VGaWxlVXBsb2FkU3luYycsXG4gIHN0YW5kYWxvbmU6IHRydWVcbn0pXG5leHBvcnQgY2xhc3MgRGJ4RmlyZWJhc2VTdG9yYWdlRmlsZVVwbG9hZFN5bmNEaXJlY3RpdmUgaW1wbGVtZW50cyBPbkRlc3Ryb3kge1xuICBwcml2YXRlIHJlYWRvbmx5IF9hbGxvd2VkU3ViID0gbmV3IFN1YnNjcmlwdGlvbk9iamVjdCgpO1xuICBwcml2YXRlIHJlYWRvbmx5IF9tdWx0aVN1YiA9IG5ldyBTdWJzY3JpcHRpb25PYmplY3QoKTtcbiAgcHJpdmF0ZSByZWFkb25seSBfZmlsZXNTdWIgPSBuZXcgU3Vic2NyaXB0aW9uT2JqZWN0KCk7XG5cbiAgcmVhZG9ubHkgdXBsb2FkU3RvcmUgPSBpbmplY3QoRGJ4RmlyZWJhc2VTdG9yYWdlRmlsZVVwbG9hZFN0b3JlKTtcbiAgcmVhZG9ubHkgdXBsb2FkQ29tcG9uZW50ID0gaW5qZWN0KERieEZpbGVVcGxvYWRDb21wb25lbnQpO1xuXG4gIGNvbnN0cnVjdG9yKCkge1xuICAgIHRoaXMuX2FsbG93ZWRTdWIuc3Vic2NyaXB0aW9uID0gdGhpcy51cGxvYWRTdG9yZS5maWxlVHlwZXNBbGxvd2VkJC5zdWJzY3JpYmUoKHgpID0+IHRoaXMudXBsb2FkQ29tcG9uZW50LnNldEFjY2VwdCh4KSk7XG4gICAgdGhpcy5fbXVsdGlTdWIuc3Vic2NyaXB0aW9uID0gdGhpcy51cGxvYWRTdG9yZS5pc011bHRpVXBsb2FkQWxsb3dlZCQuc3Vic2NyaWJlKCh4KSA9PiB0aGlzLnVwbG9hZENvbXBvbmVudC5zZXRNdWx0aXBsZSh4KSk7XG4gICAgdGhpcy5fZmlsZXNTdWIuc3Vic2NyaXB0aW9uID0gdGhpcy51cGxvYWRDb21wb25lbnQuZmlsZXNDaGFuZ2VkLnN1YnNjcmliZSgoZmlsZXMpID0+IHRoaXMudXBsb2FkU3RvcmUuc2V0RmlsZXMoZmlsZXMubWF0Y2hSZXN1bHQuYWNjZXB0ZWQpKTtcbiAgfVxuXG4gIG5nT25EZXN0cm95KCk6IHZvaWQge1xuICAgIHRoaXMuX2FsbG93ZWRTdWIuZGVzdHJveSgpO1xuICAgIHRoaXMuX211bHRpU3ViLmRlc3Ryb3koKTtcbiAgICB0aGlzLl9maWxlc1N1Yi5kZXN0cm95KCk7XG4gIH1cbn1cbiJdfQ==
@@ -0,0 +1,4 @@
1
+ export * from './container';
2
+ export * from './store';
3
+ export * from './storagefile.upload.module';
4
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9wYWNrYWdlcy9kYngtZmlyZWJhc2Uvc3JjL2xpYi9tb2R1bGVzL3N0b3JhZ2VmaWxlL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLGNBQWMsYUFBYSxDQUFDO0FBQzVCLGNBQWMsU0FBUyxDQUFDO0FBQ3hCLGNBQWMsNkJBQTZCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBmcm9tICcuL2NvbnRhaW5lcic7XG5leHBvcnQgKiBmcm9tICcuL3N0b3JlJztcbmV4cG9ydCAqIGZyb20gJy4vc3RvcmFnZWZpbGUudXBsb2FkLm1vZHVsZSc7XG4iXX0=
@@ -0,0 +1,83 @@
1
+ import { DbxActionLoadingContextDirective, DbxActionModule, DbxActionSnackbarErrorDirective, DbxFileUploadComponent, DbxFileUploadActionSyncDirective, DbxLoadingComponent } from '@dereekb/dbx-web';
2
+ import { DbxFirebaseStorageFileUploadActionHandlerDirective, DbxFirebaseStorageFileUploadStoreDirective, DbxFirebaseStorageFileUploadSyncDirective, DbxFirebaseStorageFileUploadInitializeDocumentDirective } from './container';
3
+ import { NgModule } from '@angular/core';
4
+ import { DbxFirebaseStorageFileDocumentStoreDirective } from './store/storagefile.document.store.directive';
5
+ import { DbxFirebaseStorageFileCollectionStoreDirective } from './store';
6
+ import * as i0 from "@angular/core";
7
+ export const importsAndExports = [
8
+ // dbx-core/dbx-web modules/components
9
+ DbxActionModule,
10
+ DbxLoadingComponent,
11
+ DbxActionLoadingContextDirective,
12
+ DbxActionSnackbarErrorDirective,
13
+ DbxFileUploadComponent,
14
+ DbxFileUploadActionSyncDirective,
15
+ // containers
16
+ DbxFirebaseStorageFileUploadActionHandlerDirective,
17
+ DbxFirebaseStorageFileUploadStoreDirective,
18
+ DbxFirebaseStorageFileUploadSyncDirective,
19
+ DbxFirebaseStorageFileUploadInitializeDocumentDirective,
20
+ // stores
21
+ DbxFirebaseStorageFileCollectionStoreDirective,
22
+ DbxFirebaseStorageFileDocumentStoreDirective
23
+ ];
24
+ /**
25
+ * Convenience module for importing various modules/components that are relevant to the storage file upload feature.
26
+ *
27
+ * - DbxActionModule
28
+ * - DbxFileUploadComponent
29
+ * - DbxFirebaseStorageFileUploadActionHandlerDirective
30
+ * - DbxFirebaseStorageFileUploadStoreDirective
31
+ * - DbxFirebaseStorageFileUploadSyncDirective
32
+ * - DbxFirebaseStorageFileUploadInitializeDocumentDirective
33
+ * - DbxFirebaseStorageFileUploadStore
34
+ */
35
+ export class DbxFirebaseStorageFileUploadModule {
36
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: DbxFirebaseStorageFileUploadModule, deps: [], target: i0.ɵɵFactoryTarget.NgModule });
37
+ static ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "14.0.0", version: "18.2.13", ngImport: i0, type: DbxFirebaseStorageFileUploadModule, imports: [
38
+ // dbx-core/dbx-web modules/components
39
+ DbxActionModule,
40
+ DbxLoadingComponent,
41
+ DbxActionLoadingContextDirective,
42
+ DbxActionSnackbarErrorDirective,
43
+ DbxFileUploadComponent,
44
+ DbxFileUploadActionSyncDirective,
45
+ // containers
46
+ DbxFirebaseStorageFileUploadActionHandlerDirective,
47
+ DbxFirebaseStorageFileUploadStoreDirective,
48
+ DbxFirebaseStorageFileUploadSyncDirective,
49
+ DbxFirebaseStorageFileUploadInitializeDocumentDirective,
50
+ // stores
51
+ DbxFirebaseStorageFileCollectionStoreDirective,
52
+ DbxFirebaseStorageFileDocumentStoreDirective], exports: [
53
+ // dbx-core/dbx-web modules/components
54
+ DbxActionModule,
55
+ DbxLoadingComponent,
56
+ DbxActionLoadingContextDirective,
57
+ DbxActionSnackbarErrorDirective,
58
+ DbxFileUploadComponent,
59
+ DbxFileUploadActionSyncDirective,
60
+ // containers
61
+ DbxFirebaseStorageFileUploadActionHandlerDirective,
62
+ DbxFirebaseStorageFileUploadStoreDirective,
63
+ DbxFirebaseStorageFileUploadSyncDirective,
64
+ DbxFirebaseStorageFileUploadInitializeDocumentDirective,
65
+ // stores
66
+ DbxFirebaseStorageFileCollectionStoreDirective,
67
+ DbxFirebaseStorageFileDocumentStoreDirective] });
68
+ static ɵinj = i0.ɵɵngDeclareInjector({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: DbxFirebaseStorageFileUploadModule, imports: [
69
+ // dbx-core/dbx-web modules/components
70
+ DbxActionModule,
71
+ DbxLoadingComponent,
72
+ DbxFileUploadComponent,
73
+ // dbx-core/dbx-web modules/components
74
+ DbxActionModule] });
75
+ }
76
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: DbxFirebaseStorageFileUploadModule, decorators: [{
77
+ type: NgModule,
78
+ args: [{
79
+ imports: importsAndExports,
80
+ exports: importsAndExports
81
+ }]
82
+ }] });
83
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3RvcmFnZWZpbGUudXBsb2FkLm1vZHVsZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uL3BhY2thZ2VzL2RieC1maXJlYmFzZS9zcmMvbGliL21vZHVsZXMvc3RvcmFnZWZpbGUvc3RvcmFnZWZpbGUudXBsb2FkLm1vZHVsZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsZ0NBQWdDLEVBQUUsZUFBZSxFQUFFLCtCQUErQixFQUFFLHNCQUFzQixFQUFFLGdDQUFnQyxFQUFFLG1CQUFtQixFQUFFLE1BQU0sa0JBQWtCLENBQUM7QUFDck0sT0FBTyxFQUFFLGtEQUFrRCxFQUFFLDBDQUEwQyxFQUFFLHlDQUF5QyxFQUFFLHVEQUF1RCxFQUFFLE1BQU0sYUFBYSxDQUFDO0FBQ2pPLE9BQU8sRUFBRSxRQUFRLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDekMsT0FBTyxFQUFFLDRDQUE0QyxFQUFFLE1BQU0sOENBQThDLENBQUM7QUFDNUcsT0FBTyxFQUFFLDhDQUE4QyxFQUFFLE1BQU0sU0FBUyxDQUFDOztBQUV6RSxNQUFNLENBQUMsTUFBTSxpQkFBaUIsR0FBRztJQUMvQixzQ0FBc0M7SUFDdEMsZUFBZTtJQUNmLG1CQUFtQjtJQUNuQixnQ0FBZ0M7SUFDaEMsK0JBQStCO0lBQy9CLHNCQUFzQjtJQUN0QixnQ0FBZ0M7SUFDaEMsYUFBYTtJQUNiLGtEQUFrRDtJQUNsRCwwQ0FBMEM7SUFDMUMseUNBQXlDO0lBQ3pDLHVEQUF1RDtJQUN2RCxTQUFTO0lBQ1QsOENBQThDO0lBQzlDLDRDQUE0QztDQUM3QyxDQUFDO0FBRUY7Ozs7Ozs7Ozs7R0FVRztBQUtILE1BQU0sT0FBTyxrQ0FBa0M7d0dBQWxDLGtDQUFrQzt5R0FBbEMsa0NBQWtDO1lBaEM3QyxzQ0FBc0M7WUFDdEMsZUFBZTtZQUNmLG1CQUFtQjtZQUNuQixnQ0FBZ0M7WUFDaEMsK0JBQStCO1lBQy9CLHNCQUFzQjtZQUN0QixnQ0FBZ0M7WUFDaEMsYUFBYTtZQUNiLGtEQUFrRDtZQUNsRCwwQ0FBMEM7WUFDMUMseUNBQXlDO1lBQ3pDLHVEQUF1RDtZQUN2RCxTQUFTO1lBQ1QsOENBQThDO1lBQzlDLDRDQUE0QztZQWQ1QyxzQ0FBc0M7WUFDdEMsZUFBZTtZQUNmLG1CQUFtQjtZQUNuQixnQ0FBZ0M7WUFDaEMsK0JBQStCO1lBQy9CLHNCQUFzQjtZQUN0QixnQ0FBZ0M7WUFDaEMsYUFBYTtZQUNiLGtEQUFrRDtZQUNsRCwwQ0FBMEM7WUFDMUMseUNBQXlDO1lBQ3pDLHVEQUF1RDtZQUN2RCxTQUFTO1lBQ1QsOENBQThDO1lBQzlDLDRDQUE0Qzt5R0FrQmpDLGtDQUFrQztZQWhDN0Msc0NBQXNDO1lBQ3RDLGVBQWU7WUFDZixtQkFBbUI7WUFHbkIsc0JBQXNCO1lBTHRCLHNDQUFzQztZQUN0QyxlQUFlOzs0RkErQkosa0NBQWtDO2tCQUo5QyxRQUFRO21CQUFDO29CQUNSLE9BQU8sRUFBRSxpQkFBaUI7b0JBQzFCLE9BQU8sRUFBRSxpQkFBaUI7aUJBQzNCIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgRGJ4QWN0aW9uTG9hZGluZ0NvbnRleHREaXJlY3RpdmUsIERieEFjdGlvbk1vZHVsZSwgRGJ4QWN0aW9uU25hY2tiYXJFcnJvckRpcmVjdGl2ZSwgRGJ4RmlsZVVwbG9hZENvbXBvbmVudCwgRGJ4RmlsZVVwbG9hZEFjdGlvblN5bmNEaXJlY3RpdmUsIERieExvYWRpbmdDb21wb25lbnQgfSBmcm9tICdAZGVyZWVrYi9kYngtd2ViJztcbmltcG9ydCB7IERieEZpcmViYXNlU3RvcmFnZUZpbGVVcGxvYWRBY3Rpb25IYW5kbGVyRGlyZWN0aXZlLCBEYnhGaXJlYmFzZVN0b3JhZ2VGaWxlVXBsb2FkU3RvcmVEaXJlY3RpdmUsIERieEZpcmViYXNlU3RvcmFnZUZpbGVVcGxvYWRTeW5jRGlyZWN0aXZlLCBEYnhGaXJlYmFzZVN0b3JhZ2VGaWxlVXBsb2FkSW5pdGlhbGl6ZURvY3VtZW50RGlyZWN0aXZlIH0gZnJvbSAnLi9jb250YWluZXInO1xuaW1wb3J0IHsgTmdNb2R1bGUgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IERieEZpcmViYXNlU3RvcmFnZUZpbGVEb2N1bWVudFN0b3JlRGlyZWN0aXZlIH0gZnJvbSAnLi9zdG9yZS9zdG9yYWdlZmlsZS5kb2N1bWVudC5zdG9yZS5kaXJlY3RpdmUnO1xuaW1wb3J0IHsgRGJ4RmlyZWJhc2VTdG9yYWdlRmlsZUNvbGxlY3Rpb25TdG9yZURpcmVjdGl2ZSB9IGZyb20gJy4vc3RvcmUnO1xuXG5leHBvcnQgY29uc3QgaW1wb3J0c0FuZEV4cG9ydHMgPSBbXG4gIC8vIGRieC1jb3JlL2RieC13ZWIgbW9kdWxlcy9jb21wb25lbnRzXG4gIERieEFjdGlvbk1vZHVsZSxcbiAgRGJ4TG9hZGluZ0NvbXBvbmVudCxcbiAgRGJ4QWN0aW9uTG9hZGluZ0NvbnRleHREaXJlY3RpdmUsXG4gIERieEFjdGlvblNuYWNrYmFyRXJyb3JEaXJlY3RpdmUsXG4gIERieEZpbGVVcGxvYWRDb21wb25lbnQsXG4gIERieEZpbGVVcGxvYWRBY3Rpb25TeW5jRGlyZWN0aXZlLFxuICAvLyBjb250YWluZXJzXG4gIERieEZpcmViYXNlU3RvcmFnZUZpbGVVcGxvYWRBY3Rpb25IYW5kbGVyRGlyZWN0aXZlLFxuICBEYnhGaXJlYmFzZVN0b3JhZ2VGaWxlVXBsb2FkU3RvcmVEaXJlY3RpdmUsXG4gIERieEZpcmViYXNlU3RvcmFnZUZpbGVVcGxvYWRTeW5jRGlyZWN0aXZlLFxuICBEYnhGaXJlYmFzZVN0b3JhZ2VGaWxlVXBsb2FkSW5pdGlhbGl6ZURvY3VtZW50RGlyZWN0aXZlLFxuICAvLyBzdG9yZXNcbiAgRGJ4RmlyZWJhc2VTdG9yYWdlRmlsZUNvbGxlY3Rpb25TdG9yZURpcmVjdGl2ZSxcbiAgRGJ4RmlyZWJhc2VTdG9yYWdlRmlsZURvY3VtZW50U3RvcmVEaXJlY3RpdmVcbl07XG5cbi8qKlxuICogQ29udmVuaWVuY2UgbW9kdWxlIGZvciBpbXBvcnRpbmcgdmFyaW91cyBtb2R1bGVzL2NvbXBvbmVudHMgdGhhdCBhcmUgcmVsZXZhbnQgdG8gdGhlIHN0b3JhZ2UgZmlsZSB1cGxvYWQgZmVhdHVyZS5cbiAqXG4gKiAtIERieEFjdGlvbk1vZHVsZVxuICogLSBEYnhGaWxlVXBsb2FkQ29tcG9uZW50XG4gKiAtIERieEZpcmViYXNlU3RvcmFnZUZpbGVVcGxvYWRBY3Rpb25IYW5kbGVyRGlyZWN0aXZlXG4gKiAtIERieEZpcmViYXNlU3RvcmFnZUZpbGVVcGxvYWRTdG9yZURpcmVjdGl2ZVxuICogLSBEYnhGaXJlYmFzZVN0b3JhZ2VGaWxlVXBsb2FkU3luY0RpcmVjdGl2ZVxuICogLSBEYnhGaXJlYmFzZVN0b3JhZ2VGaWxlVXBsb2FkSW5pdGlhbGl6ZURvY3VtZW50RGlyZWN0aXZlXG4gKiAtIERieEZpcmViYXNlU3RvcmFnZUZpbGVVcGxvYWRTdG9yZVxuICovXG5ATmdNb2R1bGUoe1xuICBpbXBvcnRzOiBpbXBvcnRzQW5kRXhwb3J0cyxcbiAgZXhwb3J0czogaW1wb3J0c0FuZEV4cG9ydHNcbn0pXG5leHBvcnQgY2xhc3MgRGJ4RmlyZWJhc2VTdG9yYWdlRmlsZVVwbG9hZE1vZHVsZSB7fVxuIl19
@@ -0,0 +1,6 @@
1
+ export * from './storagefile.collection.store';
2
+ export * from './storagefile.collection.store.directive';
3
+ export * from './storagefile.document.store';
4
+ export * from './storagefile.document.store.directive';
5
+ export * from './storagefile.upload.store';
6
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi8uLi9wYWNrYWdlcy9kYngtZmlyZWJhc2Uvc3JjL2xpYi9tb2R1bGVzL3N0b3JhZ2VmaWxlL3N0b3JlL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLGNBQWMsZ0NBQWdDLENBQUM7QUFDL0MsY0FBYywwQ0FBMEMsQ0FBQztBQUN6RCxjQUFjLDhCQUE4QixDQUFDO0FBQzdDLGNBQWMsd0NBQXdDLENBQUM7QUFDdkQsY0FBYyw0QkFBNEIsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCAqIGZyb20gJy4vc3RvcmFnZWZpbGUuY29sbGVjdGlvbi5zdG9yZSc7XG5leHBvcnQgKiBmcm9tICcuL3N0b3JhZ2VmaWxlLmNvbGxlY3Rpb24uc3RvcmUuZGlyZWN0aXZlJztcbmV4cG9ydCAqIGZyb20gJy4vc3RvcmFnZWZpbGUuZG9jdW1lbnQuc3RvcmUnO1xuZXhwb3J0ICogZnJvbSAnLi9zdG9yYWdlZmlsZS5kb2N1bWVudC5zdG9yZS5kaXJlY3RpdmUnO1xuZXhwb3J0ICogZnJvbSAnLi9zdG9yYWdlZmlsZS51cGxvYWQuc3RvcmUnO1xuIl19
@@ -0,0 +1,22 @@
1
+ import { Directive } from '@angular/core';
2
+ import { DbxFirebaseCollectionStoreDirective, provideDbxFirebaseCollectionStoreDirective } from '../../../model/modules/store';
3
+ import { StorageFileCollectionStore } from './storagefile.collection.store';
4
+ import * as i0 from "@angular/core";
5
+ import * as i1 from "./storagefile.collection.store";
6
+ export class DbxFirebaseStorageFileCollectionStoreDirective extends DbxFirebaseCollectionStoreDirective {
7
+ constructor(store) {
8
+ super(store);
9
+ this.setConstraints([]);
10
+ }
11
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: DbxFirebaseStorageFileCollectionStoreDirective, deps: [{ token: i1.StorageFileCollectionStore }], target: i0.ɵɵFactoryTarget.Directive });
12
+ static ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "18.2.13", type: DbxFirebaseStorageFileCollectionStoreDirective, isStandalone: true, selector: "[dbxFirebaseStorageFileCollection]", providers: provideDbxFirebaseCollectionStoreDirective(DbxFirebaseStorageFileCollectionStoreDirective, StorageFileCollectionStore), usesInheritance: true, ngImport: i0 });
13
+ }
14
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: DbxFirebaseStorageFileCollectionStoreDirective, decorators: [{
15
+ type: Directive,
16
+ args: [{
17
+ selector: '[dbxFirebaseStorageFileCollection]',
18
+ providers: provideDbxFirebaseCollectionStoreDirective(DbxFirebaseStorageFileCollectionStoreDirective, StorageFileCollectionStore),
19
+ standalone: true
20
+ }]
21
+ }], ctorParameters: () => [{ type: i1.StorageFileCollectionStore }] });
22
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3RvcmFnZWZpbGUuY29sbGVjdGlvbi5zdG9yZS5kaXJlY3RpdmUuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi8uLi9wYWNrYWdlcy9kYngtZmlyZWJhc2Uvc3JjL2xpYi9tb2R1bGVzL3N0b3JhZ2VmaWxlL3N0b3JlL3N0b3JhZ2VmaWxlLmNvbGxlY3Rpb24uc3RvcmUuZGlyZWN0aXZlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxTQUFTLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDMUMsT0FBTyxFQUFFLG1DQUFtQyxFQUFFLDBDQUEwQyxFQUFFLE1BQU0sOEJBQThCLENBQUM7QUFDL0gsT0FBTyxFQUFFLDBCQUEwQixFQUFFLE1BQU0sZ0NBQWdDLENBQUM7OztBQVE1RSxNQUFNLE9BQU8sOENBQStDLFNBQVEsbUNBQWlHO0lBQ25LLFlBQVksS0FBaUM7UUFDM0MsS0FBSyxDQUFDLEtBQUssQ0FBQyxDQUFDO1FBQ2IsSUFBSSxDQUFDLGNBQWMsQ0FBQyxFQUFFLENBQUMsQ0FBQztJQUMxQixDQUFDO3dHQUpVLDhDQUE4Qzs0RkFBOUMsOENBQThDLGlGQUg5QywwQ0FBMEMsQ0FBQyw4Q0FBOEMsRUFBRSwwQkFBMEIsQ0FBQzs7NEZBR3RILDhDQUE4QztrQkFMMUQsU0FBUzttQkFBQztvQkFDVCxRQUFRLEVBQUUsb0NBQW9DO29CQUM5QyxTQUFTLEVBQUUsMENBQTBDLGlEQUFpRCwwQkFBMEIsQ0FBQztvQkFDakksVUFBVSxFQUFFLElBQUk7aUJBQ2pCIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgRGlyZWN0aXZlIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBEYnhGaXJlYmFzZUNvbGxlY3Rpb25TdG9yZURpcmVjdGl2ZSwgcHJvdmlkZURieEZpcmViYXNlQ29sbGVjdGlvblN0b3JlRGlyZWN0aXZlIH0gZnJvbSAnLi4vLi4vLi4vbW9kZWwvbW9kdWxlcy9zdG9yZSc7XG5pbXBvcnQgeyBTdG9yYWdlRmlsZUNvbGxlY3Rpb25TdG9yZSB9IGZyb20gJy4vc3RvcmFnZWZpbGUuY29sbGVjdGlvbi5zdG9yZSc7XG5pbXBvcnQgeyBTdG9yYWdlRmlsZSwgU3RvcmFnZUZpbGVEb2N1bWVudCB9IGZyb20gJ0BkZXJlZWtiL2ZpcmViYXNlJztcblxuQERpcmVjdGl2ZSh7XG4gIHNlbGVjdG9yOiAnW2RieEZpcmViYXNlU3RvcmFnZUZpbGVDb2xsZWN0aW9uXScsXG4gIHByb3ZpZGVyczogcHJvdmlkZURieEZpcmViYXNlQ29sbGVjdGlvblN0b3JlRGlyZWN0aXZlKERieEZpcmViYXNlU3RvcmFnZUZpbGVDb2xsZWN0aW9uU3RvcmVEaXJlY3RpdmUsIFN0b3JhZ2VGaWxlQ29sbGVjdGlvblN0b3JlKSxcbiAgc3RhbmRhbG9uZTogdHJ1ZVxufSlcbmV4cG9ydCBjbGFzcyBEYnhGaXJlYmFzZVN0b3JhZ2VGaWxlQ29sbGVjdGlvblN0b3JlRGlyZWN0aXZlIGV4dGVuZHMgRGJ4RmlyZWJhc2VDb2xsZWN0aW9uU3RvcmVEaXJlY3RpdmU8U3RvcmFnZUZpbGUsIFN0b3JhZ2VGaWxlRG9jdW1lbnQsIFN0b3JhZ2VGaWxlQ29sbGVjdGlvblN0b3JlPiB7XG4gIGNvbnN0cnVjdG9yKHN0b3JlOiBTdG9yYWdlRmlsZUNvbGxlY3Rpb25TdG9yZSkge1xuICAgIHN1cGVyKHN0b3JlKTtcbiAgICB0aGlzLnNldENvbnN0cmFpbnRzKFtdKTtcbiAgfVxufVxuIl19
@@ -0,0 +1,18 @@
1
+ import { Injectable, inject } from '@angular/core';
2
+ import { AbstractDbxFirebaseCollectionStore, firebaseCollectionStoreCreateFunction } from '../../../model/modules/store';
3
+ import { StorageFileFirestoreCollections, StorageFileFunctions } from '@dereekb/firebase';
4
+ import * as i0 from "@angular/core";
5
+ import * as i1 from "@dereekb/firebase";
6
+ export class StorageFileCollectionStore extends AbstractDbxFirebaseCollectionStore {
7
+ storageFileFunctions = inject(StorageFileFunctions);
8
+ constructor(collections) {
9
+ super({ firestoreCollection: collections.storageFileCollection });
10
+ }
11
+ initializeAllStorageFilesFromUpload = firebaseCollectionStoreCreateFunction(this, this.storageFileFunctions.storageFile.createStorageFile.allFromUpload);
12
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: StorageFileCollectionStore, deps: [{ token: i1.StorageFileFirestoreCollections }], target: i0.ɵɵFactoryTarget.Injectable });
13
+ static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: StorageFileCollectionStore });
14
+ }
15
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: StorageFileCollectionStore, decorators: [{
16
+ type: Injectable
17
+ }], ctorParameters: () => [{ type: i1.StorageFileFirestoreCollections }] });
18
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3RvcmFnZWZpbGUuY29sbGVjdGlvbi5zdG9yZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uLy4uL3BhY2thZ2VzL2RieC1maXJlYmFzZS9zcmMvbGliL21vZHVsZXMvc3RvcmFnZWZpbGUvc3RvcmUvc3RvcmFnZWZpbGUuY29sbGVjdGlvbi5zdG9yZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsVUFBVSxFQUFFLE1BQU0sRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUNuRCxPQUFPLEVBQUUsa0NBQWtDLEVBQUUscUNBQXFDLEVBQUUsTUFBTSw4QkFBOEIsQ0FBQztBQUN6SCxPQUFPLEVBQUUsK0JBQStCLEVBQW9DLG9CQUFvQixFQUFFLE1BQU0sbUJBQW1CLENBQUM7OztBQUc1SCxNQUFNLE9BQU8sMEJBQTJCLFNBQVEsa0NBQW9FO0lBQ3pHLG9CQUFvQixHQUFHLE1BQU0sQ0FBQyxvQkFBb0IsQ0FBQyxDQUFDO0lBRTdELFlBQVksV0FBNEM7UUFDdEQsS0FBSyxDQUFDLEVBQUUsbUJBQW1CLEVBQUUsV0FBVyxDQUFDLHFCQUFxQixFQUFFLENBQUMsQ0FBQztJQUNwRSxDQUFDO0lBRVEsbUNBQW1DLEdBQUcscUNBQXFDLENBQUMsSUFBSSxFQUFFLElBQUksQ0FBQyxvQkFBb0IsQ0FBQyxXQUFXLENBQUMsaUJBQWlCLENBQUMsYUFBYSxDQUFDLENBQUM7d0dBUHZKLDBCQUEwQjs0R0FBMUIsMEJBQTBCOzs0RkFBMUIsMEJBQTBCO2tCQUR0QyxVQUFVIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgSW5qZWN0YWJsZSwgaW5qZWN0IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBBYnN0cmFjdERieEZpcmViYXNlQ29sbGVjdGlvblN0b3JlLCBmaXJlYmFzZUNvbGxlY3Rpb25TdG9yZUNyZWF0ZUZ1bmN0aW9uIH0gZnJvbSAnLi4vLi4vLi4vbW9kZWwvbW9kdWxlcy9zdG9yZSc7XG5pbXBvcnQgeyBTdG9yYWdlRmlsZUZpcmVzdG9yZUNvbGxlY3Rpb25zLCBTdG9yYWdlRmlsZSwgU3RvcmFnZUZpbGVEb2N1bWVudCwgU3RvcmFnZUZpbGVGdW5jdGlvbnMgfSBmcm9tICdAZGVyZWVrYi9maXJlYmFzZSc7XG5cbkBJbmplY3RhYmxlKClcbmV4cG9ydCBjbGFzcyBTdG9yYWdlRmlsZUNvbGxlY3Rpb25TdG9yZSBleHRlbmRzIEFic3RyYWN0RGJ4RmlyZWJhc2VDb2xsZWN0aW9uU3RvcmU8U3RvcmFnZUZpbGUsIFN0b3JhZ2VGaWxlRG9jdW1lbnQ+IHtcbiAgcmVhZG9ubHkgc3RvcmFnZUZpbGVGdW5jdGlvbnMgPSBpbmplY3QoU3RvcmFnZUZpbGVGdW5jdGlvbnMpO1xuXG4gIGNvbnN0cnVjdG9yKGNvbGxlY3Rpb25zOiBTdG9yYWdlRmlsZUZpcmVzdG9yZUNvbGxlY3Rpb25zKSB7XG4gICAgc3VwZXIoeyBmaXJlc3RvcmVDb2xsZWN0aW9uOiBjb2xsZWN0aW9ucy5zdG9yYWdlRmlsZUNvbGxlY3Rpb24gfSk7XG4gIH1cblxuICByZWFkb25seSBpbml0aWFsaXplQWxsU3RvcmFnZUZpbGVzRnJvbVVwbG9hZCA9IGZpcmViYXNlQ29sbGVjdGlvblN0b3JlQ3JlYXRlRnVuY3Rpb24odGhpcywgdGhpcy5zdG9yYWdlRmlsZUZ1bmN0aW9ucy5zdG9yYWdlRmlsZS5jcmVhdGVTdG9yYWdlRmlsZS5hbGxGcm9tVXBsb2FkKTtcbn1cbiJdfQ==