quasar-ui-danx 0.4.53 → 0.4.55

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/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "quasar-ui-danx",
3
- "version": "0.4.53",
3
+ "version": "0.4.55",
4
4
  "author": "Dan <dan@flytedesk.com>",
5
5
  "description": "DanX Vue / Quasar component library",
6
6
  "license": "MIT",
@@ -10,7 +10,7 @@
10
10
  <component
11
11
  :is="icon || typeOptions.icon"
12
12
  class="transition-all"
13
- :class="iconClass + ' ' + typeOptions.iconClass"
13
+ :class="resolvedIconClass"
14
14
  />
15
15
  <slot>
16
16
  <div
@@ -60,6 +60,8 @@ import {
60
60
  FaSolidCircleCheck as ConfirmIcon,
61
61
  FaSolidCircleXmark as CancelIcon,
62
62
  FaSolidCopy as CopyIcon,
63
+ FaSolidFileExport as ExportIcon,
64
+ FaSolidFileImport as ImportIcon,
63
65
  FaSolidPause as PauseIcon,
64
66
  FaSolidPencil as EditIcon,
65
67
  FaSolidPlay as PlayIcon,
@@ -71,7 +73,7 @@ import { computed, ref } from "vue";
71
73
  import { ActionTarget, ResourceAction } from "../../../types";
72
74
 
73
75
  export interface ActionButtonProps {
74
- type?: "trash" | "trash-red" | "create" | "edit" | "copy" | "play" | "stop" | "pause" | "refresh" | "confirm" | "cancel";
76
+ type?: "trash" | "trash-red" | "create" | "edit" | "copy" | "play" | "stop" | "pause" | "refresh" | "confirm" | "cancel" | "export" | "import";
75
77
  color?: "red" | "blue" | "sky" | "sky-invert" | "green" | "green-invert" | "lime" | "white" | "gray";
76
78
  size?: "xxs" | "xs" | "sm" | "md" | "lg";
77
79
  icon?: object | string;
@@ -160,64 +162,37 @@ const colorClass = computed(() => {
160
162
  return "";
161
163
  }
162
164
  });
165
+
166
+ const resolvedIconClass = computed(() => props.iconClass + " " + mappedSizeClass[props.size].icon);
167
+
163
168
  const typeOptions = computed(() => {
164
- const iconClass = mappedSizeClass[props.size].icon;
165
169
  switch (props.type) {
166
170
  case "trash":
167
- return {
168
- icon: TrashIcon,
169
- iconClass
170
- };
171
+ return { icon: TrashIcon };
172
+ case "export":
173
+ return { icon: ExportIcon };
174
+ case "import":
175
+ return { icon: ImportIcon };
171
176
  case "create":
172
- return {
173
- icon: CreateIcon,
174
- iconClass
175
- };
177
+ return { icon: CreateIcon };
176
178
  case "confirm":
177
- return {
178
- icon: ConfirmIcon,
179
- iconClass
180
- };
179
+ return { icon: ConfirmIcon };
181
180
  case "cancel":
182
- return {
183
- icon: CancelIcon,
184
- iconClass
185
- };
181
+ return { icon: CancelIcon };
186
182
  case "edit":
187
- return {
188
- icon: EditIcon,
189
- iconClass
190
- };
183
+ return { icon: EditIcon };
191
184
  case "copy":
192
- return {
193
- icon: CopyIcon,
194
- iconClass
195
- };
185
+ return { icon: CopyIcon };
196
186
  case "play":
197
- return {
198
- icon: PlayIcon,
199
- iconClass
200
- };
187
+ return { icon: PlayIcon };
201
188
  case "stop":
202
- return {
203
- icon: StopIcon,
204
- iconClass
205
- };
189
+ return { icon: StopIcon };
206
190
  case "pause":
207
- return {
208
- icon: PauseIcon,
209
- iconClass
210
- };
191
+ return { icon: PauseIcon };
211
192
  case "refresh":
212
- return {
213
- icon: RefreshIcon,
214
- iconClass
215
- };
193
+ return { icon: RefreshIcon };
216
194
  default:
217
- return {
218
- icon: EditIcon,
219
- iconClass
220
- };
195
+ return { icon: EditIcon };
221
196
  }
222
197
  });
223
198
 
@@ -69,6 +69,7 @@
69
69
  class="absolute-bottom w-full bg-slate-800"
70
70
  >
71
71
  <QLinearProgress
72
+ :key="'progress-' + isUploading ? 'uploading' : 'transcoding'"
72
73
  :value="isUploading ? file.progress : ((transcodingStatus?.progress || 0) / 100)"
73
74
  size="36px"
74
75
  :color="isUploading ? 'green-800' : 'blue-800'"
@@ -80,15 +81,14 @@
80
81
  class="mr-2 text-slate-50 ml-1 flex-shrink-0"
81
82
  :size="btnSize === 'xs' ? 10 : 20"
82
83
  />
83
- <div class="whitespace-nowrap overflow-hidden ellipsis">
84
- {{ isUploading ? "Uploading..." : transcodingStatus?.message }}
85
- </div>
86
- <QTooltip
87
- v-if="transcodingStatus?.message"
88
- class="text-sm"
89
- >
90
- {{ transcodingStatus?.message }}
91
- </QTooltip>
84
+ <template v-if="statusMessage">
85
+ <div class="whitespace-nowrap overflow-hidden ellipsis">
86
+ {{ statusMessage }}
87
+ </div>
88
+ <QTooltip class="text-sm">
89
+ {{ statusMessage }}
90
+ </QTooltip>
91
+ </template>
92
92
  </div>
93
93
  </QLinearProgress>
94
94
  </div>
@@ -208,6 +208,7 @@ const computedImage: ComputedRef<UploadedFile | null> = computed(() => {
208
208
  });
209
209
 
210
210
  const isUploading = computed(() => !props.file || props.file?.progress !== undefined);
211
+ const statusMessage = computed(() => isUploading.value ? "Uploading..." : transcodingStatus.value?.message);
211
212
  const previewableFiles: ComputedRef<(UploadedFile | null)[] | null> = computed(() => {
212
213
  return props.relatedFiles?.length > 0 ? uniqueBy([computedImage.value, ...props.relatedFiles], filesHaveSameUrl) : [computedImage.value];
213
214
  });
@@ -1,5 +1,6 @@
1
1
  import { uid } from "quasar";
2
2
  import { danxOptions } from "../config";
3
+ import { DateTime, parseDateTime } from "../helpers/formats";
3
4
  import {
4
5
  FileUploadAllCompleteCallback,
5
6
  FileUploadCompleteCallback,
@@ -278,11 +279,15 @@ export class FileUpload {
278
279
  */
279
280
  isTranscoding(file: UploadedFile) {
280
281
  const metaTranscodes = file?.meta?.transcodes || [];
281
-
282
282
  for (const transcodeName of Object.keys(metaTranscodes)) {
283
283
  const transcode = metaTranscodes[transcodeName];
284
+
284
285
  if (transcode.status === "Pending" || transcode.status === "In Progress") {
285
- return true;
286
+ const startedAt = parseDateTime(transcode.started_at);
287
+ const duration = startedAt && DateTime.now().diff(startedAt, "minutes");
288
+ if (!duration?.minutes || duration.minutes < 30) {
289
+ return true;
290
+ }
286
291
  }
287
292
  }
288
293
  return false;
@@ -0,0 +1,20 @@
1
+ import { AnyObject } from "../types";
2
+
3
+ export function importJson(file: File): Promise<AnyObject> {
4
+ return new Promise((resolve, reject) => {
5
+ const reader = new FileReader();
6
+ let jsonData = null;
7
+
8
+ reader.onload = async (e) => {
9
+ try {
10
+ jsonData = JSON.parse(e.target?.result as string);
11
+ resolve(jsonData);
12
+ } catch (error) {
13
+ console.error("Invalid JSON file:", error);
14
+ reject(error);
15
+ }
16
+ };
17
+
18
+ reader.readAsText(file);
19
+ });
20
+ }
@@ -10,6 +10,7 @@ export * from "./FileUpload";
10
10
  export * from "./FlashMessages";
11
11
  export * from "./formats";
12
12
  export * from "./hotkeys";
13
+ export * from "./importFile";
13
14
  export * from "./multiFileUpload";
14
15
  export * from "./objectStore";
15
16
  export * from "./request";
@@ -44,10 +44,10 @@ export function storeObject<T extends TypedObject>(newObject: T, recentlyStoredO
44
44
  // Retrieve the existing object if it already exists in the store
45
45
  const oldObject = store.get(objectKey);
46
46
 
47
- // If an old object exists, and it is newer than the new object, do not store the new object, just return the old
47
+ // If an old object exists, and it is newer than the new object, and all the child objects are not newer than the original do not store the new object,
48
+ // just return the old
48
49
  // NOTE: If the timestamp is the same, its possible the intention is to update the existing object, so DO NOT return old object in this case
49
- // @ts-expect-error __timestamp is guaranteed to be set in this case on both old and new
50
- if (oldObject && newObject.__timestamp < oldObject.__timestamp) {
50
+ if (!hasRecentUpdates(newObject, oldObject)) {
51
51
  recentlyStoredObjects[objectKey] = oldObject;
52
52
 
53
53
  // Recursively store all the children of the object as well
@@ -105,6 +105,55 @@ function storeObjectChildren<T extends TypedObject>(object: T, recentlyStoredObj
105
105
  }
106
106
  }
107
107
 
108
+ /**
109
+ * Recursively check the current object and all its child objects
110
+ * to see if any of the objects are newer than the currently stored objects
111
+ */
112
+ function hasRecentUpdates(newObject: TypedObject, oldObject: TypedObject | null) {
113
+ // If there are no timestamps to compare, assume there are updates
114
+ if (!newObject.__timestamp || !oldObject?.__timestamp) return true;
115
+
116
+ // If the new object has a newer timestamp than the old then there are updates
117
+ if (newObject.__timestamp > oldObject.__timestamp) return true;
118
+
119
+ for (const key of Object.keys(newObject)) {
120
+ const newObjectValue = newObject[key];
121
+ const oldObjectValue = oldObject[key];
122
+
123
+ // If the old object does not have this key, this is new information, therefore there are updates
124
+ if (!oldObjectValue?.__timestamp) {
125
+ return true;
126
+ }
127
+
128
+ if (Array.isArray(newObjectValue) && newObjectValue.length > 0) {
129
+ for (const newObjectItem of newObjectValue as TypedObject[]) {
130
+ // Only compare the object if it is a TypedObject
131
+ if (newObjectItem?.__type) {
132
+ const oldObjectItem = (oldObjectValue as TypedObject[]).find((v: TypedObject) => v.id === newObjectItem.id && v.__type === newObjectItem.__type);
133
+
134
+ // If the oldObject does not have this entry, then there is new information, therefore there are updates
135
+ if (!oldObjectItem?.__timestamp) {
136
+ return true;
137
+ }
138
+
139
+ // Compare the child objects to see if there are updates
140
+ if (hasRecentUpdates(newObjectItem, oldObjectItem)) {
141
+ return true;
142
+ }
143
+ }
144
+ }
145
+ } else if (newObjectValue?.__type) {
146
+ // Only compare the object if it is a TypedObject
147
+ if (hasRecentUpdates(newObjectValue, oldObjectValue)) {
148
+ return true;
149
+ }
150
+ }
151
+ }
152
+
153
+ // If no updates were found, return false
154
+ return false;
155
+ }
156
+
108
157
  /**
109
158
  * Remove an object from all lists in the store
110
159
  */