index.vue 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735
  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. <a-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. <a-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="onSearch"
  52. />
  53. </template>
  54. </BasicForm>
  55. </CollapseContainer>
  56. </PageWrapper>
  57. </template>
  58. <script lang="ts">
  59. import { type Recordable } from '@vben/types';
  60. import { computed, defineComponent, unref, ref } from 'vue';
  61. import { BasicForm, FormSchema, ApiSelect } from '/@/components/Form/index';
  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: 'field34',
  461. component: 'ApiRadioGroup',
  462. label: '远程Radio',
  463. helpMessage: ['ApiRadioGroup组件', '使用接口提供的数据生成选项'],
  464. required: true,
  465. componentProps: {
  466. api: optionsListApi,
  467. params: {
  468. count: 2,
  469. },
  470. resultField: 'list',
  471. // use name as label
  472. labelField: 'name',
  473. // use id as value
  474. valueField: 'id',
  475. },
  476. defaultValue: '1',
  477. colProps: {
  478. span: 8,
  479. },
  480. },
  481. {
  482. field: 'field35',
  483. component: 'ApiRadioGroup',
  484. label: '远程Radio',
  485. helpMessage: ['ApiRadioGroup组件', '使用接口提供的数据生成选项'],
  486. required: true,
  487. componentProps: {
  488. api: optionsListApi,
  489. params: {
  490. count: 2,
  491. },
  492. resultField: 'list',
  493. // use name as label
  494. labelField: 'name',
  495. // use id as value
  496. valueField: 'id',
  497. isBtn: true,
  498. onChange: (e, v) => {
  499. console.log('ApiRadioGroup====>:', e, v);
  500. },
  501. },
  502. colProps: {
  503. span: 8,
  504. },
  505. },
  506. // {
  507. // field: 'field36',
  508. // component: 'ApiTree',
  509. // label: '远程Tree',
  510. // helpMessage: ['ApiTree组件', '使用接口提供的数据生成选项'],
  511. // required: true,
  512. // componentProps: {
  513. // api: treeOptionsListApi,
  514. // params: {
  515. // count: 2,
  516. // },
  517. // afterFetch: (v) => {
  518. // //do something
  519. // return v;
  520. // },
  521. // resultField: 'list',
  522. // },
  523. // colProps: {
  524. // span: 8,
  525. // },
  526. // },
  527. {
  528. field: 'divider-linked',
  529. component: 'Divider',
  530. label: '字段联动',
  531. colProps: {
  532. span: 24,
  533. },
  534. },
  535. {
  536. field: 'province',
  537. component: 'Select',
  538. label: '省份',
  539. colProps: {
  540. span: 8,
  541. },
  542. componentProps: ({ formModel, formActionType }) => {
  543. return {
  544. options: provincesOptions,
  545. placeholder: '省份与城市联动',
  546. onChange: (e: any) => {
  547. // console.log(e)
  548. let citiesOptions =
  549. e == 1
  550. ? citiesOptionsData[provincesOptions[0].id]
  551. : citiesOptionsData[provincesOptions[1].id];
  552. // console.log(citiesOptions)
  553. if (e === undefined) {
  554. citiesOptions = [];
  555. }
  556. formModel.city = undefined; // reset city value
  557. const { updateSchema } = formActionType;
  558. updateSchema({
  559. field: 'city',
  560. componentProps: {
  561. options: citiesOptions,
  562. },
  563. });
  564. },
  565. };
  566. },
  567. },
  568. {
  569. field: 'city',
  570. component: 'Select',
  571. label: '城市',
  572. colProps: {
  573. span: 8,
  574. },
  575. componentProps: {
  576. options: [], // defalut []
  577. placeholder: '省份与城市联动',
  578. },
  579. },
  580. {
  581. field: 'divider-selects',
  582. component: 'Divider',
  583. label: '互斥多选',
  584. helpMessage: ['两个Select共用数据源', '但不可选择对方已选中的项目'],
  585. colProps: {
  586. span: 24,
  587. },
  588. },
  589. {
  590. field: 'selectA',
  591. component: 'Select',
  592. label: '互斥SelectA',
  593. slot: 'selectA',
  594. defaultValue: [],
  595. colProps: {
  596. span: 8,
  597. },
  598. },
  599. {
  600. field: 'selectB',
  601. component: 'Select',
  602. label: '互斥SelectB',
  603. slot: 'selectB',
  604. defaultValue: [],
  605. colProps: {
  606. span: 8,
  607. },
  608. },
  609. {
  610. field: 'divider-deconstruct',
  611. component: 'Divider',
  612. label: '字段解构',
  613. helpMessage: ['如果组件的值是 array 或者 object', '可以根据 ES6 的解构语法分别取值'],
  614. colProps: {
  615. span: 24,
  616. },
  617. },
  618. {
  619. field: '[startTime, endTime]',
  620. label: '时间范围',
  621. component: 'TimeRangePicker',
  622. componentProps: {
  623. format: 'HH:mm:ss',
  624. placeholder: ['开始时间', '结束时间'],
  625. },
  626. },
  627. {
  628. field: '[startDate, endDate]',
  629. label: '日期范围',
  630. component: 'RangePicker',
  631. componentProps: {
  632. format: 'YYYY-MM-DD',
  633. placeholder: ['开始日期', '结束日期'],
  634. },
  635. },
  636. {
  637. field: '[startDateTime, endDateTime]',
  638. label: '日期时间范围',
  639. component: 'RangePicker',
  640. componentProps: {
  641. format: 'YYYY-MM-DD HH:mm:ss',
  642. placeholder: ['开始日期、时间', '结束日期、时间'],
  643. showTime: { format: 'HH:mm:ss' },
  644. },
  645. },
  646. {
  647. field: 'divider-others',
  648. component: 'Divider',
  649. label: '其它',
  650. colProps: {
  651. span: 24,
  652. },
  653. },
  654. {
  655. field: 'field20',
  656. component: 'InputNumber',
  657. label: '字段20',
  658. required: true,
  659. colProps: {
  660. span: 8,
  661. },
  662. },
  663. {
  664. field: 'field21',
  665. component: 'Slider',
  666. label: '字段21',
  667. componentProps: {
  668. min: 0,
  669. max: 100,
  670. range: true,
  671. marks: {
  672. 20: '20°C',
  673. 60: '60°C',
  674. },
  675. },
  676. colProps: {
  677. span: 8,
  678. },
  679. },
  680. {
  681. field: 'field22',
  682. component: 'Rate',
  683. label: '字段22',
  684. defaultValue: 3,
  685. colProps: {
  686. span: 8,
  687. },
  688. componentProps: {
  689. disabled: false,
  690. allowHalf: true,
  691. },
  692. },
  693. ];
  694. export default defineComponent({
  695. components: { BasicForm, CollapseContainer, PageWrapper, ApiSelect, ASelect: Select },
  696. setup() {
  697. const check = ref(null);
  698. const { createMessage } = useMessage();
  699. const keyword = ref<string>('');
  700. const searchParams = computed<Recordable<string>>(() => {
  701. return { keyword: unref(keyword) };
  702. });
  703. function onSearch(value: string) {
  704. keyword.value = value;
  705. }
  706. return {
  707. schemas,
  708. optionsListApi,
  709. optionsA,
  710. optionsB,
  711. valueSelectA,
  712. valueSelectB,
  713. onSearch: useDebounceFn(onSearch, 300),
  714. searchParams,
  715. handleReset: () => {
  716. keyword.value = '';
  717. },
  718. handleSubmit: (values: any) => {
  719. console.log('values', values);
  720. createMessage.success('click search,values:' + JSON.stringify(values));
  721. },
  722. check,
  723. };
  724. },
  725. });
  726. </script>