MessageTemplateView.vue 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209
  1. <template>
  2. <div class="app-page">
  3. <section class="glass-card section-card">
  4. <el-form :model="filters" inline class="filter-form">
  5. <el-form-item label="模板名称">
  6. <el-input v-model="filters.name" placeholder="模板名称" clearable style="width:160px" @keyup.enter="loadData" />
  7. </el-form-item>
  8. <el-form-item label="类型">
  9. <el-select v-model="filters.type" placeholder="全部" clearable style="width:120px">
  10. <el-option label="邮件" value="邮件" />
  11. <el-option label="短信" value="短信" />
  12. <el-option label="站内信" value="站内信" />
  13. </el-select>
  14. </el-form-item>
  15. <el-form-item label="状态">
  16. <el-select v-model="filters.status" placeholder="全部" clearable style="width:120px">
  17. <el-option label="启用" value="启用" />
  18. <el-option label="停用" value="停用" />
  19. </el-select>
  20. </el-form-item>
  21. <el-form-item>
  22. <el-button type="primary" @click="loadData">查询</el-button>
  23. <el-button @click="resetFilters">重置</el-button>
  24. </el-form-item>
  25. </el-form>
  26. </section>
  27. <section class="glass-card section-card" style="padding:12px 24px">
  28. <div class="table-toolbar" style="margin-bottom:0">
  29. <div class="chip-list">
  30. <el-button type="primary" @click="openDialog()">新建模板</el-button>
  31. </div>
  32. <el-button @click="loadData">刷新</el-button>
  33. </div>
  34. </section>
  35. <section class="glass-card section-card">
  36. <el-table :data="filteredItems" style="width:100%" v-loading="loading">
  37. <el-table-column prop="name" label="模板名称" min-width="160" />
  38. <el-table-column prop="type" label="类型" width="80">
  39. <template #default="{ row }">
  40. <el-tag :type="getMessageTemplateType(row.type).type" size="small">{{ getMessageTemplateType(row.type).label }}</el-tag>
  41. </template>
  42. </el-table-column>
  43. <el-table-column prop="channel" label="渠道" width="120" />
  44. <el-table-column prop="subject" label="主题" min-width="200" show-overflow-tooltip />
  45. <el-table-column prop="variables" label="变量" min-width="200" show-overflow-tooltip />
  46. <el-table-column prop="status" label="状态" width="80">
  47. <template #default="{ row }">
  48. <el-tag :type="getMessageTemplateStatus(row.status).type" size="small">{{ getMessageTemplateStatus(row.status).label }}</el-tag>
  49. </template>
  50. </el-table-column>
  51. <el-table-column prop="updatedAt" label="更新时间" width="160" />
  52. <el-table-column label="操作" width="180" fixed="right">
  53. <template #default="{ row }">
  54. <el-button link type="primary" @click="openDialog(row)">编辑</el-button>
  55. <el-button link :type="row.status === '启用' ? 'danger' : 'primary'" @click="toggleStatus(row)">{{ row.status === '启用' ? '停用' : '启用' }}</el-button>
  56. <el-button link type="primary" @click="doCopy(row)">复制</el-button>
  57. </template>
  58. </el-table-column>
  59. <template #empty>
  60. <el-empty description="暂无数据" />
  61. </template>
  62. </el-table>
  63. </section>
  64. <el-dialog v-model="dialogVisible" :title="editingItem ? '编辑模板' : '新建模板'" width="600px" destroy-on-close>
  65. <el-form :model="formData" label-width="90px">
  66. <el-form-item label="模板名称" required>
  67. <el-input v-model="formData.name" placeholder="请输入模板名称" />
  68. </el-form-item>
  69. <el-form-item label="类型" required>
  70. <el-select v-model="formData.type" placeholder="选择类型" style="width:100%">
  71. <el-option label="邮件" value="邮件" />
  72. <el-option label="短信" value="短信" />
  73. <el-option label="站内信" value="站内信" />
  74. </el-select>
  75. </el-form-item>
  76. <el-form-item label="适用渠道">
  77. <el-select v-model="formData.channel" placeholder="选择渠道" style="width:100%">
  78. <el-option label="通用" value="通用" />
  79. <el-option label="Shopify" value="Shopify" />
  80. <el-option label="TikTok Shop" value="TikTok Shop" />
  81. </el-select>
  82. </el-form-item>
  83. <el-form-item label="模板主题" v-if="formData.type !== '短信'">
  84. <el-input v-model="formData.subject" placeholder="邮件主题/站内信标题" />
  85. </el-form-item>
  86. <el-form-item label="模板内容" required>
  87. <el-input v-model="formData.content" type="textarea" :rows="6" placeholder="模板内容,支持变量占位符:{{orderNo}}、{{buyerName}}等" />
  88. </el-form-item>
  89. <el-form-item label="可用变量">
  90. <div style="color:var(--cb-text-soft);font-size:13px">
  91. {{ variablePlaceholders }}
  92. </div>
  93. </el-form-item>
  94. <el-form-item label="备注">
  95. <el-input v-model="formData.remark" type="textarea" :rows="2" />
  96. </el-form-item>
  97. </el-form>
  98. <template #footer>
  99. <el-button @click="dialogVisible = false">取消</el-button>
  100. <el-button type="primary" @click="saveTemplate">保存</el-button>
  101. </template>
  102. </el-dialog>
  103. </div>
  104. </template>
  105. <script setup lang="ts">
  106. import { computed, onMounted, reactive, ref } from 'vue';
  107. import { ElMessage, ElMessageBox } from 'element-plus';
  108. import { getMessageTemplateType, getMessageTemplateStatus } from '@/utils/enumMappings';
  109. interface MessageTemplate {
  110. id: string;
  111. name: string;
  112. type: string;
  113. channel: string;
  114. subject: string;
  115. content: string;
  116. variables: string;
  117. status: string;
  118. updatedAt: string;
  119. remark?: string;
  120. }
  121. const items = ref<MessageTemplate[]>([
  122. { id: 'MT001', name: '订单发货通知', type: '邮件', channel: '通用', subject: '您的订单已发货 - {{orderNo}}', content: '亲爱的 {{buyerName}},\n\n您的订单 {{orderNo}} 已于今日发货,预计3-5个工作日送达。\n\n物流信息:{{trackingNo}}\n\n感谢您的购买!', variables: '{{orderNo}}, {{buyerName}}, {{trackingNo}}', status: '启用', updatedAt: '2026-04-15 10:30:00' },
  123. { id: 'MT002', name: '发货提醒短信', type: '短信', channel: '通用', subject: '', content: '【CrossBorder】亲爱的{{buyerName}},您的订单{{orderNo}}已发货,运单号{{trackingNo}},请注意查收。如有疑问请联系客服。', variables: '{{orderNo}}, {{buyerName}}, {{trackingNo}}', status: '启用', updatedAt: '2026-04-10 14:20:00' },
  124. { id: 'MT003', name: '退款确认通知', type: '邮件', channel: '通用', subject: '退款已完成 - {{orderNo}}', content: '亲爱的 {{buyerName}},\n\n您的退款申请已处理完成。\n\n退款单号:{{refundNo}}\n退款金额:{{refundAmount}}\n\n预计1-7个工作日到账,感谢您的理解。', variables: '{{orderNo}}, {{buyerName}}, {{refundNo}}, {{refundAmount}}', status: '启用', updatedAt: '2026-04-08 09:00:00' },
  125. { id: 'MT004', name: '缺货提醒', type: '站内信', channel: 'Shopify', subject: '库存不足提醒', content: '您好,SKU {{sku}} ({{productName}}) 当前库存不足,请及时补货。', variables: '{{sku}}, {{productName}}', status: '停用', updatedAt: '2026-03-28 16:45:00' }
  126. ]);
  127. const loading = ref(false);
  128. const dialogVisible = ref(false);
  129. const editingItem = ref<MessageTemplate | null>(null);
  130. const filters = ref({ name: '', type: '', status: '' });
  131. const formData = reactive({
  132. name: '',
  133. type: '',
  134. channel: '通用',
  135. subject: '',
  136. content: '',
  137. remark: ''
  138. });
  139. const variablePlaceholders = '{{orderNo}} 订单号 | {{buyerName}} 买家姓名 | {{orderAmount}} 订单金额 | {{trackingNo}} 运单号 | {{sku}} SKU | {{productName}} 商品名称';
  140. const filteredItems = computed(() => {
  141. return items.value.filter(item => {
  142. if (filters.value.name && !item.name.includes(filters.value.name)) return false;
  143. if (filters.value.type && item.type !== filters.value.type) return false;
  144. if (filters.value.status && item.status !== filters.value.status) return false;
  145. return true;
  146. });
  147. });
  148. const loadData = () => { loading.value = true; setTimeout(() => { loading.value = false; }, 300); };
  149. const resetFilters = () => { filters.value = { name: '', type: '', status: '' }; };
  150. const openDialog = (item?: MessageTemplate) => {
  151. editingItem.value = item || null;
  152. if (item) {
  153. Object.assign(formData, { name: item.name, type: item.type, channel: item.channel, subject: item.subject, content: item.content, remark: item.remark || '' });
  154. } else {
  155. Object.assign(formData, { name: '', type: '', channel: '通用', subject: '', content: '', remark: '' });
  156. }
  157. dialogVisible.value = true;
  158. };
  159. const saveTemplate = () => {
  160. if (!formData.name || !formData.type || !formData.content) {
  161. ElMessage.warning('请填写必填项');
  162. return;
  163. }
  164. if (editingItem.value) {
  165. const idx = items.value.findIndex(i => i.id === editingItem.value!.id);
  166. if (idx !== -1) items.value[idx] = { ...items.value[idx], ...formData, updatedAt: new Date().toLocaleString() };
  167. ElMessage.success('模板已更新');
  168. } else {
  169. items.value.push({ id: `MT${String(items.value.length + 1).padStart(3, '0')}`, ...formData, variables: '{{orderNo}}, {{buyerName}}', status: '启用', updatedAt: new Date().toLocaleString() });
  170. ElMessage.success('模板已创建');
  171. }
  172. dialogVisible.value = false;
  173. };
  174. const toggleStatus = async (row: MessageTemplate) => {
  175. const action = row.status === '启用' ? '停用' : '启用';
  176. await ElMessageBox.confirm(`确认${action}「${row.name}」?`);
  177. const idx = items.value.findIndex(i => i.id === row.id);
  178. if (idx !== -1) items.value[idx].status = row.status === '启用' ? '停用' : '启用';
  179. ElMessage.success(`模板已${action}`);
  180. };
  181. const doCopy = (row: MessageTemplate) => {
  182. ElMessageBox.confirm(`确认复制「${row.name}」模板?`).then(() => {
  183. items.value.push({ ...row, id: `MT${String(items.value.length + 1).padStart(3, '0')}`, name: `${row.name} (副本)`, status: '停用', updatedAt: new Date().toLocaleString() });
  184. ElMessage.success('模板已复制');
  185. }).catch(() => {});
  186. };
  187. onMounted(loadData);
  188. </script>
  189. <style scoped>
  190. .filter-form :deep(.el-form-item) { margin-bottom: 0; }
  191. </style>