@dragonmastery/dragoncore-vue 0.0.32 → 0.0.33
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/dist/{CreateTeamForm-BQewI0Gn.js → CreateTeamForm-ClisfzsU.js} +2 -2
- package/dist/{CreateTeamForm-BQewI0Gn.js.map → CreateTeamForm-ClisfzsU.js.map} +1 -1
- package/dist/{CreateTeamForm-BK8PJeXz.js → CreateTeamForm-DWAeq_Xd.js} +2 -2
- package/dist/{CreditBalanceDashboard-CRHrhlSZ.js → CreditBalanceDashboard-CJr1yOjW.js} +2 -2
- package/dist/{CreditBalanceDashboard-CRHrhlSZ.js.map → CreditBalanceDashboard-CJr1yOjW.js.map} +1 -1
- package/dist/{CreditBalanceDashboard-C2QnVfjU.js → CreditBalanceDashboard-D370HlpF.js} +4 -4
- package/dist/{CreditManagement-CAaf_rVj.js → CreditManagement-B4aHJfOo.js} +4 -4
- package/dist/{CreditManagement-7xiygPYj.js → CreditManagement-BbDEviZx.js} +2 -2
- package/dist/{CreditManagement-7xiygPYj.js.map → CreditManagement-BbDEviZx.js.map} +1 -1
- package/dist/{CreditTransactionHistory-CP7ufh7Z.js → CreditTransactionHistory-mz-QIKUG.js} +2 -2
- package/dist/{CreditTransactionHistory-CP7ufh7Z.js.map → CreditTransactionHistory-mz-QIKUG.js.map} +1 -1
- package/dist/{CustomerCreateSupportTicketForm-Cd3ijzWy.js → CustomerCreateSupportTicketForm-Cul4aVjB.js} +3 -3
- package/dist/{CustomerCreateSupportTicketForm-Cd3ijzWy.js.map → CustomerCreateSupportTicketForm-Cul4aVjB.js.map} +1 -1
- package/dist/{CustomerCreateSupportTicketForm-C3DSzhJR.js → CustomerCreateSupportTicketForm-DCPjvzyF.js} +4 -4
- package/dist/{CustomerSupportTicketDetailPage-DwpkxpYV.js → CustomerSupportTicketDetailPage-DflGaKvk.js} +3 -3
- package/dist/{CustomerSupportTicketDetailPage-DwpkxpYV.js.map → CustomerSupportTicketDetailPage-DflGaKvk.js.map} +1 -1
- package/dist/{CustomerSupportTicketList-BrTX8SUV.js → CustomerSupportTicketList-BPWQ8WjH.js} +18 -18
- package/dist/{CustomerSupportTicketSuccess-k3DX9NhV.js → CustomerSupportTicketSuccess-B4WR8Zfn.js} +2 -2
- package/dist/{CustomerSupportTicketSuccess-k3DX9NhV.js.map → CustomerSupportTicketSuccess-B4WR8Zfn.js.map} +1 -1
- package/dist/{CustomerSupportTicketSuccess-BLSMAjQN.js → CustomerSupportTicketSuccess-Bsp1d0oF.js} +3 -3
- package/dist/{EditTeamForm-CETmPEZ7.js → EditTeamForm-B4p-C-8i.js} +2 -2
- package/dist/{EditTeamForm-CETmPEZ7.js.map → EditTeamForm-B4p-C-8i.js.map} +1 -1
- package/dist/{EditTeamForm-DiZXp6vh.js → EditTeamForm-ucwkw2eL.js} +2 -2
- package/dist/{InlineAttachments-Ywf9r3jd.js → InlineAttachments-DvqCOd6U.js} +19 -9
- package/dist/InlineAttachments-DvqCOd6U.js.map +1 -0
- package/dist/{SavedFiltersPage-CTLpYU2c.js → SavedFiltersPage-Y1T18UAH.js} +19 -19
- package/dist/{SavedFiltersPage-CTLpYU2c.js.map → SavedFiltersPage-Y1T18UAH.js.map} +1 -1
- package/dist/{StaffCreateSupportTicketForm-gPZizMKW.js → StaffCreateSupportTicketForm-B1YioWlS.js} +3 -3
- package/dist/{StaffCreateSupportTicketForm-gPZizMKW.js.map → StaffCreateSupportTicketForm-B1YioWlS.js.map} +1 -1
- package/dist/{StaffCreateSupportTicketForm-BnpxlQaG.js → StaffCreateSupportTicketForm-UkE9YVWE.js} +4 -4
- package/dist/{StaffSupportTicketDetailPage-DUqLTHVT.js → StaffSupportTicketDetailPage-CMxsVxjf.js} +3 -3
- package/dist/{StaffSupportTicketDetailPage-DUqLTHVT.js.map → StaffSupportTicketDetailPage-CMxsVxjf.js.map} +1 -1
- package/dist/{StaffSupportTicketList-Qp0vQw8X.js → StaffSupportTicketList-BlGTtE1R.js} +18 -18
- package/dist/{StaffSupportTicketSuccess-BvwrpoAb.js → StaffSupportTicketSuccess-CzS9uS2h.js} +3 -3
- package/dist/{StaffSupportTicketSuccess-C9vqsC51.js → StaffSupportTicketSuccess-DTMToAq2.js} +2 -2
- package/dist/{StaffSupportTicketSuccess-C9vqsC51.js.map → StaffSupportTicketSuccess-DTMToAq2.js.map} +1 -1
- package/dist/{TeamAttachmentsTab-CnAvlIbn.js → TeamAttachmentsTab-BgtXRohH.js} +18 -18
- package/dist/{TeamList-Fb6EfybL.js → TeamList-D3NqS5DL.js} +2 -2
- package/dist/{TeamList-C0Mbkbja.js → TeamList-P7rVlBuQ.js} +2 -2
- package/dist/{TeamList-C0Mbkbja.js.map → TeamList-P7rVlBuQ.js.map} +1 -1
- package/dist/{TeamParent-XgBD3twV.js → TeamParent-CuoVw7Yk.js} +2 -2
- package/dist/{TeamParent-COSOJ-ZZ.js → TeamParent-D35VUiio.js} +2 -2
- package/dist/{TeamParent-COSOJ-ZZ.js.map → TeamParent-D35VUiio.js.map} +1 -1
- package/dist/{TimelineNoteInput-CrivEMUP.js → TimelineNoteInput-D6AbJMx5.js} +2 -2
- package/dist/{TimelineNoteInput-CrivEMUP.js.map → TimelineNoteInput-D6AbJMx5.js.map} +1 -1
- package/dist/{ViewTeam-BXSq4M0V.js → ViewTeam-7YvvwcXd.js} +2 -2
- package/dist/{ViewTeam-DGaNNd8N.js → ViewTeam-BbsTsjAy.js} +2 -2
- package/dist/{ViewTeam-DGaNNd8N.js.map → ViewTeam-BbsTsjAy.js.map} +1 -1
- package/dist/{customerSupportTicketRoutes-BjqjQc9A.js → customerSupportTicketRoutes-38JuLJGP.js} +6 -6
- package/dist/{customerSupportTicketRoutes-BjqjQc9A.js.map → customerSupportTicketRoutes-38JuLJGP.js.map} +1 -1
- package/dist/index.d.ts +765 -765
- package/dist/index.js +18 -18
- package/dist/{saved_filter-DXWnU4Zb.js → saved_filter-YIncsdws.js} +2 -2
- package/dist/{saved_filter-DXWnU4Zb.js.map → saved_filter-YIncsdws.js.map} +1 -1
- package/dist/{src-B7oHTH_y.js → src-BXO0PrFd.js} +276 -145
- package/dist/src-BXO0PrFd.js.map +1 -0
- package/dist/{staffSupportTicketRoutes-CeuduGmG.js → staffSupportTicketRoutes-BZrj4aMG.js} +6 -6
- package/dist/{staffSupportTicketRoutes-CeuduGmG.js.map → staffSupportTicketRoutes-BZrj4aMG.js.map} +1 -1
- package/dist/{teamRoutes-A7plQKi0.js → teamRoutes-Aa9aBVCa.js} +7 -7
- package/dist/{teamRoutes-A7plQKi0.js.map → teamRoutes-Aa9aBVCa.js.map} +1 -1
- package/package.json +2 -2
- package/dist/InlineAttachments-Ywf9r3jd.js.map +0 -1
- package/dist/src-B7oHTH_y.js.map +0 -1
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"CustomerSupportTicketDetailPage-DwpkxpYV.js","names":["$emit","items: TimelineItem[]"],"sources":["../src/slices/support_ticket/customer/components/CustomerActionBanner.vue","../src/slices/support_ticket/customer/components/CustomerMetadataCard.vue","../src/slices/support_ticket/customer/components/CustomerSupportTicketSubscribeControl.vue","../src/slices/support_ticket/customer/customerMetadataCardEditFormMetadata.ts","../src/slices/support_ticket/customer/components/CustomerMetadataCardEdit.vue","../src/slices/support_ticket/customer/components/CustomerTimeline.vue","../src/slices/support_ticket/customer/CustomerSupportTicketDetailPage.vue"],"sourcesContent":["<template>\n <ActionBannerAlert v-if=\"isArchived\" variant=\"neutral\" icon=\"archive\">\n This ticket is archived. Comments, edits, and attachments are locked for both customer and staff.\n </ActionBannerAlert>\n <ActionBannerAlert v-else-if=\"isApproved\" variant=\"success\" icon=\"lock\">\n This ticket was approved on {{ lockedDateFormatted }}.\n </ActionBannerAlert>\n <ActionBannerAlert v-else-if=\"isRejected\" variant=\"error\" icon=\"lock\">\n This ticket was rejected on {{ lockedDateFormatted }}.\n </ActionBannerAlert>\n</template>\n\n<script setup lang=\"ts\">\nimport type { CustomerSupportTicketReadDto } from '@dragonmastery/dragoncore-shared';\nimport { computed } from 'vue';\nimport { formatTicketDate } from '../../utils/formatTicketDate';\nimport ActionBannerAlert from '../../shared/ActionBannerAlert.vue';\n\ninterface Props {\n ticket: CustomerSupportTicketReadDto;\n}\n\nconst props = defineProps<Props>();\n\nconst isArchived = computed(() => !!props.ticket.archived_at);\n\nconst isApproved = computed(() => {\n const t = props.ticket;\n return !!t.is_locked && !!t.locked_approval_at && t.status !== 'CANCELLED';\n});\n\nconst isRejected = computed(() => {\n const t = props.ticket;\n return !!t.is_locked && !!t.locked_approval_at && t.status === 'CANCELLED';\n});\n\nconst lockedDateFormatted = computed(() => {\n const at = props.ticket.locked_approval_at;\n if (!at) return '';\n return formatTicketDate(at).formatted;\n});\n</script>\n","<template>\n <div class=\"border border-base-200 rounded-lg p-4 md:p-6\">\n <div class=\"grid grid-cols-1 md:grid-cols-2 gap-x-8 gap-y-3\">\n <MetadataField label=\"Status\">\n <SupportTicketApprovalBadge\n :approval-status=\"displayApprovalStatus\"\n size=\"sm\"\n />\n </MetadataField>\n <MetadataField label=\"Type\">\n <SupportTicketTypeBadge :type=\"ticket.type\" size=\"sm\" />\n </MetadataField>\n <MetadataField label=\"Priority\">\n <SupportTicketPriorityBadge :priority=\"ticket.priority\" size=\"sm\" />\n </MetadataField>\n <MetadataField label=\"Requester\">\n {{ ticket.created_by_display_name }}\n </MetadataField>\n <div class=\"md:col-span-2\">\n <MetadataField label=\"Description\">\n <p v-if=\"ticket.description?.trim()\" class=\"whitespace-pre-wrap text-sm\">\n {{ ticket.description }}\n </p>\n <template #empty>\n <span class=\"text-base-content/50 italic text-sm\">No description provided</span>\n </template>\n </MetadataField>\n </div>\n <MetadataField label=\"Ticket ID\" :copyable=\"true\">\n {{ ticketDisplayId }}\n </MetadataField>\n <MetadataField label=\"Created\">\n {{ createdFormatted }}\n <span v-if=\"createdRelative\" class=\"text-base-content/50\">· {{ createdRelative }}</span>\n </MetadataField>\n <template v-if=\"showCredits\">\n <MetadataField label=\"Credits\">\n {{ creditDisplay }}\n </MetadataField>\n </template>\n <template v-if=\"ticket.target_at\">\n <MetadataField label=\"Target\">\n {{ targetFormatted }}\n <span class=\"text-base-content/50\">· {{ targetRelative }}</span>\n </MetadataField>\n </template>\n <template v-if=\"ticket.completed_at\">\n <MetadataField label=\"Completed\">\n {{ completedFormatted }}\n <span class=\"text-base-content/50\">· {{ completedRelative }}</span>\n </MetadataField>\n </template>\n </div>\n <div v-if=\"canEdit\" class=\"flex justify-end gap-2 mt-4\">\n <button type=\"button\" class=\"btn btn-sm\" @click=\"$emit('edit')\">\n <svg\n xmlns=\"http://www.w3.org/2000/svg\"\n fill=\"none\"\n viewBox=\"0 0 24 24\"\n stroke-width=\"1.5\"\n stroke=\"currentColor\"\n class=\"w-4 h-4 mr-2\"\n >\n <path\n stroke-linecap=\"round\"\n stroke-linejoin=\"round\"\n d=\"M16.862 4.487l1.687-1.688a1.875 1.875 0 112.652 2.652L10.582 16.07a4.5 4.5 0 01-1.897 1.13L6 18l.8-2.685a4.5 4.5 0 011.13-1.897l8.932-8.931zm0 0L19.5 7.125M18 14v4.75A2.25 2.25 0 0115.75 21H5.25A2.25 2.25 0 013 18.75V8.25A2.25 2.25 0 015.25 6H10\"\n />\n </svg>\n Edit\n </button>\n </div>\n </div>\n</template>\n\n<script setup lang=\"ts\">\nimport { computed } from 'vue';\nimport type { CustomerSupportTicketReadDto } from '@dragonmastery/dragoncore-shared';\nimport type { SupportTicketApproval } from '@dragonmastery/dragoncore-shared';\nimport MetadataField from '../../shared/MetadataField.vue';\nimport SupportTicketApprovalBadge from '../../shared/SupportTicketApprovalBadge.vue';\nimport SupportTicketTypeBadge from '../../shared/SupportTicketTypeBadge.vue';\nimport SupportTicketPriorityBadge from '../../shared/SupportTicketPriorityBadge.vue';\nimport { formatTicketDate } from '../../utils/formatTicketDate';\nimport { formatTicketDisplayId } from '../../utils/displayIdFormatter';\nimport { formatCustomerCreditValue } from '../../utils/creditValueFormatter';\n\ninterface Props {\n ticket: CustomerSupportTicketReadDto;\n}\n\ndefineEmits<{ edit: [] }>();\n\nconst props = defineProps<Props>();\n\nconst canEdit = computed(() => !props.ticket.is_locked && !props.ticket.archived_at);\n\nconst displayApprovalStatus = computed((): SupportTicketApproval => {\n const s = props.ticket.status;\n if (s === 'PENDING') return 'PENDING';\n if (s === 'CANCELLED') return 'REJECTED';\n return 'APPROVED';\n});\n\nconst ticketDisplayId = computed(() =>\n formatTicketDisplayId(\n props.ticket.display_id,\n props.ticket.display_id_prefix,\n props.ticket.id,\n ),\n);\n\nconst createdFormatted = computed(() => {\n const at = props.ticket.created_at;\n return at ? formatTicketDate(at).formatted : '';\n});\nconst createdRelative = computed(() => {\n const at = props.ticket.created_at;\n return at ? formatTicketDate(at).relative : '';\n});\n\nconst showCredits = computed(() => {\n return !!props.ticket.is_locked && !!props.ticket.locked_approval_at;\n});\n\nconst creditDisplay = computed(() => formatCustomerCreditValue(props.ticket.credit_value));\n\nconst targetFormatted = computed(() =>\n props.ticket.target_at ? formatTicketDate(props.ticket.target_at!).formatted : '',\n);\nconst targetRelative = computed(() =>\n props.ticket.target_at ? formatTicketDate(props.ticket.target_at!).relative : '',\n);\n\nconst completedFormatted = computed(() =>\n props.ticket.completed_at ? formatTicketDate(props.ticket.completed_at!).formatted : '',\n);\nconst completedRelative = computed(() =>\n props.ticket.completed_at ? formatTicketDate(props.ticket.completed_at!).relative : '',\n);\n</script>\n","<template>\n <div\n v-if=\"ticket\"\n class=\"flex flex-wrap items-center justify-between gap-3 rounded-lg border border-base-200 bg-base-200/30 px-4 py-3\"\n >\n <div class=\"flex items-center gap-3 min-w-0\">\n <div\n class=\"flex h-9 w-9 shrink-0 items-center justify-center rounded-lg\"\n :class=\"\n isSubscribed ? 'bg-info/20 text-info' : 'bg-base-300 text-base-content/60'\n \"\n >\n <!-- Bell (subscribed) -->\n <svg\n v-if=\"isSubscribed\"\n xmlns=\"http://www.w3.org/2000/svg\"\n class=\"h-5 w-5\"\n fill=\"none\"\n viewBox=\"0 0 24 24\"\n stroke=\"currentColor\"\n stroke-width=\"2\"\n >\n <path\n stroke-linecap=\"round\"\n stroke-linejoin=\"round\"\n d=\"M15 17h5l-1.405-1.405A2.032 2.032 0 0118 14.158V11a6.002 6.002 0 00-4-5.659V5a2 2 0 10-4 0v.341C7.67 6.165 6 8.388 6 11v3.159c0 .538-.214 1.055-.595 1.436L4 17h5m6 0v1a3 3 0 11-6 0v-1m6 0H9\"\n />\n </svg>\n <!-- Bell with slash (unsubscribed) -->\n <svg\n v-else\n xmlns=\"http://www.w3.org/2000/svg\"\n class=\"h-5 w-5\"\n fill=\"none\"\n viewBox=\"0 0 24 24\"\n stroke=\"currentColor\"\n stroke-width=\"2\"\n >\n <path\n stroke-linecap=\"round\"\n stroke-linejoin=\"round\"\n d=\"M9.143 17.082a24.248 24.248 0 0 0 3.844.148m-3.844-.148a23.856 23.856 0 0 1-5.455-1.31 8.964 8.964 0 0 0 2.3-5.542m3.155 6.852a3 3 0 0 0 5.667 1.97m1.965-2.277L21 21m-4.225-4.225a23.81 23.81 0 0 0 3.536-1.003A8.967 8.967 0 0 1 18 9.75V9A6 6 0 0 0 6.53 6.53m10.245 10.245L6.53 6.53M3 3l3.53 3.53\"\n />\n </svg>\n </div>\n <p class=\"text-sm text-base-content/80 min-w-0\">\n {{\n isSubscribed\n ? 'You will receive notifications for updates.'\n : 'Get notified when there are updates or new comments.'\n }}\n </p>\n <button\n type=\"button\"\n :class=\"[\n 'btn btn-sm shrink-0 font-medium',\n isSubscribed ? 'btn-outline' : 'btn-info',\n ]\"\n :disabled=\"isToggling\"\n @click=\"handleToggle\"\n >\n <span v-if=\"isToggling\" class=\"loading loading-spinner loading-xs\"></span>\n <span v-else>{{ isSubscribed ? 'Unsubscribe' : 'Subscribe' }}</span>\n </button>\n </div>\n </div>\n</template>\n\n<script setup lang=\"ts\">\nimport type {\n CustomerSupportTicketReadDto,\n DragoncoreApi,\n} from '@dragonmastery/dragoncore-shared';\nimport { computed, ref, watch } from 'vue';\nimport { toast } from 'vue3-toastify';\nimport { useMutation } from '../../../../composables/useMutation';\nimport { extractRpcErrorMessage } from '../../../../utils/extractRpcErrorMessage';\n\ninterface Props {\n ticket: CustomerSupportTicketReadDto;\n}\n\nconst props = defineProps<Props>();\n\nconst isToggling = ref(false);\n// Local override after toggle - avoids full-page refetch\nconst subscriptionOverride = ref<boolean | null>(null);\n\nwatch(\n () => props.ticket.id,\n () => {\n subscriptionOverride.value = null;\n },\n);\n\nconst isSubscribed = computed(\n () => subscriptionOverride.value ?? props.ticket.my_subscription != null,\n);\n\nconst { mutate: toggleSubscription } = useMutation<\n DragoncoreApi,\n string,\n { subscribed: boolean }\n>((api, id) => api.supportTickets.toggleSubscription(id), {\n invalidate: /^support-tickets?:/,\n});\n\nasync function handleToggle() {\n try {\n isToggling.value = true;\n const result = await toggleSubscription(props.ticket.id);\n subscriptionOverride.value = result.subscribed;\n toast.success(\n result.subscribed\n ? 'You are now subscribed to this ticket'\n : 'You have unsubscribed from this ticket',\n );\n } catch (e) {\n toast.error(extractRpcErrorMessage(e, 'Failed to update subscription'));\n } finally {\n isToggling.value = false;\n }\n}\n</script>\n","import {\n CustomerSupportTicketUpdateSchema,\n SUPPORT_TICKET_PRIORITY_NUMBER_TO_LABEL,\n} from '@dragonmastery/dragoncore-shared';\nimport { withMetadata } from '@dragonmastery/zinia-forms-core';\nimport { z } from 'zod';\n\nexport type CustomerMetadataCardEditFormDto = z.infer<typeof CustomerSupportTicketUpdateSchema>;\n\nexport const customerMetadataCardEditFormMetadata = withMetadata(\n CustomerSupportTicketUpdateSchema,\n 'customerMetadataCardEditForm',\n {\n title: {\n label: 'Title',\n placeholder: 'Enter ticket title',\n },\n description: {\n label: 'Description',\n inputType: 'textarea',\n placeholder: 'Describe the ticket in detail',\n },\n type: { label: 'Type' },\n priority: {\n label: 'Priority',\n inputType: 'select',\n valueToLabel: SUPPORT_TICKET_PRIORITY_NUMBER_TO_LABEL,\n valueType: 'number',\n },\n },\n);\n","<template>\n <div class=\"border border-base-200 rounded-lg p-4 md:p-6\">\n <ZiniaForm\n @handle-submit=\"handleSubmit\"\n @success=\"handleSuccess\"\n @error=\"handleError\"\n title=\"\"\n subtitle=\"\"\n >\n <div class=\"grid grid-cols-1 md:grid-cols-2 gap-4\">\n <!-- Title - full width -->\n <div class=\"form-control md:col-span-2\">\n <zinia.TitleField placeholder=\"Enter ticket title\" />\n </div>\n\n <!-- Description - full width -->\n <div class=\"form-control md:col-span-2\">\n <zinia.DescriptionField class=\"w-full\" placeholder=\"Describe the ticket in detail\" />\n </div>\n\n <!-- Type -->\n <div class=\"form-control\">\n <zinia.TypeField />\n </div>\n\n <!-- Priority -->\n <div class=\"form-control\">\n <zinia.PriorityField />\n </div>\n </div>\n\n <div v-if=\"form.submitError\" class=\"alert alert-error py-2 mt-4\">\n <span class=\"text-sm\">{{ form.submitError }}</span>\n </div>\n <ZiniaFormErrorsSummary title=\"Please fix the following errors:\" />\n\n <!-- Action Buttons -->\n <div class=\"flex justify-end gap-2 mt-4\">\n <button type=\"button\" class=\"btn btn-ghost\" @click=\"handleCancel\" :disabled=\"saving\">\n Cancel\n </button>\n <ZiniaSubmitButton submitText=\"Save\" submittingText=\"Saving...\" />\n </div>\n </ZiniaForm>\n </div>\n</template>\n\n<script setup lang=\"ts\">\nimport {\n supportTicketPriorityToNumber,\n type CustomerSupportTicketReadDto,\n type CustomerSupportTicketUpdateDto,\n} from '@dragonmastery/dragoncore-shared';\nimport { useForm } from '@dragonmastery/zinia-forms-core';\nimport { computed } from 'vue';\nimport { useMutation } from '../../../../composables/useMutation';\nimport { extractRpcErrorMessage } from '../../../../utils/extractRpcErrorMessage';\nimport { customerMetadataCardEditFormMetadata } from '../customerMetadataCardEditFormMetadata';\n\ninterface Props {\n ticket: CustomerSupportTicketReadDto;\n saving?: boolean;\n}\n\nconst props = defineProps<Props>();\n\nconst emit = defineEmits<{\n success: [];\n cancel: [];\n}>();\n\nconst { mutate: updateTicket, loading: isSaving } = useMutation(\n (api, input: CustomerSupportTicketUpdateDto) => api.supportTickets.updateTicket(input),\n { invalidate: /^support-tickets?:/ },\n);\n\nconst saving = computed(() => props.saving ?? isSaving.value);\n\nconst { form, zinia, ZiniaForm, ZiniaSubmitButton, ZiniaFormErrorsSummary } = useForm(\n customerMetadataCardEditFormMetadata,\n {\n storeName: `customer-metadata-card-edit-${props.ticket.id}`,\n persistToLocalStorage: false,\n renderStyle: 'daisy_ui',\n fetchData: async () => {\n const t = props.ticket;\n return {\n id: t.id,\n title: t.title || '',\n description: t.description ?? '',\n type: t.type,\n priority: supportTicketPriorityToNumber(t.priority),\n };\n },\n },\n);\n\nfunction buildPayload(formData: Record<string, unknown>): CustomerSupportTicketUpdateDto {\n return {\n id: props.ticket.id,\n title: String(formData.title ?? '').trim(),\n description: (formData.description as string) || '',\n type: formData.type as CustomerSupportTicketUpdateDto['type'],\n priority: formData.priority as CustomerSupportTicketUpdateDto['priority'],\n };\n}\n\nasync function handleSubmit(formData: Record<string, unknown>) {\n const payload = buildPayload(formData);\n await updateTicket(payload);\n return undefined;\n}\n\nfunction handleSuccess() {\n emit('success');\n}\n\nfunction handleError(error: Error | unknown) {\n const message = extractRpcErrorMessage(error, 'Failed to update ticket');\n form.setSubmitError(message);\n}\n\nfunction handleCancel() {\n emit('cancel');\n}\n</script>\n","<template>\n <div class=\"flex flex-col gap-4 w-full\">\n <TimelineNoteInput\n ref=\"noteInputRef\"\n :show-type-toggle=\"false\"\n :disabled=\"commentsLocked ?? false\"\n :on-submit=\"onAddNote\"\n />\n <p\n v-if=\"mergedTimelineItems.length === 0\"\n class=\"text-base-content/50 text-sm text-center py-8\"\n >\n No activity yet.\n </p>\n <template v-for=\"item in mergedTimelineItems\" :key=\"item.id\">\n <TimelineItem\n v-if=\"item.type === 'customer-note'\"\n :author-name=\"item.data.authorName\"\n :created-at=\"item.data.createdAt\"\n variant=\"customer\"\n >\n {{ item.data.body ?? '' }}\n </TimelineItem>\n <TimelineSystemEvent\n v-else-if=\"item.type === 'system-event'\"\n :author=\"item.data.author\"\n :message=\"item.data.message\"\n :timestamp=\"item.data.timestamp\"\n :action=\"item.data.action\"\n :type=\"item.data.type\"\n :details=\"item.data.details\"\n :old-value=\"item.data.oldValue\"\n :new-value=\"item.data.newValue\"\n :changes=\"item.data.changes\"\n />\n </template>\n </div>\n</template>\n\n<script setup lang=\"ts\">\nimport type { CustomerSupportTicketReadDto } from '@dragonmastery/dragoncore-shared';\nimport { computed, ref } from 'vue';\nimport type { SystemEvent } from '../../utils/parseRecordVersions';\nimport TimelineItem from '../../shared/TimelineItem.vue';\nimport TimelineNoteInput from '../../shared/TimelineNoteInput.vue';\nimport TimelineSystemEvent from '../../shared/TimelineSystemEvent.vue';\n\nexport interface CustomerNoteDto {\n authorName: string;\n createdAt: string;\n body: string;\n}\n\ninterface Props {\n ticket: CustomerSupportTicketReadDto;\n notes: CustomerNoteDto[];\n systemEvents: SystemEvent[];\n /** Locks comments - only when archived (customers can add comments until archived) */\n commentsLocked?: boolean;\n /** Called when user submits a note. Must return a Promise. */\n onAddNote: (payload: {\n content: string;\n noteType: 'customer' | 'internal';\n }) => Promise<void>;\n}\n\nconst props = defineProps<Props>();\n\nconst noteInputRef = ref<InstanceType<typeof TimelineNoteInput> | null>(null);\n\ntype TimelineItem =\n | { type: 'customer-note'; timestamp: string; id: string; data: CustomerNoteDto }\n | { type: 'system-event'; timestamp: string; id: string; data: SystemEvent };\n\nconst mergedTimelineItems = computed((): TimelineItem[] => {\n const items: TimelineItem[] = [];\n\n for (const note of props.notes ?? []) {\n items.push({\n type: 'customer-note',\n timestamp: note.createdAt,\n id: `note-${note.createdAt}-${note.authorName}`,\n data: note,\n });\n }\n\n for (const event of props.systemEvents ?? []) {\n items.push({\n type: 'system-event',\n timestamp: event.timestamp,\n id: event.id,\n data: event,\n });\n }\n\n items.sort((a, b) => new Date(b.timestamp).getTime() - new Date(a.timestamp).getTime());\n return items;\n});\n\nfunction resetInput() {\n noteInputRef.value?.reset();\n}\n\ndefineExpose({ resetInput });\n</script>\n","<template>\n <!-- Loading State (only on initial load; refetch keeps content visible) -->\n <div v-if=\"isLoading && !ticket\" class=\"flex justify-center items-center p-8\">\n <span class=\"loading loading-spinner loading-lg\"></span>\n </div>\n\n <!-- Error State -->\n <div v-else-if=\"error\" class=\"alert alert-error mb-4 max-w-4xl mx-auto px-4\">\n <svg\n xmlns=\"http://www.w3.org/2000/svg\"\n class=\"stroke-current shrink-0 h-6 w-6\"\n fill=\"none\"\n viewBox=\"0 0 24 24\"\n >\n <path\n stroke-linecap=\"round\"\n stroke-linejoin=\"round\"\n stroke-width=\"2\"\n d=\"M10 14l2-2m0 0l2-2m-2 2l-2-2m2 2l2 2m7-2a9 9 0 11-18 0 9 9 0 0118 0z\"\n />\n </svg>\n <div class=\"flex-1\">\n <span>{{ error.message || 'An error occurred' }}</span>\n <div class=\"mt-2\">\n <router-link :to=\"{ name: 'CustomerSupportTicketList' }\" class=\"link link-primary\">\n Go back to ticket list\n </router-link>\n </div>\n </div>\n </div>\n\n <!-- Main Content -->\n <div v-else-if=\"ticket\" class=\"px-4 py-6 max-w-4xl mx-auto\">\n <!-- Header -->\n <div class=\"mb-6\">\n <h1 class=\"text-2xl font-bold text-base-content break-words leading-tight mt-2\">\n {{ ticket.title }}\n </h1>\n </div>\n\n <!-- Action Banner -->\n <div class=\"mb-6\">\n <CustomerActionBanner :ticket=\"ticket\" />\n </div>\n\n <!-- Subscribe / Unsubscribe -->\n <div class=\"mb-6\">\n <CustomerSupportTicketSubscribeControl :ticket=\"ticket\" />\n </div>\n\n <!-- Metadata Card (Display or Edit) -->\n <div class=\"mb-8\">\n <CustomerMetadataCard v-if=\"!isEditMode\" :ticket=\"ticket\" @edit=\"enterEditMode\" />\n <CustomerMetadataCardEdit\n v-else\n :ticket=\"ticket\"\n @success=\"handleMetadataSaveSuccess\"\n @cancel=\"exitEditMode\"\n />\n </div>\n\n <!-- Attachments (separate from timeline so adding a comment doesn't cause refetch) -->\n <div class=\"mb-8\">\n <SupportTicketAttachmentsCollapsible\n :record-id=\"ticket.id\"\n :locked=\"isCommentsLocked\"\n :editable=\"!isCommentsLocked\"\n @uploaded=\"refetchTimeline\"\n @deleted=\"refetchTimeline\"\n />\n </div>\n\n <!-- Timeline: stay mounted once we have data so adding a comment doesn't unmount/remount -->\n <div>\n <CustomerTimeline\n v-if=\"timelineData != null\"\n :ticket=\"ticket\"\n :notes=\"customerNotesForTimeline\"\n :system-events=\"filteredSystemEvents\"\n :comments-locked=\"isCommentsLocked\"\n :on-add-note=\"handleAddNote\"\n />\n <div v-else-if=\"timelineError\" class=\"alert alert-error mb-4\">\n <span>{{ timelineError?.message ?? 'Failed to load timeline' }}</span>\n <button class=\"btn btn-sm btn-outline mt-2\" @click=\"refetchTimeline\">\n Retry\n </button>\n </div>\n <div v-else class=\"flex justify-center items-center py-8\">\n <span class=\"loading loading-spinner loading-lg\"></span>\n </div>\n </div>\n </div>\n</template>\n\n<script setup lang=\"ts\">\nimport type { CustomerSupportTicketReadDto, NoteCreateDto } from '@dragonmastery/dragoncore-shared';\nimport { OPERATORS, RecordConst } from '@dragonmastery/dragoncore-shared';\nimport { computed, inject, nextTick } from 'vue';\nimport { useRoute, useRouter } from 'vue-router';\nimport { toast } from 'vue3-toastify';\nimport { BATCH_MODE } from '../../../composables/useRpcAuth';\nimport { useMutation } from '../../../composables/useMutation';\nimport { useQuery } from '../../../composables/useQuery';\nimport SupportTicketAttachmentsCollapsible from '../shared/SupportTicketAttachmentsCollapsible.vue';\nimport { parseRecordVersions, type SystemEvent } from '../utils/parseRecordVersions';\nimport CustomerActionBanner from './components/CustomerActionBanner.vue';\nimport CustomerMetadataCard from './components/CustomerMetadataCard.vue';\nimport CustomerSupportTicketSubscribeControl from './components/CustomerSupportTicketSubscribeControl.vue';\nimport CustomerMetadataCardEdit from './components/CustomerMetadataCardEdit.vue';\nimport CustomerTimeline from './components/CustomerTimeline.vue';\n\ninterface Props {\n ticket: CustomerSupportTicketReadDto | null;\n isLoading?: boolean;\n error?: Error | null;\n}\n\nconst props = defineProps<Props>();\n\nconst route = useRoute();\nconst router = useRouter();\nconst support_ticket_id = route.params.id as string;\n\n// Edit mode state\nconst isEditMode = computed(() => route.query.mode === 'edit');\n\n// Get refresh function from parent\nconst refreshTicket = inject<(() => Promise<void>) | undefined>('refreshTicket');\n\n// Comments locked only when archived (customers can add comments until archived)\nconst isCommentsLocked = computed(() => {\n if (!props.ticket) return false;\n return !!props.ticket.archived_at;\n});\n\n// Fetch customer notes and record versions in a single batched request\nconst {\n data: timelineData,\n error: timelineError,\n refetch: refetchTimeline,\n} = useQuery(\n async (api) => {\n const [customerNotes, recordVersions] = await Promise.all([\n api.notes.getNotes({\n record_id: { operator: OPERATORS.EQUALS, value: support_ticket_id },\n record_type: { operator: OPERATORS.EQUALS, value: 'support_ticket' },\n is_internal: { operator: OPERATORS.EQUALS, value: false },\n first: 100,\n sortBy: 'created_at',\n sortDirection: 'asc',\n }),\n api.recordVersions.listRecordVersionsCustomer(\n support_ticket_id,\n RecordConst.SUPPORT_TICKET, // legacy; record_types in filters takes priority\n {\n record_types: [RecordConst.SUPPORT_TICKET, RecordConst.SUPPORT_TICKET_ACTIVITY],\n first: 100,\n sortBy: 'recorded_at',\n sortDirection: 'asc',\n },\n ),\n ]);\n return { customerNotes, recordVersions };\n },\n {\n enabled: !!support_ticket_id && !!props.ticket,\n batchMode: BATCH_MODE.batch,\n trackedSegment: 'customer-support-ticket-timeline',\n },\n);\n\n// Map notes for timeline\ninterface CustomerNoteForTimeline {\n authorName: string;\n createdAt: string;\n body: string;\n}\n\nconst customerNotesForTimeline = computed((): CustomerNoteForTimeline[] => {\n const items = timelineData.value?.customerNotes?.items ?? [];\n return items\n .map((n) => ({\n authorName: n.created_by_display_name ?? n.created_by ?? '(unknown)',\n createdAt: n.created_at,\n body: n.body ?? '',\n }))\n .sort((a, b) => new Date(a.createdAt).getTime() - new Date(b.createdAt).getTime());\n});\n\n// Parse and filter system events for customer visibility (merged by API from record_types)\nconst allSystemEvents = computed(() => {\n const data = timelineData.value?.recordVersions;\n const versions = data?.items ?? [];\n const userDisplayMap = data?.user_display_map;\n const displayMap = userDisplayMap ? new Map(Object.entries(userDisplayMap)) : undefined;\n return parseRecordVersions(versions, displayMap);\n});\n\n// Filter out staff-only system events\nconst filteredSystemEvents = computed((): SystemEvent[] => {\n if (!props.ticket) return [];\n\n const events = allSystemEvents.value;\n const isPending = props.ticket.status === 'PENDING' || !props.ticket.is_locked;\n\n // Staff-only fields that customers should never see\n const STAFF_ONLY_FIELDS = new Set(['dev_lifecycle', 'delivered_value', 'start_at']);\n\n return events.filter((event) => {\n // Filter by field name if available\n if (event.fieldName) {\n // Always filter out staff-only fields\n if (STAFF_ONLY_FIELDS.has(event.fieldName)) {\n return false;\n }\n\n // Filter credit_value changes when ticket is still pending\n if (isPending && event.fieldName === 'credit_value') {\n return false;\n }\n\n return true;\n }\n\n // Fallback to message-based filtering if fieldName not available\n const message = event.message.toLowerCase();\n\n // Always filter out dev_lifecycle, delivered_value, start_at\n if (\n message.includes('dev lifecycle') ||\n message.includes('moved') ||\n message.includes('delivered value') ||\n message.includes('start date')\n ) {\n return false;\n }\n\n // Filter credit_value changes when ticket is still pending\n if (isPending && (message.includes('credits') || message.includes('credit'))) {\n return false;\n }\n\n // Allow all other events (type, priority, status, target date, completed date, etc.)\n return true;\n });\n});\n\n// Note creation mutation\nconst { mutate: createNote } = useMutation(\n (api, input: NoteCreateDto) => api.notes.createNote(input),\n { invalidate: /^notes?:/ },\n);\n\n// Handle note creation\nasync function handleAddNote(payload: {\n content: string;\n noteType: 'customer' | 'internal';\n}) {\n try {\n await createNote({\n record_id: support_ticket_id,\n record_type: 'support_ticket' as const,\n body: payload.content,\n tag: null,\n is_internal: false,\n });\n await refetchTimeline();\n toast.success('Comment added');\n } catch {\n toast.error('Failed to add comment');\n throw new Error('Failed to add comment');\n }\n}\n\n// Enter edit mode\nfunction enterEditMode() {\n router.push({\n name: route.name || 'CustomerViewSupportTicket',\n params: route.params,\n query: { ...route.query, mode: 'edit' },\n });\n}\n\n// Exit edit mode\nasync function exitEditMode() {\n await router.push({\n name: route.name || 'CustomerViewSupportTicket',\n params: route.params,\n query: { ...route.query, mode: undefined },\n });\n}\n\n// Handle metadata save success (form does mutation; we refresh and exit)\nasync function handleMetadataSaveSuccess() {\n await exitEditMode();\n await nextTick();\n toast.success('Ticket updated successfully');\n if (refreshTicket) {\n await refreshTicket();\n }\n await refetchTimeline();\n}\n</script>\n"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"CustomerSupportTicketDetailPage-DflGaKvk.js","names":["$emit","items: TimelineItem[]"],"sources":["../src/slices/support_ticket/customer/components/CustomerActionBanner.vue","../src/slices/support_ticket/customer/components/CustomerMetadataCard.vue","../src/slices/support_ticket/customer/components/CustomerSupportTicketSubscribeControl.vue","../src/slices/support_ticket/customer/customerMetadataCardEditFormMetadata.ts","../src/slices/support_ticket/customer/components/CustomerMetadataCardEdit.vue","../src/slices/support_ticket/customer/components/CustomerTimeline.vue","../src/slices/support_ticket/customer/CustomerSupportTicketDetailPage.vue"],"sourcesContent":["<template>\n <ActionBannerAlert v-if=\"isArchived\" variant=\"neutral\" icon=\"archive\">\n This ticket is archived. Comments, edits, and attachments are locked for both customer and staff.\n </ActionBannerAlert>\n <ActionBannerAlert v-else-if=\"isApproved\" variant=\"success\" icon=\"lock\">\n This ticket was approved on {{ lockedDateFormatted }}.\n </ActionBannerAlert>\n <ActionBannerAlert v-else-if=\"isRejected\" variant=\"error\" icon=\"lock\">\n This ticket was rejected on {{ lockedDateFormatted }}.\n </ActionBannerAlert>\n</template>\n\n<script setup lang=\"ts\">\nimport type { CustomerSupportTicketReadDto } from '@dragonmastery/dragoncore-shared';\nimport { computed } from 'vue';\nimport { formatTicketDate } from '../../utils/formatTicketDate';\nimport ActionBannerAlert from '../../shared/ActionBannerAlert.vue';\n\ninterface Props {\n ticket: CustomerSupportTicketReadDto;\n}\n\nconst props = defineProps<Props>();\n\nconst isArchived = computed(() => !!props.ticket.archived_at);\n\nconst isApproved = computed(() => {\n const t = props.ticket;\n return !!t.is_locked && !!t.locked_approval_at && t.status !== 'CANCELLED';\n});\n\nconst isRejected = computed(() => {\n const t = props.ticket;\n return !!t.is_locked && !!t.locked_approval_at && t.status === 'CANCELLED';\n});\n\nconst lockedDateFormatted = computed(() => {\n const at = props.ticket.locked_approval_at;\n if (!at) return '';\n return formatTicketDate(at).formatted;\n});\n</script>\n","<template>\n <div class=\"border border-base-200 rounded-lg p-4 md:p-6\">\n <div class=\"grid grid-cols-1 md:grid-cols-2 gap-x-8 gap-y-3\">\n <MetadataField label=\"Status\">\n <SupportTicketApprovalBadge\n :approval-status=\"displayApprovalStatus\"\n size=\"sm\"\n />\n </MetadataField>\n <MetadataField label=\"Type\">\n <SupportTicketTypeBadge :type=\"ticket.type\" size=\"sm\" />\n </MetadataField>\n <MetadataField label=\"Priority\">\n <SupportTicketPriorityBadge :priority=\"ticket.priority\" size=\"sm\" />\n </MetadataField>\n <MetadataField label=\"Requester\">\n {{ ticket.created_by_display_name }}\n </MetadataField>\n <div class=\"md:col-span-2\">\n <MetadataField label=\"Description\">\n <p v-if=\"ticket.description?.trim()\" class=\"whitespace-pre-wrap text-sm\">\n {{ ticket.description }}\n </p>\n <template #empty>\n <span class=\"text-base-content/50 italic text-sm\">No description provided</span>\n </template>\n </MetadataField>\n </div>\n <MetadataField label=\"Ticket ID\" :copyable=\"true\">\n {{ ticketDisplayId }}\n </MetadataField>\n <MetadataField label=\"Created\">\n {{ createdFormatted }}\n <span v-if=\"createdRelative\" class=\"text-base-content/50\">· {{ createdRelative }}</span>\n </MetadataField>\n <template v-if=\"showCredits\">\n <MetadataField label=\"Credits\">\n {{ creditDisplay }}\n </MetadataField>\n </template>\n <template v-if=\"ticket.target_at\">\n <MetadataField label=\"Target\">\n {{ targetFormatted }}\n <span class=\"text-base-content/50\">· {{ targetRelative }}</span>\n </MetadataField>\n </template>\n <template v-if=\"ticket.completed_at\">\n <MetadataField label=\"Completed\">\n {{ completedFormatted }}\n <span class=\"text-base-content/50\">· {{ completedRelative }}</span>\n </MetadataField>\n </template>\n </div>\n <div v-if=\"canEdit\" class=\"flex justify-end gap-2 mt-4\">\n <button type=\"button\" class=\"btn btn-sm\" @click=\"$emit('edit')\">\n <svg\n xmlns=\"http://www.w3.org/2000/svg\"\n fill=\"none\"\n viewBox=\"0 0 24 24\"\n stroke-width=\"1.5\"\n stroke=\"currentColor\"\n class=\"w-4 h-4 mr-2\"\n >\n <path\n stroke-linecap=\"round\"\n stroke-linejoin=\"round\"\n d=\"M16.862 4.487l1.687-1.688a1.875 1.875 0 112.652 2.652L10.582 16.07a4.5 4.5 0 01-1.897 1.13L6 18l.8-2.685a4.5 4.5 0 011.13-1.897l8.932-8.931zm0 0L19.5 7.125M18 14v4.75A2.25 2.25 0 0115.75 21H5.25A2.25 2.25 0 013 18.75V8.25A2.25 2.25 0 015.25 6H10\"\n />\n </svg>\n Edit\n </button>\n </div>\n </div>\n</template>\n\n<script setup lang=\"ts\">\nimport { computed } from 'vue';\nimport type { CustomerSupportTicketReadDto } from '@dragonmastery/dragoncore-shared';\nimport type { SupportTicketApproval } from '@dragonmastery/dragoncore-shared';\nimport MetadataField from '../../shared/MetadataField.vue';\nimport SupportTicketApprovalBadge from '../../shared/SupportTicketApprovalBadge.vue';\nimport SupportTicketTypeBadge from '../../shared/SupportTicketTypeBadge.vue';\nimport SupportTicketPriorityBadge from '../../shared/SupportTicketPriorityBadge.vue';\nimport { formatTicketDate } from '../../utils/formatTicketDate';\nimport { formatTicketDisplayId } from '../../utils/displayIdFormatter';\nimport { formatCustomerCreditValue } from '../../utils/creditValueFormatter';\n\ninterface Props {\n ticket: CustomerSupportTicketReadDto;\n}\n\ndefineEmits<{ edit: [] }>();\n\nconst props = defineProps<Props>();\n\nconst canEdit = computed(() => !props.ticket.is_locked && !props.ticket.archived_at);\n\nconst displayApprovalStatus = computed((): SupportTicketApproval => {\n const s = props.ticket.status;\n if (s === 'PENDING') return 'PENDING';\n if (s === 'CANCELLED') return 'REJECTED';\n return 'APPROVED';\n});\n\nconst ticketDisplayId = computed(() =>\n formatTicketDisplayId(\n props.ticket.display_id,\n props.ticket.display_id_prefix,\n props.ticket.id,\n ),\n);\n\nconst createdFormatted = computed(() => {\n const at = props.ticket.created_at;\n return at ? formatTicketDate(at).formatted : '';\n});\nconst createdRelative = computed(() => {\n const at = props.ticket.created_at;\n return at ? formatTicketDate(at).relative : '';\n});\n\nconst showCredits = computed(() => {\n return !!props.ticket.is_locked && !!props.ticket.locked_approval_at;\n});\n\nconst creditDisplay = computed(() => formatCustomerCreditValue(props.ticket.credit_value));\n\nconst targetFormatted = computed(() =>\n props.ticket.target_at ? formatTicketDate(props.ticket.target_at!).formatted : '',\n);\nconst targetRelative = computed(() =>\n props.ticket.target_at ? formatTicketDate(props.ticket.target_at!).relative : '',\n);\n\nconst completedFormatted = computed(() =>\n props.ticket.completed_at ? formatTicketDate(props.ticket.completed_at!).formatted : '',\n);\nconst completedRelative = computed(() =>\n props.ticket.completed_at ? formatTicketDate(props.ticket.completed_at!).relative : '',\n);\n</script>\n","<template>\n <div\n v-if=\"ticket\"\n class=\"flex flex-wrap items-center justify-between gap-3 rounded-lg border border-base-200 bg-base-200/30 px-4 py-3\"\n >\n <div class=\"flex items-center gap-3 min-w-0\">\n <div\n class=\"flex h-9 w-9 shrink-0 items-center justify-center rounded-lg\"\n :class=\"\n isSubscribed ? 'bg-info/20 text-info' : 'bg-base-300 text-base-content/60'\n \"\n >\n <!-- Bell (subscribed) -->\n <svg\n v-if=\"isSubscribed\"\n xmlns=\"http://www.w3.org/2000/svg\"\n class=\"h-5 w-5\"\n fill=\"none\"\n viewBox=\"0 0 24 24\"\n stroke=\"currentColor\"\n stroke-width=\"2\"\n >\n <path\n stroke-linecap=\"round\"\n stroke-linejoin=\"round\"\n d=\"M15 17h5l-1.405-1.405A2.032 2.032 0 0118 14.158V11a6.002 6.002 0 00-4-5.659V5a2 2 0 10-4 0v.341C7.67 6.165 6 8.388 6 11v3.159c0 .538-.214 1.055-.595 1.436L4 17h5m6 0v1a3 3 0 11-6 0v-1m6 0H9\"\n />\n </svg>\n <!-- Bell with slash (unsubscribed) -->\n <svg\n v-else\n xmlns=\"http://www.w3.org/2000/svg\"\n class=\"h-5 w-5\"\n fill=\"none\"\n viewBox=\"0 0 24 24\"\n stroke=\"currentColor\"\n stroke-width=\"2\"\n >\n <path\n stroke-linecap=\"round\"\n stroke-linejoin=\"round\"\n d=\"M9.143 17.082a24.248 24.248 0 0 0 3.844.148m-3.844-.148a23.856 23.856 0 0 1-5.455-1.31 8.964 8.964 0 0 0 2.3-5.542m3.155 6.852a3 3 0 0 0 5.667 1.97m1.965-2.277L21 21m-4.225-4.225a23.81 23.81 0 0 0 3.536-1.003A8.967 8.967 0 0 1 18 9.75V9A6 6 0 0 0 6.53 6.53m10.245 10.245L6.53 6.53M3 3l3.53 3.53\"\n />\n </svg>\n </div>\n <p class=\"text-sm text-base-content/80 min-w-0\">\n {{\n isSubscribed\n ? 'You will receive notifications for updates.'\n : 'Get notified when there are updates or new comments.'\n }}\n </p>\n <button\n type=\"button\"\n :class=\"[\n 'btn btn-sm shrink-0 font-medium',\n isSubscribed ? 'btn-outline' : 'btn-info',\n ]\"\n :disabled=\"isToggling\"\n @click=\"handleToggle\"\n >\n <span v-if=\"isToggling\" class=\"loading loading-spinner loading-xs\"></span>\n <span v-else>{{ isSubscribed ? 'Unsubscribe' : 'Subscribe' }}</span>\n </button>\n </div>\n </div>\n</template>\n\n<script setup lang=\"ts\">\nimport type {\n CustomerSupportTicketReadDto,\n DragoncoreApi,\n} from '@dragonmastery/dragoncore-shared';\nimport { computed, ref, watch } from 'vue';\nimport { toast } from 'vue3-toastify';\nimport { useMutation } from '../../../../composables/useMutation';\nimport { extractRpcErrorMessage } from '../../../../utils/extractRpcErrorMessage';\n\ninterface Props {\n ticket: CustomerSupportTicketReadDto;\n}\n\nconst props = defineProps<Props>();\n\nconst isToggling = ref(false);\n// Local override after toggle - avoids full-page refetch\nconst subscriptionOverride = ref<boolean | null>(null);\n\nwatch(\n () => props.ticket.id,\n () => {\n subscriptionOverride.value = null;\n },\n);\n\nconst isSubscribed = computed(\n () => subscriptionOverride.value ?? props.ticket.my_subscription != null,\n);\n\nconst { mutate: toggleSubscription } = useMutation<\n DragoncoreApi,\n string,\n { subscribed: boolean }\n>((api, id) => api.supportTickets.toggleSubscription(id), {\n invalidate: /^support-tickets?:/,\n});\n\nasync function handleToggle() {\n try {\n isToggling.value = true;\n const result = await toggleSubscription(props.ticket.id);\n subscriptionOverride.value = result.subscribed;\n toast.success(\n result.subscribed\n ? 'You are now subscribed to this ticket'\n : 'You have unsubscribed from this ticket',\n );\n } catch (e) {\n toast.error(extractRpcErrorMessage(e, 'Failed to update subscription'));\n } finally {\n isToggling.value = false;\n }\n}\n</script>\n","import {\n CustomerSupportTicketUpdateSchema,\n SUPPORT_TICKET_PRIORITY_NUMBER_TO_LABEL,\n} from '@dragonmastery/dragoncore-shared';\nimport { withMetadata } from '@dragonmastery/zinia-forms-core';\nimport { z } from 'zod';\n\nexport type CustomerMetadataCardEditFormDto = z.infer<typeof CustomerSupportTicketUpdateSchema>;\n\nexport const customerMetadataCardEditFormMetadata = withMetadata(\n CustomerSupportTicketUpdateSchema,\n 'customerMetadataCardEditForm',\n {\n title: {\n label: 'Title',\n placeholder: 'Enter ticket title',\n },\n description: {\n label: 'Description',\n inputType: 'textarea',\n placeholder: 'Describe the ticket in detail',\n },\n type: { label: 'Type' },\n priority: {\n label: 'Priority',\n inputType: 'select',\n valueToLabel: SUPPORT_TICKET_PRIORITY_NUMBER_TO_LABEL,\n valueType: 'number',\n },\n },\n);\n","<template>\n <div class=\"border border-base-200 rounded-lg p-4 md:p-6\">\n <ZiniaForm\n @handle-submit=\"handleSubmit\"\n @success=\"handleSuccess\"\n @error=\"handleError\"\n title=\"\"\n subtitle=\"\"\n >\n <div class=\"grid grid-cols-1 md:grid-cols-2 gap-4\">\n <!-- Title - full width -->\n <div class=\"form-control md:col-span-2\">\n <zinia.TitleField placeholder=\"Enter ticket title\" />\n </div>\n\n <!-- Description - full width -->\n <div class=\"form-control md:col-span-2\">\n <zinia.DescriptionField class=\"w-full\" placeholder=\"Describe the ticket in detail\" />\n </div>\n\n <!-- Type -->\n <div class=\"form-control\">\n <zinia.TypeField />\n </div>\n\n <!-- Priority -->\n <div class=\"form-control\">\n <zinia.PriorityField />\n </div>\n </div>\n\n <div v-if=\"form.submitError\" class=\"alert alert-error py-2 mt-4\">\n <span class=\"text-sm\">{{ form.submitError }}</span>\n </div>\n <ZiniaFormErrorsSummary title=\"Please fix the following errors:\" />\n\n <!-- Action Buttons -->\n <div class=\"flex justify-end gap-2 mt-4\">\n <button type=\"button\" class=\"btn btn-ghost\" @click=\"handleCancel\" :disabled=\"saving\">\n Cancel\n </button>\n <ZiniaSubmitButton submitText=\"Save\" submittingText=\"Saving...\" />\n </div>\n </ZiniaForm>\n </div>\n</template>\n\n<script setup lang=\"ts\">\nimport {\n supportTicketPriorityToNumber,\n type CustomerSupportTicketReadDto,\n type CustomerSupportTicketUpdateDto,\n} from '@dragonmastery/dragoncore-shared';\nimport { useForm } from '@dragonmastery/zinia-forms-core';\nimport { computed } from 'vue';\nimport { useMutation } from '../../../../composables/useMutation';\nimport { extractRpcErrorMessage } from '../../../../utils/extractRpcErrorMessage';\nimport { customerMetadataCardEditFormMetadata } from '../customerMetadataCardEditFormMetadata';\n\ninterface Props {\n ticket: CustomerSupportTicketReadDto;\n saving?: boolean;\n}\n\nconst props = defineProps<Props>();\n\nconst emit = defineEmits<{\n success: [];\n cancel: [];\n}>();\n\nconst { mutate: updateTicket, loading: isSaving } = useMutation(\n (api, input: CustomerSupportTicketUpdateDto) => api.supportTickets.updateTicket(input),\n { invalidate: /^support-tickets?:/ },\n);\n\nconst saving = computed(() => props.saving ?? isSaving.value);\n\nconst { form, zinia, ZiniaForm, ZiniaSubmitButton, ZiniaFormErrorsSummary } = useForm(\n customerMetadataCardEditFormMetadata,\n {\n storeName: `customer-metadata-card-edit-${props.ticket.id}`,\n persistToLocalStorage: false,\n renderStyle: 'daisy_ui',\n fetchData: async () => {\n const t = props.ticket;\n return {\n id: t.id,\n title: t.title || '',\n description: t.description ?? '',\n type: t.type,\n priority: supportTicketPriorityToNumber(t.priority),\n };\n },\n },\n);\n\nfunction buildPayload(formData: Record<string, unknown>): CustomerSupportTicketUpdateDto {\n return {\n id: props.ticket.id,\n title: String(formData.title ?? '').trim(),\n description: (formData.description as string) || '',\n type: formData.type as CustomerSupportTicketUpdateDto['type'],\n priority: formData.priority as CustomerSupportTicketUpdateDto['priority'],\n };\n}\n\nasync function handleSubmit(formData: Record<string, unknown>) {\n const payload = buildPayload(formData);\n await updateTicket(payload);\n return undefined;\n}\n\nfunction handleSuccess() {\n emit('success');\n}\n\nfunction handleError(error: Error | unknown) {\n const message = extractRpcErrorMessage(error, 'Failed to update ticket');\n form.setSubmitError(message);\n}\n\nfunction handleCancel() {\n emit('cancel');\n}\n</script>\n","<template>\n <div class=\"flex flex-col gap-4 w-full\">\n <TimelineNoteInput\n ref=\"noteInputRef\"\n :show-type-toggle=\"false\"\n :disabled=\"commentsLocked ?? false\"\n :on-submit=\"onAddNote\"\n />\n <p\n v-if=\"mergedTimelineItems.length === 0\"\n class=\"text-base-content/50 text-sm text-center py-8\"\n >\n No activity yet.\n </p>\n <template v-for=\"item in mergedTimelineItems\" :key=\"item.id\">\n <TimelineItem\n v-if=\"item.type === 'customer-note'\"\n :author-name=\"item.data.authorName\"\n :created-at=\"item.data.createdAt\"\n variant=\"customer\"\n >\n {{ item.data.body ?? '' }}\n </TimelineItem>\n <TimelineSystemEvent\n v-else-if=\"item.type === 'system-event'\"\n :author=\"item.data.author\"\n :message=\"item.data.message\"\n :timestamp=\"item.data.timestamp\"\n :action=\"item.data.action\"\n :type=\"item.data.type\"\n :details=\"item.data.details\"\n :old-value=\"item.data.oldValue\"\n :new-value=\"item.data.newValue\"\n :changes=\"item.data.changes\"\n />\n </template>\n </div>\n</template>\n\n<script setup lang=\"ts\">\nimport type { CustomerSupportTicketReadDto } from '@dragonmastery/dragoncore-shared';\nimport { computed, ref } from 'vue';\nimport type { SystemEvent } from '../../utils/parseRecordVersions';\nimport TimelineItem from '../../shared/TimelineItem.vue';\nimport TimelineNoteInput from '../../shared/TimelineNoteInput.vue';\nimport TimelineSystemEvent from '../../shared/TimelineSystemEvent.vue';\n\nexport interface CustomerNoteDto {\n authorName: string;\n createdAt: string;\n body: string;\n}\n\ninterface Props {\n ticket: CustomerSupportTicketReadDto;\n notes: CustomerNoteDto[];\n systemEvents: SystemEvent[];\n /** Locks comments - only when archived (customers can add comments until archived) */\n commentsLocked?: boolean;\n /** Called when user submits a note. Must return a Promise. */\n onAddNote: (payload: {\n content: string;\n noteType: 'customer' | 'internal';\n }) => Promise<void>;\n}\n\nconst props = defineProps<Props>();\n\nconst noteInputRef = ref<InstanceType<typeof TimelineNoteInput> | null>(null);\n\ntype TimelineItem =\n | { type: 'customer-note'; timestamp: string; id: string; data: CustomerNoteDto }\n | { type: 'system-event'; timestamp: string; id: string; data: SystemEvent };\n\nconst mergedTimelineItems = computed((): TimelineItem[] => {\n const items: TimelineItem[] = [];\n\n for (const note of props.notes ?? []) {\n items.push({\n type: 'customer-note',\n timestamp: note.createdAt,\n id: `note-${note.createdAt}-${note.authorName}`,\n data: note,\n });\n }\n\n for (const event of props.systemEvents ?? []) {\n items.push({\n type: 'system-event',\n timestamp: event.timestamp,\n id: event.id,\n data: event,\n });\n }\n\n items.sort((a, b) => new Date(b.timestamp).getTime() - new Date(a.timestamp).getTime());\n return items;\n});\n\nfunction resetInput() {\n noteInputRef.value?.reset();\n}\n\ndefineExpose({ resetInput });\n</script>\n","<template>\n <!-- Loading State (only on initial load; refetch keeps content visible) -->\n <div v-if=\"isLoading && !ticket\" class=\"flex justify-center items-center p-8\">\n <span class=\"loading loading-spinner loading-lg\"></span>\n </div>\n\n <!-- Error State -->\n <div v-else-if=\"error\" class=\"alert alert-error mb-4 max-w-4xl mx-auto px-4\">\n <svg\n xmlns=\"http://www.w3.org/2000/svg\"\n class=\"stroke-current shrink-0 h-6 w-6\"\n fill=\"none\"\n viewBox=\"0 0 24 24\"\n >\n <path\n stroke-linecap=\"round\"\n stroke-linejoin=\"round\"\n stroke-width=\"2\"\n d=\"M10 14l2-2m0 0l2-2m-2 2l-2-2m2 2l2 2m7-2a9 9 0 11-18 0 9 9 0 0118 0z\"\n />\n </svg>\n <div class=\"flex-1\">\n <span>{{ error.message || 'An error occurred' }}</span>\n <div class=\"mt-2\">\n <router-link :to=\"{ name: 'CustomerSupportTicketList' }\" class=\"link link-primary\">\n Go back to ticket list\n </router-link>\n </div>\n </div>\n </div>\n\n <!-- Main Content -->\n <div v-else-if=\"ticket\" class=\"px-4 py-6 max-w-4xl mx-auto\">\n <!-- Header -->\n <div class=\"mb-6\">\n <h1 class=\"text-2xl font-bold text-base-content break-words leading-tight mt-2\">\n {{ ticket.title }}\n </h1>\n </div>\n\n <!-- Action Banner -->\n <div class=\"mb-6\">\n <CustomerActionBanner :ticket=\"ticket\" />\n </div>\n\n <!-- Subscribe / Unsubscribe -->\n <div class=\"mb-6\">\n <CustomerSupportTicketSubscribeControl :ticket=\"ticket\" />\n </div>\n\n <!-- Metadata Card (Display or Edit) -->\n <div class=\"mb-8\">\n <CustomerMetadataCard v-if=\"!isEditMode\" :ticket=\"ticket\" @edit=\"enterEditMode\" />\n <CustomerMetadataCardEdit\n v-else\n :ticket=\"ticket\"\n @success=\"handleMetadataSaveSuccess\"\n @cancel=\"exitEditMode\"\n />\n </div>\n\n <!-- Attachments (separate from timeline so adding a comment doesn't cause refetch) -->\n <div class=\"mb-8\">\n <SupportTicketAttachmentsCollapsible\n :record-id=\"ticket.id\"\n :locked=\"isCommentsLocked\"\n :editable=\"!isCommentsLocked\"\n @uploaded=\"refetchTimeline\"\n @deleted=\"refetchTimeline\"\n />\n </div>\n\n <!-- Timeline: stay mounted once we have data so adding a comment doesn't unmount/remount -->\n <div>\n <CustomerTimeline\n v-if=\"timelineData != null\"\n :ticket=\"ticket\"\n :notes=\"customerNotesForTimeline\"\n :system-events=\"filteredSystemEvents\"\n :comments-locked=\"isCommentsLocked\"\n :on-add-note=\"handleAddNote\"\n />\n <div v-else-if=\"timelineError\" class=\"alert alert-error mb-4\">\n <span>{{ timelineError?.message ?? 'Failed to load timeline' }}</span>\n <button class=\"btn btn-sm btn-outline mt-2\" @click=\"refetchTimeline\">\n Retry\n </button>\n </div>\n <div v-else class=\"flex justify-center items-center py-8\">\n <span class=\"loading loading-spinner loading-lg\"></span>\n </div>\n </div>\n </div>\n</template>\n\n<script setup lang=\"ts\">\nimport type { CustomerSupportTicketReadDto, NoteCreateDto } from '@dragonmastery/dragoncore-shared';\nimport { OPERATORS, RecordConst } from '@dragonmastery/dragoncore-shared';\nimport { computed, inject, nextTick } from 'vue';\nimport { useRoute, useRouter } from 'vue-router';\nimport { toast } from 'vue3-toastify';\nimport { BATCH_MODE } from '../../../composables/useRpcAuth';\nimport { useMutation } from '../../../composables/useMutation';\nimport { useQuery } from '../../../composables/useQuery';\nimport SupportTicketAttachmentsCollapsible from '../shared/SupportTicketAttachmentsCollapsible.vue';\nimport { parseRecordVersions, type SystemEvent } from '../utils/parseRecordVersions';\nimport CustomerActionBanner from './components/CustomerActionBanner.vue';\nimport CustomerMetadataCard from './components/CustomerMetadataCard.vue';\nimport CustomerSupportTicketSubscribeControl from './components/CustomerSupportTicketSubscribeControl.vue';\nimport CustomerMetadataCardEdit from './components/CustomerMetadataCardEdit.vue';\nimport CustomerTimeline from './components/CustomerTimeline.vue';\n\ninterface Props {\n ticket: CustomerSupportTicketReadDto | null;\n isLoading?: boolean;\n error?: Error | null;\n}\n\nconst props = defineProps<Props>();\n\nconst route = useRoute();\nconst router = useRouter();\nconst support_ticket_id = route.params.id as string;\n\n// Edit mode state\nconst isEditMode = computed(() => route.query.mode === 'edit');\n\n// Get refresh function from parent\nconst refreshTicket = inject<(() => Promise<void>) | undefined>('refreshTicket');\n\n// Comments locked only when archived (customers can add comments until archived)\nconst isCommentsLocked = computed(() => {\n if (!props.ticket) return false;\n return !!props.ticket.archived_at;\n});\n\n// Fetch customer notes and record versions in a single batched request\nconst {\n data: timelineData,\n error: timelineError,\n refetch: refetchTimeline,\n} = useQuery(\n async (api) => {\n const [customerNotes, recordVersions] = await Promise.all([\n api.notes.getNotes({\n record_id: { operator: OPERATORS.EQUALS, value: support_ticket_id },\n record_type: { operator: OPERATORS.EQUALS, value: 'support_ticket' },\n is_internal: { operator: OPERATORS.EQUALS, value: false },\n first: 100,\n sortBy: 'created_at',\n sortDirection: 'asc',\n }),\n api.recordVersions.listRecordVersionsCustomer(\n support_ticket_id,\n RecordConst.SUPPORT_TICKET, // legacy; record_types in filters takes priority\n {\n record_types: [RecordConst.SUPPORT_TICKET, RecordConst.SUPPORT_TICKET_ACTIVITY],\n first: 100,\n sortBy: 'recorded_at',\n sortDirection: 'asc',\n },\n ),\n ]);\n return { customerNotes, recordVersions };\n },\n {\n enabled: !!support_ticket_id && !!props.ticket,\n batchMode: BATCH_MODE.batch,\n trackedSegment: 'customer-support-ticket-timeline',\n },\n);\n\n// Map notes for timeline\ninterface CustomerNoteForTimeline {\n authorName: string;\n createdAt: string;\n body: string;\n}\n\nconst customerNotesForTimeline = computed((): CustomerNoteForTimeline[] => {\n const items = timelineData.value?.customerNotes?.items ?? [];\n return items\n .map((n) => ({\n authorName: n.created_by_display_name ?? n.created_by ?? '(unknown)',\n createdAt: n.created_at,\n body: n.body ?? '',\n }))\n .sort((a, b) => new Date(a.createdAt).getTime() - new Date(b.createdAt).getTime());\n});\n\n// Parse and filter system events for customer visibility (merged by API from record_types)\nconst allSystemEvents = computed(() => {\n const data = timelineData.value?.recordVersions;\n const versions = data?.items ?? [];\n const userDisplayMap = data?.user_display_map;\n const displayMap = userDisplayMap ? new Map(Object.entries(userDisplayMap)) : undefined;\n return parseRecordVersions(versions, displayMap);\n});\n\n// Filter out staff-only system events\nconst filteredSystemEvents = computed((): SystemEvent[] => {\n if (!props.ticket) return [];\n\n const events = allSystemEvents.value;\n const isPending = props.ticket.status === 'PENDING' || !props.ticket.is_locked;\n\n // Staff-only fields that customers should never see\n const STAFF_ONLY_FIELDS = new Set(['dev_lifecycle', 'delivered_value', 'start_at']);\n\n return events.filter((event) => {\n // Filter by field name if available\n if (event.fieldName) {\n // Always filter out staff-only fields\n if (STAFF_ONLY_FIELDS.has(event.fieldName)) {\n return false;\n }\n\n // Filter credit_value changes when ticket is still pending\n if (isPending && event.fieldName === 'credit_value') {\n return false;\n }\n\n return true;\n }\n\n // Fallback to message-based filtering if fieldName not available\n const message = event.message.toLowerCase();\n\n // Always filter out dev_lifecycle, delivered_value, start_at\n if (\n message.includes('dev lifecycle') ||\n message.includes('moved') ||\n message.includes('delivered value') ||\n message.includes('start date')\n ) {\n return false;\n }\n\n // Filter credit_value changes when ticket is still pending\n if (isPending && (message.includes('credits') || message.includes('credit'))) {\n return false;\n }\n\n // Allow all other events (type, priority, status, target date, completed date, etc.)\n return true;\n });\n});\n\n// Note creation mutation\nconst { mutate: createNote } = useMutation(\n (api, input: NoteCreateDto) => api.notes.createNote(input),\n { invalidate: /^notes?:/ },\n);\n\n// Handle note creation\nasync function handleAddNote(payload: {\n content: string;\n noteType: 'customer' | 'internal';\n}) {\n try {\n await createNote({\n record_id: support_ticket_id,\n record_type: 'support_ticket' as const,\n body: payload.content,\n tag: null,\n is_internal: false,\n });\n await refetchTimeline();\n toast.success('Comment added');\n } catch {\n toast.error('Failed to add comment');\n throw new Error('Failed to add comment');\n }\n}\n\n// Enter edit mode\nfunction enterEditMode() {\n router.push({\n name: route.name || 'CustomerViewSupportTicket',\n params: route.params,\n query: { ...route.query, mode: 'edit' },\n });\n}\n\n// Exit edit mode\nasync function exitEditMode() {\n await router.push({\n name: route.name || 'CustomerViewSupportTicket',\n params: route.params,\n query: { ...route.query, mode: undefined },\n });\n}\n\n// Handle metadata save success (form does mutation; we refresh and exit)\nasync function handleMetadataSaveSuccess() {\n await exitEditMode();\n await nextTick();\n toast.success('Ticket updated successfully');\n if (refreshTicket) {\n await refreshTicket();\n }\n await refetchTimeline();\n}\n</script>\n"],"mappings":"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"}
|
package/dist/{CustomerSupportTicketList-BrTX8SUV.js → CustomerSupportTicketList-BPWQ8WjH.js}
RENAMED
|
@@ -3,11 +3,11 @@ import "./EnhancedRefreshTokenHandler-s8wUXtB5.js";
|
|
|
3
3
|
import "./useQueryCache-CUTrwJWX.js";
|
|
4
4
|
import "./useMutation-BXSu7_-s.js";
|
|
5
5
|
import "./useQuery-DownvLRA.js";
|
|
6
|
-
import { S as CustomerSupportTicketList_default } from "./src-
|
|
6
|
+
import { S as CustomerSupportTicketList_default } from "./src-BXO0PrFd.js";
|
|
7
7
|
import "./AppLink-FcNGKgvG.js";
|
|
8
|
-
import "./saved_filter-
|
|
8
|
+
import "./saved_filter-YIncsdws.js";
|
|
9
9
|
import "./ConfirmDialog-DjthOYU6.js";
|
|
10
|
-
import "./InlineAttachments-
|
|
10
|
+
import "./InlineAttachments-DvqCOd6U.js";
|
|
11
11
|
import "./TeamMembersTab-BigqpBDH.js";
|
|
12
12
|
import "./Appearance-shr0Aql0.js";
|
|
13
13
|
import "./Signup-D6vnyS4w.js";
|
|
@@ -21,30 +21,30 @@ import "./UserProfilePage-D7FNmca1.js";
|
|
|
21
21
|
import "./ChangePasswordPage-CR91NZP4.js";
|
|
22
22
|
import "./teamMetadata-NTjPt89L.js";
|
|
23
23
|
import "./team_memberRoutes-mO1f-Y4o.js";
|
|
24
|
-
import "./teamRoutes-
|
|
25
|
-
import "./CreateTeamForm-
|
|
26
|
-
import "./EditTeamForm-
|
|
24
|
+
import "./teamRoutes-Aa9aBVCa.js";
|
|
25
|
+
import "./CreateTeamForm-ClisfzsU.js";
|
|
26
|
+
import "./EditTeamForm-B4p-C-8i.js";
|
|
27
27
|
import "./TeamHistoryTab-5kNQ_XAq.js";
|
|
28
|
-
import "./TeamList-
|
|
28
|
+
import "./TeamList-P7rVlBuQ.js";
|
|
29
29
|
import "./TeamNotesTab-BREl3Vr0.js";
|
|
30
|
-
import "./TeamParent-
|
|
31
|
-
import "./ViewTeam-
|
|
30
|
+
import "./TeamParent-D35VUiio.js";
|
|
31
|
+
import "./ViewTeam-BbsTsjAy.js";
|
|
32
32
|
import "./teamMemberMetadata-C4urCwBU.js";
|
|
33
33
|
import "./CreateTeamMemberForm-B3jxlAsC.js";
|
|
34
34
|
import "./EditTeamMemberForm-D6vf9hp-.js";
|
|
35
35
|
import "./TeamMemberList-DyfXcR6F.js";
|
|
36
36
|
import "./TeamMemberParent-2zISixbT.js";
|
|
37
37
|
import "./ViewTeamMember-Dsl2GKI1.js";
|
|
38
|
-
import "./customerSupportTicketRoutes-
|
|
39
|
-
import "./staffSupportTicketRoutes-
|
|
38
|
+
import "./customerSupportTicketRoutes-38JuLJGP.js";
|
|
39
|
+
import "./staffSupportTicketRoutes-BZrj4aMG.js";
|
|
40
40
|
import "./TimelineSystemEvent-D5fkhkZT.js";
|
|
41
|
-
import "./CustomerCreateSupportTicketForm-
|
|
41
|
+
import "./CustomerCreateSupportTicketForm-Cul4aVjB.js";
|
|
42
42
|
import "./CustomerSupportTicketParent-D9FuZjPz.js";
|
|
43
|
-
import "./CustomerSupportTicketSuccess-
|
|
44
|
-
import "./StaffCreateSupportTicketForm-
|
|
43
|
+
import "./CustomerSupportTicketSuccess-B4WR8Zfn.js";
|
|
44
|
+
import "./StaffCreateSupportTicketForm-B1YioWlS.js";
|
|
45
45
|
import "./SupportTicketDevLifecycleBadge-D8-Cv1Np.js";
|
|
46
46
|
import "./StaffSupportTicketParent-CZllER18.js";
|
|
47
|
-
import "./StaffSupportTicketSuccess-
|
|
47
|
+
import "./StaffSupportTicketSuccess-DTMToAq2.js";
|
|
48
48
|
import "./LoginForm-CjF4NSgM.js";
|
|
49
49
|
import "./useEmailVerificationChannel-DYiMSAES.js";
|
|
50
50
|
import "./ForgotPassword-BykH9B9R.js";
|
|
@@ -57,8 +57,8 @@ import "./VerifyEmail-KCti4rzf.js";
|
|
|
57
57
|
import "./UserListPage-CCl0K7Gk.js";
|
|
58
58
|
import "./CreateUserPage-BcCIat_l.js";
|
|
59
59
|
import "./EditUserPage-Dy61CCle.js";
|
|
60
|
-
import "./CreditTransactionHistory-
|
|
61
|
-
import "./CreditBalanceDashboard-
|
|
62
|
-
import "./CreditManagement-
|
|
60
|
+
import "./CreditTransactionHistory-mz-QIKUG.js";
|
|
61
|
+
import "./CreditBalanceDashboard-CJr1yOjW.js";
|
|
62
|
+
import "./CreditManagement-BbDEviZx.js";
|
|
63
63
|
|
|
64
64
|
export { CustomerSupportTicketList_default as default };
|
package/dist/{CustomerSupportTicketSuccess-k3DX9NhV.js → CustomerSupportTicketSuccess-B4WR8Zfn.js}
RENAMED
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import { t as customerSupportPaths } from "./customerSupportTicketRoutes-
|
|
1
|
+
import { t as customerSupportPaths } from "./customerSupportTicketRoutes-38JuLJGP.js";
|
|
2
2
|
import { createCommentVNode, createElementBlock, createElementVNode, createStaticVNode, createTextVNode, createVNode, defineComponent, openBlock, resolveComponent, unref, withCtx } from "vue";
|
|
3
3
|
|
|
4
4
|
//#region src/slices/support_ticket/customer/CustomerSupportTicketSuccess.vue
|
|
@@ -51,4 +51,4 @@ var CustomerSupportTicketSuccess_default = _sfc_main;
|
|
|
51
51
|
|
|
52
52
|
//#endregion
|
|
53
53
|
export { CustomerSupportTicketSuccess_default as t };
|
|
54
|
-
//# sourceMappingURL=CustomerSupportTicketSuccess-
|
|
54
|
+
//# sourceMappingURL=CustomerSupportTicketSuccess-B4WR8Zfn.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"CustomerSupportTicketSuccess-
|
|
1
|
+
{"version":3,"file":"CustomerSupportTicketSuccess-B4WR8Zfn.js","names":[],"sources":["../src/slices/support_ticket/customer/CustomerSupportTicketSuccess.vue"],"sourcesContent":["<template>\n <div class=\"flex flex-col items-center justify-center min-h-[60vh] p-8\">\n <div class=\"text-center max-w-2xl\">\n <!-- Success Icon -->\n <div class=\"mb-6\">\n <div class=\"inline-flex items-center justify-center w-20 h-20 bg-success rounded-full\">\n <svg\n class=\"w-10 h-10 text-success-content\"\n fill=\"none\"\n stroke=\"currentColor\"\n viewBox=\"0 0 24 24\"\n xmlns=\"http://www.w3.org/2000/svg\"\n >\n <path\n stroke-linecap=\"round\"\n stroke-linejoin=\"round\"\n stroke-width=\"2\"\n d=\"M5 13l4 4L19 7\"\n ></path>\n </svg>\n </div>\n </div>\n\n <!-- Success Message -->\n <h1 class=\"text-3xl font-bold text-base-content mb-4\">\n Support Ticket Submitted Successfully!\n </h1>\n\n <p class=\"text-lg text-base-content/70 mb-8\">\n Your support ticket has been submitted and our team will review it shortly.\n </p>\n\n <!-- What Happens Next -->\n <div class=\"bg-base-200 rounded-lg p-6 mb-8\">\n <h2 class=\"text-xl font-semibold mb-4\">What happens next?</h2>\n <div class=\"space-y-3 text-left\">\n <div class=\"flex items-start gap-3\">\n <div\n class=\"flex-shrink-0 w-6 h-6 bg-primary text-primary-content rounded-full flex items-center justify-center text-sm font-semibold\"\n >\n 1\n </div>\n <div>\n <div class=\"font-medium\">Review</div>\n <div class=\"text-sm text-base-content/70\">\n Our team will review your ticket and determine the best approach\n </div>\n </div>\n </div>\n <div class=\"flex items-start gap-3\">\n <div\n class=\"flex-shrink-0 w-6 h-6 bg-primary text-primary-content rounded-full flex items-center justify-center text-sm font-semibold\"\n >\n 2\n </div>\n <div>\n <div class=\"font-medium\">Notification</div>\n <div class=\"text-sm text-base-content/70\">\n You'll receive email updates on status changes and progress\n </div>\n </div>\n </div>\n <div class=\"flex items-start gap-3\">\n <div\n class=\"flex-shrink-0 w-6 h-6 bg-primary text-primary-content rounded-full flex items-center justify-center text-sm font-semibold\"\n >\n 3\n </div>\n <div>\n <div class=\"font-medium\">Tracking</div>\n <div class=\"text-sm text-base-content/70\">\n Monitor progress in your support ticket list\n </div>\n </div>\n </div>\n </div>\n </div>\n\n <!-- Action Buttons -->\n <div class=\"flex flex-col sm:flex-row gap-4 justify-center\">\n <router-link\n :to=\"{ name: customerSupportPaths.customer_list.name }\"\n class=\"btn btn-primary btn-lg\"\n >\n View My Support Tickets\n </router-link>\n <router-link\n :to=\"{ name: customerSupportPaths.customer_create.name }\"\n class=\"btn btn-outline btn-lg\"\n >\n Submit Another Ticket\n </router-link>\n </div>\n </div>\n </div>\n</template>\n\n<script setup lang=\"ts\">\nimport { customerSupportPaths } from './customerSupportTicketRoutes';\n</script>\n"],"mappings":";;;;;;;;;;;;uBACE,mBA6FM,OA7FN,YA6FM,CA5FJ,mBA2FM,OA3FN,YA2FM;IA1FJ,mBAAA,iBAAqB;8BACrB,mBAiBM,OAAA,EAjBD,OAAM,QAAM,EAAA,CACf,mBAeM,OAAA,EAfD,OAAM,6EAA2E,EAAA,CACpF,mBAaM,OAAA;KAZJ,OAAM;KACN,MAAK;KACL,QAAO;KACP,SAAQ;KACR,OAAM;QAEN,mBAKQ,QAAA;KAJN,kBAAe;KACf,mBAAgB;KAChB,gBAAa;KACb,GAAE;;IAMV,mBAAA,oBAAwB;8BACxB,mBAEK,MAAA,EAFD,OAAM,6CAA2C,EAAC,4CAEtD,GAAA;8BAEA,mBAEI,KAAA,EAFD,OAAM,qCAAmC,EAAC,iFAE7C,GAAA;IAEA,mBAAA,sBAA0B;;IA8C1B,mBAAA,mBAAuB;IACvB,mBAaM,OAbN,YAaM,CAZJ,YAKc,wBAAA;KAJX,IAAE,EAAA,MAAU,MAAA,qBAAoB,CAAC,cAAc,MAAI;KACpD,OAAM;;4BAGR,CAAA,GAAA,OAAA,OAAA,OAAA,KAAA,CAAA,gBAFC,6BAED,GAAA,CAAA,EAAA,CAAA;;mBACA,YAKc,wBAAA;KAJX,IAAE,EAAA,MAAU,MAAA,qBAAoB,CAAC,gBAAgB,MAAI;KACtD,OAAM;;4BAGR,CAAA,GAAA,OAAA,OAAA,OAAA,KAAA,CAAA,gBAFC,2BAED,GAAA,CAAA,EAAA,CAAA"}
|
package/dist/{CustomerSupportTicketSuccess-BLSMAjQN.js → CustomerSupportTicketSuccess-Bsp1d0oF.js}
RENAMED
|
@@ -3,10 +3,10 @@ import "./EnhancedRefreshTokenHandler-s8wUXtB5.js";
|
|
|
3
3
|
import "./useQueryCache-CUTrwJWX.js";
|
|
4
4
|
import "./useMutation-BXSu7_-s.js";
|
|
5
5
|
import "./useQuery-DownvLRA.js";
|
|
6
|
-
import "./saved_filter-
|
|
6
|
+
import "./saved_filter-YIncsdws.js";
|
|
7
7
|
import "./ConfirmDialog-DjthOYU6.js";
|
|
8
8
|
import "./userAuthorized-qmzUYDa-.js";
|
|
9
|
-
import "./customerSupportTicketRoutes-
|
|
10
|
-
import { t as CustomerSupportTicketSuccess_default } from "./CustomerSupportTicketSuccess-
|
|
9
|
+
import "./customerSupportTicketRoutes-38JuLJGP.js";
|
|
10
|
+
import { t as CustomerSupportTicketSuccess_default } from "./CustomerSupportTicketSuccess-B4WR8Zfn.js";
|
|
11
11
|
|
|
12
12
|
export { CustomerSupportTicketSuccess_default as default };
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
import { t as useMutation } from "./useMutation-BXSu7_-s.js";
|
|
2
2
|
import { t as FieldsetSection_default } from "./FieldsetSection-Br_sygWW.js";
|
|
3
3
|
import { n as teamUpdateSchemaWithMetadata } from "./teamMetadata-NTjPt89L.js";
|
|
4
|
-
import { t as teamPaths } from "./teamRoutes-
|
|
4
|
+
import { t as teamPaths } from "./teamRoutes-Aa9aBVCa.js";
|
|
5
5
|
import { createBlock, createCommentVNode, createElementBlock, createElementVNode, createVNode, defineComponent, inject, openBlock, toDisplayString, unref, watch, withCtx } from "vue";
|
|
6
6
|
import { useRoute, useRouter } from "vue-router";
|
|
7
7
|
import { toast } from "vue3-toastify";
|
|
@@ -160,4 +160,4 @@ var EditTeamForm_default = _sfc_main;
|
|
|
160
160
|
|
|
161
161
|
//#endregion
|
|
162
162
|
export { EditTeamForm_default as t };
|
|
163
|
-
//# sourceMappingURL=EditTeamForm-
|
|
163
|
+
//# sourceMappingURL=EditTeamForm-B4p-C-8i.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"EditTeamForm-
|
|
1
|
+
{"version":3,"file":"EditTeamForm-B4p-C-8i.js","names":[],"sources":["../src/slices/team/EditTeamForm.vue"],"sourcesContent":["<template>\n <div v-if=\"form.isLoading\" class=\"flex justify-center items-center p-8\">\n <span class=\"loading loading-spinner loading-lg\"></span>\n </div>\n\n <div v-else-if=\"form.loadError\" class=\"alert alert-error mb-4\">\n <svg\n xmlns=\"http://www.w3.org/2000/svg\"\n class=\"stroke-current shrink-0 h-6 w-6\"\n fill=\"none\"\n viewBox=\"0 0 24 24\"\n >\n <path\n stroke-linecap=\"round\"\n stroke-linejoin=\"round\"\n stroke-width=\"2\"\n d=\"M10 14l2-2m0 0l2-2m-2 2l-2-2m2 2l2 2m7-2a9 9 0 11-18 0 9 9 0 0118 0z\"\n />\n </svg>\n <span>{{ form.loadError }}</span>\n </div>\n\n <ZiniaForm\n v-else\n @handle-submit=\"handleSubmit\"\n @success=\"handleSuccess\"\n @error=\"handleError\"\n title=\"Edit Team\"\n subtitle=\"Update team details\"\n >\n <!-- Basic Information -->\n <FieldsetSection title=\"Team Details\">\n <zinia.UniqueNameField placeholder=\"Enter unique name\" />\n <zinia.DisplayNameField placeholder=\"Enter display name\" />\n <zinia.LegalNameField placeholder=\"Enter legal name (optional)\" />\n </FieldsetSection>\n\n <FieldsetSection title=\"Description\">\n <zinia.DescriptionField placeholder=\"Enter team description\" />\n </FieldsetSection>\n\n <!-- Contact Information -->\n <FieldsetSection title=\"Contact Information\">\n <zinia.ContactNameField placeholder=\"Enter contact name\" />\n <zinia.ContactEmailField placeholder=\"Enter contact email\" />\n <zinia.ContactBusinessPhoneField placeholder=\"Enter business phone\" />\n <zinia.ContactMobilePhoneField placeholder=\"Enter mobile phone\" />\n <zinia.ContactTimeZoneField placeholder=\"Enter time zone\" />\n </FieldsetSection>\n\n <!-- Address Information -->\n <FieldsetSection title=\"Address\">\n <zinia.AddressFullField placeholder=\"Enter address\" />\n <zinia.AddressCityField placeholder=\"Enter city\" />\n <zinia.AddressZipField placeholder=\"Enter zip code\" />\n </FieldsetSection>\n\n <!-- Web Presence -->\n <FieldsetSection title=\"Web Presence\">\n <zinia.TwitterUsernameField placeholder=\"Enter Twitter username\" />\n <zinia.UrlField placeholder=\"Enter URL\" />\n </FieldsetSection>\n\n <!-- Branding -->\n <FieldsetSection title=\"Branding\">\n <zinia.LogoField placeholder=\"Enter logo URL\" />\n </FieldsetSection>\n\n <!-- Email Settings -->\n <FieldsetSection title=\"Email Settings\">\n <zinia.EmailSentFromField placeholder=\"Enter sent from email\" />\n <zinia.EmailReplyToField placeholder=\"Enter reply to email\" />\n </FieldsetSection>\n\n <!-- Form Status Messages -->\n <div v-if=\"form.submitError\" class=\"alert alert-error mb-4\">\n <svg\n xmlns=\"http://www.w3.org/2000/svg\"\n class=\"stroke-current shrink-0 h-6 w-6\"\n fill=\"none\"\n viewBox=\"0 0 24 24\"\n >\n <path\n stroke-linecap=\"round\"\n stroke-linejoin=\"round\"\n stroke-width=\"2\"\n d=\"M10 14l2-2m0 0l2-2m-2 2l-2-2m2 2l2 2m7-2a9 9 0 11-18 0 9 9 0 0118 0z\"\n />\n </svg>\n <span>{{ form.submitError }}</span>\n </div>\n\n <!-- Submit Button -->\n <div class=\"flex justify-center mt-6\">\n <ZiniaSubmitButton submitText=\"Update Team\" submittingText=\"Updating Team...\" />\n </div>\n <ZiniaFormErrorsSummary title=\"Please fix the following errors:\" />\n <ZiniaResetButton />\n </ZiniaForm>\n</template>\n\n<script setup lang=\"ts\">\nimport FieldsetSection from '../../components/ui/FieldsetSection.vue';\nimport { useMutation } from '../../composables/useMutation';\nimport { useForm } from '@dragonmastery/zinia-forms-core';\nimport type { TeamReadDto, TeamUpdateDto } from '@dragonmastery/dragoncore-shared';\nimport { inject, watch } from 'vue';\nimport { useRoute, useRouter } from 'vue-router';\nimport { toast } from 'vue3-toastify';\nimport { teamUpdateSchemaWithMetadata } from './teamMetadata';\nimport { teamPaths } from './teamRoutes';\n\ninterface Props {\n team: TeamReadDto | null;\n isLoading?: boolean;\n error?: Error | null;\n}\n\nconst props = defineProps<Props>();\n\nconst route = useRoute();\nconst router = useRouter();\nconst team_id = route.params.id as string;\n\nconst {\n form,\n zinia,\n ZiniaForm,\n ZiniaSubmitButton,\n ZiniaResetButton,\n ZiniaFormErrorsSummary,\n clearSavedFormState,\n} = useForm(teamUpdateSchemaWithMetadata, {\n storeName: `edit-team-${team_id}`,\n persistToLocalStorage: false,\n renderStyle: 'daisy_ui',\n fetchData: async () => {\n if (!props.team) {\n throw new Error('No team data found');\n }\n\n return props.team as TeamUpdateDto;\n },\n});\n\n// Get refresh function from parent\nconst refreshTeam = inject<(() => Promise<void>) | undefined>('refreshTeam');\n\n// Redirect to view if team is archived\nwatch(\n () => props.team,\n (team) => {\n if (team?.archived_at) {\n router.replace({\n name: teamPaths.view.name,\n params: { id: team_id },\n });\n }\n },\n);\n\n// Setup mutation for updating team\nconst { mutate: updateTeam } = useMutation(\n (api, input: TeamUpdateDto) => api.teams.updateTeam(input),\n { invalidate: /^teams?:/ },\n);\n\n// Handle form submission\nconst handleSubmit = async (formData: TeamUpdateDto) => {\n const updatedTeam = await updateTeam(formData);\n if (!updatedTeam) {\n throw new Error('Failed to update team');\n }\n return updatedTeam;\n};\n\n// Handle successful submission\nconst handleSuccess = async () => {\n clearSavedFormState();\n toast.success(`Team updated successfully!`);\n // Refresh team data via parent\n if (refreshTeam) {\n await refreshTeam();\n }\n};\n\n// Handle submission error\nconst handleError = (error: Error | unknown) => {\n form.setSubmitError(error instanceof Error ? error.message : 'An unknown error occurred');\n};\n</script>\n"],"mappings":"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"}
|
|
@@ -6,7 +6,7 @@ import "./FieldsetSection-Br_sygWW.js";
|
|
|
6
6
|
import "./userAuthorized-qmzUYDa-.js";
|
|
7
7
|
import "./teamMetadata-NTjPt89L.js";
|
|
8
8
|
import "./team_memberRoutes-mO1f-Y4o.js";
|
|
9
|
-
import "./teamRoutes-
|
|
10
|
-
import { t as EditTeamForm_default } from "./EditTeamForm-
|
|
9
|
+
import "./teamRoutes-Aa9aBVCa.js";
|
|
10
|
+
import { t as EditTeamForm_default } from "./EditTeamForm-B4p-C-8i.js";
|
|
11
11
|
|
|
12
12
|
export { EditTeamForm_default as default };
|
|
@@ -716,11 +716,15 @@ const _hoisted_19 = {
|
|
|
716
716
|
};
|
|
717
717
|
const _hoisted_20 = { class: "flex-1 min-w-0" };
|
|
718
718
|
const _hoisted_21 = { class: "text-sm sm:text-base font-medium truncate" };
|
|
719
|
-
const _hoisted_22 = {
|
|
720
|
-
|
|
719
|
+
const _hoisted_22 = {
|
|
720
|
+
key: 0,
|
|
721
|
+
class: "text-xs text-base-content/70 truncate"
|
|
722
|
+
};
|
|
723
|
+
const _hoisted_23 = { class: "text-xs text-base-content/60" };
|
|
721
724
|
const _hoisted_24 = ["onClick"];
|
|
722
725
|
const _hoisted_25 = ["onClick"];
|
|
723
|
-
const _hoisted_26 =
|
|
726
|
+
const _hoisted_26 = ["onClick"];
|
|
727
|
+
const _hoisted_27 = {
|
|
724
728
|
key: 3,
|
|
725
729
|
class: "text-center py-8 text-base-content/50"
|
|
726
730
|
};
|
|
@@ -792,6 +796,8 @@ const _sfc_main = /* @__PURE__ */ defineComponent({
|
|
|
792
796
|
for (const item of fileItems) files.push({
|
|
793
797
|
id: item.id,
|
|
794
798
|
name: item.original_name,
|
|
799
|
+
title: item.title ?? null,
|
|
800
|
+
description: item.description ?? null,
|
|
795
801
|
size: parseInt(item.file_size || "0"),
|
|
796
802
|
type: item.content_type || "",
|
|
797
803
|
uploadedAt: new Date(item.created_at)
|
|
@@ -1175,7 +1181,11 @@ const _sfc_main = /* @__PURE__ */ defineComponent({
|
|
|
1175
1181
|
alt: file.name,
|
|
1176
1182
|
class: "w-full h-full object-cover",
|
|
1177
1183
|
onError: handleImageError
|
|
1178
|
-
}, null, 40, _hoisted_18)) : (openBlock(), createElementBlock("div", _hoisted_19, [..._cache[9] || (_cache[9] = [createElementVNode("span", { class: "loading loading-spinner loading-xs" }, null, -1)])]))])) : (openBlock(), createElementBlock(Fragment, { key: 1 }, [createCommentVNode(" File type badge for non-images "), createElementVNode("div", { class: normalizeClass(["badge badge-sm", getFileTypeBadgeClass(file.type)]) }, toDisplayString(getFileTypeLabel(file.type)), 3)], 2112))]), createElementVNode("div", _hoisted_20, [
|
|
1184
|
+
}, null, 40, _hoisted_18)) : (openBlock(), createElementBlock("div", _hoisted_19, [..._cache[9] || (_cache[9] = [createElementVNode("span", { class: "loading loading-spinner loading-xs" }, null, -1)])]))])) : (openBlock(), createElementBlock(Fragment, { key: 1 }, [createCommentVNode(" File type badge for non-images "), createElementVNode("div", { class: normalizeClass(["badge badge-sm", getFileTypeBadgeClass(file.type)]) }, toDisplayString(getFileTypeLabel(file.type)), 3)], 2112))]), createElementVNode("div", _hoisted_20, [
|
|
1185
|
+
createElementVNode("div", _hoisted_21, toDisplayString(file.title || file.name), 1),
|
|
1186
|
+
file.description ? (openBlock(), createElementBlock("div", _hoisted_22, toDisplayString(file.description), 1)) : createCommentVNode("v-if", true),
|
|
1187
|
+
createElementVNode("div", _hoisted_23, toDisplayString(formatFileSize(file.size)) + " • " + toDisplayString(formatDate(file.uploadedAt)), 1)
|
|
1188
|
+
])]), createElementVNode("div", {
|
|
1179
1189
|
class: "flex gap-1 justify-end sm:justify-start",
|
|
1180
1190
|
onClick: _cache[2] || (_cache[2] = withModifiers(() => {}, ["stop"]))
|
|
1181
1191
|
}, [
|
|
@@ -1201,7 +1211,7 @@ const _sfc_main = /* @__PURE__ */ defineComponent({
|
|
|
1201
1211
|
"stroke-linejoin": "round",
|
|
1202
1212
|
"stroke-width": "2",
|
|
1203
1213
|
d: "M2.458 12C3.732 7.943 7.523 5 12 5c4.478 0 8.268 2.943 9.542 7-1.274 4.057-5.064 7-9.542 7-4.477 0-8.268-2.943-9.542-7z"
|
|
1204
|
-
})], -1)])], 8,
|
|
1214
|
+
})], -1)])], 8, _hoisted_24)) : createCommentVNode("v-if", true),
|
|
1205
1215
|
createElementVNode("button", {
|
|
1206
1216
|
onClick: withModifiers(($event) => downloadFile(file, $event), ["prevent"]),
|
|
1207
1217
|
class: "btn btn-sm btn-ghost btn-circle",
|
|
@@ -1218,7 +1228,7 @@ const _sfc_main = /* @__PURE__ */ defineComponent({
|
|
|
1218
1228
|
"stroke-linejoin": "round",
|
|
1219
1229
|
"stroke-width": "2",
|
|
1220
1230
|
d: "M4 16v1a3 3 0 003 3h10a3 3 0 003-3v-1m-4-4l-4 4m0 0l-4-4m4 4V4"
|
|
1221
|
-
})], -1)])], 8,
|
|
1231
|
+
})], -1)])], 8, _hoisted_25),
|
|
1222
1232
|
__props.canDelete ? (openBlock(), createElementBlock("button", {
|
|
1223
1233
|
key: 1,
|
|
1224
1234
|
onClick: withModifiers(($event) => confirmDelete(file), ["prevent"]),
|
|
@@ -1236,9 +1246,9 @@ const _sfc_main = /* @__PURE__ */ defineComponent({
|
|
|
1236
1246
|
"stroke-linejoin": "round",
|
|
1237
1247
|
"stroke-width": "2",
|
|
1238
1248
|
d: "M19 7l-.867 12.142A2 2 0 0116.138 21H7.862a2 2 0 01-1.995-1.858L5 7m5 4v6m4-6v6m1-10V4a1 1 0 00-1-1h-4a1 1 0 00-1 1v3M4 7h16"
|
|
1239
|
-
})], -1)])], 8,
|
|
1249
|
+
})], -1)])], 8, _hoisted_26)) : createCommentVNode("v-if", true)
|
|
1240
1250
|
])], 10, _hoisted_14);
|
|
1241
|
-
}), 128))])) : !unref(attachmentsLoading) && __props.recordId ? (openBlock(), createElementBlock("div",
|
|
1251
|
+
}), 128))])) : !unref(attachmentsLoading) && __props.recordId ? (openBlock(), createElementBlock("div", _hoisted_27, [..._cache[13] || (_cache[13] = [createElementVNode("svg", {
|
|
1242
1252
|
xmlns: "http://www.w3.org/2000/svg",
|
|
1243
1253
|
class: "h-12 w-12 mx-auto mb-2 opacity-50",
|
|
1244
1254
|
fill: "none",
|
|
@@ -1293,4 +1303,4 @@ var InlineAttachments_default = _sfc_main;
|
|
|
1293
1303
|
|
|
1294
1304
|
//#endregion
|
|
1295
1305
|
export { ImageModal_default as n, InlineAttachments_default as t };
|
|
1296
|
-
//# sourceMappingURL=InlineAttachments-
|
|
1306
|
+
//# sourceMappingURL=InlineAttachments-DvqCOd6U.js.map
|