fix: form fieldMappingTime improve and modelPropName support (#5335)
Some checks are pending
CI / Test (ubuntu-latest) (push) Waiting to run
CI / Test (windows-latest) (push) Waiting to run
CI / Lint (ubuntu-latest) (push) Waiting to run
CI / Lint (windows-latest) (push) Waiting to run
CI / Check (ubuntu-latest) (push) Waiting to run
CI / Check (windows-latest) (push) Waiting to run
CI / CI OK (push) Blocked by required conditions
CodeQL / Analyze (${{ matrix.language }}) (none, javascript-typescript) (push) Waiting to run
Deploy Website on push / Deploy Push Playground Ftp (push) Waiting to run
Deploy Website on push / Deploy Push Docs Ftp (push) Waiting to run
Deploy Website on push / Deploy Push Antd Ftp (push) Waiting to run
Deploy Website on push / Deploy Push Element Ftp (push) Waiting to run
Deploy Website on push / Deploy Push Naive Ftp (push) Waiting to run
Release Drafter / update_release_draft (push) Waiting to run

* 表单的fieldMappingTime支持将格式化掩码设为null以便原值映射,这样可以支持非日期时间类型的组件;
* 表单增加modelPropName设置组件的双向绑定属性名,用于支持未提前注册的双向绑定属性为非默认名称的组件。
* 增加一些经常会有人提到的组合字段演示,
This commit is contained in:
Netfan 2025-01-09 22:49:28 +08:00 committed by GitHub
parent 99c7fd72f8
commit 516d0b8dc8
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
8 changed files with 115 additions and 21 deletions

View File

@ -316,12 +316,18 @@ useVbenForm 返回的第二个参数,是一个对象,包含了一些表单
| collapsed | 是否折叠,在`showCollapseButton`为`true`时生效 | `boolean` | `false` | | collapsed | 是否折叠,在`showCollapseButton`为`true`时生效 | `boolean` | `false` |
| collapseTriggerResize | 折叠时,触发`resize`事件 | `boolean` | `false` | | collapseTriggerResize | 折叠时,触发`resize`事件 | `boolean` | `false` |
| collapsedRows | 折叠时保持的行数 | `number` | `1` | | collapsedRows | 折叠时保持的行数 | `number` | `1` |
| fieldMappingTime | 用于将表单内时间区域组件的数组值映射成 2 个字段 | `[string, [string, string], string?][]` | - | | fieldMappingTime | 用于将表单内的数组值映射成 2 个字段 | `[string, [string, string],Nullable<string>?][]` | - |
| commonConfig | 表单项的通用配置,每个配置都会传递到每个表单项,表单项可覆盖 | `FormCommonConfig` | - | | commonConfig | 表单项的通用配置,每个配置都会传递到每个表单项,表单项可覆盖 | `FormCommonConfig` | - |
| schema | 表单项的每一项配置 | `FormSchema[]` | - | | schema | 表单项的每一项配置 | `FormSchema[]` | - |
| submitOnEnter | 按下回车健时提交表单 | `boolean` | false | | submitOnEnter | 按下回车健时提交表单 | `boolean` | false |
| submitOnChange | 字段值改变时提交表单(内部防抖,这个属性一般用于表格的搜索表单) | `boolean` | false | | submitOnChange | 字段值改变时提交表单(内部防抖,这个属性一般用于表格的搜索表单) | `boolean` | false |
::: tip fieldMappingTime
此属性用于将表单内的数组值映射成 2 个字段,例如:`[['timeRange', ['startTime', 'endTime'], 'YYYY-MM-DD']]``timeRange`应当是一个至少具有2个成员的数组类型的值。Form会将`timeRange`的值前两个值分别按照格式掩码`YYYY-MM-DD`格式化后映射到`startTime`和`endTime`字段上。如果明确地将格式掩码设为null则原值映射而不进行格式化适用于非日期时间字段
:::
### TS 类型说明 ### TS 类型说明
::: details ActionButtonOptions ::: details ActionButtonOptions
@ -406,6 +412,11 @@ export interface FormCommonConfig {
* 所有表单项的label宽度 * 所有表单项的label宽度
*/ */
labelWidth?: number; labelWidth?: number;
/**
* 所有表单项的model属性名。使用自定义组件时可通过此配置指定组件的model属性名。已经在modelPropNameMap中注册的组件不受此配置影响
* @default "modelValue"
*/
modelPropName?: string;
/** /**
* 所有表单项的wrapper样式 * 所有表单项的wrapper样式
*/ */

View File

@ -368,6 +368,10 @@ export class FormApi {
} }
const [startTime, endTime] = values[field]; const [startTime, endTime] = values[field];
if (format === null) {
values[startTimeKey] = startTime;
values[endTimeKey] = endTime;
} else {
const [startTimeFormat, endTimeFormat] = Array.isArray(format) const [startTimeFormat, endTimeFormat] = Array.isArray(format)
? format ? format
: [format, format]; : [format, format];
@ -378,7 +382,7 @@ export class FormApi {
values[endTimeKey] = endTime values[endTimeKey] = endTime
? formatDate(endTime, endTimeFormat) ? formatDate(endTime, endTimeFormat)
: undefined; : undefined;
}
// delete values[field]; // delete values[field];
Reflect.deleteProperty(values, field); Reflect.deleteProperty(values, field);
}, },

View File

@ -41,6 +41,7 @@ const {
label, label,
labelClass, labelClass,
labelWidth, labelWidth,
modelPropName,
renderComponentContent, renderComponentContent,
rules, rules,
} = defineProps< } = defineProps<
@ -202,9 +203,9 @@ function fieldBindEvent(slotProps: Record<string, any>) {
const modelValue = slotProps.componentField.modelValue; const modelValue = slotProps.componentField.modelValue;
const handler = slotProps.componentField['onUpdate:modelValue']; const handler = slotProps.componentField['onUpdate:modelValue'];
const bindEventField = isString(component) const bindEventField =
? componentBindEventMap.value?.[component] modelPropName ||
: null; (isString(component) ? componentBindEventMap.value?.[component] : null);
let value = modelValue; let value = modelValue;
// antd design event // antd design event

View File

@ -98,6 +98,7 @@ const computedSchema = computed(
hideRequiredMark = false, hideRequiredMark = false,
labelClass = '', labelClass = '',
labelWidth = 100, labelWidth = 100,
modelPropName = '',
wrapperClass = '', wrapperClass = '',
} = mergeWithArrayOverride(props.commonConfig, props.globalCommonConfig); } = mergeWithArrayOverride(props.commonConfig, props.globalCommonConfig);
return (props.schema || []).map((schema, index) => { return (props.schema || []).map((schema, index) => {
@ -118,6 +119,7 @@ const computedSchema = computed(
hideLabel, hideLabel,
hideRequiredMark, hideRequiredMark,
labelWidth, labelWidth,
modelPropName,
wrapperClass, wrapperClass,
...schema, ...schema,
commonComponentProps: componentProps, commonComponentProps: componentProps,

View File

@ -4,7 +4,7 @@ import type { ZodTypeAny } from 'zod';
import type { Component, HtmlHTMLAttributes, Ref } from 'vue'; import type { Component, HtmlHTMLAttributes, Ref } from 'vue';
import type { VbenButtonProps } from '@vben-core/shadcn-ui'; import type { VbenButtonProps } from '@vben-core/shadcn-ui';
import type { ClassType } from '@vben-core/typings'; import type { ClassType, Nullable } from '@vben-core/typings';
import type { FormApi } from './form-api'; import type { FormApi } from './form-api';
@ -197,6 +197,11 @@ export interface FormCommonConfig {
* label宽度 * label宽度
*/ */
labelWidth?: number; labelWidth?: number;
/**
* model属性名
* @default "modelValue"
*/
modelPropName?: string;
/** /**
* wrapper样式 * wrapper样式
*/ */
@ -219,7 +224,7 @@ export type HandleResetFn = (
export type FieldMappingTime = [ export type FieldMappingTime = [
string, string,
[string, string], [string, string],
([string, string] | string)?, ([string, string] | Nullable<string>)?,
][]; ][];
export interface FormSchema< export interface FormSchema<
@ -330,7 +335,7 @@ export interface VbenFormProps<
*/ */
actionWrapperClass?: ClassType; actionWrapperClass?: ClassType;
/** /**
* *
*/ */
fieldMappingTime?: FieldMappingTime; fieldMappingTime?: FieldMappingTime;
/** /**

View File

@ -21,7 +21,7 @@
.form-valid-error { .form-valid-error {
/** select 选择器的样式 */ /** select 选择器的样式 */
.ant-select .ant-select-selector { .ant-select:not(.valid-success) .ant-select-selector:not(.valid-success) {
border-color: hsl(var(--destructive)) !important; border-color: hsl(var(--destructive)) !important;
} }
@ -39,6 +39,10 @@
border-color: hsl(var(--destructive)); border-color: hsl(var(--destructive));
box-shadow: 0 0 0 2px rgb(255 38 5 / 6%); box-shadow: 0 0 0 2px rgb(255 38 5 / 6%);
} }
.ant-input:not(.valid-success) {
border-color: hsl(var(--destructive)) !important;
}
} }
/** 区间选择器下面来回切换时的样式 */ /** 区间选择器下面来回切换时的样式 */

View File

@ -1,11 +1,13 @@
<script lang="ts" setup> <script lang="ts" setup>
import { h } from 'vue'; import { h, markRaw } from 'vue';
import { Page } from '@vben/common-ui'; import { Page } from '@vben/common-ui';
import { Card, Input, message } from 'ant-design-vue'; import { Card, Input, message } from 'ant-design-vue';
import { useVbenForm } from '#/adapter/form'; import { useVbenForm, z } from '#/adapter/form';
import TwoFields from './modules/two-fields.vue';
const [Form] = useVbenForm({ const [Form] = useVbenForm({
// //
@ -16,6 +18,7 @@ const [Form] = useVbenForm({
}, },
labelClass: 'w-2/6', labelClass: 'w-2/6',
}, },
fieldMappingTime: [['field4', ['phoneType', 'phoneNumber'], null]],
// //
handleSubmit: onSubmit, handleSubmit: onSubmit,
// labelinputvertical // labelinputvertical
@ -39,9 +42,10 @@ const [Form] = useVbenForm({
}), }),
}, },
{ {
component: h(Input, { placeholder: '请输入' }), component: h(Input, { placeholder: '请输入Field2' }),
fieldName: 'field2', fieldName: 'field2',
label: '自定义组件', label: '自定义组件',
modelPropName: 'value',
rules: 'required', rules: 'required',
}, },
{ {
@ -50,6 +54,27 @@ const [Form] = useVbenForm({
label: '自定义组件(slot)', label: '自定义组件(slot)',
rules: 'required', rules: 'required',
}, },
{
component: markRaw(TwoFields),
defaultValue: [undefined, ''],
disabledOnChangeListener: false,
fieldName: 'field4',
formItemClass: 'col-span-1',
label: '组合字段',
rules: z
.array(z.string().optional())
.length(2, '请选择类型并输入手机号码')
.refine((v) => !!v[0], {
message: '请选择类型',
})
.refine((v) => !!v[1] && v[1] !== '', {
message: '       输入手机号码',
})
.refine((v) => v[1]?.match(/^1[3-9]\d{9}$/), {
// 使
message: '       号码格式不正确',
}),
},
], ],
// 21 // 21
wrapperClass: 'grid-cols-1 md:grid-cols-2', wrapperClass: 'grid-cols-1 md:grid-cols-2',

View File

@ -0,0 +1,42 @@
<script lang="ts" setup>
import { Input, Select } from 'ant-design-vue';
const emit = defineEmits(['blur', 'change']);
const modelValue = defineModel<[string, string]>({
default: () => [undefined, undefined],
});
function onChange() {
emit('change', modelValue.value);
}
</script>
<template>
<div class="flex w-full gap-1">
<Select
v-model:value="modelValue[0]"
class="w-[80px]"
placeholder="类型"
allow-clear
:class="{ 'valid-success': !!modelValue[0] }"
:options="[
{ label: '个人', value: 'personal' },
{ label: '工作', value: 'work' },
{ label: '私密', value: 'private' },
]"
@blur="emit('blur')"
@change="onChange"
/>
<Input
placeholder="请输入11位手机号码"
class="flex-1"
allow-clear
:class="{ 'valid-success': modelValue[1]?.match(/^1[3-9]\d{9}$/) }"
v-model:value="modelValue[1]"
:maxlength="11"
type="tel"
@blur="emit('blur')"
@change="onChange"
/>
</div>
</template>