Skip to content

Commit

Permalink
docs: add names demo (#48488)
Browse files Browse the repository at this point in the history
* feat: add names demo

* feat: snap

* feat: add doc

* feat: add doc

* feat: add doc

* feat: remove range

* feat: remove range

* feat: review

* feat: review

* feat: review

* feat: review

* feat: review

* feat: review

* feat: hidden

* feat: resert

* feat: noStyle

* feat: doc

* feat: review

* feat: review

* feat: review

* feat: review

* feat: add en

* feat: doc

* feat: review

* feat: review

* feat: review

* feat: review

* feat: review

* feat: review

* feat: 方法

* feat: review

* feat: review

* feat: review

* feat: review

* feat: doc

* feat: review

* feat: review
  • Loading branch information
crazyair committed Apr 26, 2024
1 parent c8bd145 commit b173ce1
Show file tree
Hide file tree
Showing 2 changed files with 294 additions and 0 deletions.
147 changes: 147 additions & 0 deletions docs/blog/form-names.en-US.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,147 @@
---
title: HOC Aggregate FieldItem
date: 2024-04-26
author: crazyair
---

During the form development process, there are occasional needs for combining attributes. The UI display fields are different from the backend data structure fields. For example, when interfacing with the backend, the province and city fields are often defined as two separate fields `{ province: Beijing, city: Haidian }`, rather than a combined one `{ province: [Beijing, Haidian] }`. Therefore, it is necessary to handle the values in `initialValues` and `onFinish` as follows:

```tsx
import React from 'react';
import { Cascader, Form } from 'antd';

const data = { province: 'Beijing', city: 'Haidian' };
const options = [
{ value: 'zhejiang', label: 'Zhejiang', children: [{ value: 'hangzhou', label: 'Hangzhou' }] },
{ value: 'jiangsu', label: 'Jiangsu', children: [{ value: 'nanjing', label: 'Nanjing' }] },
];
const createUser = (values) => console.log(values);

const Demo = () => (
<Form
initialValues={{ province: [data.province, data.city] }}
onFinish={(values) => {
const { province, ...rest } = values;
createUser({ province: province[0], city: province[1], ...rest });
}}
>
<Form.Item label="Address" name="province">
<Cascader options={options} placeholder="Please select" />
</Form.Item>
</Form>
);
export default Demo;
```

## Encapsulating Aggregate Field Components

When the form is relatively simple, it's manageable, but when encountering a `Form.List` scenario, it becomes necessary to process the values using `map`, which can become quite complex. Therefore, we need to encapsulate an aggregated field component to enable a single `Form.Item` to handle multiple `name` attributes.

## Approach Summary

To implement the aggregation field functionality, we need to utilize `getValueProps`, `getValueFromEvent`, and `transform` to facilitate the transformation of data from `FormStore` and to re-insert the structure into `FormStore` upon change.

### getValueProps

By default, `Form.Item` passes the field value from `FormStore` as the `value` prop to the child component. However, with `getValueProps`, you can customize the `props` that are passed to the child component to implement transformation functionality. In an aggregation scenario, we can iterate through `names` and combine the values from `FormStore` into a single `value` that is then passed to the child component:

```tsx
getValueProps={() => ({ value: names.map((name) => form.getFieldValue(name)) })}
```

### getValueFromEvent

When the child component modifies the value, the `setFields` method is used to set the aggregated `value` returned by the child component to the corresponding `name`, thereby updating the values of `names` in `FormStore`:

```tsx
getValueFromEvent={(values) => {
form.setFields(names.map((name, index) => ({ name, value: values[index] })));
return values[0];
}}
```

### transform

In `rules`, the default provided `value` for validation originates from the value passed to the corresponding `name` when the child component changes. Additionally, it is necessary to retrieve the values of `names` from `FormStore` and use the `transform` method to modify the `value` of `rules`:

```tsx
rules={[{
transform: () => {
const values = names.map((name) => form.getFieldValue(name));
return values;
},
}]}
```

## Final Result

```tsx
import React from 'react';
import type { FormItemProps } from 'antd';
import { Cascader, Form } from 'antd';

export const AggregateFormItem = (
props: FormItemProps & { names?: FormItemProps<Record<string, any>>['name'][] },
) => {
const form = Form.useFormInstance();

const { names = [], rules = [], ...rest } = props;
const [firstName, ...resetNames] = names;
return (
<>
<Form.Item
name={firstName}
// Convert the values of names into an array passed to children
getValueProps={() => ({ value: names.map((name) => form.getFieldValue(name)) })}
getValueFromEvent={(values) => {
// Set the form store values for names
form.setFields(names.map((name, index) => ({ name, value: values[index] })));
return values[0];
}}
rules={rules.map((thisRule) => {
if (typeof thisRule === 'object') {
return {
...thisRule,
transform: () => {
// Set the values of the names fields for the rule value
const values = names.map((name) => form.getFieldValue(name));
return values;
},
};
}
return thisRule;
})}
{...rest}
/>
{/* Bind other fields so they can getFieldValue to get values and setFields to set values */}
{resetNames.map((name) => (
<Form.Item key={name?.toString()} name={name} noStyle />
))}
</>
);
};

const data = { province: 'Beijing', city: 'Haidian' };
const options = [
{ value: 'zhejiang', label: 'Zhejiang', children: [{ value: 'hangzhou', label: 'Hangzhou' }] },
{ value: 'jiangsu', label: 'Jiangsu', children: [{ value: 'nanjing', label: 'Nanjing' }] },
];
const createUser = (values) => console.log(values);

export const Demo = () => (
<Form
initialValues={data}
onFinish={(values) => {
createUser(values);
}}
>
<AggregateFormItem label="Address" names={['province', 'city']} rules={[{ required: true }]}>
<Cascader options={options} placeholder="Please select" />
</AggregateFormItem>
</Form>
);
```

## Summary

By doing so, we have implemented a feature that allows for operating multiple `names` within a `Form.Item`, making the form logic clearer and easier to maintain. Additionally, there are some edge cases in this example that have not been considered. For instance, `setFields([{ name:'city', value:'nanjing' }])` will not update the selected value of `Cascader`. To achieve a refresh effect, you need to add `Form.useWatch(values => resetNames.map(name => get(values, name)), form);`. Feel free to explore more edge cases and handle them as needed.
147 changes: 147 additions & 0 deletions docs/blog/form-names.zh-CN.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,147 @@
---
title: 封装 Form.Item 实现数组转对象
date: 2024-04-26
author: crazyair
---

在表单开发过程中,偶尔会遇到组合属性的需求。UI 展示字段与后端返回数据结构字段有所不同。比如说,跟后端对接接口,定义省市字段经常是 2 个字段 `{ province: Beijing, city: Haidian }`,而不是 `{ province:[Beijing,Haidian] }`,因此则需要在 `initialValues` 以及 `onFinish` 处理值,如下:

```tsx
import React from 'react';
import { Cascader, Form } from 'antd';

const data = { province: 'Beijing', city: 'Haidian' };
const options = [
{ value: 'zhejiang', label: 'Zhejiang', children: [{ value: 'hangzhou', label: 'Hangzhou' }] },
{ value: 'jiangsu', label: 'Jiangsu', children: [{ value: 'nanjing', label: 'Nanjing' }] },
];
const createUser = (values) => console.log(values);

const Demo = () => (
<Form
initialValues={{ province: [data.province, data.city] }}
onFinish={(values) => {
const { province, ...rest } = values;
createUser({ province: province[0], city: province[1], ...rest });
}}
>
<Form.Item label="Address" name="province">
<Cascader options={options} placeholder="Please select" />
</Form.Item>
</Form>
);
export default Demo;
```

## 封装聚合字段组件

当表单比较简单还好,如果遇到 `Form.List` 场景,就需要 `map` 处理值,将变的很复杂。于是我们需要封装聚合字段组件,实现一个 `Form.Item` 可以写多个 `name`

## 思路整理

要实现聚合字段功能,我们需要用到 `getValueProps` `getValueFromEvent` `transform`,从而实现数据从 `FormStore` 中的转化,以及变更时重新传入 `FormStore` 结构中。

### getValueProps

默认情况下,`Form.Item` 会将 `FormStore` 中的字段值作为 `value` prop 传递给子组件。而通过 `getValueProps` 可以自定义传入给子组件的 `props` 从而实现转化功能。在聚合场景中,我们可以遍历 `names` 分别将 `FormStore` 中的值组合为一个 `value` 传递给子组件:

```tsx
getValueProps={() => ({ value: names.map((name) => form.getFieldValue(name)) })}
```

### getValueFromEvent

当子组件修改值时,使用 `setFields` 方法将子组件返回的聚合 `value` 分别设置给对应的 `name`,从而实现更新 `FormStore``names` 的值:

```tsx
getValueFromEvent={(values) => {
form.setFields(names.map((name, index) => ({ name, value: values[index] })));
return values[0];
}}
```

### transform

`rules` 中校验默认提供的 `value` 来源于子组件变更时传递给 `name` 对应的值,还需要从 `FormStore` 获取 `names` 的值使用 `transform` 方法修改 `rules``value`

```tsx
rules={[{
transform: () => {
const values = names.map((name) => form.getFieldValue(name));
return values;
},
}]}
```

## 最终效果

```tsx
import React from 'react';
import type { FormItemProps } from 'antd';
import { Cascader, Form } from 'antd';

export const AggregateFormItem = (
props: FormItemProps & { names?: FormItemProps<Record<string, any>>['name'][] },
) => {
const form = Form.useFormInstance();

const { names = [], rules = [], ...rest } = props;
const [firstName, ...resetNames] = names;
return (
<>
<Form.Item
name={firstName}
// 将 names 的值转成数组传给 children
getValueProps={() => ({ value: names.map((name) => form.getFieldValue(name)) })}
getValueFromEvent={(values) => {
// 将 form store 分别设置给 names
form.setFields(names.map((name, index) => ({ name, value: values[index] })));
return values[0];
}}
rules={rules.map((thisRule) => {
if (typeof thisRule === 'object') {
return {
...thisRule,
transform: () => {
// 将 names 字段的值设置给 rule value
const values = names.map((name) => form.getFieldValue(name));
return values;
},
};
}
return thisRule;
})}
{...rest}
/>
{/* 绑定其他字段,使其可以 getFieldValue 获取值、setFields 设置值 */}
{resetNames.map((name) => (
<Form.Item key={name?.toString()} name={name} noStyle />
))}
</>
);
};

const data = { province: 'Beijing', city: 'Haidian' };
const options = [
{ value: 'zhejiang', label: 'Zhejiang', children: [{ value: 'hangzhou', label: 'Hangzhou' }] },
{ value: 'jiangsu', label: 'Jiangsu', children: [{ value: 'nanjing', label: 'Nanjing' }] },
];
const createUser = (values) => console.log(values);

export const Demo = () => (
<Form
initialValues={data}
onFinish={(values) => {
createUser(values);
}}
>
<AggregateFormItem label="Address" names={['province', 'city']} rules={[{ required: true }]}>
<Cascader options={options} placeholder="Please select" />
</AggregateFormItem>
</Form>
);
```

## 总结

通过这种方式,我们实现了一个可以在 `Form.Item` 中操作多个 `name` 的功能,使得表单逻辑更加清晰和易于维护。另外此示例还有些边界场景没有考虑,比如 `setFields([{ name:'city' value:'nanjing' }])` 不会更新 `Cascader` 选中的值,需要增加 `Form.useWatch(values => resetNames.map(name => get(values, name)), form);` 达到刷新效果等。更多的边界问题就交给你去试试吧~

0 comments on commit b173ce1

Please sign in to comment.