template.vue 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490
  1. <template>
  2. <div class="app-container">
  3. <!-- 搜索栏 -->
  4. <el-form :model="queryParams" ref="queryForm" :inline="true" v-show="showSearch" label-width="80px">
  5. <el-form-item label="模板名称" prop="templateName">
  6. <el-input v-model="queryParams.templateName" placeholder="请输入模板名称" clearable @keyup.enter.native="handleQuery" />
  7. </el-form-item>
  8. <el-form-item label="状态" prop="status">
  9. <el-select v-model="queryParams.status" placeholder="模板状态" clearable>
  10. <el-option v-for="dict in statusOptions" :key="dict.value" :label="dict.label" :value="dict.value" />
  11. </el-select>
  12. </el-form-item>
  13. <el-form-item>
  14. <el-button type="primary" icon="el-icon-search" @click="handleQuery">搜索</el-button>
  15. <el-button icon="el-icon-refresh" @click="resetQuery">重置</el-button>
  16. </el-form-item>
  17. </el-form>
  18. <!-- 操作按钮 -->
  19. <el-row :gutter="10" class="mb8">
  20. <el-col :span="1.5">
  21. <el-button type="primary" plain icon="el-icon-plus" @click="handleAdd">新增模板</el-button>
  22. </el-col>
  23. <right-toolbar :showSearch.sync="showSearch" @queryTable="getList" />
  24. </el-row>
  25. <!-- 表格 -->
  26. <el-table v-loading="loading" :data="templateList" @selection-change="handleSelectionChange">
  27. <el-table-column type="selection" width="55" align="center" />
  28. <el-table-column label="模板ID" align="center" prop="templateId" width="80" />
  29. <el-table-column label="模板名称" align="center" prop="templateName" show-overflow-tooltip />
  30. <el-table-column label="模板描述" align="center" prop="templateDesc" show-overflow-tooltip />
  31. <el-table-column label="版本" align="center" prop="version" width="80" />
  32. <el-table-column label="状态" align="center" prop="status" width="100">
  33. <template slot-scope="scope">
  34. <el-tag :type="scope.row.status === 1 ? 'success' : 'danger'">
  35. {{ scope.row.status === 1 ? '启用' : '禁用' }}
  36. </el-tag>
  37. </template>
  38. </el-table-column>
  39. <el-table-column label="创建时间" align="center" prop="createTime" width="180" />
  40. <el-table-column label="操作" align="center" class-name="small-padding fixed-width" width="280">
  41. <template slot-scope="scope">
  42. <el-button size="mini" type="text" icon="el-icon-edit" @click="handleNodeDesign(scope.row)">节点设计</el-button>
  43. <el-button size="mini" type="text" icon="el-icon-edit" @click="handleUpdate(scope.row)">修改</el-button>
  44. <el-button size="mini" type="text" icon="el-icon-delete" @click="handleDelete(scope.row)">删除</el-button>
  45. </template>
  46. </el-table-column>
  47. </el-table>
  48. <pagination v-show="total > 0" :total="total" :page.sync="queryParams.pageNum" :limit.sync="queryParams.pageSize" @pagination="getList" />
  49. <!-- 新增/修改模板对话框 -->
  50. <el-dialog :title="title" :visible.sync="open" width="600px" append-to-body>
  51. <el-form ref="form" :model="form" :rules="rules" label-width="100px">
  52. <el-form-item label="模板名称" prop="templateName">
  53. <el-input v-model="form.templateName" placeholder="请输入模板名称" />
  54. </el-form-item>
  55. <el-form-item label="模板描述" prop="templateDesc">
  56. <el-input v-model="form.templateDesc" type="textarea" placeholder="请输入模板描述" />
  57. </el-form-item>
  58. <el-form-item label="状态" prop="status">
  59. <el-radio-group v-model="form.status">
  60. <el-radio v-for="dict in statusOptions" :key="dict.value" :label="dict.value">{{ dict.label }}</el-radio>
  61. </el-radio-group>
  62. </el-form-item>
  63. </el-form>
  64. <div slot="footer" class="dialog-footer">
  65. <el-button @click="cancel">取 消</el-button>
  66. <el-button type="primary" @click="submitForm">确 定</el-button>
  67. </div>
  68. </el-dialog>
  69. <!-- 节点设计对话框 -->
  70. <el-dialog :title="'节点设计 - ' + currentTemplate.templateName" :visible.sync="nodeDesignOpen" width="1100px" append-to-body @opened="loadTemplateNodes">
  71. <div class="node-panel">
  72. <el-button type="primary" size="small" icon="el-icon-plus" @click="handleAddNode">添加节点</el-button>
  73. <el-table :data="templateNodes" style="margin-top: 10px;" row-key="nodeId">
  74. <el-table-column label="节点ID" align="center" prop="nodeId" width="80" />
  75. <el-table-column label="排序" align="center" prop="sortOrder" width="60" />
  76. <el-table-column label="节点名称" align="center" prop="nodeName" />
  77. <el-table-column label="节点类型" align="center" prop="nodeType" width="120">
  78. <template slot-scope="scope">
  79. <el-tag :type="getNodeTypeTag(scope.row.nodeType)">{{ getNodeTypeName(scope.row.nodeType) }}</el-tag>
  80. </template>
  81. </el-table-column>
  82. <el-table-column label="上一节点" align="center" width="140">
  83. <template slot-scope="scope">
  84. <span v-if="scope.row.prevNodeId">{{ getPrevNodeLabel(scope.row.prevNodeId) }}</span>
  85. <span v-else>-</span>
  86. </template>
  87. </el-table-column>
  88. <el-table-column label="操作" align="center" width="180">
  89. <template slot-scope="scope">
  90. <el-button size="mini" type="text" icon="el-icon-edit" @click="handleEditNode(scope.row)">编辑</el-button>
  91. <el-button size="mini" type="text" icon="el-icon-s-order" @click="handleNodeRules(scope.row)">管理规则</el-button>
  92. <el-button size="mini" type="text" icon="el-icon-delete" @click="handleDeleteNode(scope.row)">删除</el-button>
  93. </template>
  94. </el-table-column>
  95. </el-table>
  96. </div>
  97. </el-dialog>
  98. <!-- 新增/编辑节点对话框 -->
  99. <el-dialog :title="nodeFormTitle" :visible.sync="nodeFormOpen" width="550px" append-to-body>
  100. <el-form ref="nodeForm" :model="nodeForm" label-width="100px" size="small">
  101. <el-form-item label="节点名称" prop="nodeName">
  102. <el-input v-model="nodeForm.nodeName" placeholder="请输入节点名称" />
  103. </el-form-item>
  104. <el-form-item label="节点类型" prop="nodeType">
  105. <el-select v-model="nodeForm.nodeType" placeholder="请选择节点类型">
  106. <el-option v-for="dict in nodeTypeOptions" :key="dict.dictValue" :label="dict.dictLabel" :value="dict.dictValue" />
  107. </el-select>
  108. </el-form-item>
  109. <el-form-item label="排序" prop="sortOrder">
  110. <el-input-number v-model="nodeForm.sortOrder" :min="0" />
  111. </el-form-item>
  112. <el-form-item label="上一节点" prop="prevNodeId">
  113. <el-select v-model="nodeForm.prevNodeId" placeholder="请选择上一节点" clearable>
  114. <el-option v-for="node in templateNodes" :key="node.nodeId" :label="node.nodeName + ' (ID:' + node.nodeId + ')'" :value="node.nodeId" />
  115. </el-select>
  116. </el-form-item>
  117. <!-- <el-form-item label="下一节点ID" prop="nextNodeId">
  118. <el-input-number v-model="nodeForm.nextNodeId" :min="0" placeholder="可选" />
  119. </el-form-item> -->
  120. <el-form-item label="节点配置" prop="nodeConfig">
  121. <el-input v-model="nodeForm.nodeConfig" type="textarea" :rows="4" placeholder="JSON格式配置" />
  122. </el-form-item>
  123. </el-form>
  124. <div slot="footer" class="dialog-footer">
  125. <el-button @click="nodeFormOpen = false">取 消</el-button>
  126. <el-button type="primary" @click="submitNodeForm">保存节点</el-button>
  127. </div>
  128. </el-dialog>
  129. <!-- 节点规则管理对话框 -->
  130. <el-dialog :title="'管理规则 - ' + currentRuleNode.nodeName" :visible.sync="ruleDialogOpen" width="800px" append-to-body @opened="loadNodeRules">
  131. <div class="rule-actions mb8">
  132. <el-button type="primary" size="small" icon="el-icon-plus" @click="handleAddRule">新增规则</el-button>
  133. </div>
  134. <el-table :data="nodeRules" border>
  135. <el-table-column label="排序" align="center" prop="sortOrder" width="60" />
  136. <el-table-column label="规则名称" align="center" prop="ruleName" />
  137. <el-table-column label="间隔天数" align="center" prop="sendDay" width="80" />
  138. <el-table-column label="间隔时间" align="center" prop="sendTime" width="100" />
  139. <el-table-column label="操作" align="center" width="120">
  140. <template slot-scope="scope">
  141. <el-button size="mini" type="text" icon="el-icon-edit" @click="handleEditRule(scope.row)">编辑</el-button>
  142. <el-button size="mini" type="text" icon="el-icon-delete" @click="handleDeleteRule(scope.row)">删除</el-button>
  143. </template>
  144. </el-table-column>
  145. </el-table>
  146. <!-- 规则编辑区域 -->
  147. <div v-if="ruleFormVisible" class="rule-form-panel" style="margin-top: 20px; border: 1px solid #dcdfe6; border-radius: 4px; padding: 15px;">
  148. <h4>{{ ruleFormTitle }}</h4>
  149. <el-form ref="ruleForm" :model="ruleForm" label-width="100px" size="small">
  150. <el-form-item label="规则名称" prop="ruleName">
  151. <el-input v-model="ruleForm.ruleName" placeholder="规则名称,仅内部可见" />
  152. </el-form-item>
  153. <el-form-item label="间隔时间" prop="sendTime">
  154. <el-time-picker
  155. v-model="ruleForm.sendTime"
  156. value-format="HH:mm"
  157. format="HH:mm"
  158. :picker-options="{ selectableRange: ['00:00:00 - 23:59:59'] }"
  159. placeholder="选择时间"
  160. style="width: 160px;">
  161. </el-time-picker>
  162. </el-form-item>
  163. <el-form-item label="间隔天数" prop="sendDay">
  164. <el-input-number v-model="ruleForm.sendDay" :min="0" placeholder="第几天" />
  165. </el-form-item>
  166. <!-- <el-form-item label="内容类别" prop="contentType">
  167. <el-input v-model="ruleForm.contentType" placeholder="请输入内容类别" />
  168. </el-form-item> -->
  169. <el-form-item label="发送提示词" prop="prompt">
  170. <el-input v-model="ruleForm.prompt" type="textarea" :rows="3" placeholder="发送提示词" />
  171. </el-form-item>
  172. <el-form-item label="条件" prop="rule">
  173. <el-select v-model="ruleForm.rule" placeholder="请选择条件" style="width: 30%;">
  174. <el-option label="已回复消息" value="replied" />
  175. <el-option label="未回复消息" value="unreplied" />
  176. </el-select>
  177. </el-form-item>
  178. <el-form-item label="排序" prop="sortOrder">
  179. <el-input-number v-model="ruleForm.sortOrder" :min="0" />
  180. </el-form-item>
  181. <el-form-item>
  182. <el-button type="primary" @click="submitRuleForm">保存规则</el-button>
  183. <el-button @click="ruleFormVisible = false">取消</el-button>
  184. </el-form-item>
  185. </el-form>
  186. </div>
  187. </el-dialog>
  188. </div>
  189. </template>
  190. <script>
  191. import { listTemplate, getTemplate, addTemplate, updateTemplate, delTemplate } from '../../api/aiAddwxSop/api'
  192. import { listTemplateNode, addTemplateNode, updateTemplateNode, delTemplateNode } from '../../api/aiAddwxSop/api'
  193. import { listTemplateNodeRule, addTemplateNodeRule, updateTemplateNodeRule, delTemplateNodeRule } from '../../api/aiAddwxSop/api'
  194. export default {
  195. name: 'AiAddwxSopTemplate',
  196. data() {
  197. return {
  198. loading: true,
  199. showSearch: true,
  200. total: 0,
  201. templateList: [],
  202. templateNodes: [],
  203. ids: [],
  204. title: '',
  205. open: false,
  206. nodeDesignOpen: false,
  207. nodeFormOpen: false,
  208. currentTemplate: {},
  209. ruleDialogOpen: false,
  210. currentRuleNode: {},
  211. nodeTypeOptions: [],
  212. nodeRules: [],
  213. ruleForm: {},
  214. ruleFormVisible: false,
  215. queryParams: {
  216. pageNum: 1,
  217. pageSize: 10,
  218. templateName: null,
  219. status: null
  220. },
  221. form: {},
  222. nodeForm: {},
  223. rules: {
  224. templateName: [{ required: true, message: '模板名称不能为空', trigger: 'blur' }]
  225. },
  226. statusOptions: [
  227. { label: '启用', value: 1 },
  228. { label: '禁用', value: 0 }
  229. ]
  230. }
  231. },
  232. computed: {
  233. nodeFormTitle() {
  234. return this.nodeForm.nodeId ? '编辑节点' : '新增节点'
  235. },
  236. ruleFormTitle() {
  237. return this.ruleForm.ruleId ? '编辑规则' : '新增规则'
  238. }
  239. },
  240. created() {
  241. this.getDicts('ai_add_wechat_sop_node').then(response => {
  242. this.nodeTypeOptions = response.data
  243. })
  244. this.getList()
  245. },
  246. methods: {
  247. getList() {
  248. this.loading = true
  249. listTemplate(this.queryParams).then(response => {
  250. this.templateList = response.rows
  251. this.total = response.total
  252. this.loading = false
  253. })
  254. },
  255. handleQuery() {
  256. this.queryParams.pageNum = 1
  257. this.getList()
  258. },
  259. resetQuery() {
  260. this.resetForm('queryForm')
  261. this.handleQuery()
  262. },
  263. handleSelectionChange(selection) {
  264. this.ids = selection.map(item => item.templateId)
  265. },
  266. cancel() {
  267. this.open = false
  268. this.reset()
  269. },
  270. reset() {
  271. this.form = {}
  272. this.resetForm('form')
  273. },
  274. handleAdd() {
  275. this.reset()
  276. this.open = true
  277. this.title = '新增模板'
  278. this.form.status = 1
  279. },
  280. handleUpdate(row) {
  281. this.reset()
  282. this.open = true
  283. this.title = '修改模板'
  284. this.form = { ...row }
  285. },
  286. handleDelete(row) {
  287. const templateIds = row.templateId
  288. this.$confirm('确认删除模板"' + row.templateName + '"吗?', '警告', {
  289. confirmButtonText: '确定',
  290. cancelButtonText: '取消',
  291. type: 'warning'
  292. }).then(() => {
  293. return delTemplate(templateIds)
  294. }).then(() => {
  295. this.getList()
  296. this.msgSuccess('删除成功')
  297. })
  298. },
  299. submitForm() {
  300. this.$refs['form'].validate(valid => {
  301. if (valid) {
  302. if (this.form.templateId != null) {
  303. updateTemplate(this.form).then(() => {
  304. this.msgSuccess('修改成功')
  305. this.open = false
  306. this.getList()
  307. })
  308. } else {
  309. addTemplate(this.form).then(() => {
  310. this.msgSuccess('新增成功')
  311. this.open = false
  312. this.getList()
  313. })
  314. }
  315. }
  316. })
  317. },
  318. // 节点设计
  319. handleNodeDesign(row) {
  320. this.currentTemplate = row
  321. this.nodeDesignOpen = true
  322. this.nodeForm = {}
  323. this.nodeForm.templateId = row.templateId
  324. },
  325. loadTemplateNodes() {
  326. listTemplateNode(this.currentTemplate.templateId).then(response => {
  327. this.templateNodes = response.data || []
  328. })
  329. },
  330. getNodeTypeName(type) {
  331. const dict = this.nodeTypeOptions.find(d => d.dictValue === type)
  332. return dict ? dict.dictLabel : type
  333. },
  334. getNodeTypeTag(type) {
  335. const map = {
  336. 'receive': 'success',
  337. 'medication': 'warning',
  338. 'effect_feedback': 'info',
  339. 'transfer_ai': 'danger'
  340. }
  341. return map[type] || ''
  342. },
  343. getPrevNodeLabel(prevNodeId) {
  344. const node = this.templateNodes.find(n => n.nodeId === prevNodeId)
  345. return node ? node.nodeName + ' (ID:' + node.nodeId + ')' : 'ID:' + prevNodeId
  346. },
  347. handleAddNode() {
  348. this.nodeForm = {
  349. templateId: this.currentTemplate.templateId,
  350. sortOrder: this.templateNodes.length + 1
  351. }
  352. this.nodeFormOpen = true
  353. },
  354. handleEditNode(row) {
  355. this.nodeForm = { ...row, templateId: this.currentTemplate.templateId }
  356. this.nodeFormOpen = true
  357. },
  358. handleDeleteNode(row) {
  359. this.$confirm('确认删除节点"' + row.nodeName + '"吗?', '警告', {
  360. confirmButtonText: '确定',
  361. cancelButtonText: '取消',
  362. type: 'warning'
  363. }).then(() => {
  364. return delTemplateNode(row.nodeId)
  365. }).then(() => {
  366. this.loadTemplateNodes()
  367. this.msgSuccess('删除成功')
  368. })
  369. },
  370. submitNodeForm() {
  371. if (this.nodeForm.nodeId) {
  372. updateTemplateNode(this.nodeForm).then(() => {
  373. this.msgSuccess('修改成功')
  374. this.loadTemplateNodes()
  375. this.nodeForm = {}
  376. this.nodeFormOpen = false
  377. })
  378. } else {
  379. const templateName = (this.currentTemplate.templateName || 'node').replace(/\s+/g, '_')
  380. const nodeType = this.nodeForm.nodeType || 'unknown'
  381. const timestamp = Date.now()
  382. this.nodeForm.nodeKey = templateName + '_' + nodeType + '_' + timestamp
  383. addTemplateNode(this.nodeForm).then(() => {
  384. this.msgSuccess('新增成功')
  385. this.loadTemplateNodes()
  386. this.nodeForm = {}
  387. this.nodeFormOpen = false
  388. })
  389. }
  390. },
  391. // 节点规则管理
  392. handleNodeRules(row) {
  393. this.currentRuleNode = row
  394. this.ruleDialogOpen = true
  395. this.ruleForm = {}
  396. this.ruleFormVisible = false
  397. },
  398. loadNodeRules() {
  399. listTemplateNodeRule(this.currentRuleNode.nodeId).then(response => {
  400. this.nodeRules = response.data || []
  401. })
  402. },
  403. getMsgTypeName(type) {
  404. const map = { 1: '普通', 2: '课程', 4: 'AI触达', 5: '打标签', 20: '直播间' }
  405. return map[type] || type
  406. },
  407. getMsgTypeTag(type) {
  408. const map = { 1: '', 2: 'success', 4: 'warning', 5: 'danger', 20: 'info' }
  409. return map[type] || ''
  410. },
  411. handleAddRule() {
  412. this.ruleForm = {
  413. nodeId: this.currentRuleNode.nodeId,
  414. msgType: 1,
  415. sortOrder: this.nodeRules.length + 1
  416. }
  417. this.ruleFormVisible = true
  418. },
  419. handleEditRule(row) {
  420. this.ruleForm = { ...row }
  421. if (row.contentJson) {
  422. try {
  423. const parsed = JSON.parse(row.contentJson)
  424. this.ruleForm.prompt = parsed.prompt || ''
  425. this.ruleForm.rule = parsed.rule || ''
  426. } catch (e) {
  427. this.ruleForm.prompt = ''
  428. this.ruleForm.rule = ''
  429. }
  430. }
  431. this.ruleFormVisible = true
  432. },
  433. handleDeleteRule(row) {
  434. this.$confirm('确认删除规则"' + row.ruleName + '"吗?', '警告', {
  435. confirmButtonText: '确定',
  436. cancelButtonText: '取消',
  437. type: 'warning'
  438. }).then(() => {
  439. return delTemplateNodeRule(row.ruleId)
  440. }).then(() => {
  441. this.loadNodeRules()
  442. this.msgSuccess('删除成功')
  443. })
  444. },
  445. submitRuleForm() {
  446. this.ruleForm.contentJson = JSON.stringify({
  447. prompt: this.ruleForm.prompt || '',
  448. rule: this.ruleForm.rule || ''
  449. })
  450. if (this.ruleForm.ruleId) {
  451. updateTemplateNodeRule(this.ruleForm).then(() => {
  452. this.msgSuccess('修改成功')
  453. this.loadNodeRules()
  454. this.ruleForm = {}
  455. this.ruleFormVisible = false
  456. })
  457. } else {
  458. addTemplateNodeRule(this.ruleForm).then(() => {
  459. this.msgSuccess('新增成功')
  460. this.loadNodeRules()
  461. this.ruleForm = {}
  462. this.ruleFormVisible = false
  463. })
  464. }
  465. }
  466. }
  467. }
  468. </script>
  469. <style scoped>
  470. .node-panel {
  471. border: 1px solid #dcdfe6;
  472. border-radius: 4px;
  473. padding: 15px;
  474. min-height: 300px;
  475. }
  476. .node-panel h4 {
  477. margin: 0 0 10px 0;
  478. }
  479. </style>