antd二次封装函数式组件(Searchgroup)

本人长年累月做中台项目,接触的需求都是根据各种姿势查询数据,考虑到高频繁的使用,二话不说,必须封装来方便复用。其实之前已经用类组件写过一个并投入使用过几个项目,但感觉写得并不清爽, 于是趁着空余时间重新封装(设计)一下Searchgroup(搜索框组)组件,属于复合组件
本文涉及的技术栈主要有antd,react(最好16.8+)

截图antd官网
抛开样式问题,我们大概要做到效果如上图,根据条件查询并可清空条件,列出一些需求点
  • 搜索条件配置简单,包括Input,Select,DatePicker等,甚至可配置自定义类型的组件
  • 点击查询可搜集到所有条件的值
  • 点击清空可清空并可自定义callback(如可重置条件后自动搜索一次)
  • 支持条件默认值配置
  • 支持校验
  • 条件框排版可配置
  • 改变某个条件,能触发事件(如改变语言条件,其它条件清空或做其它动作)
// usage
    <Searchgroup 
          config={[
            { name: 'name', label: 'name', type: 'input'},
            { name: 'sex', label: 'sex', type: 'select', options: {
              '男': 1,
              '女': 2,
            }},
            { name: 'age', label: 'age', type: 'inputNumber'},
            { name: 'address', label: 'address'},
            { name: 'hobby', label: 'hobby'},
            { name: 'birthday', label: 'birthday',type: 'datePicker'},
            { name: 'job', label: 'job', rules: [{ required: true, message: 'Please input your job!'},
            { name: 'lang', label: 'lang', type: 'select', options: {
              'Chinese': '1',
              'English': '2',
            }}
          ]}
          col={3}
          onSearch={handleSearch}
          onClear={handleClear}
        />

配置属性在文章底部

// Searchgroup.js
import React, { forwardRef } from 'react'
import { Button, Input, Select, InputNumber, DatePicker, Form  } from 'antd'
import PropTypes from 'prop-types'

const { Option } = Select
const FormItem = Form.Item

// 默认条件布局
const defaultFormItemLayout = {
    labelCol: { span: 6 },
    wrapperCol: { span: 18 },
}

// Form.create()包裹,目的使用内置的方法收集、清空、校验
const Index = Form.create()(props => {
    const { config, col = 3, form, onSearch, onClear, resetSearch = true,formItemLayout = defaultFormItemLayout } = props
    const { getFieldDecorator, validateFieldsAndScroll, resetFields } = form
    const handleSearch = () => {
        validateFieldsAndScroll((err, values) => {
            onSearch && onSearch(values)
        })
    }
    const handleClear = () => {
        resetFields()
        if(onClear) {
            onClear()
            resetSearch && handleSearch()
        }
    }
    return (
        <>
          <div className="jantd-searchgroup">
              {
                  config.map((p,i) => {
                      const { name, label, initialValue, rules,  ...restProps} = p
                      return (                         
                            <FormItem label={name} key={i} className="jantd-searchgroup-col" style={{width: 100/col + '%'}} {...formItemLayout}> 
                                {
                                    getFieldDecorator(label, {
                                        initialValue,
                                        rules,
                                    })(<C {...restProps} />)
                                }
                            </FormItem>                                                      
                      )                      
                  })
              }
              <FormItem label=' ' className="jantd-searchgroup-col jantd-searchgroup-btns" style={{width: 100/col + '%'}} {...formItemLayout}>
                <Button type="primary" onClick={handleSearch}>search</Button>
                <Button onClick={handleClear}>clear</Button>
              </FormItem>
              
          </div>
        </>      
    )
})


// 不同类型组件,因为被FormItem包裹,需要支持ref,react16.3之前只能用class,16.8及以后函数组件可以用forwardRef包裹
const C = forwardRef((props, ref) => {
    const baseProps = {
        ...props,
    }
    const { type, options }  = props
    const createC = type => {
        switch(type) {
            case 'select':
                return (
                    <Select {...baseProps}>
                    {
                        Object.keys(options).map(p => <Option key={options[p]}>{p}</Option>)
                    }
                    </Select>
                )
            case 'inputNumber':
                return <InputNumber {...baseProps} />
            case 'datePicker':
                return <DatePicker {...baseProps} />
            default:
                return <Input {...baseProps} />
        }       
        
    }
    return (
        <span ref={ref}>
            {createC(type)}
        </span>
    )
})

Index.propTypes = {
    config: PropTypes.array.isRequired,
    col: PropTypes.number,
    onSearch: PropTypes.func,
    onClear: PropTypes.func,
    resetSearch: PropTypes.bool,
    formItemLayout: PropTypes.object,
}

export default Index

组件的主要设计逻辑,搜集和清空条件逻辑是利用了antd的form表单的方法,当然逐个条件组件onChange上交收集值也是没问题的(我上个版本就是这样做),但考虑到校验功能在Form有现成的配置逻辑,在这里就重写了,排版方面用百分比浮动的方法

目前效果.png

上面一口气先完成了主要功能,会发现配置了默认值的条件,点击清空后并不是设为空置,而是重设为默认值,这点需要改造,不使用resetFields,而去遍历配置逐个set为undefined

    const handleClear = () => {
        // 删除这个方法
        // resetFields()
        config.forEach(c => setFieldsValue({[c.label]: undefined }))
        if(onClear) {
            onClear()
            resetSearch && handleSearch()
        }
    }

然后需要改变语言lang后自动改变sex选项到options,需这样配置使用

// usage
// ...
  const [ sex, setSex ] = useState('1')
  const sexOptions = {
    '1': {
      '男': 1,
      '女': 2,
    },
    '2': {
      'male': 1,
      'female': 2,
    }
  }
// ...
<Searchgroup
  config={[
   { name: 'sex', label: 'sex', type: 'select', options: sexOptions[sex]},
   { name: 'lang', label: 'lang', type: 'select', options: {
      'Chinese': '1',
      'English': '2',
    }, onChange: v => setSex(v)}
  ]}
/>

接着实现自定义组件作搜索条件,注意的是自定义组件需符合FormItem输入输出的约定
antd官网.png
//  自定义组件
const MyInputs = props => {
  const { value = {}, onChange } = props
  return (
    <div style={{display: 'flex'}}>
      <Input value={value.n} onChange={e => onChange({
        ...value,
        n: e.target.value
      })} />
      <Select value={value.b} onChange={v => onChange({
        ...value,
        b: v,
      })}>
        <Option key='a'>A</Option>
        <Option key='b'>B</Option>
      </Select>
    </div>
  )
}

// usage
// ...
<Searchgroup 
  config={[
   { name: 'inputs', label: 'inputs', render: MyInputs}
 ]}
/>

// Searchgroup
// ...
// 不同类型组件
const C = forwardRef((props, ref) => {
    const baseProps = {
        ...props,
    }
    const { type, render, options }  = props
    if(render) {
        const C = render
        return <C {...baseProps} />
    }
// ...
一开始上面的组件类型type只有Input、Select、DatePicker、InputNumber配置,假如条件类型丰富一点就不够用了,而现在加上了自定义组件的配置方法,只要符合规范的自定义组件写法约定,Upload、Checkbox或者更加复杂的交互组件也可以先从自定义属性配置用起,往后会根据情况新增组件类型的直接配置

然后贴上样式代码
/* 这次用cra脚手架写的,懒得配置less,下面的样式实际要加上命名空间 */
.ant-select, .ant-input-number, .ant-calendar-picker {
  width: 100% !important;
}

.jantd-searchgroup {
  width: 100%;
}

.jantd-searchgroup-col {
  float: left;
  display: flex !important;
  align-items: center;
}

.jantd-searchgroup-btns label {
  opacity: 0;
}

.jantd-searchgroup-btns button {
  margin: 0 8px;
}

优化

如无意外,每次onChange其中一个条件都会引起所有条件组件的re-render,因为react设计哲学是不会干涉重复的业务渲染,这需要手动优化,我选择用React.memo,这个api对标类组件的PureComponent,都是自动的shallowEqual。
第一个参数是组件,第二个参数是优化函数,判断是否需要re-render,返回true即不re-render

// Searchgroup.js
// ...
const areEqual = (prevProps, nextProps) => {
   // 值相等就不需要重新渲染
    if(prevProps.value === nextProps.value) {
        return true
    }
    return false
}
// 不同类型组件
const C = memo(forwardRef((props, ref) => {
// ...
}), areEqual)

加上memo之后发现没多余的re-render,优化成功,基本功能和优化就完成了,有空再完善

配置

config: 各条件的配置(name: 显示的名字,label: 提交的key,type:条件的组件类型,initialValue:默认值,rules:校验规则 ,render:自定义组件(符合FormItem自定义组件规范约定)...)
col:每行的个数,默认3
onSearch:点击搜索触发的回调
onClear:点击清空触发的回调
resetSearch: 点击清空重新搜索,默认true
formItemLayout: 条件的布局,应用于FormItem


更新

2020-02-05
暂定为体验版,使用了umi/father打包(支持umd、cjs、es三种格式),已将Searchgroup加入组件库,安装方法:
npm/cnpm i j-antd -S
yarn add j-antd -S
使用方法
import { Searchgroup } from 'j-antd'

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,837评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,551评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,417评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,448评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,524评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,554评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,569评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,316评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,766评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,077评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,240评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,912评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,560评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,176评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,425评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,114评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,114评论 2 352

推荐阅读更多精彩内容