@dragonmastery/dragoncore-vue 0.0.26 → 0.0.28

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (81) hide show
  1. package/dist/{ConsentFlowStep-Dwg2HpnI.js → ConsentFlowStep-DstxorHZ.js} +1 -1
  2. package/dist/{ConsentFlowStep-Dwg2HpnI.js.map → ConsentFlowStep-DstxorHZ.js.map} +1 -1
  3. package/dist/{ConsentRequired-Cp9QGWiV.js → ConsentRequired-ELUidmNv.js} +2 -2
  4. package/dist/{ConsentRequired-Cp9QGWiV.js.map → ConsentRequired-ELUidmNv.js.map} +1 -1
  5. package/dist/{CreateTeamForm-C-ktK6hn.js → CreateTeamForm-CmVZdqOQ.js} +2 -2
  6. package/dist/{CreateTeamForm-CZRTmCOx.js → CreateTeamForm-DRfZ74on.js} +2 -2
  7. package/dist/{CreateTeamForm-CZRTmCOx.js.map → CreateTeamForm-DRfZ74on.js.map} +1 -1
  8. package/dist/{CreditBalanceDashboard-BHLCoCbK.js → CreditBalanceDashboard-BRY56-9w.js} +2 -2
  9. package/dist/{CreditBalanceDashboard-BHLCoCbK.js.map → CreditBalanceDashboard-BRY56-9w.js.map} +1 -1
  10. package/dist/{CreditBalanceDashboard-Bzxo_Rhr.js → CreditBalanceDashboard-CwXWREgJ.js} +4 -4
  11. package/dist/{CreditManagement-Ds6bCB1y.js → CreditManagement-BAuzw-NE.js} +4 -4
  12. package/dist/{CreditManagement-C_Va9qmV.js → CreditManagement-D-bsc1US.js} +2 -2
  13. package/dist/{CreditManagement-C_Va9qmV.js.map → CreditManagement-D-bsc1US.js.map} +1 -1
  14. package/dist/{CreditTransactionHistory-JVPQPiLB.js → CreditTransactionHistory-plIaRscn.js} +2 -2
  15. package/dist/{CreditTransactionHistory-JVPQPiLB.js.map → CreditTransactionHistory-plIaRscn.js.map} +1 -1
  16. package/dist/{CustomerCreateSupportTicketForm-BfEiv6I8.js → CustomerCreateSupportTicketForm-Co6C_P5o.js} +2 -2
  17. package/dist/{CustomerCreateSupportTicketForm-BfEiv6I8.js.map → CustomerCreateSupportTicketForm-Co6C_P5o.js.map} +1 -1
  18. package/dist/{CustomerCreateSupportTicketForm-BI5LQ9sF.js → CustomerCreateSupportTicketForm-D72blvMU.js} +3 -3
  19. package/dist/{CustomerSupportTicketDetailPage-CI0UmP2U.js → CustomerSupportTicketDetailPage-CUkf9swo.js} +2 -2
  20. package/dist/{CustomerSupportTicketDetailPage-CI0UmP2U.js.map → CustomerSupportTicketDetailPage-CUkf9swo.js.map} +1 -1
  21. package/dist/{CustomerSupportTicketList-Bf9ImaIe.js → CustomerSupportTicketList-CB_Y1lVj.js} +19 -19
  22. package/dist/{CustomerSupportTicketSuccess-QgmDYuA_.js → CustomerSupportTicketSuccess-B87Zth-g.js} +3 -3
  23. package/dist/{CustomerSupportTicketSuccess-CmXVwmtE.js → CustomerSupportTicketSuccess-DVqoR5-o.js} +2 -2
  24. package/dist/{CustomerSupportTicketSuccess-CmXVwmtE.js.map → CustomerSupportTicketSuccess-DVqoR5-o.js.map} +1 -1
  25. package/dist/DefaultReferralTeamPage-BHail7YF.js +207 -0
  26. package/dist/DefaultReferralTeamPage-BHail7YF.js.map +1 -0
  27. package/dist/{EditTeamForm-DYvHGBRF.js → EditTeamForm-BEOkUaKG.js} +2 -2
  28. package/dist/{EditTeamForm-DYvHGBRF.js.map → EditTeamForm-BEOkUaKG.js.map} +1 -1
  29. package/dist/{EditTeamForm-BLqalA9l.js → EditTeamForm-DhutyI9c.js} +2 -2
  30. package/dist/{LoginForm-DPmrJ_wm.js → LoginForm-D1Mx2vAY.js} +59 -37
  31. package/dist/LoginForm-D1Mx2vAY.js.map +1 -0
  32. package/dist/{LoginForm-By2I6TuV.js → LoginForm-p2fJiTtw.js} +1 -1
  33. package/dist/{SavedFiltersPage-bB4jbQgI.js → SavedFiltersPage-CvBKztlD.js} +20 -20
  34. package/dist/{SavedFiltersPage-bB4jbQgI.js.map → SavedFiltersPage-CvBKztlD.js.map} +1 -1
  35. package/dist/{SignupConsentFlow-Dd3QAkVF.js → SignupConsentFlow-clxBjJlU.js} +2 -2
  36. package/dist/{SignupConsentFlow-Dd3QAkVF.js.map → SignupConsentFlow-clxBjJlU.js.map} +1 -1
  37. package/dist/{SignupRequirementsPage-DUQ63ZuW.js → SignupRequirementsPage-CohJluxQ.js} +1 -1
  38. package/dist/{SignupRequirementsPage-DUQ63ZuW.js.map → SignupRequirementsPage-CohJluxQ.js.map} +1 -1
  39. package/dist/{StaffCreateSupportTicketForm-LuHzqt2w.js → StaffCreateSupportTicketForm-Cm595v_4.js} +2 -2
  40. package/dist/{StaffCreateSupportTicketForm-LuHzqt2w.js.map → StaffCreateSupportTicketForm-Cm595v_4.js.map} +1 -1
  41. package/dist/{StaffCreateSupportTicketForm-ByvHC20-.js → StaffCreateSupportTicketForm-DBhhJyXE.js} +3 -3
  42. package/dist/{StaffSupportTicketDetailPage--idGVu5P.js → StaffSupportTicketDetailPage-B63QXyum.js} +2 -2
  43. package/dist/{StaffSupportTicketDetailPage--idGVu5P.js.map → StaffSupportTicketDetailPage-B63QXyum.js.map} +1 -1
  44. package/dist/{StaffSupportTicketList-irgGqZUp.js → StaffSupportTicketList-2TbMweMK.js} +19 -19
  45. package/dist/{StaffSupportTicketSuccess-C6PI7mxk.js → StaffSupportTicketSuccess-DKzJs74k.js} +3 -3
  46. package/dist/{StaffSupportTicketSuccess-BoQZY9Qt.js → StaffSupportTicketSuccess-DgULDGIj.js} +2 -2
  47. package/dist/{StaffSupportTicketSuccess-BoQZY9Qt.js.map → StaffSupportTicketSuccess-DgULDGIj.js.map} +1 -1
  48. package/dist/{SupportStaffPage-c2ouJwJk.js → SupportStaffPage-CkFLlle4.js} +1 -1
  49. package/dist/{SupportStaffPage-c2ouJwJk.js.map → SupportStaffPage-CkFLlle4.js.map} +1 -1
  50. package/dist/{SupportTicketMaintenancePage-B07avInx.js → SupportTicketMaintenancePage-CEKi8xQB.js} +1 -1
  51. package/dist/{SupportTicketMaintenancePage-B07avInx.js.map → SupportTicketMaintenancePage-CEKi8xQB.js.map} +1 -1
  52. package/dist/{TeamAttachmentsTab-B9fDWDrw.js → TeamAttachmentsTab-D0SJplvU.js} +19 -19
  53. package/dist/{TeamList-lMtyRe75.js → TeamList-DU6CFPUY.js} +2 -2
  54. package/dist/{TeamList-CBJOtPQR.js → TeamList-gppM0GOD.js} +2 -2
  55. package/dist/{TeamList-CBJOtPQR.js.map → TeamList-gppM0GOD.js.map} +1 -1
  56. package/dist/{TeamParent-Ckbc8M17.js → TeamParent-CuASTHKr.js} +2 -2
  57. package/dist/{TeamParent-CYKznys5.js → TeamParent-YPtenk3l.js} +2 -2
  58. package/dist/{TeamParent-CYKznys5.js.map → TeamParent-YPtenk3l.js.map} +1 -1
  59. package/dist/{TimelineNoteInput--q4ENAgu.js → TimelineNoteInput-DXZhcUkH.js} +1 -1
  60. package/dist/{TimelineNoteInput--q4ENAgu.js.map → TimelineNoteInput-DXZhcUkH.js.map} +1 -1
  61. package/dist/{VerifyEmail-DdUKxJnk.js → VerifyEmail-CLDngljq.js} +1 -1
  62. package/dist/{VerifyEmail-CEbvnWLl.js → VerifyEmail-CWUhRA1o.js} +9 -4
  63. package/dist/VerifyEmail-CWUhRA1o.js.map +1 -0
  64. package/dist/{ViewTeam-zP_Jah0c.js → ViewTeam-Bvvfik4P.js} +2 -2
  65. package/dist/{ViewTeam-BgZzH7lK.js → ViewTeam-CRmIplCt.js} +2 -2
  66. package/dist/{ViewTeam-BgZzH7lK.js.map → ViewTeam-CRmIplCt.js.map} +1 -1
  67. package/dist/{customerSupportTicketRoutes-DZ6mSNIQ.js → customerSupportTicketRoutes-Cy4fp4wx.js} +6 -6
  68. package/dist/{customerSupportTicketRoutes-DZ6mSNIQ.js.map → customerSupportTicketRoutes-Cy4fp4wx.js.map} +1 -1
  69. package/dist/index.d.ts +183 -136
  70. package/dist/index.js +20 -20
  71. package/dist/{saved_filter-B0MGrzy2.js → saved_filter-jeZd2rlb.js} +2 -2
  72. package/dist/{saved_filter-B0MGrzy2.js.map → saved_filter-jeZd2rlb.js.map} +1 -1
  73. package/dist/{src-Bx44Vnfl.js → src-zjaOyP9b.js} +51 -13
  74. package/dist/{src-Bx44Vnfl.js.map → src-zjaOyP9b.js.map} +1 -1
  75. package/dist/{staffSupportTicketRoutes-CKdCYj3S.js → staffSupportTicketRoutes-L4CU5dcu.js} +6 -6
  76. package/dist/{staffSupportTicketRoutes-CKdCYj3S.js.map → staffSupportTicketRoutes-L4CU5dcu.js.map} +1 -1
  77. package/dist/{teamRoutes-BYpld9yI.js → teamRoutes-CtNcFZjR.js} +7 -7
  78. package/dist/{teamRoutes-BYpld9yI.js.map → teamRoutes-CtNcFZjR.js.map} +1 -1
  79. package/package.json +2 -2
  80. package/dist/LoginForm-DPmrJ_wm.js.map +0 -1
  81. package/dist/VerifyEmail-CEbvnWLl.js.map +0 -1
@@ -1 +1 @@
1
- {"version":3,"file":"CustomerSupportTicketDetailPage-CI0UmP2U.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":";;;;;;;;;;;;;;;;;;;;;;EAsBA,MAAM,QAAQ;EAEd,MAAM,aAAa,eAAe,CAAC,CAAC,MAAM,OAAO,YAAY;EAE7D,MAAM,aAAa,eAAe;GAChC,MAAM,IAAI,MAAM;AAChB,UAAO,CAAC,CAAC,EAAE,aAAa,CAAC,CAAC,EAAE,sBAAsB,EAAE,WAAW;IAC/D;EAEF,MAAM,aAAa,eAAe;GAChC,MAAM,IAAI,MAAM;AAChB,UAAO,CAAC,CAAC,EAAE,aAAa,CAAC,CAAC,EAAE,sBAAsB,EAAE,WAAW;IAC/D;EAEF,MAAM,sBAAsB,eAAe;GACzC,MAAM,KAAK,MAAM,OAAO;AACxB,OAAI,CAAC,GAAI,QAAO;AAChB,UAAO,iBAAiB,GAAG,CAAC;IAC5B;;UAvCyB,WAAA,SAAA,WAAA,EAAzB,YAEoB,2BAAA;;IAFiB,SAAQ;IAAU,MAAK;;2BAE5D,CAAA,GAAA,OAAA,OAAA,OAAA,KAAA,CAAA,gBAFsE,uGAEtE,GAAA,CAAA,EAAA,CAAA;;SAC8B,WAAA,SAAA,WAAA,EAA9B,YAEoB,2BAAA;;IAFsB,SAAQ;IAAU,MAAK;;2BACnC,CAAA,gBAD0C,kCAC1C,gBAAG,oBAAA,MAAmB,GAAG,MACvD,EAAA,CAAA,CAAA;;SAC8B,WAAA,SAAA,WAAA,EAA9B,YAEoB,2BAAA;;IAFsB,SAAQ;IAAQ,MAAK;;2BACjC,CAAA,gBADwC,kCACxC,gBAAG,oBAAA,MAAmB,GAAG,MACvD,EAAA,CAAA,CAAA;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;ECoFF,MAAM,QAAQ;EAEd,MAAM,UAAU,eAAe,CAAC,MAAM,OAAO,aAAa,CAAC,MAAM,OAAO,YAAY;EAEpF,MAAM,wBAAwB,eAAsC;GAClE,MAAM,IAAI,MAAM,OAAO;AACvB,OAAI,MAAM,UAAW,QAAO;AAC5B,OAAI,MAAM,YAAa,QAAO;AAC9B,UAAO;IACP;EAEF,MAAM,kBAAkB,eACtB,sBACE,MAAM,OAAO,YACb,MAAM,OAAO,mBACb,MAAM,OAAO,GACd,CACF;EAED,MAAM,mBAAmB,eAAe;GACtC,MAAM,KAAK,MAAM,OAAO;AACxB,UAAO,KAAK,iBAAiB,GAAG,CAAC,YAAY;IAC7C;EACF,MAAM,kBAAkB,eAAe;GACrC,MAAM,KAAK,MAAM,OAAO;AACxB,UAAO,KAAK,iBAAiB,GAAG,CAAC,WAAW;IAC5C;EAEF,MAAM,cAAc,eAAe;AACjC,UAAO,CAAC,CAAC,MAAM,OAAO,aAAa,CAAC,CAAC,MAAM,OAAO;IAClD;EAEF,MAAM,gBAAgB,eAAe,0BAA0B,MAAM,OAAO,aAAa,CAAC;EAE1F,MAAM,kBAAkB,eACtB,MAAM,OAAO,YAAY,iBAAiB,MAAM,OAAO,UAAW,CAAC,YAAY,GAChF;EACD,MAAM,iBAAiB,eACrB,MAAM,OAAO,YAAY,iBAAiB,MAAM,OAAO,UAAW,CAAC,WAAW,GAC/E;EAED,MAAM,qBAAqB,eACzB,MAAM,OAAO,eAAe,iBAAiB,MAAM,OAAO,aAAc,CAAC,YAAY,GACtF;EACD,MAAM,oBAAoB,eACxB,MAAM,OAAO,eAAe,iBAAiB,MAAM,OAAO,aAAc,CAAC,WAAW,GACrF;;uBA1IC,mBAuEM,OAvEN,cAuEM,CAtEJ,mBAkDM,OAlDN,cAkDM;IAjDJ,YAKgB,uBAAA,EALD,OAAM,UAAQ,EAAA;4BAIzB,CAHF,YAGE,oCAAA;MAFC,mBAAiB,sBAAA;MAClB,MAAK;;;;IAGT,YAEgB,uBAAA,EAFD,OAAM,QAAM,EAAA;4BAC+B,CAAxD,YAAwD,gCAAA;MAA/B,MAAM,QAAA,OAAO;MAAM,MAAK;;;;IAEnD,YAEgB,uBAAA,EAFD,OAAM,YAAU,EAAA;4BACuC,CAApE,YAAoE,oCAAA;MAAvC,UAAU,QAAA,OAAO;MAAU,MAAK;;;;IAE/D,YAEgB,uBAAA,EAFD,OAAM,aAAW,EAAA;4BACM,CAAA,gCAAjC,QAAA,OAAO,wBAAuB,EAAA,EAAA,CAAA,CAAA;;;IAEnC,mBASM,OATN,cASM,CARJ,YAOgB,uBAAA,EAPD,OAAM,eAAa,EAAA;KAIrB,OAAK,cACkE,CAAA,GAAA,OAAA,OAAA,OAAA,KAAA,CAAhF,mBAAgF,QAAA,EAA1E,OAAM,uCAAqC,EAAC,2BAAuB,GAAA,CAAA,EAAA,CAAA;4BAFvE,CAFK,QAAA,OAAO,aAAa,MAAI,IAAA,WAAA,EAAjC,mBAEI,KAFJ,cAEI,gBADC,QAAA,OAAO,YAAW,EAAA,EAAA,IAAA,mBAAA,QAAA,KAAA,CAAA,CAAA;;;IAO3B,YAEgB,uBAAA;KAFD,OAAM;KAAa,UAAU;;4BACrB,CAAA,gCAAlB,gBAAA,MAAe,EAAA,EAAA,CAAA,CAAA;;;IAEpB,YAGgB,uBAAA,EAHD,OAAM,WAAS,EAAA;4BACN,CAAA,gCAAnB,iBAAA,MAAgB,GAAG,KACtB,EAAA,EAAY,gBAAA,SAAA,WAAA,EAAZ,mBAAwF,QAAxF,cAA0D,OAAE,gBAAG,gBAAA,MAAe,EAAA,EAAA,IAAA,mBAAA,QAAA,KAAA,CAAA,CAAA;;;IAEhE,YAAA,SAAA,WAAA,EACd,YAEgB,uBAAA;;KAFD,OAAM;;4BACA,CAAA,gCAAhB,cAAA,MAAa,EAAA,EAAA,CAAA,CAAA;;;IAGJ,QAAA,OAAO,aAAA,WAAA,EACrB,YAGgB,uBAAA;;KAHD,OAAM;;4BACE,CAAA,gCAAlB,gBAAA,MAAe,GAAG,KACrB,EAAA,EAAA,mBAAgE,QAAhE,cAAmC,OAAE,gBAAG,eAAA,MAAc,EAAA,EAAA,CAAA,CAAA;;;IAG1C,QAAA,OAAO,gBAAA,WAAA,EACrB,YAGgB,uBAAA;;KAHD,OAAM;;4BACK,CAAA,gCAArB,mBAAA,MAAkB,GAAG,KACxB,EAAA,EAAA,mBAAmE,QAAnE,cAAmC,OAAE,gBAAG,kBAAA,MAAiB,EAAA,EAAA,CAAA,CAAA;;;OAIpD,QAAA,SAAA,WAAA,EAAX,mBAkBM,OAlBN,cAkBM,CAjBJ,mBAgBS,UAAA;IAhBD,MAAK;IAAS,OAAM;IAAc,SAAK,OAAA,OAAA,OAAA,MAAA,WAAEA,KAAAA,MAAK,OAAA;qCACpD,mBAaM,OAAA;IAZJ,OAAM;IACN,MAAK;IACL,SAAQ;IACR,gBAAa;IACb,QAAO;IACP,OAAM;OAEN,mBAIE,QAAA;IAHA,kBAAe;IACf,mBAAgB;IAChB,GAAE;6BAEA,UAER,GAAA,CAAA,EAAA,CAAA,CAAA,CAAA,IAAA,mBAAA,QAAA,KAAA,CAAA,CAAA;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;ECYN,MAAM,QAAQ;EAEd,MAAM,aAAa,IAAI,MAAM;EAE7B,MAAM,uBAAuB,IAAoB,KAAK;AAEtD,cACQ,MAAM,OAAO,UACb;AACJ,wBAAqB,QAAQ;IAEhC;EAED,MAAM,eAAe,eACb,qBAAqB,SAAS,MAAM,OAAO,mBAAmB,KACrE;EAED,MAAM,EAAE,QAAQ,uBAAuB,aAIpC,KAAK,OAAO,IAAI,eAAe,mBAAmB,GAAG,EAAE,EACxD,YAAY,sBACb,CAAC;EAEF,eAAe,eAAe;AAC5B,OAAI;AACF,eAAW,QAAQ;IACnB,MAAM,SAAS,MAAM,mBAAmB,MAAM,OAAO,GAAG;AACxD,yBAAqB,QAAQ,OAAO;AACpC,UAAM,QACJ,OAAO,aACH,0CACA,yCACL;YACM,GAAG;AACV,UAAM,MAAM,uBAAuB,GAAG,gCAAgC,CAAC;aAC/D;AACR,eAAW,QAAQ;;;;UAtHb,QAAA,UAAA,WAAA,EADR,mBAgEM,OAhEN,cAgEM,CA5DJ,mBA2DM,OA3DN,cA2DM;IA1DJ,mBAsCM,OAAA,EArCJ,OAAK,eAAA,CAAC,gEACa,aAAA,QAAY,yBAAA,mCAAA,CAAA,EAAA,GAI/B,mBAAA,sBAA0B,EAElB,aAAA,SAAA,WAAA,EADR,mBAcM,OAdN,cAcM,CAAA,GAAA,OAAA,OAAA,OAAA,KAAA,CALJ,mBAIE,QAAA;KAHA,kBAAe;KACf,mBAAgB;KAChB,GAAE;sCAIN,mBAcM,UAAA,EAAA,KAAA,GAAA,EAAA,CAfN,mBAAA,mCAAuC,EAAA,OAAA,OAAA,OAAA,KACvC,mBAcM,OAAA;KAZJ,OAAM;KACN,OAAM;KACN,MAAK;KACL,SAAQ;KACR,QAAO;KACP,gBAAa;QAEb,mBAIE,QAAA;KAHA,kBAAe;KACf,mBAAgB;KAChB,GAAE;;IAIR,mBAMI,KANJ,cAMI,gBAJA,aAAA,QAAA,gDAAA,uDAAA;IAKJ,mBAWS,UAAA;KAVP,MAAK;KACJ,OAAK,eAAA,CAAA,mCAA2D,aAAA,QAAY,gBAAA,WAAA,CAAA;KAI5E,UAAU,WAAA;KACV,SAAO;QAEI,WAAA,SAAA,WAAA,EAAZ,mBAA0E,QAA1E,aAA0E,KAAA,WAAA,EAC1E,mBAAoE,QAAA,cAAA,gBAApD,aAAA,QAAY,gBAAA,YAAA,EAAA,EAAA,EAAA,EAAA,IAAA,aAAA;;;;;;;;;ACrDpC,MAAa,uCAAuC,aAClD,mCACA,gCACA;CACE,OAAO;EACL,OAAO;EACP,aAAa;EACd;CACD,aAAa;EACX,OAAO;EACP,WAAW;EACX,aAAa;EACd;CACD,MAAM,EAAE,OAAO,QAAQ;CACvB,UAAU;EACR,OAAO;EACP,WAAW;EACX,cAAc;EACd,WAAW;EACZ;CACF,CACF;;;;;;;;;;;;;;;;;;;;;;;;;ECkCD,MAAM,QAAQ;EAEd,MAAM,OAAO;EAKb,MAAM,EAAE,QAAQ,cAAc,SAAS,aAAa,aACjD,KAAK,UAA0C,IAAI,eAAe,aAAa,MAAM,EACtF,EAAE,YAAY,sBAAsB,CACrC;EAED,MAAM,SAAS,eAAe,MAAM,UAAU,SAAS,MAAM;EAE7D,MAAM,EAAE,MAAM,OAAO,WAAW,mBAAmB,2BAA2B,QAC5E,sCACA;GACE,WAAW,+BAA+B,MAAM,OAAO;GACvD,uBAAuB;GACvB,aAAa;GACb,WAAW,YAAY;IACrB,MAAM,IAAI,MAAM;AAChB,WAAO;KACL,IAAI,EAAE;KACN,OAAO,EAAE,SAAS;KAClB,aAAa,EAAE,eAAe;KAC9B,MAAM,EAAE;KACR,UAAU,8BAA8B,EAAE,SAAS;KACpD;;GAEJ,CACF;EAED,SAAS,aAAa,UAAmE;AACvF,UAAO;IACL,IAAI,MAAM,OAAO;IACjB,OAAO,OAAO,SAAS,SAAS,GAAG,CAAC,MAAM;IAC1C,aAAc,SAAS,eAA0B;IACjD,MAAM,SAAS;IACf,UAAU,SAAS;IACpB;;EAGH,eAAe,aAAa,UAAmC;AAE7D,SAAM,aADU,aAAa,SAAS,CACX;;EAI7B,SAAS,gBAAgB;AACvB,QAAK,UAAU;;EAGjB,SAAS,YAAY,OAAwB;GAC3C,MAAM,UAAU,uBAAuB,OAAO,0BAA0B;AACxE,QAAK,eAAe,QAAQ;;EAG9B,SAAS,eAAe;AACtB,QAAK,SAAS;;;uBA1Hd,mBA2CM,OA3CN,cA2CM,CA1CJ,YAyCY,MAAA,UAAA,EAAA;IAxCT,gBAAe;IACf,WAAS;IACT,SAAO;IACR,OAAM;IACN,UAAS;;2BAsBH;KApBN,mBAoBM,OApBN,cAoBM;MAnBJ,mBAAA,uBAA2B;MAC3B,mBAEM,OAFN,cAEM,CADJ,YAAqD,MAAA,MAAA,CAAA,YAAA,EAAnC,aAAY,sBAAoB,CAAA,CAAA,CAAA;MAGpD,mBAAA,6BAAiC;MACjC,mBAEM,OAFN,cAEM,CADJ,YAAqF,MAAA,MAAA,CAAA,kBAAA;OAA7D,OAAM;OAAS,aAAY;;MAGrD,mBAAA,SAAa;MACb,mBAEM,OAFN,cAEM,CADJ,YAAmB,MAAA,MAAA,CAAA,UAAA,CAAA,CAAA;MAGrB,mBAAA,aAAiB;MACjB,mBAEM,OAFN,cAEM,CADJ,YAAuB,MAAA,MAAA,CAAA,cAAA,CAAA,CAAA;;KAIhB,MAAA,KAAI,CAAC,eAAA,WAAA,EAAhB,mBAEM,OAFN,cAEM,CADJ,mBAAmD,QAAnD,cAAmD,gBAA1B,MAAA,KAAI,CAAC,YAAW,EAAA,EAAA,CAAA,CAAA,IAAA,mBAAA,QAAA,KAAA;KAE3C,YAAmE,MAAA,uBAAA,EAAA,EAA3C,OAAM,oCAAkC,CAAA;KAEhE,mBAAA,mBAAuB;KACvB,mBAKM,OALN,cAKM,CAJJ,mBAES,UAAA;MAFD,MAAK;MAAS,OAAM;MAAiB,SAAO;MAAe,UAAU,OAAA;QAAQ,YAErF,GAAA,cAAA,EACA,YAAkE,MAAA,kBAAA,EAAA;MAA/C,YAAW;MAAO,gBAAe;;;;;;;;;;;;;;;;;;;;;;;;;;;ECyB5D,MAAM,QAAQ;EAEd,MAAM,eAAe,IAAmD,KAAK;EAM7E,MAAM,sBAAsB,eAA+B;GACzD,MAAMC,QAAwB,EAAE;AAEhC,QAAK,MAAM,QAAQ,MAAM,SAAS,EAAE,CAClC,OAAM,KAAK;IACT,MAAM;IACN,WAAW,KAAK;IAChB,IAAI,QAAQ,KAAK,UAAU,GAAG,KAAK;IACnC,MAAM;IACP,CAAC;AAGJ,QAAK,MAAM,SAAS,MAAM,gBAAgB,EAAE,CAC1C,OAAM,KAAK;IACT,MAAM;IACN,WAAW,MAAM;IACjB,IAAI,MAAM;IACV,MAAM;IACP,CAAC;AAGJ,SAAM,MAAM,GAAG,MAAM,IAAI,KAAK,EAAE,UAAU,CAAC,SAAS,GAAG,IAAI,KAAK,EAAE,UAAU,CAAC,SAAS,CAAC;AACvF,UAAO;IACP;EAEF,SAAS,aAAa;AACpB,gBAAa,OAAO,OAAO;;AAG7B,WAAa,EAAE,YAAY,CAAC;;uBAtG1B,mBAmCM,OAnCN,cAmCM;IAlCJ,YAKE,2BAAA;cAJI;KAAJ,KAAI;KACH,oBAAkB;KAClB,UAAU,QAAA,kBAAc;KACxB,aAAW,QAAA;;IAGN,oBAAA,MAAoB,WAAM,KAAA,WAAA,EADlC,mBAKI,KALJ,cAGC,qBAED,IAAA,mBAAA,QAAA,KAAA;sBACA,mBAqBW,UAAA,MAAA,WArBc,oBAAA,QAAR,SAAI;6DAA+B,KAAK,IAAA,GAE/C,KAAK,SAAI,mBAAA,WAAA,EADjB,YAOe,sBAAA;;MALZ,eAAa,KAAK,KAAK;MACvB,cAAY,KAAK,KAAK;MACvB,SAAQ;;6BAEkB,CAAA,gCAAvB,KAAK,KAAK,QAAI,GAAA,EAAA,EAAA,CAAA,CAAA;;gDAGN,KAAK,SAAI,kBAAA,WAAA,EADtB,YAWE,6BAAA;;MATC,QAAQ,KAAK,KAAK;MAClB,SAAS,KAAK,KAAK;MACnB,WAAW,KAAK,KAAK;MACrB,QAAQ,KAAK,KAAK;MAClB,MAAM,KAAK,KAAK;MAChB,SAAS,KAAK,KAAK;MACnB,aAAW,KAAK,KAAK;MACrB,aAAW,KAAK,KAAK;MACrB,SAAS,KAAK,KAAK;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;ECqF5B,MAAM,QAAQ;EAEd,MAAM,QAAQ,UAAU;EACxB,MAAM,SAAS,WAAW;EAC1B,MAAM,oBAAoB,MAAM,OAAO;EAGvC,MAAM,aAAa,eAAe,MAAM,MAAM,SAAS,OAAO;EAG9D,MAAM,gBAAgB,OAA0C,gBAAgB;EAGhF,MAAM,mBAAmB,eAAe;AACtC,OAAI,CAAC,MAAM,OAAQ,QAAO;AAC1B,UAAO,CAAC,CAAC,MAAM,OAAO;IACtB;EAGF,MAAM,EACJ,MAAM,cACN,OAAO,eACP,SAAS,oBACP,SACF,OAAO,QAAQ;GACb,MAAM,CAAC,eAAe,kBAAkB,MAAM,QAAQ,IAAI,CACxD,IAAI,MAAM,SAAS;IACjB,WAAW;KAAE,UAAU,UAAU;KAAQ,OAAO;KAAmB;IACnE,aAAa;KAAE,UAAU,UAAU;KAAQ,OAAO;KAAkB;IACpE,aAAa;KAAE,UAAU,UAAU;KAAQ,OAAO;KAAO;IACzD,OAAO;IACP,QAAQ;IACR,eAAe;IAChB,CAAC,EACF,IAAI,eAAe,2BACjB,mBACA,YAAY,gBACZ;IACE,cAAc,CAAC,YAAY,gBAAgB,YAAY,wBAAwB;IAC/E,OAAO;IACP,QAAQ;IACR,eAAe;IAChB,CACF,CACF,CAAC;AACF,UAAO;IAAE;IAAe;IAAgB;KAE1C;GACE,SAAS,CAAC,CAAC,qBAAqB,CAAC,CAAC,MAAM;GACxC,WAAW,WAAW;GACtB,gBAAgB;GACjB,CACF;EASD,MAAM,2BAA2B,eAA0C;AAEzE,WADc,aAAa,OAAO,eAAe,SAAS,EAAE,EAEzD,KAAK,OAAO;IACX,YAAY,EAAE,2BAA2B,EAAE,cAAc;IACzD,WAAW,EAAE;IACb,MAAM,EAAE,QAAQ;IACjB,EAAC,CACD,MAAM,GAAG,MAAM,IAAI,KAAK,EAAE,UAAU,CAAC,SAAS,GAAG,IAAI,KAAK,EAAE,UAAU,CAAC,SAAS,CAAC;IACpF;EAGF,MAAM,kBAAkB,eAAe;GACrC,MAAM,OAAO,aAAa,OAAO;GACjC,MAAM,WAAW,MAAM,SAAS,EAAE;GAClC,MAAM,iBAAiB,MAAM;AAE7B,UAAO,oBAAoB,UADR,iBAAiB,IAAI,IAAI,OAAO,QAAQ,eAAe,CAAC,GAAG,OAC9B;IAChD;EAGF,MAAM,uBAAuB,eAA8B;AACzD,OAAI,CAAC,MAAM,OAAQ,QAAO,EAAE;GAE5B,MAAM,SAAS,gBAAgB;GAC/B,MAAM,YAAY,MAAM,OAAO,WAAW,aAAa,CAAC,MAAM,OAAO;GAGrE,MAAM,oBAAoB,IAAI,IAAI;IAAC;IAAiB;IAAmB;IAAW,CAAC;AAEnF,UAAO,OAAO,QAAQ,UAAU;AAE9B,QAAI,MAAM,WAAW;AAEnB,SAAI,kBAAkB,IAAI,MAAM,UAAU,CACxC,QAAO;AAIT,SAAI,aAAa,MAAM,cAAc,eACnC,QAAO;AAGT,YAAO;;IAIT,MAAM,UAAU,MAAM,QAAQ,aAAa;AAG3C,QACE,QAAQ,SAAS,gBAAgB,IACjC,QAAQ,SAAS,QAAQ,IACzB,QAAQ,SAAS,kBAAkB,IACnC,QAAQ,SAAS,aAAY,CAE7B,QAAO;AAIT,QAAI,cAAc,QAAQ,SAAS,UAAU,IAAI,QAAQ,SAAS,SAAS,EACzE,QAAO;AAIT,WAAO;KACP;IACF;EAGF,MAAM,EAAE,QAAQ,eAAe,aAC5B,KAAK,UAAyB,IAAI,MAAM,WAAW,MAAM,EAC1D,EAAE,YAAY,YAAY,CAC3B;EAGD,eAAe,cAAc,SAG1B;AACD,OAAI;AACF,UAAM,WAAW;KACf,WAAW;KACX,aAAa;KACb,MAAM,QAAQ;KACd,KAAK;KACL,aAAa;KACd,CAAC;AACF,UAAM,iBAAiB;AACvB,UAAM,QAAQ,gBAAgB;WACxB;AACN,UAAM,MAAM,wBAAwB;AACpC,UAAM,IAAI,MAAM,wBAAwB;;;EAK5C,SAAS,gBAAgB;AACvB,UAAO,KAAK;IACV,MAAM,MAAM,QAAQ;IACpB,QAAQ,MAAM;IACd,OAAO;KAAE,GAAG,MAAM;KAAO,MAAM;KAAQ;IACxC,CAAC;;EAIJ,eAAe,eAAe;AAC5B,SAAM,OAAO,KAAK;IAChB,MAAM,MAAM,QAAQ;IACpB,QAAQ,MAAM;IACd,OAAO;KAAE,GAAG,MAAM;KAAO,MAAM;KAAW;IAC3C,CAAC;;EAIJ,eAAe,4BAA4B;AACzC,SAAM,cAAc;AACpB,SAAM,UAAU;AAChB,SAAM,QAAQ,8BAA8B;AAC5C,OAAI,cACF,OAAM,eAAe;AAEvB,SAAM,iBAAiB;;;;2DA5SvB,mBAAA,wEAA4E,EACjE,QAAA,aAAS,CAAK,QAAA,UAAA,WAAA,EAAzB,mBAEM,OAFN,YAEM,CAAA,GAAA,OAAA,OAAA,OAAA,KAAA,CADJ,mBAAwD,QAAA,EAAlD,OAAM,sCAAoC,EAAA,MAAA,GAAA,CAAA,EAAA,CAAA,IAIlC,QAAA,SAAA,WAAA,EAAhB,mBAsBM,UAAA,EAAA,KAAA,GAAA,EAAA,CAvBN,mBAAA,gBAAoB,EACpB,mBAsBM,OAtBN,YAsBM,CAAA,OAAA,OAAA,OAAA,KArBJ,mBAYM,OAAA;IAXJ,OAAM;IACN,OAAM;IACN,MAAK;IACL,SAAQ;OAER,mBAKE,QAAA;IAJA,kBAAe;IACf,mBAAgB;IAChB,gBAAa;IACb,GAAE;cAGN,mBAOM,OAPN,YAOM,CANJ,mBAAuD,QAAA,MAAA,gBAA9C,QAAA,MAAM,WAAO,oBAAA,EAAA,EAAA,EACtB,mBAIM,OAJN,YAIM,CAHJ,YAEc,wBAAA;IAFA,IAAI,EAAA,MAAA,6BAAqC;IAAE,OAAM;;2BAE/D,CAAA,GAAA,OAAA,OAAA,OAAA,KAAA,CAAA,gBAFmF,4BAEnF,GAAA,CAAA,EAAA,CAAA;;uBAMU,QAAA,UAAA,WAAA,EAAhB,mBA4DM,UAAA,EAAA,KAAA,GAAA,EAAA,CA7DN,mBAAA,iBAAqB,EACrB,mBA4DM,OA5DN,YA4DM;IA3DJ,mBAAA,WAAe;IACf,mBAIM,OAJN,YAIM,CAHJ,mBAEK,MAFL,YAEK,gBADA,QAAA,OAAO,MAAK,EAAA,EAAA,CAAA,CAAA;IAInB,mBAAA,kBAAsB;IACtB,mBAEM,OAFN,YAEM,CADJ,YAAyC,8BAAA,EAAlB,QAAQ,QAAA,QAAM,EAAA,MAAA,GAAA,CAAA,SAAA,CAAA,CAAA,CAAA;IAGvC,mBAAA,4BAAgC;IAChC,mBAEM,OAFN,YAEM,CADJ,YAA0D,+CAAA,EAAlB,QAAQ,QAAA,QAAM,EAAA,MAAA,GAAA,CAAA,SAAA,CAAA,CAAA,CAAA;IAGxD,mBAAA,oCAAwC;IACxC,mBAQM,OARN,aAQM,CAAA,CAPyB,WAAA,SAAA,WAAA,EAA7B,YAAkF,8BAAA;;KAAxC,QAAQ,QAAA;KAAS,QAAM;6CACjE,YAKE,kCAAA;;KAHC,QAAQ,QAAA;KACR,WAAS;KACT,UAAQ;;IAIb,mBAAA,mFAAuF;IACvF,mBAQM,OARN,aAQM,CAPJ,YAME,6CAAA;KALC,aAAW,QAAA,OAAO;KAClB,QAAQ,iBAAA;KACR,UAAQ,CAAG,iBAAA;KACX,YAAU,MAAA,gBAAe;KACzB,WAAS,MAAA,gBAAe;;;;;;;;IAI7B,mBAAA,yFAA6F;IAC7F,mBAkBM,OAAA,MAAA,CAhBI,MAAA,aAAY,IAAA,QAAA,WAAA,EADpB,YAOE,0BAAA;;KALC,QAAQ,QAAA;KACR,OAAO,yBAAA;KACP,iBAAe,qBAAA;KACf,mBAAiB,iBAAA;KACjB,eAAa;;;;;;UAEA,MAAA,cAAa,IAAA,WAAA,EAA7B,mBAKM,OALN,aAKM,CAJJ,mBAAsE,QAAA,MAAA,gBAA7D,MAAA,cAAa,EAAE,WAAO,0BAAA,EAAA,EAAA,EAC/B,mBAES,UAAA;KAFD,OAAM;KAA+B,SAAK,OAAA,OAAA,OAAA,MAAA,GAAA,SAAE,MAAA,gBAAA,IAAA,MAAA,gBAAA,CAAA,GAAA,KAAe;OAAE,UAErE,CAAA,CAAA,KAAA,WAAA,EAEF,mBAEM,OAFN,aAEM,CAAA,GAAA,OAAA,OAAA,OAAA,KAAA,CADJ,mBAAwD,QAAA,EAAlD,OAAM,sCAAoC,EAAA,MAAA,GAAA,CAAA,EAAA,CAAA,EAAA,CAAA"}
1
+ {"version":3,"file":"CustomerSupportTicketDetailPage-CUkf9swo.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":";;;;;;;;;;;;;;;;;;;;;;EAsBA,MAAM,QAAQ;EAEd,MAAM,aAAa,eAAe,CAAC,CAAC,MAAM,OAAO,YAAY;EAE7D,MAAM,aAAa,eAAe;GAChC,MAAM,IAAI,MAAM;AAChB,UAAO,CAAC,CAAC,EAAE,aAAa,CAAC,CAAC,EAAE,sBAAsB,EAAE,WAAW;IAC/D;EAEF,MAAM,aAAa,eAAe;GAChC,MAAM,IAAI,MAAM;AAChB,UAAO,CAAC,CAAC,EAAE,aAAa,CAAC,CAAC,EAAE,sBAAsB,EAAE,WAAW;IAC/D;EAEF,MAAM,sBAAsB,eAAe;GACzC,MAAM,KAAK,MAAM,OAAO;AACxB,OAAI,CAAC,GAAI,QAAO;AAChB,UAAO,iBAAiB,GAAG,CAAC;IAC5B;;UAvCyB,WAAA,SAAA,WAAA,EAAzB,YAEoB,2BAAA;;IAFiB,SAAQ;IAAU,MAAK;;2BAE5D,CAAA,GAAA,OAAA,OAAA,OAAA,KAAA,CAAA,gBAFsE,uGAEtE,GAAA,CAAA,EAAA,CAAA;;SAC8B,WAAA,SAAA,WAAA,EAA9B,YAEoB,2BAAA;;IAFsB,SAAQ;IAAU,MAAK;;2BACnC,CAAA,gBAD0C,kCAC1C,gBAAG,oBAAA,MAAmB,GAAG,MACvD,EAAA,CAAA,CAAA;;SAC8B,WAAA,SAAA,WAAA,EAA9B,YAEoB,2BAAA;;IAFsB,SAAQ;IAAQ,MAAK;;2BACjC,CAAA,gBADwC,kCACxC,gBAAG,oBAAA,MAAmB,GAAG,MACvD,EAAA,CAAA,CAAA;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;ECoFF,MAAM,QAAQ;EAEd,MAAM,UAAU,eAAe,CAAC,MAAM,OAAO,aAAa,CAAC,MAAM,OAAO,YAAY;EAEpF,MAAM,wBAAwB,eAAsC;GAClE,MAAM,IAAI,MAAM,OAAO;AACvB,OAAI,MAAM,UAAW,QAAO;AAC5B,OAAI,MAAM,YAAa,QAAO;AAC9B,UAAO;IACP;EAEF,MAAM,kBAAkB,eACtB,sBACE,MAAM,OAAO,YACb,MAAM,OAAO,mBACb,MAAM,OAAO,GACd,CACF;EAED,MAAM,mBAAmB,eAAe;GACtC,MAAM,KAAK,MAAM,OAAO;AACxB,UAAO,KAAK,iBAAiB,GAAG,CAAC,YAAY;IAC7C;EACF,MAAM,kBAAkB,eAAe;GACrC,MAAM,KAAK,MAAM,OAAO;AACxB,UAAO,KAAK,iBAAiB,GAAG,CAAC,WAAW;IAC5C;EAEF,MAAM,cAAc,eAAe;AACjC,UAAO,CAAC,CAAC,MAAM,OAAO,aAAa,CAAC,CAAC,MAAM,OAAO;IAClD;EAEF,MAAM,gBAAgB,eAAe,0BAA0B,MAAM,OAAO,aAAa,CAAC;EAE1F,MAAM,kBAAkB,eACtB,MAAM,OAAO,YAAY,iBAAiB,MAAM,OAAO,UAAW,CAAC,YAAY,GAChF;EACD,MAAM,iBAAiB,eACrB,MAAM,OAAO,YAAY,iBAAiB,MAAM,OAAO,UAAW,CAAC,WAAW,GAC/E;EAED,MAAM,qBAAqB,eACzB,MAAM,OAAO,eAAe,iBAAiB,MAAM,OAAO,aAAc,CAAC,YAAY,GACtF;EACD,MAAM,oBAAoB,eACxB,MAAM,OAAO,eAAe,iBAAiB,MAAM,OAAO,aAAc,CAAC,WAAW,GACrF;;uBA1IC,mBAuEM,OAvEN,cAuEM,CAtEJ,mBAkDM,OAlDN,cAkDM;IAjDJ,YAKgB,uBAAA,EALD,OAAM,UAAQ,EAAA;4BAIzB,CAHF,YAGE,oCAAA;MAFC,mBAAiB,sBAAA;MAClB,MAAK;;;;IAGT,YAEgB,uBAAA,EAFD,OAAM,QAAM,EAAA;4BAC+B,CAAxD,YAAwD,gCAAA;MAA/B,MAAM,QAAA,OAAO;MAAM,MAAK;;;;IAEnD,YAEgB,uBAAA,EAFD,OAAM,YAAU,EAAA;4BACuC,CAApE,YAAoE,oCAAA;MAAvC,UAAU,QAAA,OAAO;MAAU,MAAK;;;;IAE/D,YAEgB,uBAAA,EAFD,OAAM,aAAW,EAAA;4BACM,CAAA,gCAAjC,QAAA,OAAO,wBAAuB,EAAA,EAAA,CAAA,CAAA;;;IAEnC,mBASM,OATN,cASM,CARJ,YAOgB,uBAAA,EAPD,OAAM,eAAa,EAAA;KAIrB,OAAK,cACkE,CAAA,GAAA,OAAA,OAAA,OAAA,KAAA,CAAhF,mBAAgF,QAAA,EAA1E,OAAM,uCAAqC,EAAC,2BAAuB,GAAA,CAAA,EAAA,CAAA;4BAFvE,CAFK,QAAA,OAAO,aAAa,MAAI,IAAA,WAAA,EAAjC,mBAEI,KAFJ,cAEI,gBADC,QAAA,OAAO,YAAW,EAAA,EAAA,IAAA,mBAAA,QAAA,KAAA,CAAA,CAAA;;;IAO3B,YAEgB,uBAAA;KAFD,OAAM;KAAa,UAAU;;4BACrB,CAAA,gCAAlB,gBAAA,MAAe,EAAA,EAAA,CAAA,CAAA;;;IAEpB,YAGgB,uBAAA,EAHD,OAAM,WAAS,EAAA;4BACN,CAAA,gCAAnB,iBAAA,MAAgB,GAAG,KACtB,EAAA,EAAY,gBAAA,SAAA,WAAA,EAAZ,mBAAwF,QAAxF,cAA0D,OAAE,gBAAG,gBAAA,MAAe,EAAA,EAAA,IAAA,mBAAA,QAAA,KAAA,CAAA,CAAA;;;IAEhE,YAAA,SAAA,WAAA,EACd,YAEgB,uBAAA;;KAFD,OAAM;;4BACA,CAAA,gCAAhB,cAAA,MAAa,EAAA,EAAA,CAAA,CAAA;;;IAGJ,QAAA,OAAO,aAAA,WAAA,EACrB,YAGgB,uBAAA;;KAHD,OAAM;;4BACE,CAAA,gCAAlB,gBAAA,MAAe,GAAG,KACrB,EAAA,EAAA,mBAAgE,QAAhE,cAAmC,OAAE,gBAAG,eAAA,MAAc,EAAA,EAAA,CAAA,CAAA;;;IAG1C,QAAA,OAAO,gBAAA,WAAA,EACrB,YAGgB,uBAAA;;KAHD,OAAM;;4BACK,CAAA,gCAArB,mBAAA,MAAkB,GAAG,KACxB,EAAA,EAAA,mBAAmE,QAAnE,cAAmC,OAAE,gBAAG,kBAAA,MAAiB,EAAA,EAAA,CAAA,CAAA;;;OAIpD,QAAA,SAAA,WAAA,EAAX,mBAkBM,OAlBN,cAkBM,CAjBJ,mBAgBS,UAAA;IAhBD,MAAK;IAAS,OAAM;IAAc,SAAK,OAAA,OAAA,OAAA,MAAA,WAAEA,KAAAA,MAAK,OAAA;qCACpD,mBAaM,OAAA;IAZJ,OAAM;IACN,MAAK;IACL,SAAQ;IACR,gBAAa;IACb,QAAO;IACP,OAAM;OAEN,mBAIE,QAAA;IAHA,kBAAe;IACf,mBAAgB;IAChB,GAAE;6BAEA,UAER,GAAA,CAAA,EAAA,CAAA,CAAA,CAAA,IAAA,mBAAA,QAAA,KAAA,CAAA,CAAA;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;ECYN,MAAM,QAAQ;EAEd,MAAM,aAAa,IAAI,MAAM;EAE7B,MAAM,uBAAuB,IAAoB,KAAK;AAEtD,cACQ,MAAM,OAAO,UACb;AACJ,wBAAqB,QAAQ;IAEhC;EAED,MAAM,eAAe,eACb,qBAAqB,SAAS,MAAM,OAAO,mBAAmB,KACrE;EAED,MAAM,EAAE,QAAQ,uBAAuB,aAIpC,KAAK,OAAO,IAAI,eAAe,mBAAmB,GAAG,EAAE,EACxD,YAAY,sBACb,CAAC;EAEF,eAAe,eAAe;AAC5B,OAAI;AACF,eAAW,QAAQ;IACnB,MAAM,SAAS,MAAM,mBAAmB,MAAM,OAAO,GAAG;AACxD,yBAAqB,QAAQ,OAAO;AACpC,UAAM,QACJ,OAAO,aACH,0CACA,yCACL;YACM,GAAG;AACV,UAAM,MAAM,uBAAuB,GAAG,gCAAgC,CAAC;aAC/D;AACR,eAAW,QAAQ;;;;UAtHb,QAAA,UAAA,WAAA,EADR,mBAgEM,OAhEN,cAgEM,CA5DJ,mBA2DM,OA3DN,cA2DM;IA1DJ,mBAsCM,OAAA,EArCJ,OAAK,eAAA,CAAC,gEACa,aAAA,QAAY,yBAAA,mCAAA,CAAA,EAAA,GAI/B,mBAAA,sBAA0B,EAElB,aAAA,SAAA,WAAA,EADR,mBAcM,OAdN,cAcM,CAAA,GAAA,OAAA,OAAA,OAAA,KAAA,CALJ,mBAIE,QAAA;KAHA,kBAAe;KACf,mBAAgB;KAChB,GAAE;sCAIN,mBAcM,UAAA,EAAA,KAAA,GAAA,EAAA,CAfN,mBAAA,mCAAuC,EAAA,OAAA,OAAA,OAAA,KACvC,mBAcM,OAAA;KAZJ,OAAM;KACN,OAAM;KACN,MAAK;KACL,SAAQ;KACR,QAAO;KACP,gBAAa;QAEb,mBAIE,QAAA;KAHA,kBAAe;KACf,mBAAgB;KAChB,GAAE;;IAIR,mBAMI,KANJ,cAMI,gBAJA,aAAA,QAAA,gDAAA,uDAAA;IAKJ,mBAWS,UAAA;KAVP,MAAK;KACJ,OAAK,eAAA,CAAA,mCAA2D,aAAA,QAAY,gBAAA,WAAA,CAAA;KAI5E,UAAU,WAAA;KACV,SAAO;QAEI,WAAA,SAAA,WAAA,EAAZ,mBAA0E,QAA1E,aAA0E,KAAA,WAAA,EAC1E,mBAAoE,QAAA,cAAA,gBAApD,aAAA,QAAY,gBAAA,YAAA,EAAA,EAAA,EAAA,EAAA,IAAA,aAAA;;;;;;;;;ACrDpC,MAAa,uCAAuC,aAClD,mCACA,gCACA;CACE,OAAO;EACL,OAAO;EACP,aAAa;EACd;CACD,aAAa;EACX,OAAO;EACP,WAAW;EACX,aAAa;EACd;CACD,MAAM,EAAE,OAAO,QAAQ;CACvB,UAAU;EACR,OAAO;EACP,WAAW;EACX,cAAc;EACd,WAAW;EACZ;CACF,CACF;;;;;;;;;;;;;;;;;;;;;;;;;ECkCD,MAAM,QAAQ;EAEd,MAAM,OAAO;EAKb,MAAM,EAAE,QAAQ,cAAc,SAAS,aAAa,aACjD,KAAK,UAA0C,IAAI,eAAe,aAAa,MAAM,EACtF,EAAE,YAAY,sBAAsB,CACrC;EAED,MAAM,SAAS,eAAe,MAAM,UAAU,SAAS,MAAM;EAE7D,MAAM,EAAE,MAAM,OAAO,WAAW,mBAAmB,2BAA2B,QAC5E,sCACA;GACE,WAAW,+BAA+B,MAAM,OAAO;GACvD,uBAAuB;GACvB,aAAa;GACb,WAAW,YAAY;IACrB,MAAM,IAAI,MAAM;AAChB,WAAO;KACL,IAAI,EAAE;KACN,OAAO,EAAE,SAAS;KAClB,aAAa,EAAE,eAAe;KAC9B,MAAM,EAAE;KACR,UAAU,8BAA8B,EAAE,SAAS;KACpD;;GAEJ,CACF;EAED,SAAS,aAAa,UAAmE;AACvF,UAAO;IACL,IAAI,MAAM,OAAO;IACjB,OAAO,OAAO,SAAS,SAAS,GAAG,CAAC,MAAM;IAC1C,aAAc,SAAS,eAA0B;IACjD,MAAM,SAAS;IACf,UAAU,SAAS;IACpB;;EAGH,eAAe,aAAa,UAAmC;AAE7D,SAAM,aADU,aAAa,SAAS,CACX;;EAI7B,SAAS,gBAAgB;AACvB,QAAK,UAAU;;EAGjB,SAAS,YAAY,OAAwB;GAC3C,MAAM,UAAU,uBAAuB,OAAO,0BAA0B;AACxE,QAAK,eAAe,QAAQ;;EAG9B,SAAS,eAAe;AACtB,QAAK,SAAS;;;uBA1Hd,mBA2CM,OA3CN,cA2CM,CA1CJ,YAyCY,MAAA,UAAA,EAAA;IAxCT,gBAAe;IACf,WAAS;IACT,SAAO;IACR,OAAM;IACN,UAAS;;2BAsBH;KApBN,mBAoBM,OApBN,cAoBM;MAnBJ,mBAAA,uBAA2B;MAC3B,mBAEM,OAFN,cAEM,CADJ,YAAqD,MAAA,MAAA,CAAA,YAAA,EAAnC,aAAY,sBAAoB,CAAA,CAAA,CAAA;MAGpD,mBAAA,6BAAiC;MACjC,mBAEM,OAFN,cAEM,CADJ,YAAqF,MAAA,MAAA,CAAA,kBAAA;OAA7D,OAAM;OAAS,aAAY;;MAGrD,mBAAA,SAAa;MACb,mBAEM,OAFN,cAEM,CADJ,YAAmB,MAAA,MAAA,CAAA,UAAA,CAAA,CAAA;MAGrB,mBAAA,aAAiB;MACjB,mBAEM,OAFN,cAEM,CADJ,YAAuB,MAAA,MAAA,CAAA,cAAA,CAAA,CAAA;;KAIhB,MAAA,KAAI,CAAC,eAAA,WAAA,EAAhB,mBAEM,OAFN,cAEM,CADJ,mBAAmD,QAAnD,cAAmD,gBAA1B,MAAA,KAAI,CAAC,YAAW,EAAA,EAAA,CAAA,CAAA,IAAA,mBAAA,QAAA,KAAA;KAE3C,YAAmE,MAAA,uBAAA,EAAA,EAA3C,OAAM,oCAAkC,CAAA;KAEhE,mBAAA,mBAAuB;KACvB,mBAKM,OALN,cAKM,CAJJ,mBAES,UAAA;MAFD,MAAK;MAAS,OAAM;MAAiB,SAAO;MAAe,UAAU,OAAA;QAAQ,YAErF,GAAA,cAAA,EACA,YAAkE,MAAA,kBAAA,EAAA;MAA/C,YAAW;MAAO,gBAAe;;;;;;;;;;;;;;;;;;;;;;;;;;;ECyB5D,MAAM,QAAQ;EAEd,MAAM,eAAe,IAAmD,KAAK;EAM7E,MAAM,sBAAsB,eAA+B;GACzD,MAAMC,QAAwB,EAAE;AAEhC,QAAK,MAAM,QAAQ,MAAM,SAAS,EAAE,CAClC,OAAM,KAAK;IACT,MAAM;IACN,WAAW,KAAK;IAChB,IAAI,QAAQ,KAAK,UAAU,GAAG,KAAK;IACnC,MAAM;IACP,CAAC;AAGJ,QAAK,MAAM,SAAS,MAAM,gBAAgB,EAAE,CAC1C,OAAM,KAAK;IACT,MAAM;IACN,WAAW,MAAM;IACjB,IAAI,MAAM;IACV,MAAM;IACP,CAAC;AAGJ,SAAM,MAAM,GAAG,MAAM,IAAI,KAAK,EAAE,UAAU,CAAC,SAAS,GAAG,IAAI,KAAK,EAAE,UAAU,CAAC,SAAS,CAAC;AACvF,UAAO;IACP;EAEF,SAAS,aAAa;AACpB,gBAAa,OAAO,OAAO;;AAG7B,WAAa,EAAE,YAAY,CAAC;;uBAtG1B,mBAmCM,OAnCN,cAmCM;IAlCJ,YAKE,2BAAA;cAJI;KAAJ,KAAI;KACH,oBAAkB;KAClB,UAAU,QAAA,kBAAc;KACxB,aAAW,QAAA;;IAGN,oBAAA,MAAoB,WAAM,KAAA,WAAA,EADlC,mBAKI,KALJ,cAGC,qBAED,IAAA,mBAAA,QAAA,KAAA;sBACA,mBAqBW,UAAA,MAAA,WArBc,oBAAA,QAAR,SAAI;6DAA+B,KAAK,IAAA,GAE/C,KAAK,SAAI,mBAAA,WAAA,EADjB,YAOe,sBAAA;;MALZ,eAAa,KAAK,KAAK;MACvB,cAAY,KAAK,KAAK;MACvB,SAAQ;;6BAEkB,CAAA,gCAAvB,KAAK,KAAK,QAAI,GAAA,EAAA,EAAA,CAAA,CAAA;;gDAGN,KAAK,SAAI,kBAAA,WAAA,EADtB,YAWE,6BAAA;;MATC,QAAQ,KAAK,KAAK;MAClB,SAAS,KAAK,KAAK;MACnB,WAAW,KAAK,KAAK;MACrB,QAAQ,KAAK,KAAK;MAClB,MAAM,KAAK,KAAK;MAChB,SAAS,KAAK,KAAK;MACnB,aAAW,KAAK,KAAK;MACrB,aAAW,KAAK,KAAK;MACrB,SAAS,KAAK,KAAK;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;ECqF5B,MAAM,QAAQ;EAEd,MAAM,QAAQ,UAAU;EACxB,MAAM,SAAS,WAAW;EAC1B,MAAM,oBAAoB,MAAM,OAAO;EAGvC,MAAM,aAAa,eAAe,MAAM,MAAM,SAAS,OAAO;EAG9D,MAAM,gBAAgB,OAA0C,gBAAgB;EAGhF,MAAM,mBAAmB,eAAe;AACtC,OAAI,CAAC,MAAM,OAAQ,QAAO;AAC1B,UAAO,CAAC,CAAC,MAAM,OAAO;IACtB;EAGF,MAAM,EACJ,MAAM,cACN,OAAO,eACP,SAAS,oBACP,SACF,OAAO,QAAQ;GACb,MAAM,CAAC,eAAe,kBAAkB,MAAM,QAAQ,IAAI,CACxD,IAAI,MAAM,SAAS;IACjB,WAAW;KAAE,UAAU,UAAU;KAAQ,OAAO;KAAmB;IACnE,aAAa;KAAE,UAAU,UAAU;KAAQ,OAAO;KAAkB;IACpE,aAAa;KAAE,UAAU,UAAU;KAAQ,OAAO;KAAO;IACzD,OAAO;IACP,QAAQ;IACR,eAAe;IAChB,CAAC,EACF,IAAI,eAAe,2BACjB,mBACA,YAAY,gBACZ;IACE,cAAc,CAAC,YAAY,gBAAgB,YAAY,wBAAwB;IAC/E,OAAO;IACP,QAAQ;IACR,eAAe;IAChB,CACF,CACF,CAAC;AACF,UAAO;IAAE;IAAe;IAAgB;KAE1C;GACE,SAAS,CAAC,CAAC,qBAAqB,CAAC,CAAC,MAAM;GACxC,WAAW,WAAW;GACtB,gBAAgB;GACjB,CACF;EASD,MAAM,2BAA2B,eAA0C;AAEzE,WADc,aAAa,OAAO,eAAe,SAAS,EAAE,EAEzD,KAAK,OAAO;IACX,YAAY,EAAE,2BAA2B,EAAE,cAAc;IACzD,WAAW,EAAE;IACb,MAAM,EAAE,QAAQ;IACjB,EAAC,CACD,MAAM,GAAG,MAAM,IAAI,KAAK,EAAE,UAAU,CAAC,SAAS,GAAG,IAAI,KAAK,EAAE,UAAU,CAAC,SAAS,CAAC;IACpF;EAGF,MAAM,kBAAkB,eAAe;GACrC,MAAM,OAAO,aAAa,OAAO;GACjC,MAAM,WAAW,MAAM,SAAS,EAAE;GAClC,MAAM,iBAAiB,MAAM;AAE7B,UAAO,oBAAoB,UADR,iBAAiB,IAAI,IAAI,OAAO,QAAQ,eAAe,CAAC,GAAG,OAC9B;IAChD;EAGF,MAAM,uBAAuB,eAA8B;AACzD,OAAI,CAAC,MAAM,OAAQ,QAAO,EAAE;GAE5B,MAAM,SAAS,gBAAgB;GAC/B,MAAM,YAAY,MAAM,OAAO,WAAW,aAAa,CAAC,MAAM,OAAO;GAGrE,MAAM,oBAAoB,IAAI,IAAI;IAAC;IAAiB;IAAmB;IAAW,CAAC;AAEnF,UAAO,OAAO,QAAQ,UAAU;AAE9B,QAAI,MAAM,WAAW;AAEnB,SAAI,kBAAkB,IAAI,MAAM,UAAU,CACxC,QAAO;AAIT,SAAI,aAAa,MAAM,cAAc,eACnC,QAAO;AAGT,YAAO;;IAIT,MAAM,UAAU,MAAM,QAAQ,aAAa;AAG3C,QACE,QAAQ,SAAS,gBAAgB,IACjC,QAAQ,SAAS,QAAQ,IACzB,QAAQ,SAAS,kBAAkB,IACnC,QAAQ,SAAS,aAAY,CAE7B,QAAO;AAIT,QAAI,cAAc,QAAQ,SAAS,UAAU,IAAI,QAAQ,SAAS,SAAS,EACzE,QAAO;AAIT,WAAO;KACP;IACF;EAGF,MAAM,EAAE,QAAQ,eAAe,aAC5B,KAAK,UAAyB,IAAI,MAAM,WAAW,MAAM,EAC1D,EAAE,YAAY,YAAY,CAC3B;EAGD,eAAe,cAAc,SAG1B;AACD,OAAI;AACF,UAAM,WAAW;KACf,WAAW;KACX,aAAa;KACb,MAAM,QAAQ;KACd,KAAK;KACL,aAAa;KACd,CAAC;AACF,UAAM,iBAAiB;AACvB,UAAM,QAAQ,gBAAgB;WACxB;AACN,UAAM,MAAM,wBAAwB;AACpC,UAAM,IAAI,MAAM,wBAAwB;;;EAK5C,SAAS,gBAAgB;AACvB,UAAO,KAAK;IACV,MAAM,MAAM,QAAQ;IACpB,QAAQ,MAAM;IACd,OAAO;KAAE,GAAG,MAAM;KAAO,MAAM;KAAQ;IACxC,CAAC;;EAIJ,eAAe,eAAe;AAC5B,SAAM,OAAO,KAAK;IAChB,MAAM,MAAM,QAAQ;IACpB,QAAQ,MAAM;IACd,OAAO;KAAE,GAAG,MAAM;KAAO,MAAM;KAAW;IAC3C,CAAC;;EAIJ,eAAe,4BAA4B;AACzC,SAAM,cAAc;AACpB,SAAM,UAAU;AAChB,SAAM,QAAQ,8BAA8B;AAC5C,OAAI,cACF,OAAM,eAAe;AAEvB,SAAM,iBAAiB;;;;2DA5SvB,mBAAA,wEAA4E,EACjE,QAAA,aAAS,CAAK,QAAA,UAAA,WAAA,EAAzB,mBAEM,OAFN,YAEM,CAAA,GAAA,OAAA,OAAA,OAAA,KAAA,CADJ,mBAAwD,QAAA,EAAlD,OAAM,sCAAoC,EAAA,MAAA,GAAA,CAAA,EAAA,CAAA,IAIlC,QAAA,SAAA,WAAA,EAAhB,mBAsBM,UAAA,EAAA,KAAA,GAAA,EAAA,CAvBN,mBAAA,gBAAoB,EACpB,mBAsBM,OAtBN,YAsBM,CAAA,OAAA,OAAA,OAAA,KArBJ,mBAYM,OAAA;IAXJ,OAAM;IACN,OAAM;IACN,MAAK;IACL,SAAQ;OAER,mBAKE,QAAA;IAJA,kBAAe;IACf,mBAAgB;IAChB,gBAAa;IACb,GAAE;cAGN,mBAOM,OAPN,YAOM,CANJ,mBAAuD,QAAA,MAAA,gBAA9C,QAAA,MAAM,WAAO,oBAAA,EAAA,EAAA,EACtB,mBAIM,OAJN,YAIM,CAHJ,YAEc,wBAAA;IAFA,IAAI,EAAA,MAAA,6BAAqC;IAAE,OAAM;;2BAE/D,CAAA,GAAA,OAAA,OAAA,OAAA,KAAA,CAAA,gBAFmF,4BAEnF,GAAA,CAAA,EAAA,CAAA;;uBAMU,QAAA,UAAA,WAAA,EAAhB,mBA4DM,UAAA,EAAA,KAAA,GAAA,EAAA,CA7DN,mBAAA,iBAAqB,EACrB,mBA4DM,OA5DN,YA4DM;IA3DJ,mBAAA,WAAe;IACf,mBAIM,OAJN,YAIM,CAHJ,mBAEK,MAFL,YAEK,gBADA,QAAA,OAAO,MAAK,EAAA,EAAA,CAAA,CAAA;IAInB,mBAAA,kBAAsB;IACtB,mBAEM,OAFN,YAEM,CADJ,YAAyC,8BAAA,EAAlB,QAAQ,QAAA,QAAM,EAAA,MAAA,GAAA,CAAA,SAAA,CAAA,CAAA,CAAA;IAGvC,mBAAA,4BAAgC;IAChC,mBAEM,OAFN,YAEM,CADJ,YAA0D,+CAAA,EAAlB,QAAQ,QAAA,QAAM,EAAA,MAAA,GAAA,CAAA,SAAA,CAAA,CAAA,CAAA;IAGxD,mBAAA,oCAAwC;IACxC,mBAQM,OARN,aAQM,CAAA,CAPyB,WAAA,SAAA,WAAA,EAA7B,YAAkF,8BAAA;;KAAxC,QAAQ,QAAA;KAAS,QAAM;6CACjE,YAKE,kCAAA;;KAHC,QAAQ,QAAA;KACR,WAAS;KACT,UAAQ;;IAIb,mBAAA,mFAAuF;IACvF,mBAQM,OARN,aAQM,CAPJ,YAME,6CAAA;KALC,aAAW,QAAA,OAAO;KAClB,QAAQ,iBAAA;KACR,UAAQ,CAAG,iBAAA;KACX,YAAU,MAAA,gBAAe;KACzB,WAAS,MAAA,gBAAe;;;;;;;;IAI7B,mBAAA,yFAA6F;IAC7F,mBAkBM,OAAA,MAAA,CAhBI,MAAA,aAAY,IAAA,QAAA,WAAA,EADpB,YAOE,0BAAA;;KALC,QAAQ,QAAA;KACR,OAAO,yBAAA;KACP,iBAAe,qBAAA;KACf,mBAAiB,iBAAA;KACjB,eAAa;;;;;;UAEA,MAAA,cAAa,IAAA,WAAA,EAA7B,mBAKM,OALN,aAKM,CAJJ,mBAAsE,QAAA,MAAA,gBAA7D,MAAA,cAAa,EAAE,WAAO,0BAAA,EAAA,EAAA,EAC/B,mBAES,UAAA;KAFD,OAAM;KAA+B,SAAK,OAAA,OAAA,OAAA,MAAA,GAAA,SAAE,MAAA,gBAAA,IAAA,MAAA,gBAAA,CAAA,GAAA,KAAe;OAAE,UAErE,CAAA,CAAA,KAAA,WAAA,EAEF,mBAEM,OAFN,aAEM,CAAA,GAAA,OAAA,OAAA,OAAA,KAAA,CADJ,mBAAwD,QAAA,EAAlD,OAAM,sCAAoC,EAAA,MAAA,GAAA,CAAA,EAAA,CAAA,EAAA,CAAA"}
@@ -3,9 +3,9 @@ import "./EnhancedRefreshTokenHandler-C6tZCcfX.js";
3
3
  import "./useQueryCache-alzaRWEb.js";
4
4
  import "./useMutation-BLNuJoYl.js";
5
5
  import "./useQuery-BzUGEOj0.js";
6
- import { x as CustomerSupportTicketList_default } from "./src-Bx44Vnfl.js";
6
+ import { S as CustomerSupportTicketList_default } from "./src-zjaOyP9b.js";
7
7
  import "./AppLink-FcNGKgvG.js";
8
- import "./saved_filter-B0MGrzy2.js";
8
+ import "./saved_filter-jeZd2rlb.js";
9
9
  import "./ConfirmDialog-DjthOYU6.js";
10
10
  import "./InlineAttachments-DAn_QknY.js";
11
11
  import "./TeamMembersTab-BigqpBDH.js";
@@ -20,31 +20,31 @@ import "./UserProfilePage-FNLYK9kj.js";
20
20
  import "./ChangePasswordPage-nr0B06HB.js";
21
21
  import "./teamMetadata-26Mwjb2i.js";
22
22
  import "./team_memberRoutes-Cxgte_vj.js";
23
- import "./teamRoutes-BYpld9yI.js";
24
- import "./CreateTeamForm-CZRTmCOx.js";
25
- import "./EditTeamForm-DYvHGBRF.js";
23
+ import "./teamRoutes-CtNcFZjR.js";
24
+ import "./CreateTeamForm-DRfZ74on.js";
25
+ import "./EditTeamForm-BEOkUaKG.js";
26
26
  import "./TeamHistoryTab-p3hDxCc3.js";
27
- import "./TeamList-CBJOtPQR.js";
27
+ import "./TeamList-gppM0GOD.js";
28
28
  import "./TeamNotesTab-DPw9YEwK.js";
29
- import "./TeamParent-CYKznys5.js";
30
- import "./ViewTeam-BgZzH7lK.js";
29
+ import "./TeamParent-YPtenk3l.js";
30
+ import "./ViewTeam-CRmIplCt.js";
31
31
  import "./teamMemberMetadata-DX0W-B7p.js";
32
32
  import "./CreateTeamMemberForm-ITp4XFn9.js";
33
33
  import "./EditTeamMemberForm-Fyf8Zxfh.js";
34
34
  import "./TeamMemberList-D0-dM5kI.js";
35
35
  import "./TeamMemberParent-CJGWXjuM.js";
36
36
  import "./ViewTeamMember-Cf5yXdv6.js";
37
- import "./customerSupportTicketRoutes-DZ6mSNIQ.js";
38
- import "./staffSupportTicketRoutes-CKdCYj3S.js";
37
+ import "./customerSupportTicketRoutes-Cy4fp4wx.js";
38
+ import "./staffSupportTicketRoutes-L4CU5dcu.js";
39
39
  import "./TimelineSystemEvent-Ch1sZiyO.js";
40
- import "./CustomerCreateSupportTicketForm-BfEiv6I8.js";
40
+ import "./CustomerCreateSupportTicketForm-Co6C_P5o.js";
41
41
  import "./CustomerSupportTicketParent-BaKfkSlU.js";
42
- import "./CustomerSupportTicketSuccess-CmXVwmtE.js";
43
- import "./StaffCreateSupportTicketForm-LuHzqt2w.js";
42
+ import "./CustomerSupportTicketSuccess-DVqoR5-o.js";
43
+ import "./StaffCreateSupportTicketForm-Cm595v_4.js";
44
44
  import "./SupportTicketDevLifecycleBadge-BYKZjEv6.js";
45
45
  import "./StaffSupportTicketParent-yoC-_Lku.js";
46
- import "./StaffSupportTicketSuccess-BoQZY9Qt.js";
47
- import "./LoginForm-DPmrJ_wm.js";
46
+ import "./StaffSupportTicketSuccess-DgULDGIj.js";
47
+ import "./LoginForm-D1Mx2vAY.js";
48
48
  import "./useEmailVerificationChannel-QuMSgzzM.js";
49
49
  import "./Signup-BCVZZCR_.js";
50
50
  import "./ForgotPassword-Dd-E3_o1.js";
@@ -53,12 +53,12 @@ import "./Logout-Bdktl4NZ.js";
53
53
  import "./mfaSchema-C6PatIbY.js";
54
54
  import "./MfaSetup-Bjc3v0hs.js";
55
55
  import "./MfaVerify-uJlPz8xg.js";
56
- import "./VerifyEmail-CEbvnWLl.js";
56
+ import "./VerifyEmail-CWUhRA1o.js";
57
57
  import "./UserListPage-A0_eNpQ1.js";
58
58
  import "./CreateUserPage-1WiLNGr_.js";
59
59
  import "./EditUserPage-BBzGmOrx.js";
60
- import "./CreditTransactionHistory-JVPQPiLB.js";
61
- import "./CreditBalanceDashboard-BHLCoCbK.js";
62
- import "./CreditManagement-C_Va9qmV.js";
60
+ import "./CreditTransactionHistory-plIaRscn.js";
61
+ import "./CreditBalanceDashboard-BRY56-9w.js";
62
+ import "./CreditManagement-D-bsc1US.js";
63
63
 
64
64
  export { CustomerSupportTicketList_default as default };
@@ -3,10 +3,10 @@ import "./EnhancedRefreshTokenHandler-C6tZCcfX.js";
3
3
  import "./useQueryCache-alzaRWEb.js";
4
4
  import "./useMutation-BLNuJoYl.js";
5
5
  import "./useQuery-BzUGEOj0.js";
6
- import "./saved_filter-B0MGrzy2.js";
6
+ import "./saved_filter-jeZd2rlb.js";
7
7
  import "./ConfirmDialog-DjthOYU6.js";
8
8
  import "./userAuthorized-klLUHGxT.js";
9
- import "./customerSupportTicketRoutes-DZ6mSNIQ.js";
10
- import { t as CustomerSupportTicketSuccess_default } from "./CustomerSupportTicketSuccess-CmXVwmtE.js";
9
+ import "./customerSupportTicketRoutes-Cy4fp4wx.js";
10
+ import { t as CustomerSupportTicketSuccess_default } from "./CustomerSupportTicketSuccess-DVqoR5-o.js";
11
11
 
12
12
  export { CustomerSupportTicketSuccess_default as default };
@@ -1,4 +1,4 @@
1
- import { t as customerSupportPaths } from "./customerSupportTicketRoutes-DZ6mSNIQ.js";
1
+ import { t as customerSupportPaths } from "./customerSupportTicketRoutes-Cy4fp4wx.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-CmXVwmtE.js.map
54
+ //# sourceMappingURL=CustomerSupportTicketSuccess-DVqoR5-o.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"CustomerSupportTicketSuccess-CmXVwmtE.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"}
1
+ {"version":3,"file":"CustomerSupportTicketSuccess-DVqoR5-o.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"}
@@ -0,0 +1,207 @@
1
+ import "./useRpcAuth-CJtq1dqM.js";
2
+ import "./EnhancedRefreshTokenHandler-C6tZCcfX.js";
3
+ import "./useQueryCache-alzaRWEb.js";
4
+ import { t as useMutation } from "./useMutation-BLNuJoYl.js";
5
+ import { t as useQuery } from "./useQuery-BzUGEOj0.js";
6
+ import { t as extractRpcErrorMessage } from "./extractRpcErrorMessage-CAaeVysa.js";
7
+ import { Fragment, computed, createCommentVNode, createElementBlock, createElementVNode, createTextVNode, createVNode, defineComponent, openBlock, reactive, ref, renderList, toDisplayString, unref, vModelText, watch, withCtx, withDirectives, withKeys, withModifiers } from "vue";
8
+ import { toast } from "vue3-toastify";
9
+ import { DefaultReferralTeamFormSchema } from "@dragonmastery/dragoncore-shared";
10
+ import { useForm, withMetadata } from "@dragonmastery/zinia-forms-core";
11
+
12
+ //#region src/slices/admin/features/default_referral_team/defaultReferralTeamSchema.ts
13
+ const defaultReferralTeamSchemaWithMetadata = withMetadata(DefaultReferralTeamFormSchema, "defaultReferralTeamSchema", { default_team_id: {
14
+ inputType: "select",
15
+ label: "Default team",
16
+ placeholder: "Select team",
17
+ helpText: "When a referral event arrives with an unknown tag, it will be routed to this team. Select \"None\" to reject unknown tags."
18
+ } });
19
+
20
+ //#endregion
21
+ //#region src/slices/admin/features/default_referral_team/DefaultReferralTeamPage.vue
22
+ const _hoisted_1 = { class: "mt-2" };
23
+ const _hoisted_2 = {
24
+ key: 0,
25
+ class: "flex justify-center py-8"
26
+ };
27
+ const _hoisted_3 = {
28
+ key: 0,
29
+ class: "alert alert-error mt-2"
30
+ };
31
+ const _hoisted_4 = { class: "mt-4 flex justify-end" };
32
+ const _hoisted_5 = { class: "mt-12" };
33
+ const _hoisted_6 = {
34
+ key: 0,
35
+ class: "overflow-x-auto"
36
+ };
37
+ const _hoisted_7 = { class: "table table-zebra" };
38
+ const _hoisted_8 = { class: "flex items-center gap-1" };
39
+ const _hoisted_9 = ["onUpdate:modelValue", "onKeydown"];
40
+ const _hoisted_10 = ["onClick"];
41
+ const _hoisted_11 = ["disabled", "onClick"];
42
+ const _hoisted_12 = {
43
+ key: 0,
44
+ class: "loading loading-spinner loading-sm"
45
+ };
46
+ const _hoisted_13 = {
47
+ key: 1,
48
+ class: "text-base-content/60"
49
+ };
50
+ const _sfc_main = /* @__PURE__ */ defineComponent({
51
+ __name: "DefaultReferralTeamPage",
52
+ setup(__props) {
53
+ const { data: defaultTeamIdData, loading: defaultLoading, refetch: refetchDefault } = useQuery((api) => api.appSettings.getDefaultTeamIdForUnknownReferrals(), {
54
+ cacheKey: "admin-default-referral-team",
55
+ staleTime: 30 * 1e3
56
+ });
57
+ const { data: teamsData, loading: teamsLoading, refetch: refetchTeams } = useQuery((api) => api.teams.listTeams({ first: 100 }), {
58
+ cacheKey: "admin-teams-for-default-referral",
59
+ staleTime: 60 * 1e3
60
+ });
61
+ const referralTagInputs = reactive({});
62
+ const savingTeamId = ref(null);
63
+ watch(() => teamsData.value?.items ?? [], (teams) => {
64
+ for (const team of teams) referralTagInputs[team.id] = team.referral_tag ?? "";
65
+ }, { immediate: true });
66
+ const loadTeams = async () => {
67
+ while (teamsLoading.value) await new Promise((resolve) => setTimeout(resolve, 100));
68
+ return [{
69
+ value: "",
70
+ label: "None (reject unknown tags)"
71
+ }, ...(teamsData.value?.items ?? []).map((team) => ({
72
+ value: team.id,
73
+ label: `${team.display_name || team.unique_name || team.id}${team.referral_tag ? ` (${team.referral_tag})` : ""}`
74
+ }))];
75
+ };
76
+ const { form, zinia, ZiniaForm, ZiniaSubmitButton } = useForm(defaultReferralTeamSchemaWithMetadata, {
77
+ storeName: "admin-default-referral-team-form",
78
+ persistToLocalStorage: false,
79
+ renderStyle: "daisy_ui",
80
+ fetchData: async () => {
81
+ while (defaultLoading.value) await new Promise((resolve) => setTimeout(resolve, 100));
82
+ return { default_team_id: defaultTeamIdData.value?.default_team_id ?? null };
83
+ },
84
+ dataLoaders: { teams: loadTeams }
85
+ });
86
+ const isLoading = computed(() => defaultLoading.value || teamsLoading.value || form.isLoading);
87
+ const { mutate: updateDefaultTeam } = useMutation((api, input) => api.appSettings.updateDefaultTeamIdForUnknownReferrals(input), { invalidate: /admin-default-referral-team/ });
88
+ const { mutate: setReferralTag } = useMutation((api, input) => api.teams.setReferralTag(input), { invalidate: /admin-teams-for-default-referral/ });
89
+ /** Normalize a team name to a URL-friendly referral tag (e.g. "Acme Corp & Co." → "acme-corp-co") */
90
+ function normalizeToReferralTag(name) {
91
+ return name.toLowerCase().trim().replace(/[^a-z0-9\s-]/g, "").replace(/\s+/g, "-").replace(/-+/g, "-").replace(/^-|-$/g, "");
92
+ }
93
+ function normalizeReferralTag(team) {
94
+ const normalized = normalizeToReferralTag(team.display_name || team.unique_name || team.id);
95
+ if (normalized) referralTagInputs[team.id] = normalized;
96
+ }
97
+ async function saveReferralTag(teamId) {
98
+ const raw = referralTagInputs[teamId]?.trim() ?? "";
99
+ const referral_tag = raw === "" ? null : raw;
100
+ const team = teamsData.value?.items?.find((t) => t.id === teamId);
101
+ if (team && referral_tag === (team.referral_tag ?? null)) return;
102
+ savingTeamId.value = teamId;
103
+ try {
104
+ await setReferralTag({
105
+ team_id: teamId,
106
+ referral_tag
107
+ });
108
+ toast.success("Referral tag saved");
109
+ referralTagInputs[teamId] = referral_tag ?? "";
110
+ await refetchTeams();
111
+ } catch (err) {
112
+ toast.error(extractRpcErrorMessage(err, "Failed to save referral tag"));
113
+ } finally {
114
+ savingTeamId.value = null;
115
+ }
116
+ }
117
+ async function handleSubmit(formData) {
118
+ await updateDefaultTeam({ default_team_id: formData.default_team_id?.trim() || null });
119
+ return { ok: true };
120
+ }
121
+ async function handleSuccess() {
122
+ toast.success("Default team saved");
123
+ await refetchDefault();
124
+ }
125
+ function handleError(error) {
126
+ const message = error instanceof Error ? error.message : String(error);
127
+ form.setSubmitError(extractRpcErrorMessage(error, message));
128
+ toast.error(form.submitError ?? "Failed to save");
129
+ }
130
+ return (_ctx, _cache) => {
131
+ return openBlock(), createElementBlock("div", _hoisted_1, [
132
+ _cache[5] || (_cache[5] = createElementVNode("div", { class: "flex justify-between items-center mb-4" }, [createElementVNode("h1", { class: "text-2xl font-bold" }, "Default Team for Unknown Referrals")], -1)),
133
+ _cache[6] || (_cache[6] = createElementVNode("p", { class: "text-base-content/70 mb-4" }, " When a referral event arrives with an unknown referral tag, it will be routed to this team. Select a team to use as the default, or clear to reject unknown tags. ", -1)),
134
+ isLoading.value ? (openBlock(), createElementBlock("div", _hoisted_2, [..._cache[0] || (_cache[0] = [createElementVNode("span", { class: "loading loading-spinner loading-lg" }, null, -1)])])) : (openBlock(), createElementBlock(Fragment, { key: 1 }, [
135
+ createVNode(unref(ZiniaForm), {
136
+ onHandleSubmit: handleSubmit,
137
+ onSuccess: handleSuccess,
138
+ onError: handleError,
139
+ title: "",
140
+ subtitle: ""
141
+ }, {
142
+ default: withCtx(() => [
143
+ createVNode(unref(zinia).DefaultTeamIdField, {
144
+ "select-options": unref(form).extraData.teams ?? [],
145
+ placeholder: "Select team"
146
+ }, null, 8, ["select-options"]),
147
+ unref(form).submitError ? (openBlock(), createElementBlock("div", _hoisted_3, [createElementVNode("span", null, toDisplayString(unref(form).submitError), 1)])) : createCommentVNode("v-if", true),
148
+ createElementVNode("div", _hoisted_4, [createVNode(unref(ZiniaSubmitButton), {
149
+ submitText: "Save",
150
+ submittingText: "Saving..."
151
+ })])
152
+ ]),
153
+ _: 1
154
+ }),
155
+ createCommentVNode(" Referral tags per team "),
156
+ createElementVNode("div", _hoisted_5, [
157
+ _cache[3] || (_cache[3] = createElementVNode("h2", { class: "text-xl font-semibold mb-2" }, "Referral Tags per Team", -1)),
158
+ _cache[4] || (_cache[4] = createElementVNode("p", { class: "text-base-content/70 mb-4" }, " Set a referral tag for each team. Other apps use this tag when pushing events. If empty, the team's unique name or path is used as fallback. ", -1)),
159
+ unref(teamsData)?.items?.length ? (openBlock(), createElementBlock("div", _hoisted_6, [createElementVNode("table", _hoisted_7, [_cache[2] || (_cache[2] = createElementVNode("thead", null, [createElementVNode("tr", null, [
160
+ createElementVNode("th", null, "Team"),
161
+ createElementVNode("th", null, "Referral tag"),
162
+ createElementVNode("th")
163
+ ])], -1)), createElementVNode("tbody", null, [(openBlock(true), createElementBlock(Fragment, null, renderList(unref(teamsData).items, (team) => {
164
+ return openBlock(), createElementBlock("tr", { key: team.id }, [
165
+ createElementVNode("td", null, toDisplayString(team.display_name || team.unique_name || team.id), 1),
166
+ createElementVNode("td", null, [createElementVNode("div", _hoisted_8, [withDirectives(createElementVNode("input", {
167
+ "onUpdate:modelValue": ($event) => referralTagInputs[team.id] = $event,
168
+ type: "text",
169
+ class: "input input-bordered input-sm w-full max-w-xs",
170
+ placeholder: "e.g. my-team",
171
+ onKeydown: withKeys(withModifiers(($event) => saveReferralTag(team.id), ["prevent"]), ["enter"])
172
+ }, null, 40, _hoisted_9), [[vModelText, referralTagInputs[team.id]]]), createElementVNode("button", {
173
+ type: "button",
174
+ class: "btn btn-sm btn-ghost btn-square",
175
+ title: "Normalize to URL-friendly tag from team name",
176
+ onClick: ($event) => normalizeReferralTag(team)
177
+ }, [..._cache[1] || (_cache[1] = [createElementVNode("svg", {
178
+ xmlns: "http://www.w3.org/2000/svg",
179
+ class: "h-4 w-4",
180
+ fill: "none",
181
+ viewBox: "0 0 24 24",
182
+ stroke: "currentColor"
183
+ }, [createElementVNode("path", {
184
+ "stroke-linecap": "round",
185
+ "stroke-linejoin": "round",
186
+ "stroke-width": "2",
187
+ d: "M4 4v5h.582m15.356 2A8.001 8.001 0 004.582 9m0 0H9m11 11v-5h-.581m0 0a8.003 8.003 0 01-15.357-2m15.357 2H15"
188
+ })], -1)])], 8, _hoisted_10)])]),
189
+ createElementVNode("td", null, [createElementVNode("button", {
190
+ type: "button",
191
+ class: "btn btn-sm btn-primary",
192
+ disabled: savingTeamId.value === team.id || referralTagInputs[team.id] === (team.referral_tag ?? ""),
193
+ onClick: ($event) => saveReferralTag(team.id)
194
+ }, [savingTeamId.value === team.id ? (openBlock(), createElementBlock("span", _hoisted_12)) : createCommentVNode("v-if", true), createTextVNode(" " + toDisplayString(savingTeamId.value === team.id ? "Saving..." : "Save"), 1)], 8, _hoisted_11)])
195
+ ]);
196
+ }), 128))])])])) : (openBlock(), createElementBlock("p", _hoisted_13, "No teams found."))
197
+ ])
198
+ ], 64))
199
+ ]);
200
+ };
201
+ }
202
+ });
203
+ var DefaultReferralTeamPage_default = _sfc_main;
204
+
205
+ //#endregion
206
+ export { DefaultReferralTeamPage_default as default };
207
+ //# sourceMappingURL=DefaultReferralTeamPage-BHail7YF.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"DefaultReferralTeamPage-BHail7YF.js","names":[],"sources":["../src/slices/admin/features/default_referral_team/defaultReferralTeamSchema.ts","../src/slices/admin/features/default_referral_team/DefaultReferralTeamPage.vue"],"sourcesContent":["import type { DefaultReferralTeamFormDto } from '@dragonmastery/dragoncore-shared';\nimport { DefaultReferralTeamFormSchema } from '@dragonmastery/dragoncore-shared';\nimport { withMetadata } from '@dragonmastery/zinia-forms-core';\n\nexport type { DefaultReferralTeamFormDto };\n\nexport const defaultReferralTeamSchemaWithMetadata = withMetadata(\n DefaultReferralTeamFormSchema,\n 'defaultReferralTeamSchema',\n {\n default_team_id: {\n inputType: 'select',\n label: 'Default team',\n placeholder: 'Select team',\n helpText:\n 'When a referral event arrives with an unknown tag, it will be routed to this team. Select \"None\" to reject unknown tags.',\n },\n },\n);\n","<template>\n <div class=\"mt-2\">\n <div class=\"flex justify-between items-center mb-4\">\n <h1 class=\"text-2xl font-bold\">Default Team for Unknown Referrals</h1>\n </div>\n\n <p class=\"text-base-content/70 mb-4\">\n When a referral event arrives with an unknown referral tag, it will be routed to this team.\n Select a team to use as the default, or clear to reject unknown tags.\n </p>\n\n <div v-if=\"isLoading\" class=\"flex justify-center py-8\">\n <span class=\"loading loading-spinner loading-lg\"></span>\n </div>\n\n <template v-else>\n <ZiniaForm\n @handle-submit=\"handleSubmit\"\n @success=\"handleSuccess\"\n @error=\"handleError\"\n title=\"\"\n subtitle=\"\"\n >\n <zinia.DefaultTeamIdField\n :select-options=\"form.extraData.teams ?? []\"\n placeholder=\"Select team\"\n />\n\n <div v-if=\"form.submitError\" class=\"alert alert-error mt-2\">\n <span>{{ form.submitError }}</span>\n </div>\n\n <div class=\"mt-4 flex justify-end\">\n <ZiniaSubmitButton\n submitText=\"Save\"\n submittingText=\"Saving...\"\n />\n </div>\n </ZiniaForm>\n\n <!-- Referral tags per team -->\n <div class=\"mt-12\">\n <h2 class=\"text-xl font-semibold mb-2\">Referral Tags per Team</h2>\n <p class=\"text-base-content/70 mb-4\">\n Set a referral tag for each team. Other apps use this tag when pushing events.\n If empty, the team's unique name or path is used as fallback.\n </p>\n <div v-if=\"teamsData?.items?.length\" class=\"overflow-x-auto\">\n <table class=\"table table-zebra\">\n <thead>\n <tr>\n <th>Team</th>\n <th>Referral tag</th>\n <th></th>\n </tr>\n </thead>\n <tbody>\n <tr v-for=\"team in teamsData.items\" :key=\"team.id\">\n <td>{{ team.display_name || team.unique_name || team.id }}</td>\n <td>\n <div class=\"flex items-center gap-1\">\n <input\n v-model=\"referralTagInputs[team.id]\"\n type=\"text\"\n class=\"input input-bordered input-sm w-full max-w-xs\"\n placeholder=\"e.g. my-team\"\n @keydown.enter.prevent=\"saveReferralTag(team.id)\"\n />\n <button\n type=\"button\"\n class=\"btn btn-sm btn-ghost btn-square\"\n :title=\"'Normalize to URL-friendly tag from team name'\"\n @click=\"normalizeReferralTag(team)\"\n >\n <svg xmlns=\"http://www.w3.org/2000/svg\" class=\"h-4 w-4\" fill=\"none\" viewBox=\"0 0 24 24\" stroke=\"currentColor\">\n <path stroke-linecap=\"round\" stroke-linejoin=\"round\" stroke-width=\"2\" d=\"M4 4v5h.582m15.356 2A8.001 8.001 0 004.582 9m0 0H9m11 11v-5h-.581m0 0a8.003 8.003 0 01-15.357-2m15.357 2H15\" />\n </svg>\n </button>\n </div>\n </td>\n <td>\n <button\n type=\"button\"\n class=\"btn btn-sm btn-primary\"\n :disabled=\"savingTeamId === team.id || referralTagInputs[team.id] === (team.referral_tag ?? '')\"\n @click=\"saveReferralTag(team.id)\"\n >\n <span v-if=\"savingTeamId === team.id\" class=\"loading loading-spinner loading-sm\" />\n {{ savingTeamId === team.id ? 'Saving...' : 'Save' }}\n </button>\n </td>\n </tr>\n </tbody>\n </table>\n </div>\n <p v-else class=\"text-base-content/60\">No teams found.</p>\n </div>\n </template>\n </div>\n</template>\n\n<script setup lang=\"ts\">\nimport type { DefaultReferralTeamFormDto } from './defaultReferralTeamSchema';\nimport { defaultReferralTeamSchemaWithMetadata } from './defaultReferralTeamSchema';\nimport { useForm } from '@dragonmastery/zinia-forms-core';\nimport { computed, reactive, ref, watch } from 'vue';\nimport { toast } from 'vue3-toastify';\nimport { useMutation } from '../../../../composables/useMutation';\nimport { useQuery } from '../../../../composables/useQuery';\nimport { extractRpcErrorMessage } from '../../../../utils/extractRpcErrorMessage';\n\nconst {\n data: defaultTeamIdData,\n loading: defaultLoading,\n refetch: refetchDefault,\n} = useQuery((api) => api.appSettings.getDefaultTeamIdForUnknownReferrals(), {\n cacheKey: 'admin-default-referral-team',\n staleTime: 30 * 1000,\n});\n\nconst {\n data: teamsData,\n loading: teamsLoading,\n refetch: refetchTeams,\n} = useQuery(\n (api) => api.teams.listTeams({ first: 100 }),\n {\n cacheKey: 'admin-teams-for-default-referral',\n staleTime: 60 * 1000,\n },\n);\n\nconst referralTagInputs = reactive<Record<string, string>>({});\nconst savingTeamId = ref<string | null>(null);\n\nwatch(\n () => teamsData.value?.items ?? [],\n (teams) => {\n for (const team of teams) {\n referralTagInputs[team.id] = team.referral_tag ?? '';\n }\n },\n { immediate: true },\n);\n\nconst loadTeams = async () => {\n while (teamsLoading.value) {\n await new Promise((resolve) => setTimeout(resolve, 100));\n }\n const teams = teamsData.value?.items ?? [];\n const options = [\n { value: '', label: 'None (reject unknown tags)' },\n ...teams.map((team) => ({\n value: team.id,\n label: `${team.display_name || team.unique_name || team.id}${team.referral_tag ? ` (${team.referral_tag})` : ''}`,\n })),\n ];\n return options;\n};\n\nconst {\n form,\n zinia,\n ZiniaForm,\n ZiniaSubmitButton,\n} = useForm(defaultReferralTeamSchemaWithMetadata, {\n storeName: 'admin-default-referral-team-form',\n persistToLocalStorage: false,\n renderStyle: 'daisy_ui',\n fetchData: async () => {\n while (defaultLoading.value) {\n await new Promise((resolve) => setTimeout(resolve, 100));\n }\n const data = defaultTeamIdData.value;\n return {\n default_team_id: data?.default_team_id ?? null,\n };\n },\n dataLoaders: {\n teams: loadTeams,\n },\n});\n\nconst isLoading = computed(\n () => defaultLoading.value || teamsLoading.value || form.isLoading,\n);\n\nconst { mutate: updateDefaultTeam } = useMutation(\n (api, input: DefaultReferralTeamFormDto) =>\n api.appSettings.updateDefaultTeamIdForUnknownReferrals(input),\n { invalidate: /admin-default-referral-team/ },\n);\n\nconst { mutate: setReferralTag } = useMutation(\n (api, input: { team_id: string; referral_tag: string | null }) =>\n api.teams.setReferralTag(input),\n { invalidate: /admin-teams-for-default-referral/ },\n);\n\n/** Normalize a team name to a URL-friendly referral tag (e.g. \"Acme Corp & Co.\" → \"acme-corp-co\") */\nfunction normalizeToReferralTag(name: string): string {\n return name\n .toLowerCase()\n .trim()\n .replace(/[^a-z0-9\\s-]/g, '') // strip special chars except spaces and hyphens\n .replace(/\\s+/g, '-') // spaces → hyphens\n .replace(/-+/g, '-') // collapse multiple hyphens\n .replace(/^-|-$/g, ''); // trim leading/trailing hyphens\n}\n\nfunction normalizeReferralTag(team: { id: string; display_name?: string | null; unique_name?: string | null }) {\n const name = team.display_name || team.unique_name || team.id;\n const normalized = normalizeToReferralTag(name);\n if (normalized) {\n referralTagInputs[team.id] = normalized;\n }\n}\n\nasync function saveReferralTag(teamId: string) {\n const raw = referralTagInputs[teamId]?.trim() ?? '';\n const referral_tag = raw === '' ? null : raw;\n const team = teamsData.value?.items?.find((t) => t.id === teamId);\n if (team && referral_tag === (team.referral_tag ?? null)) return;\n\n savingTeamId.value = teamId;\n try {\n await setReferralTag({ team_id: teamId, referral_tag });\n toast.success('Referral tag saved');\n referralTagInputs[teamId] = referral_tag ?? '';\n await refetchTeams();\n } catch (err) {\n toast.error(extractRpcErrorMessage(err, 'Failed to save referral tag'));\n } finally {\n savingTeamId.value = null;\n }\n}\n\nasync function handleSubmit(formData: DefaultReferralTeamFormDto) {\n const default_team_id = formData.default_team_id?.trim() || null;\n await updateDefaultTeam({ default_team_id });\n return { ok: true };\n}\n\nasync function handleSuccess() {\n toast.success('Default team saved');\n await refetchDefault();\n}\n\nfunction handleError(error: Error | unknown) {\n const message = error instanceof Error ? error.message : String(error);\n form.setSubmitError(extractRpcErrorMessage(error, message));\n toast.error(form.submitError ?? 'Failed to save');\n}\n</script>\n"],"mappings":"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"}
@@ -1,7 +1,7 @@
1
1
  import { t as useMutation } from "./useMutation-BLNuJoYl.js";
2
2
  import { t as FieldsetSection_default } from "./FieldsetSection-CH1jAwcc.js";
3
3
  import { n as teamUpdateSchemaWithMetadata } from "./teamMetadata-26Mwjb2i.js";
4
- import { t as teamPaths } from "./teamRoutes-BYpld9yI.js";
4
+ import { t as teamPaths } from "./teamRoutes-CtNcFZjR.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-DYvHGBRF.js.map
163
+ //# sourceMappingURL=EditTeamForm-BEOkUaKG.js.map