editNode.vue 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498
  1. <template>
  2. <div class="system-edit-dic-container">
  3. <el-dialog :title="(ruleForm.nodeId !== 0 ? '修改' : '添加') + '数据节点'" v-model="isShowDialog" width="769px">
  4. <el-form :model="ruleForm" ref="formRef" :rules="rules" label-width="110px">
  5. <el-form-item label="数据节点标识" prop="key">
  6. <el-input v-model="ruleForm.key" placeholder="请输入数据节点名称" :disabled="detail.lockKey == 1 && ruleForm.nodeId !== 0" />
  7. </el-form-item>
  8. <el-form-item label="数据节点名称" prop="name">
  9. <el-input v-model="ruleForm.name" placeholder="请输入数据节点名称" />
  10. </el-form-item>
  11. <el-form-item label="是否主键" prop="isPk">
  12. <el-radio-group v-model="ruleForm.isPk">
  13. <el-radio :label="0">否</el-radio>
  14. <el-radio :label="1">是</el-radio>
  15. </el-radio-group>
  16. </el-form-item>
  17. <el-form-item label="数据类型" prop="dataType" v-if="detail.from == 1 || detail.from == 2">
  18. <el-select v-model="ruleForm.dataType" filterable placeholder="请选择数据类型">
  19. <el-option v-for="item in tabData" :key="item.value" :label="item.label" :value="item.value" />
  20. </el-select>
  21. </el-form-item>
  22. <el-form-item label="取值项" prop="value" v-if="detail.from == 1">
  23. <el-input v-model="ruleForm.value" placeholder="请输入取值项" class="w-35" /><el-button type="success" @click="onTest">选择值</el-button>
  24. </el-form-item>
  25. <el-form-item label="取值项" prop="value" v-if="detail.from == 4">
  26. <el-select v-model="ruleForm.value" filterable placeholder="请选择数取值项" @change="getNodeList">
  27. <el-option v-for="item in propertyData" :key="item.key" :label="item.name" :value="item.key">
  28. <span style="float: left">{{ item.key }}</span>
  29. <span style="float: right; font-size: 13px">{{ item.name }}</span>
  30. </el-option>
  31. </el-select>
  32. </el-form-item>
  33. <el-form-item label="取值项" prop="value" v-if="detail.from == 2">
  34. <el-select v-model="ruleForm.value" filterable placeholder="请选择数取值项" @change="getDbList">
  35. <el-option v-for="item in dbData" :key="item.Name" :label="item.Comment" :value="item.Name">
  36. <span style="float: left">{{ item.Name }}</span>
  37. <span style="float: right; font-size: 13px">{{ item.Comment }}</span>
  38. </el-option>
  39. </el-select>
  40. </el-form-item>
  41. <el-divider content-position="left">规则表达式</el-divider>
  42. <div v-for="(item, index) in rule" :key="index">
  43. <el-form-item label="正则表达式">
  44. <el-input v-model="item.expression" placeholder="请输入规则表达式" />
  45. </el-form-item>
  46. <el-form-item label="替换内容">
  47. <el-input v-model="item.replace" placeholder="请输入替换内容" class="w-35" />
  48. <!-- <el-input v-model="rule[index].params.name" placeholder="请输入键值" class="w-35" />
  49. <el-input v-model="rule[index].params.value" placeholder="请输入值" class="w-35" /> -->
  50. <div class="conicon">
  51. <el-icon @click="delRule(index)" v-if="index > 0">
  52. <Delete />
  53. </el-icon>
  54. </div>
  55. </el-form-item>
  56. </div>
  57. <div style="padding: 10px">
  58. <el-button type="primary" class="addbutton" @click="addRule">增加</el-button>
  59. </div>
  60. </el-form>
  61. <template #footer>
  62. <span class="dialog-footer">
  63. <el-button @click="onCancel">取 消</el-button>
  64. <el-button type="primary" @click="onSubmit">{{ ruleForm.nodeId !== 0 ? '修 改' : '添 加' }}</el-button>
  65. </span>
  66. </template>
  67. </el-dialog>
  68. <el-dialog v-model="dialogVisible" title="点击蓝色key值进行选择" width="40%">
  69. <jsontree :data="jsonPathData" @handlePath="handlePath"></jsontree>
  70. <template #footer>
  71. <span class="dialog-footer">
  72. <el-button @click="dialogVisible = false">关闭</el-button>
  73. </span>
  74. </template>
  75. </el-dialog>
  76. </div>
  77. </template>
  78. <script lang="ts">
  79. import { reactive, toRefs, defineComponent, ref, unref } from 'vue';
  80. import api from '/@/api/datahub';
  81. import 'vue3-json-viewer/dist/index.css';
  82. import jsontree from '/@/components/jsontree/index.vue';
  83. import { ElMessage } from 'element-plus';
  84. import { Delete, } from '@element-plus/icons-vue';
  85. interface RuleFormState {
  86. nodeId: number;
  87. name: string;
  88. from: number;
  89. isPk: number;
  90. key: string;
  91. dataType: string;
  92. value: string;
  93. description: string;
  94. status: number;
  95. }
  96. interface DicState {
  97. isShowDialog: boolean;
  98. ruleForm: RuleFormState;
  99. rules: {};
  100. }
  101. export default defineComponent({
  102. name: 'Edit',
  103. components: { Delete, jsontree },
  104. setup(prop, { emit }) {
  105. const formRef = ref<HTMLElement | null>(null);
  106. const state = reactive<DicState>({
  107. isShowDialog: false,
  108. dialogVisible: false,
  109. jsonPathData: [],
  110. jsonData: '',
  111. propertyData: [],
  112. dbData: [],
  113. config: {},
  114. detail: {},
  115. tabData: [
  116. {
  117. label: 'int',
  118. value: 'int',
  119. },
  120. {
  121. label: 'long',
  122. value: 'long',
  123. },
  124. {
  125. label: 'float',
  126. value: 'float',
  127. },
  128. {
  129. label: 'double',
  130. value: 'double',
  131. },
  132. {
  133. label: 'string',
  134. value: 'string',
  135. },
  136. {
  137. label: 'boolean',
  138. value: 'boolean',
  139. },
  140. {
  141. label: 'date',
  142. value: 'date',
  143. },
  144. ],
  145. ruledata: [
  146. {
  147. expression: '',
  148. replace: '',
  149. // params: {
  150. // name: '',
  151. // value: '',
  152. // },
  153. },
  154. ],
  155. rule: [
  156. {
  157. expression: '',
  158. replace: '',
  159. // params: {
  160. // name: '',
  161. // value: '',
  162. // },
  163. },
  164. ],
  165. ruleForm: {
  166. nodeId: 0,
  167. isPk: 0,
  168. name: '',
  169. key: '',
  170. dataType: '',
  171. value: '',
  172. rule: [],
  173. description: '',
  174. },
  175. rules: {
  176. key: [{ required: true, message: '数据节点标识不能为空', trigger: 'blur' }],
  177. name: [{ required: true, message: '数据节点名称不能为空', trigger: 'blur' }],
  178. isPk: [{ required: true, message: '请选择是否主键', trigger: 'blur' }],
  179. dataType: [{ required: true, message: '数据节点类型不能为空', trigger: 'blur' }],
  180. value: [{ required: true, message: '数据节点取值项不能为空', trigger: 'blur' }],
  181. },
  182. });
  183. const onTest = () => {
  184. if (state.detail.from == 1) {
  185. api.common.api(state.detail.sourceId).then((res: any) => {
  186. state.jsonData = JSON.parse(res.data);
  187. var jsonstr = jsonPath([], state.jsonData, '');
  188. state.jsonPathData = jsonstr;
  189. state.dialogVisible = true;
  190. });
  191. } else if (state.detail.from == 4) {
  192. api.common.devapi(state.detail.sourceId).then((res: any) => {
  193. state.jsonData = JSON.parse(res.data);
  194. var jsonstr = jsonPath([], state.jsonData, '');
  195. state.jsonPathData = jsonstr;
  196. state.dialogVisible = true;
  197. });
  198. }
  199. };
  200. const handlePath = (path) => {
  201. let data = path.slice(1);
  202. state.ruleForm.value = data;
  203. state.dialogVisible = false;
  204. };
  205. const delRule = (index) => {
  206. state.rule.splice(index, 1);
  207. };
  208. const addRule = () => {
  209. state.rule.push({
  210. expression: '',
  211. replace: '',
  212. // params: {
  213. // name: '',
  214. // value: '',
  215. // },
  216. });
  217. };
  218. // 打开弹窗
  219. const openDialog = (row: RuleFormState | null) => {
  220. resetForm();
  221. if (row?.nodeId) {
  222. state.ruleForm = row;
  223. var data = JSON.parse(row.rule);
  224. data.forEach((item, index) => {
  225. state.rule[index].expression = item.expression;
  226. state.rule[index].replace = item.replace;
  227. // state.rule[index].params.name = Object.keys(item.params);
  228. // state.rule[index].params.value = item.params[Object.keys(item.params)];
  229. });
  230. }
  231. api.common.detail(row.sourceId).then((res: any) => {
  232. state.detail = res.data;
  233. if (res.data.from == 4) {
  234. //propertyData
  235. api.node.getpropertyList({ key: res.data.deviceConfig.productKey }).then((re: any) => {
  236. state.propertyData = re;
  237. });
  238. }
  239. if (res.data.from == 2) {
  240. //propertyData
  241. api.common.getfields(row.sourceId).then((re: any) => {
  242. state.dbData = re.data;
  243. });
  244. }
  245. });
  246. state.ruleForm = row;
  247. state.isShowDialog = true;
  248. };
  249. const resetForm = () => {
  250. state.ruleForm = {
  251. nodeId: 0,
  252. name: '',
  253. isPk: 0,
  254. from: 1,
  255. key: '',
  256. rule: [],
  257. description: '',
  258. };
  259. };
  260. // 关闭弹窗
  261. const closeDialog = () => {
  262. state.isShowDialog = false;
  263. };
  264. // 取消
  265. const onCancel = () => {
  266. closeDialog();
  267. };
  268. // 新增
  269. const onSubmit = () => {
  270. const formWrap = unref(formRef) as any;
  271. if (!formWrap) return;
  272. formWrap.validate((valid: boolean) => {
  273. if (valid) {
  274. //修改rule数据
  275. // state.rule.forEach((item, index) => {
  276. // item.params[item.params.name] = item.params.value;
  277. // delete item.params.name;
  278. // delete item.params.value;
  279. // });
  280. state.ruleForm.rule = state.rule;
  281. if (state.ruleForm.nodeId !== 0) {
  282. //修改
  283. api.node.edit(state.ruleForm).then(() => {
  284. ElMessage.success('数据节点类型修改成功');
  285. closeDialog(); // 关闭弹窗
  286. emit('typeList');
  287. });
  288. } else {
  289. //添加
  290. api.node.add(state.ruleForm).then(() => {
  291. ElMessage.success('数据节点类型添加成功');
  292. closeDialog(); // 关闭弹窗
  293. emit('typeList');
  294. });
  295. }
  296. }
  297. });
  298. };
  299. const getNodeList = (e) => {
  300. state.propertyData.forEach((item) => {
  301. if (item.key === e) {
  302. state.ruleForm.dataType = item.valueType.type;
  303. }
  304. });
  305. };
  306. const getDbList = (e) => {
  307. state.ruleForm.dataType = state.dbData[e].Type;
  308. }
  309. const onKeyclick = () => {
  310. };
  311. const jsonPath = (arr, json, basePath) => {
  312. // 生成jsonpath
  313. const type = validateType(json);
  314. if (type === 'object') {
  315. for (let key in json) {
  316. const item = {
  317. key,
  318. path: `${basePath}.${key}`,
  319. };
  320. const childType = validateType(json[key]);
  321. item.type = childType;
  322. if (childType === 'object' || childType === 'array') {
  323. item.leaf = true;
  324. item.children = [];
  325. jsonPath(item.children, json[key], item.path);
  326. } else {
  327. item.leaf = false;
  328. item.value = json[key];
  329. }
  330. arr.push(item);
  331. }
  332. } else if (type === 'array') {
  333. json.forEach((item, index) => {
  334. const childType = validateType(item);
  335. const obj = {
  336. key: index,
  337. };
  338. obj.type = childType;
  339. obj.path = `${basePath}.${index}`;
  340. if (childType === 'object' || childType === 'array') {
  341. (obj.leaf = true), (obj.children = []);
  342. jsonPath(obj.children, item, obj.path);
  343. } else {
  344. obj.value = item;
  345. obj.leaf = false;
  346. }
  347. arr.push(obj);
  348. });
  349. }
  350. return arr;
  351. };
  352. // 校验JSON数据类型
  353. const validateType = (val) => {
  354. let type = typeof val;
  355. if (type === 'object') {
  356. if (Array.isArray(val)) {
  357. return 'array';
  358. } else if (val === null) {
  359. return 'null';
  360. } else {
  361. return 'object';
  362. }
  363. } else {
  364. switch (type) {
  365. case 'boolean':
  366. return 'boolean';
  367. case 'string':
  368. return 'string';
  369. case 'number':
  370. return 'number';
  371. default:
  372. return 'error';
  373. }
  374. }
  375. };
  376. const getOrgIdArr = (parents, childNode, treeData) => {
  377. for (var key in treeData) {
  378. // 父节点查询条件
  379. if (key === childNode) {
  380. // 如果找到结果,保存当前节点
  381. parents.push(key);
  382. break;
  383. } else {
  384. if (treeData[key] instanceof Object) {
  385. parents.push(key);
  386. //没找到,遍历该节点的子节点
  387. getOrgIdArr(parents, childNode, treeData[key]);
  388. }
  389. }
  390. }
  391. return parents;
  392. };
  393. return {
  394. jsonPath,
  395. handlePath,
  396. validateType,
  397. getNodeList,
  398. getDbList,
  399. onKeyclick,
  400. getOrgIdArr,
  401. addRule,
  402. onTest,
  403. delRule,
  404. openDialog,
  405. closeDialog,
  406. onCancel,
  407. onSubmit,
  408. formRef,
  409. ...toRefs(state),
  410. };
  411. },
  412. });
  413. </script>
  414. <style>
  415. .el-input__wrapper {
  416. width: 98%;
  417. }
  418. .box-content {
  419. border: 1px solid #e8e8e8;
  420. margin: 10px;
  421. padding: 10px;
  422. }
  423. .content-f {
  424. display: flex;
  425. margin-bottom: 10px;
  426. }
  427. .content-f .el-input__wrapper {
  428. margin-right: 5px;
  429. }
  430. .addbutton {
  431. width: 100%;
  432. margin-top: 10px;
  433. }
  434. .conicon {
  435. width: 55px;
  436. height: 25px;
  437. font-size: 28px;
  438. line-height: 28px;
  439. cursor: pointer;
  440. }
  441. .jv-key {
  442. cursor: pointer;
  443. color: #0034f1;
  444. }
  445. </style>