quasar-ui-danx 0.0.10 → 0.0.12
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 +8 -2
- package/src/components/ActionTable/ActionTable.vue +143 -0
- package/src/components/ActionTable/BatchActionMenu.vue +60 -0
- package/src/components/ActionTable/EmptyTableState.vue +33 -0
- package/src/components/ActionTable/Filters/CollapsableFiltersSidebar.vue +36 -0
- package/src/components/ActionTable/Filters/FilterGroupItem.vue +28 -0
- package/src/components/ActionTable/Filters/FilterGroupList.vue +76 -0
- package/src/components/ActionTable/Filters/FilterListToggle.vue +50 -0
- package/src/components/ActionTable/Filters/FilterableField.vue +143 -0
- package/src/components/ActionTable/Filters/index.ts +5 -0
- package/src/components/ActionTable/Form/Fields/BooleanField.vue +37 -0
- package/src/components/ActionTable/Form/Fields/ConfirmPasswordField.vue +46 -0
- package/src/components/ActionTable/Form/Fields/DateField.vue +59 -0
- package/src/components/ActionTable/Form/Fields/DateRangeField.vue +110 -0
- package/src/components/ActionTable/Form/Fields/DateTimeField.vue +50 -0
- package/src/components/ActionTable/Form/Fields/DateTimePicker.vue +59 -0
- package/src/components/ActionTable/Form/Fields/EditableDiv.vue +39 -0
- package/src/components/ActionTable/Form/Fields/FieldLabel.vue +32 -0
- package/src/components/ActionTable/Form/Fields/FileUploadButton.vue +78 -0
- package/src/components/ActionTable/Form/Fields/InlineDateTimeField.vue +44 -0
- package/src/components/ActionTable/Form/Fields/IntegerField.vue +26 -0
- package/src/components/ActionTable/Form/Fields/LabelValueBlock.vue +22 -0
- package/src/components/ActionTable/Form/Fields/LabeledInput.vue +63 -0
- package/src/components/ActionTable/Form/Fields/MultiFileField.vue +91 -0
- package/src/components/ActionTable/Form/Fields/MultiKeywordField.vue +57 -0
- package/src/components/ActionTable/Form/Fields/NewPasswordField.vue +39 -0
- package/src/components/ActionTable/Form/Fields/NumberField.vue +94 -0
- package/src/components/ActionTable/Form/Fields/NumberRangeField.vue +140 -0
- package/src/components/ActionTable/Form/Fields/SelectDrawer.vue +136 -0
- package/src/components/ActionTable/Form/Fields/SelectField.vue +318 -0
- package/src/components/ActionTable/Form/Fields/SelectWithChildrenField.vue +81 -0
- package/src/components/ActionTable/Form/Fields/SingleFileField.vue +78 -0
- package/src/components/ActionTable/Form/Fields/TextField.vue +82 -0
- package/src/components/ActionTable/Form/Fields/WysiwygField.vue +46 -0
- package/src/components/ActionTable/Form/Fields/index.ts +23 -0
- package/src/components/ActionTable/Form/RenderedForm.vue +76 -0
- package/src/components/ActionTable/Form/index.ts +2 -0
- package/src/components/ActionTable/RenderComponentColumn.vue +22 -0
- package/src/components/ActionTable/TableSummaryRow.vue +95 -0
- package/src/components/ActionTable/index.ts +10 -0
- package/src/components/ActionTable/listActions.ts +362 -0
- package/src/components/ActionTable/listHelpers.ts +74 -0
- package/src/components/ActionTable/tableColumns.ts +72 -0
- package/src/components/DragAndDrop/HandleDraggable.vue +29 -29
- package/src/components/DragAndDrop/ListItemDraggable.vue +10 -10
- package/src/components/DragAndDrop/index.ts +0 -1
- package/src/components/DragAndDrop/listDragAndDrop.ts +1 -1
- package/src/components/Utility/CollapsableSidebar.vue +119 -0
- package/src/components/Utility/ContentDrawer.vue +70 -0
- package/src/components/Utility/Dialogs/ConfirmDialog.vue +132 -0
- package/src/components/Utility/Dialogs/FullScreenDialog.vue +46 -0
- package/src/components/Utility/Dialogs/FullscreenCarouselDialog.vue +105 -0
- package/src/components/Utility/Dialogs/InfoDialog.vue +92 -0
- package/src/components/Utility/Dialogs/InputDialog.vue +35 -0
- package/src/components/Utility/ImagePreview.vue +192 -0
- package/src/components/Utility/Popover/PopoverMenu.vue +64 -0
- package/src/components/Utility/Transitions/ListTransition.vue +50 -0
- package/src/components/Utility/Transitions/SlideTransition.vue +63 -0
- package/src/components/Utility/Transitions/StaggeredListTransition.vue +97 -0
- package/src/components/Utility/index.ts +11 -0
- package/src/components/index.ts +3 -0
- package/src/helpers/FileUpload.ts +295 -0
- package/src/helpers/FlashMessages.ts +79 -0
- package/src/helpers/array.ts +37 -0
- package/src/helpers/compatibility.ts +64 -0
- package/src/helpers/date.ts +5 -0
- package/src/helpers/download.ts +200 -0
- package/src/helpers/downloadPdf.ts +92 -0
- package/src/helpers/files.ts +52 -0
- package/src/helpers/formats.ts +183 -0
- package/src/helpers/http.ts +62 -0
- package/src/helpers/index.ts +12 -1
- package/src/helpers/multiFileUpload.ts +68 -0
- package/src/helpers/singleFileUpload.ts +54 -0
- package/src/helpers/storage.ts +8 -0
- package/src/index.esm.js +3 -4
- package/src/svg/FilterIcon.svg +7 -0
- package/src/svg/ImageIcon.svg +30 -0
- package/src/svg/PdfIcon.svg +21 -0
- package/src/svg/PercentIcon.svg +13 -0
- package/src/svg/TrashIcon.svg +15 -0
- package/src/svg/XIcon.svg +18 -0
- package/src/svg/index.ts +8 -0
- package/src/vendor/tinymce-config.ts +1 -0
- package/src/vue-plugin.js +7 -4
- package/tsconfig.json +14 -13
- package/src/components/DragAndDrop/Icons/index.ts +0 -2
- /package/src/{components/DragAndDrop/Icons → svg}/DragHandleDotsIcon.svg +0 -0
- /package/src/{components/DragAndDrop/Icons → svg}/DragHandleIcon.svg +0 -0
@@ -0,0 +1,22 @@
|
|
1
|
+
<template>
|
2
|
+
<Component
|
3
|
+
:is="component.is"
|
4
|
+
v-bind="component.props"
|
5
|
+
@action="$emit('action', $event)"
|
6
|
+
>{{ component.value || component.props?.text || rowProps.value }}</Component>
|
7
|
+
</template>
|
8
|
+
<script setup>
|
9
|
+
import { computed } from "vue";
|
10
|
+
|
11
|
+
defineEmits(["action"]);
|
12
|
+
const props = defineProps({
|
13
|
+
rowProps: {
|
14
|
+
type: Object,
|
15
|
+
required: true
|
16
|
+
}
|
17
|
+
});
|
18
|
+
|
19
|
+
const component = computed(() => {
|
20
|
+
return props.rowProps.col.component(props.rowProps.row);
|
21
|
+
});
|
22
|
+
</script>
|
@@ -0,0 +1,95 @@
|
|
1
|
+
<template>
|
2
|
+
<q-tr
|
3
|
+
class="sticky-column-1 transition-all sticky-row"
|
4
|
+
:class="{'!bg-neutral-plus-7': !selectedCount, '!bg-blue-base text-white selected': selectedCount, 'opacity-50': loading}"
|
5
|
+
>
|
6
|
+
<q-td
|
7
|
+
:colspan="stickyColspan"
|
8
|
+
class="font-bold transition-all"
|
9
|
+
:class="{'!bg-neutral-plus-7 !pl-5': !selectedCount, '!bg-blue-base text-white !pl-4': selectedCount}"
|
10
|
+
>
|
11
|
+
<div class="flex flex-nowrap items-center">
|
12
|
+
<div
|
13
|
+
v-if="selectedCount"
|
14
|
+
class="flex items-center"
|
15
|
+
>
|
16
|
+
<ClearIcon
|
17
|
+
class="w-6 mr-3"
|
18
|
+
@click="$emit('clear')"
|
19
|
+
/>
|
20
|
+
{{ fNumber(selectedCount) }} {{ selectedLabel }}
|
21
|
+
</div>
|
22
|
+
<div v-else-if="itemCount">
|
23
|
+
{{ fNumber(itemCount) }} {{ label }}
|
24
|
+
</div>
|
25
|
+
<q-spinner
|
26
|
+
v-if="loading"
|
27
|
+
class="ml-3"
|
28
|
+
size="18"
|
29
|
+
/>
|
30
|
+
</div>
|
31
|
+
</q-td>
|
32
|
+
<q-td
|
33
|
+
v-for="column in summaryColumns"
|
34
|
+
:key="column.name"
|
35
|
+
:align="column.align || 'left'"
|
36
|
+
>
|
37
|
+
<template v-if="summary">
|
38
|
+
{{ formatValue(column) }}
|
39
|
+
</template>
|
40
|
+
</q-td>
|
41
|
+
</q-tr>
|
42
|
+
</template>
|
43
|
+
<script setup>
|
44
|
+
import { fNumber } from '@ui/helpers/formats';
|
45
|
+
import { XCircleIcon as ClearIcon } from '@heroicons/vue/solid';
|
46
|
+
import { computed } from 'vue';
|
47
|
+
|
48
|
+
defineEmits(['clear']);
|
49
|
+
const props = defineProps({
|
50
|
+
loading: Boolean,
|
51
|
+
label: {
|
52
|
+
type: String,
|
53
|
+
default: 'Rows'
|
54
|
+
},
|
55
|
+
selectedLabel: {
|
56
|
+
type: String,
|
57
|
+
default: 'Selected'
|
58
|
+
},
|
59
|
+
selectedCount: {
|
60
|
+
type: Number,
|
61
|
+
default: 0
|
62
|
+
},
|
63
|
+
itemCount: {
|
64
|
+
type: Number,
|
65
|
+
default: 0
|
66
|
+
},
|
67
|
+
summary: {
|
68
|
+
type: Object,
|
69
|
+
default: null
|
70
|
+
},
|
71
|
+
columns: {
|
72
|
+
type: Array,
|
73
|
+
required: true
|
74
|
+
},
|
75
|
+
stickyColspan: {
|
76
|
+
type: Number,
|
77
|
+
default: 2
|
78
|
+
}
|
79
|
+
});
|
80
|
+
|
81
|
+
const summaryColumns = computed(() => {
|
82
|
+
// The sticky columns are where we display the selection count and should not be included in the summary columns
|
83
|
+
return props.columns.slice(props.stickyColspan - 1);
|
84
|
+
});
|
85
|
+
|
86
|
+
function formatValue(column) {
|
87
|
+
const value = props.summary[column.name];
|
88
|
+
if (value === undefined) return '';
|
89
|
+
|
90
|
+
if (column.format) {
|
91
|
+
return column.format(value);
|
92
|
+
}
|
93
|
+
return value;
|
94
|
+
}
|
95
|
+
</script>
|
@@ -0,0 +1,10 @@
|
|
1
|
+
export * from "./Filters";
|
2
|
+
export * from "./Form";
|
3
|
+
export * from "./listActions";
|
4
|
+
export * from "./listHelpers";
|
5
|
+
export * from "./tableColumns";
|
6
|
+
export { default as ActionTable } from "./ActionTable.vue";
|
7
|
+
export { default as BatchActionMenu } from "./BatchActionMenu.vue";
|
8
|
+
export { default as EmptyTableState } from "./EmptyTableState.vue";
|
9
|
+
export { default as RenderComponentColumn } from "./RenderComponentColumn.vue";
|
10
|
+
export { default as TableSummaryRow } from "./TableSummaryRow.vue";
|
@@ -0,0 +1,362 @@
|
|
1
|
+
import { getItem, setItem } from "@ui/helpers";
|
2
|
+
import { computed, ref, watch } from "vue";
|
3
|
+
import { getFilterFromUrl, mapSortBy, waitForRef } from "./listHelpers";
|
4
|
+
|
5
|
+
export function useListActions(name, {
|
6
|
+
listRoute,
|
7
|
+
filterFieldOptionsRoute,
|
8
|
+
summaryRoute = null,
|
9
|
+
moreRoute = null,
|
10
|
+
applyActionRoute = null,
|
11
|
+
applyBatchActionRoute = null,
|
12
|
+
itemDetailsRoute = null,
|
13
|
+
columns = null,
|
14
|
+
filterGroups = null,
|
15
|
+
refreshFilters = false,
|
16
|
+
urlPattern = null,
|
17
|
+
filterDefaults = {}
|
18
|
+
}) {
|
19
|
+
const PAGE_SETTINGS_KEY = `${name}-pagination-settings`;
|
20
|
+
const pagedItems = ref(null);
|
21
|
+
const filter = ref({});
|
22
|
+
const showFilters = ref(getItem(`${name}-show-filters`, true));
|
23
|
+
const selectedRows = ref([]);
|
24
|
+
const isLoadingList = ref(false);
|
25
|
+
const isLoadingSummary = ref(false);
|
26
|
+
const summary = ref(null);
|
27
|
+
|
28
|
+
const filterActiveCount = computed(() => Object.keys(filter.value).filter(key => filter.value[key] !== undefined).length);
|
29
|
+
|
30
|
+
const PAGING_DEFAULT = {
|
31
|
+
sortBy: null,
|
32
|
+
descending: false,
|
33
|
+
page: 1,
|
34
|
+
rowsNumber: 0,
|
35
|
+
rowsPerPage: 50
|
36
|
+
};
|
37
|
+
const quasarPagination = ref(PAGING_DEFAULT);
|
38
|
+
|
39
|
+
const pager = computed(() => ({
|
40
|
+
perPage: quasarPagination.value.rowsPerPage,
|
41
|
+
page: quasarPagination.value.page,
|
42
|
+
filter: filter.value,
|
43
|
+
sort: columns ? mapSortBy(quasarPagination.value, columns) : undefined
|
44
|
+
}));
|
45
|
+
|
46
|
+
// When any part of the filter changes, get the new list of creatives
|
47
|
+
watch(pager, () => {
|
48
|
+
saveSettings();
|
49
|
+
loadList();
|
50
|
+
});
|
51
|
+
watch(filter, () => {
|
52
|
+
saveSettings();
|
53
|
+
loadSummary();
|
54
|
+
});
|
55
|
+
watch(selectedRows, loadSummary);
|
56
|
+
|
57
|
+
if (refreshFilters) {
|
58
|
+
watch(filter, loadFilterFieldOptions);
|
59
|
+
}
|
60
|
+
|
61
|
+
async function loadList() {
|
62
|
+
isLoadingList.value = true;
|
63
|
+
setPagedItems(await listRoute(pager.value));
|
64
|
+
isLoadingList.value = false;
|
65
|
+
}
|
66
|
+
|
67
|
+
async function loadSummary() {
|
68
|
+
if (summaryRoute) {
|
69
|
+
isLoadingSummary.value = true;
|
70
|
+
const summaryFilter = { id: null, ...filter.value };
|
71
|
+
if (selectedRows.value.length) {
|
72
|
+
summaryFilter.id = selectedRows.value.map((row) => row.id);
|
73
|
+
}
|
74
|
+
summary.value = await summaryRoute(summaryFilter);
|
75
|
+
isLoadingSummary.value = false;
|
76
|
+
}
|
77
|
+
}
|
78
|
+
|
79
|
+
// Filter fields are the field values available for the currently applied filter on Creative Groups
|
80
|
+
// (ie: all states available under the current filter)
|
81
|
+
const filterFieldOptions = ref({});
|
82
|
+
const isLoadingFilters = ref(false);
|
83
|
+
|
84
|
+
watch(() => showFilters.value, (show) => {
|
85
|
+
setItem(`${name}-show-filters`, show);
|
86
|
+
});
|
87
|
+
|
88
|
+
async function loadFilterFieldOptions() {
|
89
|
+
isLoadingFilters.value = true;
|
90
|
+
filterFieldOptions.value = await filterFieldOptionsRoute(filter.value);
|
91
|
+
isLoadingFilters.value = false;
|
92
|
+
}
|
93
|
+
|
94
|
+
// A flat list of valid filterable field names
|
95
|
+
const validFilterKeys = computed(() => filterGroups?.value?.map(group => group.fields.map(field => field.name)).flat());
|
96
|
+
|
97
|
+
/**
|
98
|
+
* Watches for a filter URL parameter and applies the filter if it is set.
|
99
|
+
*/
|
100
|
+
function applyFilterFromUrl(url) {
|
101
|
+
if (url.match(urlPattern)) {
|
102
|
+
const urlFilter = getFilterFromUrl(url, validFilterKeys.value);
|
103
|
+
|
104
|
+
if (Object.keys(urlFilter).length > 0) {
|
105
|
+
filter.value = urlFilter;
|
106
|
+
}
|
107
|
+
}
|
108
|
+
}
|
109
|
+
|
110
|
+
// Set the reactive pager to map from the Laravel pagination to Quasar pagination
|
111
|
+
// and automatically update the list of ads
|
112
|
+
function setPagedItems(items) {
|
113
|
+
if (Array.isArray(items)) {
|
114
|
+
pagedItems.value = { data: items, meta: { total: items.length } };
|
115
|
+
} else {
|
116
|
+
pagedItems.value = items;
|
117
|
+
if (items?.meta && items.meta.total !== quasarPagination.value.rowsNumber) {
|
118
|
+
quasarPagination.value.rowsNumber = items.meta.total;
|
119
|
+
}
|
120
|
+
}
|
121
|
+
}
|
122
|
+
|
123
|
+
/**
|
124
|
+
* Resets the filter and pagination settings to their defaults.
|
125
|
+
*/
|
126
|
+
function resetPaging() {
|
127
|
+
quasarPagination.value = PAGING_DEFAULT;
|
128
|
+
}
|
129
|
+
|
130
|
+
/**
|
131
|
+
* Updates a row in the paged items list with the new item data. Uses the item's id to find the row.
|
132
|
+
*
|
133
|
+
* @param updatedItem
|
134
|
+
*/
|
135
|
+
function setItemInPagedList(updatedItem) {
|
136
|
+
const data = pagedItems.value?.data?.map(item => (item.id === updatedItem.id && (item.updated_at === null || item.updated_at <= updatedItem.updated_at)) ? updatedItem : item);
|
137
|
+
pagedItems.value = { ...pagedItems.value, data };
|
138
|
+
}
|
139
|
+
|
140
|
+
/**
|
141
|
+
* Loads more items into the list.
|
142
|
+
* @param index
|
143
|
+
* @param perPage
|
144
|
+
*/
|
145
|
+
async function loadMore(index, perPage = undefined) {
|
146
|
+
const newItems = await moreRoute({
|
147
|
+
page: index + 1,
|
148
|
+
perPage,
|
149
|
+
filter: filter.value
|
150
|
+
});
|
151
|
+
|
152
|
+
if (newItems && newItems.length > 0) {
|
153
|
+
pagedItems.value.data = [...pagedItems.value.data, ...newItems];
|
154
|
+
return true;
|
155
|
+
}
|
156
|
+
|
157
|
+
return false;
|
158
|
+
}
|
159
|
+
|
160
|
+
/**
|
161
|
+
* Refreshes the list, summary, and filter field options.
|
162
|
+
* @returns {Promise<Awaited<void>[]>}
|
163
|
+
*/
|
164
|
+
async function refreshAll() {
|
165
|
+
return Promise.all([loadList(), loadSummary(), loadFilterFieldOptions()]);
|
166
|
+
}
|
167
|
+
|
168
|
+
/**
|
169
|
+
* Loads the filter and pagination settings from local storage.
|
170
|
+
*/
|
171
|
+
function loadSettings() {
|
172
|
+
const settings = getItem(PAGE_SETTINGS_KEY);
|
173
|
+
|
174
|
+
// Load the filter settings from local storage
|
175
|
+
if (settings) {
|
176
|
+
filter.value = { ...settings.filter, ...filter.value };
|
177
|
+
quasarPagination.value = settings.quasarPagination;
|
178
|
+
} else {
|
179
|
+
// If no local storage settings, apply the default filters
|
180
|
+
filter.value = { ...filterDefaults, ...filter.value };
|
181
|
+
}
|
182
|
+
|
183
|
+
// Load the URL filters if they are set
|
184
|
+
applyFilterFromUrl(window.location.href);
|
185
|
+
|
186
|
+
setTimeout(() => {
|
187
|
+
if (!isLoadingList.value) {
|
188
|
+
loadList();
|
189
|
+
}
|
190
|
+
|
191
|
+
if (!isLoadingSummary.value) {
|
192
|
+
loadSummary();
|
193
|
+
}
|
194
|
+
|
195
|
+
if (!isLoadingFilters.value) {
|
196
|
+
loadFilterFieldOptions();
|
197
|
+
}
|
198
|
+
}, 1);
|
199
|
+
}
|
200
|
+
|
201
|
+
/**
|
202
|
+
* Saves the current filter and pagination settings to local storage.
|
203
|
+
*/
|
204
|
+
async function saveSettings() {
|
205
|
+
const settings = {
|
206
|
+
filter: filter.value,
|
207
|
+
quasarPagination: { ...quasarPagination.value, page: 1 }
|
208
|
+
};
|
209
|
+
// save in local storage
|
210
|
+
setItem(PAGE_SETTINGS_KEY, settings);
|
211
|
+
}
|
212
|
+
|
213
|
+
/**
|
214
|
+
* Applies an action to an item.
|
215
|
+
*/
|
216
|
+
const isApplyingActionToItem = ref(null);
|
217
|
+
let actionResultCount = 0;
|
218
|
+
|
219
|
+
async function applyAction(item, input, itemData = {}) {
|
220
|
+
isApplyingActionToItem.value = item;
|
221
|
+
const resultNumber = ++actionResultCount;
|
222
|
+
setItemInPagedList({ ...item, ...input, ...itemData });
|
223
|
+
const result = await applyActionRoute(item, input);
|
224
|
+
if (result.success) {
|
225
|
+
// Only render the most recent campaign changes
|
226
|
+
if (resultNumber !== actionResultCount) return;
|
227
|
+
|
228
|
+
// Update the updated item in the previously loaded list if it exists
|
229
|
+
setItemInPagedList(result.item);
|
230
|
+
|
231
|
+
// Update the active item if it is the same as the updated item
|
232
|
+
if (activeItem.value?.id === result.item.id) {
|
233
|
+
activeItem.value = { ...activeItem.value, ...result.item };
|
234
|
+
}
|
235
|
+
}
|
236
|
+
isApplyingActionToItem.value = null;
|
237
|
+
return result;
|
238
|
+
}
|
239
|
+
|
240
|
+
/**
|
241
|
+
* Applies an action to all selected items.
|
242
|
+
*/
|
243
|
+
const isApplyingBatchAction = ref(false);
|
244
|
+
|
245
|
+
async function applyBatchAction(input) {
|
246
|
+
isApplyingBatchAction.value = true;
|
247
|
+
const batchFilter = { id: selectedRows.value.map(r => r.id) };
|
248
|
+
const result = await applyBatchActionRoute(batchFilter, input);
|
249
|
+
isApplyingBatchAction.value = false;
|
250
|
+
await refreshAll();
|
251
|
+
|
252
|
+
return result;
|
253
|
+
}
|
254
|
+
|
255
|
+
// The active ad for viewing / editing in the Ad Form
|
256
|
+
const activeItem = ref(null);
|
257
|
+
// Controls the tab on the Ad Form
|
258
|
+
const formTab = ref("general");
|
259
|
+
|
260
|
+
// Whenever the active item changes, fill the additional item details
|
261
|
+
// (ie: tasks, verifications, creatives, etc.)
|
262
|
+
if (itemDetailsRoute) {
|
263
|
+
watch(() => activeItem.value, async (newItem, oldItem) => {
|
264
|
+
if (newItem && oldItem?.id !== newItem.id) {
|
265
|
+
const result = await itemDetailsRoute(newItem);
|
266
|
+
|
267
|
+
// Only set the ad details if we are the response for the currently loaded item
|
268
|
+
// NOTE: race conditions might allow the finished loading item to be different to the currently
|
269
|
+
// requested item
|
270
|
+
if (result?.id === activeItem.value?.id) {
|
271
|
+
activeItem.value = result;
|
272
|
+
}
|
273
|
+
}
|
274
|
+
});
|
275
|
+
}
|
276
|
+
|
277
|
+
/**
|
278
|
+
* Opens the item's form with the given item and tab
|
279
|
+
*
|
280
|
+
* @param item
|
281
|
+
* @param tab
|
282
|
+
*/
|
283
|
+
function openItemForm(item, tab) {
|
284
|
+
activeItem.value = item;
|
285
|
+
formTab.value = tab;
|
286
|
+
}
|
287
|
+
|
288
|
+
/**
|
289
|
+
* Gets the next item in the list at the given offset (ie: 1 or -1) from the current position in the list of the
|
290
|
+
* selected item. If the next item is on a previous or next page, it will load the page first then select the item
|
291
|
+
* @param offset
|
292
|
+
* @returns {Promise<void>}
|
293
|
+
*/
|
294
|
+
async function getNextItem(offset) {
|
295
|
+
const index = pagedItems.value?.data.findIndex(i => i.id === activeItem.value.id);
|
296
|
+
if (index === undefined) return;
|
297
|
+
let nextIndex = index + offset;
|
298
|
+
|
299
|
+
// Load the previous page if the offset is before index 0
|
300
|
+
if (nextIndex < 0) {
|
301
|
+
if (quasarPagination.value.page > 1) {
|
302
|
+
quasarPagination.value = { ...quasarPagination.value, page: quasarPagination.value.page - 1 };
|
303
|
+
await waitForRef(isLoadingList, false);
|
304
|
+
nextIndex = pagedItems.value.data.length - 1;
|
305
|
+
} else {
|
306
|
+
// There are no more previous pages
|
307
|
+
return;
|
308
|
+
}
|
309
|
+
}
|
310
|
+
|
311
|
+
// Load the next page if the offset is past the last index
|
312
|
+
if (nextIndex >= pagedItems.value.data.length) {
|
313
|
+
if (quasarPagination.value.page < pagedItems.value.meta.last_page) {
|
314
|
+
quasarPagination.value = { ...quasarPagination.value, page: quasarPagination.value.page + 1 };
|
315
|
+
await waitForRef(isLoadingList, false);
|
316
|
+
nextIndex = 0;
|
317
|
+
} else {
|
318
|
+
// There are no more next pages
|
319
|
+
return;
|
320
|
+
}
|
321
|
+
}
|
322
|
+
|
323
|
+
activeItem.value = pagedItems.value.data[nextIndex];
|
324
|
+
}
|
325
|
+
|
326
|
+
// Async load the settings for this Action List
|
327
|
+
setTimeout(loadSettings, 1);
|
328
|
+
|
329
|
+
return {
|
330
|
+
// State
|
331
|
+
pagedItems,
|
332
|
+
filter,
|
333
|
+
filterActiveCount,
|
334
|
+
showFilters,
|
335
|
+
summary,
|
336
|
+
filterFieldOptions,
|
337
|
+
selectedRows,
|
338
|
+
isLoadingList,
|
339
|
+
isLoadingFilters,
|
340
|
+
isLoadingSummary,
|
341
|
+
pager,
|
342
|
+
quasarPagination,
|
343
|
+
isApplyingActionToItem,
|
344
|
+
isApplyingBatchAction,
|
345
|
+
activeItem,
|
346
|
+
formTab,
|
347
|
+
columns,
|
348
|
+
filterGroups,
|
349
|
+
|
350
|
+
// Actions
|
351
|
+
loadSummary,
|
352
|
+
resetPaging,
|
353
|
+
loadList,
|
354
|
+
loadMore,
|
355
|
+
refreshAll,
|
356
|
+
applyAction,
|
357
|
+
applyBatchAction,
|
358
|
+
getNextItem,
|
359
|
+
openItemForm,
|
360
|
+
applyFilterFromUrl
|
361
|
+
};
|
362
|
+
}
|
@@ -0,0 +1,74 @@
|
|
1
|
+
import { getUrlParam } from "@ui/helpers";
|
2
|
+
import { onMounted, watch } from "vue";
|
3
|
+
|
4
|
+
export function registerStickyScrolling(tableRef) {
|
5
|
+
onMounted(() => {
|
6
|
+
const scrollEl = tableRef.value.$el.getElementsByClassName("q-table__middle")[0];
|
7
|
+
scrollEl.addEventListener("scroll", onScroll);
|
8
|
+
|
9
|
+
function onScroll({ target }) {
|
10
|
+
// Add / remove scroll y class based on whether we're scrolling vertically
|
11
|
+
if (target.scrollTop > 0) {
|
12
|
+
scrollEl.classList.add("is-scrolling-y");
|
13
|
+
} else {
|
14
|
+
scrollEl.classList.remove("is-scrolling-y");
|
15
|
+
}
|
16
|
+
|
17
|
+
// Add / remove scroll x class based on whether we're scrolling horizontally
|
18
|
+
if (target.scrollLeft > 0) {
|
19
|
+
scrollEl.classList.add("is-scrolling-x");
|
20
|
+
} else {
|
21
|
+
scrollEl.classList.remove("is-scrolling-x");
|
22
|
+
}
|
23
|
+
}
|
24
|
+
});
|
25
|
+
}
|
26
|
+
|
27
|
+
export function mapSortBy(pagination, columns) {
|
28
|
+
if (!pagination.sortBy) return null;
|
29
|
+
|
30
|
+
const column = columns.find(c => c.name === pagination.sortBy);
|
31
|
+
return [
|
32
|
+
{
|
33
|
+
column: column.sortBy || column.name,
|
34
|
+
expression: column.sortByExpression || undefined,
|
35
|
+
order: pagination.descending ? "desc" : "asc"
|
36
|
+
}
|
37
|
+
];
|
38
|
+
}
|
39
|
+
|
40
|
+
/**
|
41
|
+
* Wait for a ref to have a value and then resolve the promise
|
42
|
+
*
|
43
|
+
* @param ref
|
44
|
+
* @param value
|
45
|
+
* @returns {Promise<void>}
|
46
|
+
*/
|
47
|
+
export function waitForRef(ref, value) {
|
48
|
+
return new Promise<void>((resolve) => {
|
49
|
+
watch(ref, (newValue) => {
|
50
|
+
if (newValue === value) {
|
51
|
+
resolve();
|
52
|
+
}
|
53
|
+
});
|
54
|
+
});
|
55
|
+
}
|
56
|
+
|
57
|
+
/**
|
58
|
+
* Returns the filter from the URL if it is set
|
59
|
+
* @param url
|
60
|
+
* @param allowedKeys
|
61
|
+
*/
|
62
|
+
export function getFilterFromUrl(url, allowedKeys = null) {
|
63
|
+
const filter = {};
|
64
|
+
const urlFilter = getUrlParam("filter", url);
|
65
|
+
if (urlFilter) {
|
66
|
+
const fields = JSON.parse(urlFilter);
|
67
|
+
Object.keys(fields).forEach((key) => {
|
68
|
+
if (!allowedKeys || allowedKeys.includes(key)) {
|
69
|
+
filter[key] = fields[key];
|
70
|
+
}
|
71
|
+
});
|
72
|
+
}
|
73
|
+
return filter;
|
74
|
+
}
|
@@ -0,0 +1,72 @@
|
|
1
|
+
import { getItem, setItem } from "@ui/helpers";
|
2
|
+
import { computed, ref, watch } from "vue";
|
3
|
+
|
4
|
+
export function useTableColumns(name, columns, options = { titleMinWidth: 120, titleMaxWidth: 200 }) {
|
5
|
+
const COLUMN_ORDER_KEY = `${name}-column-order`;
|
6
|
+
const VISIBLE_COLUMNS_KEY = `${name}-visible-columns`;
|
7
|
+
const TITLE_COLUMNS_KEY = `${name}-title-columns`;
|
8
|
+
const TITLE_WIDTH_KEY = `${name}-title-width`;
|
9
|
+
|
10
|
+
// The list that defines the order the columns should appear in
|
11
|
+
const columnOrder = ref(getItem(COLUMN_ORDER_KEY) || []);
|
12
|
+
|
13
|
+
// Manages visible columns on the table
|
14
|
+
const hiddenColumnNames = ref(getItem(VISIBLE_COLUMNS_KEY, columns.filter(c => c.category !== "General" || c.name === "status").map(c => c.name)));
|
15
|
+
|
16
|
+
// Title columns will have their name appear on the first column of the table as part of the records' title
|
17
|
+
const titleColumnNames = ref(getItem(TITLE_COLUMNS_KEY, []));
|
18
|
+
|
19
|
+
// The width of the title column
|
20
|
+
const titleWidth = ref(getItem(TITLE_WIDTH_KEY, options.titleMinWidth));
|
21
|
+
|
22
|
+
/**
|
23
|
+
* When the title column is resized, update the titleWidth
|
24
|
+
* @param val
|
25
|
+
*/
|
26
|
+
function onResizeTitleColumn(val) {
|
27
|
+
titleWidth.value = Math.max(Math.min(val.distance + val.startDropZoneSize, options.titleMaxWidth), options.titleMinWidth);
|
28
|
+
}
|
29
|
+
|
30
|
+
// Columns that should be locked to the left side of the table
|
31
|
+
const lockedColumns = computed(() => orderedColumns.value.slice(0, 1));
|
32
|
+
|
33
|
+
// The resolved list of columns in the order they should appear in
|
34
|
+
const orderedColumns = computed(() => [...columns].sort((a, b) => {
|
35
|
+
const aIndex = columnOrder.value.indexOf(a.name);
|
36
|
+
const bIndex = columnOrder.value.indexOf(b.name);
|
37
|
+
return aIndex === -1 ? 1 : bIndex === -1 ? -1 : aIndex - bIndex;
|
38
|
+
}));
|
39
|
+
|
40
|
+
// The ordered list of columns. The ordering of this list is editable and will be stored in localStorage
|
41
|
+
const sortableColumns = computed({
|
42
|
+
get() {
|
43
|
+
return orderedColumns.value.slice(1);
|
44
|
+
},
|
45
|
+
set(newColumns) {
|
46
|
+
columnOrder.value = [...lockedColumns.value.map(c => c.name), ...newColumns.map(c => c.name)];
|
47
|
+
setItem(COLUMN_ORDER_KEY, columnOrder.value);
|
48
|
+
}
|
49
|
+
});
|
50
|
+
|
51
|
+
// The list of columns that are visible. To edit the visible columns, edit the hiddenColumnNames list
|
52
|
+
const visibleColumns = computed(() => orderedColumns.value.filter(c => !hiddenColumnNames.value.includes(c.name)));
|
53
|
+
|
54
|
+
// The list of columns that should be included in the title of a row
|
55
|
+
const orderedTitleColumns = computed(() => orderedColumns.value.filter(c => titleColumnNames.value.includes(c.name)));
|
56
|
+
|
57
|
+
// Save changes to the list of hidden columns in localStorage
|
58
|
+
watch(() => hiddenColumnNames.value, () => setItem(VISIBLE_COLUMNS_KEY, hiddenColumnNames.value));
|
59
|
+
watch(() => titleColumnNames.value, () => setItem(TITLE_COLUMNS_KEY, titleColumnNames.value));
|
60
|
+
watch(() => titleWidth.value, () => setItem(TITLE_WIDTH_KEY, titleWidth.value));
|
61
|
+
|
62
|
+
return {
|
63
|
+
sortableColumns,
|
64
|
+
lockedColumns,
|
65
|
+
visibleColumns,
|
66
|
+
hiddenColumnNames,
|
67
|
+
titleColumnNames,
|
68
|
+
titleWidth,
|
69
|
+
orderedTitleColumns,
|
70
|
+
onResizeTitleColumn
|
71
|
+
};
|
72
|
+
}
|