index.vue 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804
  1. <template>
  2. <PageWrapper title="表单基础示例" contentFullHeight>
  3. <CollapseContainer title="基础示例">
  4. <BasicForm
  5. autoFocusFirstItem
  6. :labelWidth="200"
  7. :schemas="schemas"
  8. :actionColOptions="{ span: 24 }"
  9. @submit="handleSubmit"
  10. @reset="handleReset"
  11. >
  12. <template #selectA="{ model, field }">
  13. <Select
  14. :options="optionsA"
  15. mode="multiple"
  16. v-model:value="model[field]"
  17. @change="valueSelectA = model[field]"
  18. allowClear
  19. />
  20. </template>
  21. <template #selectB="{ model, field }">
  22. <Select
  23. :options="optionsB"
  24. mode="multiple"
  25. v-model:value="model[field]"
  26. @change="valueSelectB = model[field]"
  27. allowClear
  28. />
  29. </template>
  30. <template #localSearch="{ model, field }">
  31. <ApiSelect
  32. :api="optionsListApi"
  33. showSearch
  34. v-model:value="model[field]"
  35. optionFilterProp="label"
  36. resultField="list"
  37. labelField="name"
  38. valueField="id"
  39. />
  40. </template>
  41. <template #remoteSearch="{ model, field }">
  42. <ApiSelect
  43. :api="optionsListApi"
  44. showSearch
  45. v-model:value="model[field]"
  46. :filterOption="false"
  47. resultField="list"
  48. labelField="name"
  49. valueField="id"
  50. :params="searchParams"
  51. @search="useDebounceFn(onSearch, 300)"
  52. />
  53. </template>
  54. </BasicForm>
  55. </CollapseContainer>
  56. </PageWrapper>
  57. </template>
  58. <script lang="ts" setup>
  59. import { type Recordable } from '@vben/types';
  60. import { computed, unref, ref } from 'vue';
  61. import { BasicForm, FormSchema, ApiSelect } from '@/components/Form';
  62. import { CollapseContainer } from '@/components/Container';
  63. import { useMessage } from '@/hooks/web/useMessage';
  64. import { PageWrapper } from '@/components/Page';
  65. import { optionsListApi } from '@/api/demo/select';
  66. import { useDebounceFn } from '@vueuse/core';
  67. import { treeOptionsListApi } from '@/api/demo/tree';
  68. import { Select, type SelectProps } from 'ant-design-vue';
  69. import { cloneDeep } from 'lodash-es';
  70. import { areaRecord } from '@/api/demo/cascader';
  71. import { uploadApi } from '@/api/sys/upload';
  72. const valueSelectA = ref<string[]>([]);
  73. const valueSelectB = ref<string[]>([]);
  74. const options = ref<Required<SelectProps>['options']>([]);
  75. for (let i = 1; i < 10; i++) options.value.push({ label: '选项' + i, value: `${i}` });
  76. const optionsA = computed(() => {
  77. return cloneDeep(unref(options)).map((op) => {
  78. op.disabled = unref(valueSelectB).indexOf(op.value as string) !== -1;
  79. return op;
  80. });
  81. });
  82. const optionsB = computed(() => {
  83. return cloneDeep(unref(options)).map((op) => {
  84. op.disabled = unref(valueSelectA).indexOf(op.value as string) !== -1;
  85. return op;
  86. });
  87. });
  88. const provincesOptions = [
  89. {
  90. id: 'guangdong',
  91. label: '广东省',
  92. value: '1',
  93. key: '1',
  94. },
  95. {
  96. id: 'jiangsu',
  97. label: '江苏省',
  98. value: '2',
  99. key: '2',
  100. },
  101. ];
  102. const citiesOptionsData = {
  103. guangdong: [
  104. {
  105. label: '珠海市',
  106. value: '1',
  107. key: '1',
  108. },
  109. {
  110. label: '深圳市',
  111. value: '2',
  112. key: '2',
  113. },
  114. {
  115. label: '广州市',
  116. value: '3',
  117. key: '3',
  118. },
  119. ],
  120. jiangsu: [
  121. {
  122. label: '南京市',
  123. value: '1',
  124. key: '1',
  125. },
  126. {
  127. label: '无锡市',
  128. value: '2',
  129. key: '2',
  130. },
  131. {
  132. label: '苏州市',
  133. value: '3',
  134. key: '3',
  135. },
  136. ],
  137. };
  138. const schemas: FormSchema[] = [
  139. {
  140. field: 'divider-basic',
  141. component: 'Divider',
  142. label: '基础字段',
  143. colProps: {
  144. span: 24,
  145. },
  146. },
  147. {
  148. field: 'field1',
  149. component: 'Input',
  150. label: '字段1',
  151. colProps: {
  152. span: 8,
  153. },
  154. // componentProps:{},
  155. // can func
  156. componentProps: ({ schema, formModel }) => {
  157. console.log('form:', schema);
  158. console.log('formModel:', formModel);
  159. return {
  160. placeholder: '自定义placeholder',
  161. onChange: (e: any) => {
  162. console.log(e);
  163. },
  164. };
  165. },
  166. renderComponentContent: () => {
  167. return {
  168. prefix: () => 'pSlot',
  169. suffix: () => 'sSlot',
  170. };
  171. },
  172. },
  173. {
  174. field: 'field2',
  175. component: 'Input',
  176. label: '带后缀',
  177. defaultValue: '111',
  178. colProps: {
  179. span: 8,
  180. },
  181. componentProps: {
  182. onChange: (e: any) => {
  183. console.log(e);
  184. },
  185. },
  186. suffix: '天',
  187. },
  188. {
  189. field: 'fieldsc',
  190. component: 'Upload',
  191. label: '上传',
  192. colProps: {
  193. span: 8,
  194. },
  195. rules: [{ required: true, message: '请选择上传文件' }],
  196. componentProps: {
  197. api: uploadApi,
  198. },
  199. },
  200. {
  201. field: 'field3',
  202. component: 'DatePicker',
  203. label: '字段3',
  204. colProps: {
  205. span: 8,
  206. },
  207. },
  208. {
  209. field: 'field4',
  210. component: 'Select',
  211. label: '字段4',
  212. colProps: {
  213. span: 8,
  214. },
  215. componentProps: {
  216. options: [
  217. {
  218. label: '选项1',
  219. value: '1',
  220. key: '1',
  221. },
  222. {
  223. label: '选项2',
  224. value: '2',
  225. key: '2',
  226. },
  227. ],
  228. },
  229. },
  230. {
  231. field: 'field5',
  232. component: 'CheckboxGroup',
  233. label: '字段5',
  234. colProps: {
  235. span: 8,
  236. },
  237. componentProps: {
  238. options: [
  239. {
  240. label: '选项1',
  241. value: '1',
  242. },
  243. {
  244. label: '选项2',
  245. value: '2',
  246. },
  247. ],
  248. },
  249. },
  250. {
  251. field: 'field7',
  252. component: 'RadioGroup',
  253. label: '字段7',
  254. colProps: {
  255. span: 8,
  256. },
  257. componentProps: {
  258. options: [
  259. {
  260. label: '选项1',
  261. value: '1',
  262. },
  263. {
  264. label: '选项2',
  265. value: '2',
  266. },
  267. ],
  268. },
  269. },
  270. {
  271. field: 'field8',
  272. component: 'Checkbox',
  273. label: '字段8',
  274. colProps: {
  275. span: 8,
  276. },
  277. renderComponentContent: 'Check',
  278. },
  279. {
  280. field: 'field9',
  281. component: 'Switch',
  282. label: '字段9',
  283. colProps: {
  284. span: 8,
  285. },
  286. },
  287. {
  288. field: 'field10',
  289. component: 'RadioButtonGroup',
  290. label: '字段10',
  291. colProps: {
  292. span: 8,
  293. },
  294. componentProps: {
  295. options: [
  296. {
  297. label: '选项1',
  298. value: '1',
  299. },
  300. {
  301. label: '选项2',
  302. value: '2',
  303. },
  304. ],
  305. onChange: (e, v) => {
  306. console.log('RadioButtonGroup====>:', e, v);
  307. },
  308. },
  309. },
  310. {
  311. field: 'field11',
  312. component: 'Cascader',
  313. label: '字段11',
  314. colProps: {
  315. span: 8,
  316. },
  317. componentProps: {
  318. options: [
  319. {
  320. value: 'zhejiang',
  321. label: 'Zhejiang',
  322. children: [
  323. {
  324. value: 'hangzhou',
  325. label: 'Hangzhou',
  326. children: [
  327. {
  328. value: 'xihu',
  329. label: 'West Lake',
  330. },
  331. ],
  332. },
  333. ],
  334. },
  335. {
  336. value: 'jiangsu',
  337. label: 'Jiangsu',
  338. children: [
  339. {
  340. value: 'nanjing',
  341. label: 'Nanjing',
  342. children: [
  343. {
  344. value: 'zhonghuamen',
  345. label: 'Zhong Hua Men',
  346. },
  347. ],
  348. },
  349. ],
  350. },
  351. ],
  352. },
  353. },
  354. {
  355. field: 'divider-api-select',
  356. component: 'Divider',
  357. label: '远程下拉演示',
  358. colProps: {
  359. span: 24,
  360. },
  361. },
  362. {
  363. field: 'field30',
  364. component: 'ApiSelect',
  365. label: '懒加载远程下拉',
  366. required: true,
  367. componentProps: {
  368. // more details see /src/components/Form/src/components/ApiSelect.vue
  369. api: optionsListApi,
  370. params: {
  371. id: 1,
  372. },
  373. resultField: 'list',
  374. // use name as label
  375. labelField: 'name',
  376. // use id as value
  377. valueField: 'id',
  378. // not request untill to select
  379. immediate: true,
  380. onChange: (e, v) => {
  381. console.log('ApiSelect====>:', e, v);
  382. },
  383. // atfer request callback
  384. onOptionsChange: (options) => {
  385. console.log('get options', options.length, options);
  386. },
  387. },
  388. colProps: {
  389. span: 8,
  390. },
  391. defaultValue: '0',
  392. },
  393. {
  394. field: 'field8',
  395. component: 'ApiCascader',
  396. label: '联动ApiCascader',
  397. required: true,
  398. colProps: {
  399. span: 8,
  400. },
  401. componentProps: {
  402. api: areaRecord,
  403. apiParamKey: 'parentCode',
  404. dataField: 'data',
  405. labelField: 'name',
  406. valueField: 'code',
  407. initFetchParams: {
  408. parentCode: '',
  409. },
  410. isLeaf: (record) => {
  411. return !(record.levelType < 3);
  412. },
  413. onChange: (e, ...v) => {
  414. console.log('ApiCascader====>:', e, v);
  415. },
  416. },
  417. },
  418. {
  419. field: 'field31',
  420. component: 'Input',
  421. label: '下拉本地搜索',
  422. helpMessage: ['ApiSelect组件', '远程数据源本地搜索', '只发起一次请求获取所有选项'],
  423. required: true,
  424. slot: 'localSearch',
  425. colProps: {
  426. span: 8,
  427. },
  428. defaultValue: '0',
  429. },
  430. {
  431. field: 'field32',
  432. component: 'Input',
  433. label: '下拉远程搜索',
  434. helpMessage: ['ApiSelect组件', '将关键词发送到接口进行远程搜索'],
  435. required: true,
  436. slot: 'remoteSearch',
  437. colProps: {
  438. span: 8,
  439. },
  440. defaultValue: '0',
  441. },
  442. {
  443. field: 'field33',
  444. component: 'ApiTreeSelect',
  445. label: '远程下拉树',
  446. helpMessage: ['ApiTreeSelect组件', '使用接口提供的数据生成选项'],
  447. required: true,
  448. componentProps: {
  449. api: treeOptionsListApi,
  450. resultField: 'list',
  451. onChange: (e, v) => {
  452. console.log('ApiTreeSelect====>:', e, v);
  453. },
  454. },
  455. colProps: {
  456. span: 8,
  457. },
  458. },
  459. {
  460. field: 'field33',
  461. component: 'ApiTreeSelect',
  462. label: '远程懒加载下拉树',
  463. helpMessage: ['ApiTreeSelect组件', '使用接口提供的数据生成选项'],
  464. required: true,
  465. componentProps: {
  466. api: () => {
  467. return new Promise((resolve) => {
  468. resolve([
  469. {
  470. title: 'Parent Node',
  471. value: '0-0',
  472. },
  473. ]);
  474. });
  475. },
  476. async: true,
  477. onChange: (e, v) => {
  478. console.log('ApiTreeSelect====>:', e, v);
  479. },
  480. onLoadData: ({ treeData, resolve, treeNode }) => {
  481. console.log('treeNode====>:', treeNode);
  482. setTimeout(() => {
  483. const children: Recordable[] = [
  484. { title: `Child Node ${treeNode.eventKey}-0`, value: `${treeNode.eventKey}-0` },
  485. { title: `Child Node ${treeNode.eventKey}-1`, value: `${treeNode.eventKey}-1` },
  486. ];
  487. children.forEach((item) => {
  488. item.isLeaf = false;
  489. item.children = [];
  490. });
  491. treeNode.dataRef.children = children;
  492. treeData.value = [...treeData.value];
  493. resolve();
  494. return;
  495. }, 300);
  496. },
  497. },
  498. colProps: {
  499. span: 8,
  500. },
  501. },
  502. {
  503. field: 'field34',
  504. component: 'ApiRadioGroup',
  505. label: '远程Radio',
  506. helpMessage: ['ApiRadioGroup组件', '使用接口提供的数据生成选项'],
  507. required: true,
  508. componentProps: {
  509. api: optionsListApi,
  510. params: {
  511. count: 2,
  512. },
  513. resultField: 'list',
  514. // use name as label
  515. labelField: 'name',
  516. // use id as value
  517. valueField: 'id',
  518. },
  519. defaultValue: '1',
  520. colProps: {
  521. span: 8,
  522. },
  523. },
  524. {
  525. field: 'field35',
  526. component: 'ApiRadioGroup',
  527. label: '远程Radio',
  528. helpMessage: ['ApiRadioGroup组件', '使用接口提供的数据生成选项'],
  529. required: true,
  530. componentProps: {
  531. api: optionsListApi,
  532. params: {
  533. count: 2,
  534. },
  535. resultField: 'list',
  536. // use name as label
  537. labelField: 'name',
  538. // use id as value
  539. valueField: 'id',
  540. isBtn: true,
  541. onChange: (e, v) => {
  542. console.log('ApiRadioGroup====>:', e, v);
  543. },
  544. },
  545. colProps: {
  546. span: 8,
  547. },
  548. },
  549. {
  550. field: 'field36',
  551. component: 'ApiTree',
  552. label: '远程Tree',
  553. helpMessage: ['ApiTree组件', '使用接口提供的数据生成选项'],
  554. required: true,
  555. componentProps: {
  556. api: treeOptionsListApi,
  557. params: {
  558. count: 2,
  559. },
  560. afterFetch: (v) => {
  561. //do something
  562. return v;
  563. },
  564. resultField: 'list',
  565. },
  566. colProps: {
  567. span: 8,
  568. },
  569. },
  570. {
  571. label: '远程穿梭框',
  572. field: 'field37',
  573. component: 'ApiTransfer',
  574. componentProps: {
  575. render: (item) => item.label,
  576. api: async () => {
  577. return Promise.resolve(citiesOptionsData.guangdong);
  578. },
  579. },
  580. defaultValue: ['1'],
  581. required: true,
  582. },
  583. {
  584. field: 'divider-linked',
  585. component: 'Divider',
  586. label: '字段联动',
  587. colProps: {
  588. span: 24,
  589. },
  590. },
  591. {
  592. field: 'province',
  593. component: 'Select',
  594. label: '省份',
  595. colProps: {
  596. span: 8,
  597. },
  598. componentProps: ({ formModel, formActionType }) => {
  599. return {
  600. options: provincesOptions,
  601. placeholder: '省份与城市联动',
  602. onChange: (e: any) => {
  603. // console.log(e)
  604. let citiesOptions =
  605. e == 1
  606. ? citiesOptionsData[provincesOptions[0].id]
  607. : citiesOptionsData[provincesOptions[1].id];
  608. // console.log(citiesOptions)
  609. if (e === undefined) {
  610. citiesOptions = [];
  611. }
  612. formModel.city = undefined; // reset city value
  613. const { updateSchema } = formActionType;
  614. updateSchema({
  615. field: 'city',
  616. componentProps: {
  617. options: citiesOptions,
  618. },
  619. });
  620. },
  621. };
  622. },
  623. },
  624. {
  625. field: 'city',
  626. component: 'Select',
  627. label: '城市',
  628. colProps: {
  629. span: 8,
  630. },
  631. componentProps: {
  632. options: [], // defalut []
  633. placeholder: '省份与城市联动',
  634. },
  635. },
  636. {
  637. field: 'divider-selects',
  638. component: 'Divider',
  639. label: '互斥多选',
  640. helpMessage: ['两个Select共用数据源', '但不可选择对方已选中的项目'],
  641. colProps: {
  642. span: 24,
  643. },
  644. },
  645. {
  646. field: 'selectA',
  647. component: 'Select',
  648. label: '互斥SelectA',
  649. slot: 'selectA',
  650. defaultValue: [],
  651. colProps: {
  652. span: 8,
  653. },
  654. },
  655. {
  656. field: 'selectB',
  657. component: 'Select',
  658. label: '互斥SelectB',
  659. slot: 'selectB',
  660. defaultValue: [],
  661. colProps: {
  662. span: 8,
  663. },
  664. },
  665. {
  666. field: 'divider-deconstruct',
  667. component: 'Divider',
  668. label: '字段解构',
  669. helpMessage: ['如果组件的值是 array 或者 object', '可以根据 ES6 的解构语法分别取值'],
  670. colProps: {
  671. span: 24,
  672. },
  673. },
  674. {
  675. field: '[startTime, endTime]',
  676. label: '时间范围',
  677. component: 'TimeRangePicker',
  678. componentProps: {
  679. format: 'HH:mm:ss',
  680. placeholder: ['开始时间', '结束时间'],
  681. },
  682. },
  683. {
  684. field: '[startDate, endDate]',
  685. label: '日期范围',
  686. component: 'RangePicker',
  687. componentProps: {
  688. format: 'YYYY-MM-DD',
  689. placeholder: ['开始日期', '结束日期'],
  690. },
  691. },
  692. {
  693. field: '[startDateTime, endDateTime]',
  694. label: '日期时间范围',
  695. component: 'RangePicker',
  696. componentProps: {
  697. format: 'YYYY-MM-DD HH:mm:ss',
  698. placeholder: ['开始日期、时间', '结束日期、时间'],
  699. showTime: { format: 'HH:mm:ss' },
  700. },
  701. },
  702. {
  703. field: 'divider-others',
  704. component: 'Divider',
  705. label: '其它',
  706. colProps: {
  707. span: 24,
  708. },
  709. },
  710. {
  711. field: 'field20',
  712. component: 'InputNumber',
  713. label: '字段20',
  714. required: true,
  715. colProps: {
  716. span: 8,
  717. },
  718. },
  719. {
  720. field: 'field21',
  721. component: 'Slider',
  722. label: '字段21',
  723. componentProps: {
  724. min: 0,
  725. max: 100,
  726. range: true,
  727. marks: {
  728. 20: '20°C',
  729. 60: '60°C',
  730. },
  731. },
  732. colProps: {
  733. span: 8,
  734. },
  735. },
  736. {
  737. field: 'field22',
  738. component: 'Rate',
  739. label: '字段22',
  740. defaultValue: 3,
  741. colProps: {
  742. span: 8,
  743. },
  744. componentProps: {
  745. disabled: false,
  746. allowHalf: true,
  747. },
  748. },
  749. {
  750. field: 'field23',
  751. component: 'ImageUpload',
  752. label: '上传图片',
  753. required: true,
  754. defaultValue: [
  755. 'https://zos.alipayobjects.com/rmsportal/jkjgkEfvpUPVyRjUImniVslZfWPnJuuZ.png',
  756. ],
  757. componentProps: {
  758. api: uploadApi,
  759. accept: ['png', 'jpeg', 'jpg'],
  760. maxSize: 2,
  761. maxNumber: 1,
  762. },
  763. // rules: [
  764. // {
  765. // required: true,
  766. // trigger: 'change',
  767. // validator(_, value) {
  768. // if (isArray(value) && value.length > 0) {
  769. // return Promise.resolve();
  770. // } else {
  771. // return Promise.reject('请选择上传图片');
  772. // }
  773. // },
  774. // },
  775. // ],
  776. },
  777. ];
  778. const { createMessage } = useMessage();
  779. const keyword = ref<string>('');
  780. const searchParams = computed<Recordable<string>>(() => {
  781. return { keyword: unref(keyword) };
  782. });
  783. function onSearch(value: string) {
  784. keyword.value = value;
  785. }
  786. function handleReset() {
  787. keyword.value = '';
  788. }
  789. function handleSubmit(values: any) {
  790. console.log('values', values);
  791. createMessage.success('click search,values:' + JSON.stringify(values));
  792. }
  793. </script>