当前位置: 首页 > article >正文

基于Ant-Design-Vue设计的配置化表单

适用vue 3.4 版本以上

在日常的前端开发中,表单开发必不可少,尤其是在遇到一些大型的复杂的表单,这往往会变成一个痛点。于是就想着开发一个可以list来配置的表单组件。

先上组件代码

<!-- 该组件 VUE 版本在 3.4 以上可使用-->
<template>
  <Form
    ref="formRef"
    :model="model"
    v-bind='form'>
    <Row :gutter="[8, 8]">
      <template v-for="config in newConfigs" :key="config.key">
        <Col v-bind="config.wrapperCol" v-if="config.when(model)">
          <Form.Item v-bind="config.forImteAttrs"
            :label-col="config.labelCol">
            <component
              :is="config.type"
              v-bind="config.componentAttrs"
              :config="config"
              :raw="config.raw"
              :options="config.componentAttrs.options || options?.[config.key]"
              v-model:[config.model]="model[config.key]"
              v-on='{
                change: (e: any) => {
                  change(e, config.key);
                  config.componentAttrs.onChange?.({
                    update: change,
                    value: model
                  })
                }
              }'>
              {{ config.componentAttrs.buttonText || null }}
            </component>
          </Form.Item>
        </Col>
      </template>
      <slot />
    </Row>
  </Form>
</template>

<script setup lang="ts">
import {
  Col,
  Form,
  FormInstance,
  Row,
} from 'ant-design-vue';
import {
  computed, Ref,
  toRefs,
  useAttrs,
} from 'vue';
import { FormItemConfig } from './ConfigFormType';

const formRef = defineModel<Ref<null | undefined> | FormInstance>('formRef');
const model = defineModel<any>();

interface IConfigFormProps {
  layout?: 'inline' | 'vertical' | 'horizontal';
  FormGrid?: {};
  labelCol?: any;
  wrapperCol?: any;
  form?: any;
  configs: FormItemConfig[];
  options?: { [index: string]: any[] }
  readonly?: boolean;
}
const props = defineProps<IConfigFormProps>();
const {
  configs, FormGrid, labelCol, options, wrapperCol, readonly,
} = toRefs(props);

const generateUUID = () => 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, (c) => {
  // eslint-disable-next-line no-bitwise
  const r = Math.random() * 16 | 0;
  // eslint-disable-next-line no-bitwise, no-mixed-operators
  const v = c === 'x' ? r : (r & 0x3) | 0x8;
  return v.toString(16);
});

const gloablAttrs = useAttrs();
const newConfigs = computed(() => configs.value.map((config: any) => {
  const {
    grid, key, type, attrs, itemModel, when, children, ...rest
  } = config;
  if (key) {
    rest.name = key;
  }
  if (!config.wrapperCol) {
    rest.wrapperCol = wrapperCol.value;
  }
  if (!config.labelCol) {
    rest.labelCol = labelCol.value;
  }
  const forImteAttrs: Partial<any> = rest;
  const componentAttrs = attrs || {};
  return {
    layout: { ...FormGrid, ...grid },
    name: [key || rest.name],
    key: key || rest.name || generateUUID(),
    when: when || (() => true),
    type,
    children,
    model: itemModel!,
    forImteAttrs,
    wrapperCol: rest.wrapperCol,
    labelCol: rest.labelCol,
    componentAttrs,
    readonly: gloablAttrs.readonly || readonly.value,
    disabled: gloablAttrs.disabled,
    raw: config,
  };
}));

interface IConfigFormEmits {
  (e: 'finish', value: any): void;
  (e: 'change', value: any): void;
}
const emit = defineEmits<IConfigFormEmits>();
const change = (e: any, key: string) => emit('change', { value: e, key });
</script>

具体使用

<ConfigForm
	:configs="configComputed"
	v-model:formRef="formRef"
	v-model="formState"
	@finish="fetchInfo"
	layout="inline" />

configs
用来生成表单的具体配置,类型如下

// FormItemConfig.ts
import { ColProps } from 'ant-design-vue';
import { Slot } from 'vue';

export type FormItemConfig = {
  title?: string;
  type: any;
  layout?: 'horizontal' | 'vertical' | 'inline';
  key?: string;
  name?: string;
  label?: string;
  rules?: any[] | any;
  colon?: boolean;
  options?: any[];
  hasFeedback?: boolean;
  help?: string | Slot;
  labelAlign?: 'left' | 'right';
  labelCol?: ColProps;
  required?: boolean;
  tooltip?: string | Slot;
  validateFirst?: boolean;
  validateStatus?: 'success' | 'warning' | 'error' | 'validating';
  validateTrigger?: string | string[];
  wrapperCol?: ColProps;
  extra?: string | Slot | any;
  model?: any;
  // eslint-disable-next-line no-unused-vars
  when?: (values: Partial<any>, currentValue?: Partial<any>) => boolean;
  itemModel?: string;
  attrs?: Partial<any>;
  children?: FormItemConfig[];
};

configs配置
1、attrs属性为绑定到具体表单组件的属性,例如Select,Input之类的
2、其余大部分属性绑定到<Form.Item>组件上,故可以使用label,rules之类的属性,
3、itemModel一般情况下为value,type为Switch组件时,需根据实际情况变化
4、type属性是直接注入< components >组件的is属性上,所以一些自定义的组件也可以使用,注入到type组件中

import {
  AutoComplete, Button, Spin,
} from 'ant-design-vue';
import {ref } from 'vue';

const autoCompleteLoading = ref<boolean>(false)
const configs = [
    {
      title: 'XXX',
      key: 'name',
      type: autoCompleteLoading.value ? Spin : AutoComplete,
      label: 'item 1',
      rules: { required: true, message: '请输入XXX', trigger: 'blur' },
      itemModel: 'value',
      attrs: {
        options: [{ label: 'item1', value: 'value1' }],
        allowClear: true,
        placeholder: 'XXXXXXXX',
        style: 'width: 280px',
        filterOption: (inputValue: string, option: any) => option.value
          .toLowerCase()
          .indexOf(inputValue.toLowerCase()) >= 0,
      },
    },
    {
      title: 'XXXXXXX',
      type: Button,
      attrs: {
        type: 'primary',
        htmlType: 'submit',
        buttonText: '查询',
      },
    },
   	];

也可以配合计算属性和响应式函数动态响应表单

// config.ts
import { computed, Ref } from 'vue';
import {
  AutoComplete, Button, Spin,
} from 'ant-design-vue';


export function useConfigComputed(
	autoCompleteLoading:Ref<boolean>,
	dataNameList: Ref<{ label: string, value: any }[]>
) {
  return computed(() => [
    {
      title: 'XXX',
      key: 'name',
      type: autoCompleteLoading.value ? Spin : AutoComplete,
      label: 'item 1',
      rules: { required: true, message: '请输入XXX', trigger: 'blur' },
      itemModel: 'value',
      attrs: {
        options: [{ label: 'item1', value: 'value1' }],
        allowClear: true,
        placeholder: 'XXXXXXXX',
        style: 'width: 280px',
        filterOption: (inputValue: string, option: any) => option.value
          .toLowerCase()
          .indexOf(inputValue.toLowerCase()) >= 0,
      },
    },
    {
      title: 'XXXXXXX',
      type: Button,
      attrs: {
        type: 'primary',
        htmlType: 'submit',
        buttonText: '查询',
      },
    },
  ]);
}


// index.vue <script setup lang='ts'>
const autoCompleteLoading = ref<boolean>(false);
const dataNameList = ref<{ label: string, value: any }[]>([]);
const configs = useConfigComputed(
	autoCompleteLoading,
	dataNameList,
)

世纪的使用情况


http://www.kler.cn/a/300134.html

相关文章:

  • nvm版本安装
  • html、js、css实现爱心效果
  • 备赛蓝桥杯之第十五届职业院校组省赛第二题:分享点滴
  • 探索 SolidJS:一款高速的前端框架
  • Mysql数据库锁
  • 数据结构之堆排序
  • curl和ping
  • 【leetcode C++】 动态规划
  • lamp的脚本部署
  • 【CAPL实战】解决调用LIN函数不生效问题
  • [笔记]电参数测量的现有方案
  • Cesium 展示——视角漫游飞行效果
  • Linux系统本地化部署Dify并安装Ollama运行llava大语言模型详细教程
  • uniapp和vue3中使用vConsole在H5中开启移动端调试
  • R语言论文插图模板第9期—滑珠散点图
  • 一文彻底了解DNS协议工作原理,恐怕没有比这更通俗易懂的了吧?
  • HAProxy--高性能反向代理
  • MySQL 数据库:原理、应用与发展
  • 【CanMV K230 AI视觉】人脸姿态(脸部朝向)
  • Unity Hub自动安装指定版本Unity的Android开发环境
  • 【机器学习】生成对抗网络(Generative Adversarial Networks, GANs)详解
  • 14.2 k8s中我们都需要监控哪些组件
  • 结构型设计模式-外观(facade)模式 - python实现
  • js逆向基础10面向对象继承2
  • 240907-Gradio插入Mermaid流程图并自适应浏览器高度
  • 2.2.3 UDP的可靠传输协议QUIC 2