@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.
- package/esm2022/lib/auth/login/login.button.component.mjs +1 -1
- package/esm2022/lib/auth/login/login.email.content.component.mjs +2 -2
- package/esm2022/lib/development/development.scheduler.list.component.mjs +1 -1
- package/esm2022/lib/firestore/firebase.firestore.providers.mjs +21 -2
- package/esm2022/lib/model/loader/collection.loader.instance.mjs +65 -5
- package/esm2022/lib/model/modules/store/index.mjs +2 -1
- package/esm2022/lib/model/modules/store/store.collection.crud.mjs +28 -0
- package/esm2022/lib/model/modules/store/store.collection.directive.mjs +20 -2
- package/esm2022/lib/model/modules/store/store.collection.mjs +14 -1
- package/esm2022/lib/modules/index.mjs +2 -1
- package/esm2022/lib/modules/storagefile/container/index.mjs +7 -0
- package/esm2022/lib/modules/storagefile/container/storagefile.upload.action.handler.directive.mjs +118 -0
- package/esm2022/lib/modules/storagefile/container/storagefile.upload.error.mjs +12 -0
- package/esm2022/lib/modules/storagefile/container/storagefile.upload.handler.mjs +260 -0
- package/esm2022/lib/modules/storagefile/container/storagefile.upload.initialize.document.directive.mjs +39 -0
- package/esm2022/lib/modules/storagefile/container/storagefile.upload.store.directive.mjs +38 -0
- package/esm2022/lib/modules/storagefile/container/storagefile.upload.sync.directive.mjs +36 -0
- package/esm2022/lib/modules/storagefile/index.mjs +4 -0
- package/esm2022/lib/modules/storagefile/storagefile.upload.module.mjs +83 -0
- package/esm2022/lib/modules/storagefile/store/index.mjs +6 -0
- package/esm2022/lib/modules/storagefile/store/storagefile.collection.store.directive.mjs +22 -0
- package/esm2022/lib/modules/storagefile/store/storagefile.collection.store.mjs +18 -0
- package/esm2022/lib/modules/storagefile/store/storagefile.document.store.directive.mjs +21 -0
- package/esm2022/lib/modules/storagefile/store/storagefile.document.store.mjs +22 -0
- package/esm2022/lib/modules/storagefile/store/storagefile.upload.store.mjs +63 -0
- package/esm2022/lib/storage/firebase.storage.module.mjs +2 -2
- package/fesm2022/dereekb-dbx-firebase.mjs +839 -30
- package/fesm2022/dereekb-dbx-firebase.mjs.map +1 -1
- package/lib/firestore/firebase.firestore.providers.d.ts +14 -1
- package/lib/model/loader/collection.loader.instance.d.ts +26 -1
- package/lib/model/modules/store/index.d.ts +1 -0
- package/lib/model/modules/store/store.collection.crud.d.ts +21 -0
- package/lib/model/modules/store/store.collection.d.ts +19 -2
- package/lib/model/modules/store/store.collection.directive.d.ts +12 -2
- package/lib/modules/index.d.ts +1 -0
- package/lib/modules/storagefile/container/index.d.ts +6 -0
- package/lib/modules/storagefile/container/storagefile.upload.action.handler.directive.d.ts +47 -0
- package/lib/modules/storagefile/container/storagefile.upload.error.d.ts +9 -0
- package/lib/modules/storagefile/container/storagefile.upload.handler.d.ts +183 -0
- package/lib/modules/storagefile/container/storagefile.upload.initialize.document.directive.d.ts +14 -0
- package/lib/modules/storagefile/container/storagefile.upload.store.directive.d.ts +20 -0
- package/lib/modules/storagefile/container/storagefile.upload.sync.directive.d.ts +18 -0
- package/lib/modules/storagefile/index.d.ts +3 -0
- package/lib/modules/storagefile/storagefile.upload.module.d.ts +29 -0
- package/lib/modules/storagefile/store/index.d.ts +5 -0
- package/lib/modules/storagefile/store/storagefile.collection.store.d.ts +10 -0
- package/lib/modules/storagefile/store/storagefile.collection.store.directive.d.ts +9 -0
- package/lib/modules/storagefile/store/storagefile.document.store.d.ts +15 -0
- package/lib/modules/storagefile/store/storagefile.document.store.directive.d.ts +9 -0
- package/lib/modules/storagefile/store/storagefile.upload.store.d.ts +166 -0
- package/lib/storage/firebase.storage.module.d.ts +1 -1
- 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==
|