complex-params.vue 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215
  1. <template>
  2. <CusDialog
  3. :show="show"
  4. :title="titleCpt"
  5. @onClose="$emit('update:show', false)"
  6. width="80%"
  7. height="80%"
  8. @onSubmit="onSubmit"
  9. :loading="state.loading"
  10. >
  11. <div class="__cus-dialog-form">
  12. <CusForm ref="ref_form" label-width="100px">
  13. <CusFormColumn
  14. :span="24"
  15. required
  16. type="textarea"
  17. v-model:param="state.valueStr"
  18. :rows="12"
  19. :rules="[
  20. {
  21. handle: (val) => {
  22. try {
  23. const a = JSON.parse(val)
  24. return true
  25. } catch (e) {
  26. return false
  27. }
  28. },
  29. message: 'JSON结构异常'
  30. }
  31. ]"
  32. />
  33. </CusForm>
  34. <div class="params-content">
  35. <div class="request-params">
  36. <div style="margin-bottom: 10px; display: flex">
  37. <CusButton type="main" title="新增" @click="state.params.data.push({type: '2'})"/>
  38. <div class="tips" style="margin: 0 10px">
  39. 为了保证数据结构准确,提交时将默认触发“<span style="color: var(--cus-main-color)">逆向生成JSON结构</span>”
  40. </div>
  41. <CusButton type="main" title="正向解析JSON结构" style="margin-left: auto" @click="jsonToTable"/>
  42. <CusButton type="main" title="逆向生成JSON结构" @click="tableToJson"/>
  43. </div>
  44. <CusTable
  45. :data="state.params.data"
  46. :table-head="state.params.tableHead"
  47. :no-page="true"
  48. >
  49. <template #key-column-value="{scope}">
  50. <CusFormColumn
  51. class="__cus-table-form-column"
  52. :span="24"
  53. v-model:param="scope.row.key"
  54. />
  55. </template>
  56. <template #type-column-value="{scope}">
  57. <CusFormColumn
  58. class="__cus-table-form-column"
  59. :span="24"
  60. v-model:param="scope.row.type"
  61. link="select"
  62. :options="DictionaryStore.fzCsList"
  63. :clearable="false"
  64. @change="(val) => scope.row.value = ''"
  65. />
  66. </template>
  67. <template #value-column-value="{scope}">
  68. <CusFormColumn
  69. class="__cus-table-form-column"
  70. :span="24"
  71. v-model:param="scope.row.value"
  72. />
  73. </template>
  74. <template #label-column-value="{scope}">
  75. <CusFormColumn
  76. class="__cus-table-form-column"
  77. :span="24"
  78. v-model:param="scope.row.label"
  79. />
  80. </template>
  81. <template #do-column-value="{scope}">
  82. <CusButton type="table-del" @click="state.params.data.splice(scope.$index, 1)"/>
  83. </template>
  84. </CusTable>
  85. </div>
  86. </div>
  87. </div>
  88. </CusDialog>
  89. </template>
  90. <script setup lang="ts">
  91. import {computed, getCurrentInstance, nextTick, reactive, ref, watch} from "vue";
  92. import {useDictionaryStore} from "@/stores";
  93. import {ElMessage, ElMessageBox} from "element-plus";
  94. const emit = defineEmits(['update:show', 'complex'])
  95. const {proxy} = getCurrentInstance()
  96. const DictionaryStore = useDictionaryStore()
  97. const props = defineProps({
  98. show: {default: false},
  99. transfer: {}
  100. })
  101. const state: any = reactive({
  102. valueStr: '',
  103. loading: false,
  104. params: {
  105. tableHead: [
  106. {value: "key", label: "键名"},
  107. {value: "type", label: "类型", width: 160},
  108. {value: "value", label: `取值(动态取值:\$['变量名'])`},
  109. {value: "label", label: "含义"},
  110. {value: "do", label: "操作", width: 120, fixed: 'right'},
  111. ],
  112. data: []
  113. }
  114. })
  115. const ref_form = ref()
  116. const titleCpt = computed(() => {
  117. let t = '复杂参数规则'
  118. return t
  119. })
  120. const onSubmit = () => {
  121. if (state.params.data.length === 0) {
  122. ElMessage.error('请完善参数信息!')
  123. return
  124. }
  125. tableToJson()
  126. setTimeout(() => {
  127. let flag = state.params.data.every(v => proxy.$util.isValue(v.key) && proxy.$util.isValue(v.value) && proxy.$util.isValue(v.label))
  128. if (!flag) {
  129. ElMessage.error('请完善参数信息!')
  130. return
  131. }
  132. ref_form.value.submit().then(() => {
  133. ElMessageBox.confirm("是否提交?", "提示", {
  134. confirmButtonText: "确定",
  135. cancelButtonText: "取消",
  136. type: "warning",
  137. } as any).then(() => {
  138. emit('update:show', false)
  139. emit('complex', {
  140. str: state.valueStr,
  141. table: state.params.data
  142. })
  143. }).catch(() => {})
  144. }).catch((e) => {
  145. ElMessage({
  146. message: e[0].message,
  147. grouping: true,
  148. type: 'warning',
  149. })
  150. })
  151. }, 100)
  152. }
  153. const jsonToTable = () => {
  154. try {
  155. const json = JSON.parse(state.valueStr)
  156. state.params.data = []
  157. Object.entries(json).forEach(([k, v]) => {
  158. state.params.data.push({
  159. key: k, value: typeof v === 'object' ? JSON.stringify(v): v, type: '2'
  160. })
  161. })
  162. } catch (e) {
  163. ElMessage.error('JSON结构异常!')
  164. }
  165. }
  166. const tableToJson = () => {
  167. const obj = {}
  168. state.params.data.forEach(v => {
  169. obj[v.key] = v.value
  170. })
  171. state.valueStr = JSON.stringify(obj)
  172. }
  173. watch(() => props.show, (n) => {
  174. if (n) {
  175. state.loading = false
  176. initDictionary()
  177. state.valueStr = props.transfer.value ? JSON.parse(JSON.stringify(props.transfer.value)) : ''
  178. state.params.data = props.transfer.valueTable ? [...props.transfer.valueTable] : []
  179. nextTick(() => {
  180. ref_form.value.reset()
  181. })
  182. }
  183. })
  184. const initDictionary = () => {
  185. DictionaryStore.initDict('fz_cs')
  186. }
  187. </script>
  188. <style lang="scss" scoped>
  189. .__cus-dialog-form {
  190. height: 100%;
  191. display: flex;
  192. flex-direction: column;
  193. }
  194. .params-content {
  195. flex: 1;
  196. overflow: hidden;
  197. .request-params {
  198. width: 100%;
  199. height: 100%;
  200. display: flex;
  201. flex-direction: column;
  202. }
  203. }
  204. .tips {
  205. background-color: rgba(var(--cus-main-color-rgb), 0.1);
  206. border-radius: 6px;
  207. padding: 7px;
  208. margin-bottom: 10px;
  209. width: 100%;
  210. }
  211. </style>