index.vue 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441
  1. <template>
  2. <div class="by-form">
  3. <el-form
  4. :model="formData"
  5. :label-width="formOption.labelWidth"
  6. :inline="formOption.inline || false"
  7. :rules="rules"
  8. :labelPosition="formOption.labelPosition || 'top'"
  9. ref="byForm"
  10. :disabled="formOption.disabled || false">
  11. <template v-for="i in formConfig" :key="i.model">
  12. <el-form-item
  13. :label="i.label"
  14. :prop="i.prop"
  15. v-if="(i.isShow || i.isShow == undefined) && i.type"
  16. :style="
  17. i.type == 'title'
  18. ? 'width:100%'
  19. : i.itemWidth
  20. ? 'width:' + i.itemWidth + '%'
  21. : formOption.itemWidth
  22. ? 'width:' + formOption.itemWidth + '%'
  23. : '100%'
  24. "
  25. :class="i.type == 'json' ? (i.isHide ? 'by-form-json dn' : 'by-form-json') : i.isHide ? 'dn' : ''">
  26. <el-input
  27. v-if="i.type == 'input'"
  28. v-model="formData[i.prop]"
  29. :placeholder="i.placeholder || $t('common.pleaseEnter')"
  30. @input="(e) => commonsEmit(e, i)"
  31. @change="(e) => commonsEmitChange(e, i)"
  32. :type="i.itemType ? i.itemType : 'text'"
  33. :disabled="i.disabled ? i.disabled : false"
  34. :max="i.max"
  35. :min="i.min"
  36. :maxlength="i.maxlength"
  37. :readonly="i.readonly ? i.readonly : false"
  38. :style="i.style" />
  39. <el-input
  40. v-if="i.type == 'selectInput'"
  41. v-model="formData[i.prop]"
  42. :placeholder="i.placeholder || $t('common.pleaseEnter')"
  43. @input="(e) => commonsEmit(e, i)"
  44. :type="i.itemType ? i.itemType : 'text'"
  45. :disabled="i.disabled ? i.disabled : false"
  46. :max="i.max"
  47. :min="i.min"
  48. :maxlength="i.maxlength"
  49. :readonly="i.readonly ? i.readonly : false">
  50. <template #prepend>
  51. <el-select
  52. v-model="formData[i.selectProp]"
  53. :placeholder="i.selectPlaceholder || $t('common.pleaseSelect')"
  54. @change="(e) => commonsEmit(e, i)"
  55. :disabled="i.disabled ? i.disabled : false"
  56. :readonly="i.readonly ? i.readonly : false"
  57. style="width: 80px">
  58. <el-option :label="j.title || j.name || j.label" :value="j.id || j.value" v-for="j in i.data" :key="j.id"> </el-option>
  59. </el-select>
  60. </template>
  61. </el-input>
  62. <el-select
  63. v-model="formData[i.prop]"
  64. :multiple="i.multiple || false"
  65. v-else-if="i.type == 'select'"
  66. :placeholder="i.placeholder || $t('common.pleaseSelect')"
  67. @change="(e) => commonsEmit(e, i)"
  68. :disabled="i.disabled ? i.disabled : false"
  69. :clearable="i.clearable ? i.clearable : false"
  70. :filterable="i.filterable ? true : false"
  71. :style="i.style"
  72. :readonly="i.readonly ? i.readonly : false"
  73. style="width: 100%">
  74. <el-option :label="j.dictValue || j.title || j.name || j.label" :value="j.dictKey || j.id || j.value" v-for="j in i.data" :key="j.id"> </el-option>
  75. </el-select>
  76. <el-tree-select
  77. v-model="formData[i.prop]"
  78. v-else-if="i.type == 'treeSelect'"
  79. :data="i.data"
  80. :readonly="i.readonly ? i.readonly : false"
  81. :props="{
  82. value: i.propsTreeValue || 'id',
  83. label: i.propsTreeLabel || 'label',
  84. children: i.propsTreeChildren || 'children',
  85. }"
  86. value-key="id"
  87. :placeholder="i.placeholder || $t('common.pleaseSelect')"
  88. :disabled="i.disabled ? i.disabled : false"
  89. check-strictly
  90. :style="i.style"
  91. style="width: 100%" />
  92. <el-date-picker
  93. v-model="formData[i.prop]"
  94. :readonly="i.readonly ? i.readonly : false"
  95. v-else-if="i.type == 'date'"
  96. :type="i.itemType"
  97. :placeholder="i.placeholder || $t('common.pleaseSelectTime')"
  98. @change="(e) => commonsEmit(e, i)"
  99. :disabled="i.disabled ? i.disabled : false"
  100. :format="i.format ? i.format : dateFormatInit(i.itemType)"
  101. :value-format="i.format ? i.format : dateFormatInit(i.itemType)"
  102. style="width: 100%" />
  103. <el-switch
  104. :disabled="i.disabled ? i.disabled : false"
  105. v-else-if="i.type == 'switch'"
  106. :readonly="i.readonly ? i.readonly : false"
  107. v-model="formData[i.prop]" />
  108. <el-checkbox-group
  109. v-else-if="i.type == 'checkbox'"
  110. v-model="formData[i.prop]"
  111. :readonly="i.readonly ? i.readonly : false"
  112. :disabled="i.disabled ? i.disabled : false">
  113. <el-checkbox v-for="j in i.data" :key="j.id || j.value" :label="j.id || j.value" name="type">
  114. {{ j.name || j.label }}
  115. </el-checkbox>
  116. </el-checkbox-group>
  117. <el-radio-group
  118. v-else-if="i.type == 'radio'"
  119. v-model="formData[i.prop]"
  120. :readonly="i.readonly ? i.readonly : false"
  121. :disabled="i.disabled ? i.disabled : false">
  122. <el-radio :border="i.border ? i.border : false" v-for="j in i.data" :key="j.id || j.value" :label="j.id || j.value" name="type">
  123. {{ j.name || j.label }}
  124. </el-radio>
  125. </el-radio-group>
  126. <el-input-number
  127. v-else-if="i.type == 'number'"
  128. v-model="formData[i.prop]"
  129. :readonly="i.readonly ? i.readonly : false"
  130. :placeholder="i.placeholder || $t('common.pleaseEnter')"
  131. @change="(e) => commonsEmit(e, i)"
  132. :disabled="i.disabled ? i.disabled : false"
  133. :min="i.min ? i.min : 0"
  134. :max="i.max ? i.max : 9999999999"
  135. :step="i.step ? i.step : 1"
  136. :precision="i.precision !== '' ? i.precision : 2"
  137. :controls="i.controls === false ? false : true"
  138. :style="i.style"
  139. onmousewheel="return false;"
  140. style="width: 100%">
  141. </el-input-number>
  142. <el-tree
  143. v-else-if="i.type == 'tree'"
  144. :data="i.data"
  145. :props="i.props"
  146. :readonly="i.readonly ? i.readonly : false"
  147. :show-checkbox="i.showCheckbox || true">
  148. </el-tree>
  149. <el-cascader
  150. v-else-if="i.type == 'cascader'"
  151. :options="i.data"
  152. :props="i.props"
  153. :readonly="i.readonly ? i.readonly : false"
  154. :placeholder="i.placeholder || $t('common.pleaseSelect')"
  155. @change="(e) => commonsEmit(e, i)"
  156. :disabled="i.disabled ? i.disabled : false"
  157. :style="i.style">
  158. </el-cascader>
  159. <div class="form-title" v-else-if="i.type == 'title'">
  160. {{ i.title }}
  161. </div>
  162. <slot :name="i.slotName" v-else-if="i.type == 'slot'"> {{ i.slotName }}插槽占位符 </slot>
  163. <div class="upload" v-else-if="i.type == 'upload'">
  164. <el-upload
  165. v-model="formData[i.prop]"
  166. action="https://winfaster.obs.cn-south-1.myhuaweicloud.com"
  167. :data="uploadData"
  168. list-type="picture-card"
  169. :on-remove="handleRemove"
  170. :on-success="handleSuccess"
  171. :before-upload="handleBeforeUpload">
  172. <el-icon><Plus /></el-icon>
  173. </el-upload>
  174. </div>
  175. <div v-else-if="i.type == 'table'" class="by-form-table" style="width: 100%">
  176. <el-table :data="formData[i.prop]" style="width: 100%">
  177. <el-table-column :prop="j.prop" :label="j.label" :width="i.width" v-for="(j, jindex) in i.column">
  178. <template #default="scope" v-if="j.type">
  179. <component
  180. @change="(e) => formTableChange(e, scope, j)"
  181. v-model="scope.row[j.prop]"
  182. :is="formTableObj[j.type]"
  183. :placeholder="j.placeholder || $t('common.pleaseEnter')"
  184. :type="j.type == 'number' ? 'number' : 'text'">
  185. <el-option :label="n.title || n.name || n.label" :value="n.id || n.value" v-for="n in j.data" :key="n.id" v-if="j.type == 'select'">
  186. </el-option>
  187. </component>
  188. </template>
  189. </el-table-column>
  190. </el-table>
  191. </div>
  192. <div v-else-if="i.type == 'json'">
  193. <byForm :formConfig="i.json" :formOption="formOption" v-model="formData[i.prop]" ref="byform" :rules="rules"> </byForm>
  194. </div>
  195. </el-form-item>
  196. </template>
  197. </el-form>
  198. </div>
  199. </template>
  200. <script>
  201. export default {
  202. name: "byForm",
  203. };
  204. </script>
  205. <script setup>
  206. defineProps({
  207. modelValue: {
  208. type: Object,
  209. default: false,
  210. },
  211. formConfig: {
  212. type: Array,
  213. default: false,
  214. },
  215. disabled: {
  216. type: Boolean,
  217. default: false,
  218. },
  219. formOption: {
  220. type: Object,
  221. default: false,
  222. },
  223. rules: {
  224. type: Object,
  225. default: false,
  226. },
  227. });
  228. const formTableChange = (e, scope, column) => {
  229. if (column.fn) {
  230. column.fn(e, scope);
  231. }
  232. };
  233. const formTableObj = {
  234. number: "el-input",
  235. text: "el-input",
  236. select: "el-select",
  237. input: "el-input",
  238. };
  239. const fileList = ref([]);
  240. const fileListCopy = ref([]);
  241. const uploadData = ref({});
  242. //文件上传相关方法
  243. const handleSuccess = (res, file, files) => {
  244. emit("update:modelValue", formData.value);
  245. };
  246. const handleRemove = (file) => {
  247. const index = fileListCopy.value.findIndex((x) => x.uid === file.uid || x.id === file.id);
  248. fileListCopy.value.splice(index, 1);
  249. };
  250. const handleBeforeUpload = async (file) => {
  251. const res = await proxy.post("/fileInfo/getSing", { fileName: file.name });
  252. uploadData.value = res.uploadBody;
  253. fileListCopy.value.push({
  254. id: res.id,
  255. fileName: res.fileName,
  256. path: res.fileUrl,
  257. url: res.fileUrl,
  258. uid: file.uid,
  259. });
  260. };
  261. const { proxy } = getCurrentInstance();
  262. const emit = defineEmits(["update:modelValue"]);
  263. const formData = computed(() => {
  264. return proxy.modelValue;
  265. });
  266. const commonsEmit = (prop, item) => {
  267. if (item.fn) {
  268. item.fn(prop);
  269. }
  270. emit("update:modelValue", formData.value);
  271. };
  272. const commonsEmitChange = (prop, item) => {
  273. if (item.type == "input") {
  274. formData.value[item.prop] = prop.trim();
  275. }
  276. if (item.fn) {
  277. item.fn(prop);
  278. }
  279. emit("update:modelValue", formData.value);
  280. };
  281. const loadInit = () => {
  282. const v = this;
  283. for (let i = 0; i < proxy.formConfig.length; i++) {
  284. const element = proxy.formConfig[i];
  285. if (element.isLoad) {
  286. commonGetdata(element.isLoad, i);
  287. }
  288. }
  289. };
  290. const dateFormatInit = (itemType) => {
  291. const formatObj = {
  292. year: "YYYY",
  293. month: "YYYY-MM",
  294. date: "YYYY-MM-DD HH:mm:ss",
  295. dates: "YYYY-MM-DD",
  296. datetime: "YYYY-MM-DD HH:mm:ss",
  297. monthrange: "YYYY-MM-DD HH:mm:ss",
  298. datetimerange: "YYYY-MM-DD HH:mm:ss",
  299. daterange: "YYYY-MM-DD HH:mm:ss",
  300. };
  301. return formatObj[itemType];
  302. };
  303. //公用递归,保证key,val统一
  304. const commonRecursive = (arr, labelKey, labelVal, childrenName) => {
  305. for (let i = 0; i < arr.length; i++) {
  306. if (labelKey == "stringArray") {
  307. arr[i] = {
  308. label: arr[i],
  309. value: arr[i],
  310. id: arr[i],
  311. title: arr[i],
  312. };
  313. } else {
  314. arr[i].title = arr[i].label = arr[i][labelKey];
  315. arr[i].id = arr[i].value = arr[i][labelVal];
  316. }
  317. if (childrenName) {
  318. arr[i].children = arr[i][childrenName];
  319. }
  320. arr[i].checked = false;
  321. typeof arr[i][labelVal] == String ? (arr[i].key = arr[i][labelVal]) : (arr[i].key = JSON.stringify(arr[i][labelVal]));
  322. if (childrenName) {
  323. this.commonRecursive(arr[i][childrenName], labelKey, labelVal, childrenName);
  324. }
  325. }
  326. };
  327. //请求form表单所需数据字典
  328. const commonGetdata = (isLoad, i) => {
  329. proxy[isLoad.method](isLoad.url, isLoad.req).then((message) => {
  330. if (getFormat(isLoad.resUrl, message) == undefined) {
  331. console.log("请查看isLoad配置是否正确url:" + isLoad.url);
  332. return;
  333. }
  334. proxy.formConfig[i].data = getFormat(isLoad.resUrl, message);
  335. if (isLoad.labelKey) {
  336. commonRecursive(proxy.formConfig[i].data, isLoad.labelKey, isLoad.labelVal, isLoad.childrenName);
  337. }
  338. });
  339. };
  340. //根据resurl获取数据
  341. const getFormat = (formatStr, props) => {
  342. if (!formatStr) return props;
  343. return formatStr.split(".").reduce((total, cur) => (!total ? "" : total[cur]), props);
  344. };
  345. //初始化所有表单
  346. const formDataInit = () => {
  347. var map = {
  348. input: "",
  349. radio: null,
  350. select: null,
  351. checkbox: [],
  352. date: "",
  353. datetime: "",
  354. daterange: [],
  355. datetimerange: [],
  356. year: null,
  357. month: null,
  358. switch: false,
  359. inputNumber: 0,
  360. cascader: [],
  361. Solt: null,
  362. Transfer: [],
  363. Upload: { path: null, id: null, name: null },
  364. password: "",
  365. treeSelect: "",
  366. json: {},
  367. };
  368. const formDataCopy = { ...formData.value };
  369. for (let i = 0; i < proxy.formConfig.length; i++) {
  370. const element = proxy.formConfig[i];
  371. if (formDataCopy[element.prop] || element.type === "slot") {
  372. continue;
  373. }
  374. if (map[element.itemType] != undefined) {
  375. formData.value[element.prop] = map[element.itemType];
  376. } else {
  377. formData.value[element.prop] = element.multiple ? [] : map[element.type];
  378. }
  379. }
  380. emit("update:modelValue", formData.value);
  381. };
  382. const handleSubmit = async (onSubmit) => {
  383. try {
  384. const flag = await proxy.$refs["byForm"].validate();
  385. if (flag) {
  386. const form = { ...formData.value };
  387. proxy.formConfig.map((item) => {
  388. if (item.type == "json") {
  389. form[item.prop] = JSON.stringify(form[item.prop]);
  390. }
  391. });
  392. emit("update:modelValue", form);
  393. onSubmit();
  394. return true;
  395. }
  396. } catch (err) {
  397. console.log("请检查表单!", err);
  398. return false;
  399. }
  400. };
  401. const byform = ref(null); // 延迟使用,因为还没有返回跟挂载
  402. onMounted(() => {});
  403. defineExpose({
  404. handleSubmit,
  405. });
  406. formDataInit();
  407. loadInit();
  408. </script>
  409. <style>
  410. .form-title {
  411. font-size: 14px;
  412. font-weight: bold;
  413. color: #6c88f1;
  414. width: 120px;
  415. border-bottom: 1px solid #6c88f1;
  416. }
  417. .by-form .el-form--inline .el-form-item {
  418. margin-right: 0px;
  419. box-sizing: border-box;
  420. }
  421. .by-form .el-form--inline > .el-form-item {
  422. padding: 0 10px;
  423. }
  424. /* .el-form--inline.el-form--label-top{
  425. justify-content: space-between;
  426. } */
  427. .dn {
  428. display: none !important;
  429. }
  430. .by-form-json .by-form .el-form .el-form-item {
  431. margin-bottom: 18px;
  432. }
  433. .by-form-json {
  434. padding: 0px !important;
  435. }
  436. </style>