From 2f93efb5305be0ed30b1b718934168ce73b81ecf Mon Sep 17 00:00:00 2001 From: electroluxcode <3451613934@qq.com> Date: Tue, 3 Sep 2024 23:25:20 +0800 Subject: [PATCH 1/4] feat: multiple options && activeIconSetPath about cascader comp --- .../cascader-view/cascader-view.tsx | 171 +++++++++++++----- src/components/cascader/cascader.tsx | 116 +++++++++++- src/components/cascader/prompt.tsx | 2 +- src/components/check-list/check-list-item.tsx | 22 ++- src/components/check-list/check-list.tsx | 13 +- src/components/check-list/context.tsx | 7 + 6 files changed, 279 insertions(+), 52 deletions(-) diff --git a/src/components/cascader-view/cascader-view.tsx b/src/components/cascader-view/cascader-view.tsx index 206f782ecc..55af2f9056 100644 --- a/src/components/cascader-view/cascader-view.tsx +++ b/src/components/cascader-view/cascader-view.tsx @@ -13,6 +13,7 @@ import Skeleton from '../skeleton' import { useUpdateEffect } from 'ahooks' import { useFieldNames } from '../../hooks' import type { FieldNamesType, BaseOptionType } from '../../hooks' +import { cloneDeep } from 'lodash' const classPrefix = `adm-cascader-view` @@ -32,14 +33,16 @@ export type CascaderValueExtend = { export type CascaderViewProps = { options: CascaderOption[] - value?: CascaderValue[] - defaultValue?: CascaderValue[] - onChange?: (value: CascaderValue[], extend: CascaderValueExtend) => void + value?: CascaderValue[] | any + defaultValue?: CascaderValue[] | any + onChange?: (value: CascaderValue[] | any, extend: CascaderValueExtend) => void placeholder?: string | ((index: number) => string) onTabsChange?: (index: number) => void activeIcon?: ReactNode loading?: boolean + multiple?: boolean fieldNames?: FieldNamesType + activeIconSetPath?: boolean } & NativeProps<'--height'> const defaultProps = { @@ -48,7 +51,6 @@ const defaultProps = { export const CascaderView: FC = p => { const props = mergeProps(defaultProps, p) - const { locale } = useConfig() const [labelName, valueName, childrenName, disabledName] = useFieldNames( props.fieldNames @@ -57,49 +59,94 @@ export const CascaderView: FC = p => { valueName, childrenName, }) - const [value, setValue] = usePropsValue({ ...props, onChange: val => { - props.onChange?.(val, generateValueExtend(val)) + if (props.multiple) { + props.onChange?.(val, {} as any) + } else { + props.onChange?.(val, generateValueExtend(val as string[])) + } }, }) const [tabActiveIndex, setTabActiveIndex] = useState(0) const levels = useMemo(() => { - const ret: { - selected: CascaderOption | undefined - options: CascaderOption[] - }[] = [] - - let currentOptions = props.options - let reachedEnd = false - for (const v of value) { - const target = currentOptions.find(option => option[valueName] === v) - ret.push({ - selected: target, - options: currentOptions, - }) - if (!target || !target[childrenName]) { - reachedEnd = true - break + if (props.multiple) { + const ret: { + selected: CascaderOption | undefined + options: CascaderOption[] + }[] = [] + + let currentOptions = props.options + let reachedEnd = false + + for (const v of value) { + let target + ;(v as string[]).forEach(e => { + const temp = currentOptions.find(option => + e.includes(option[valueName]) + ) + if (temp) { + target = temp + } + }) + ret.push({ + selected: target, + options: currentOptions, + }) + if (!target || !target[childrenName]) { + reachedEnd = true + break + } + currentOptions = target[childrenName] } - currentOptions = target[childrenName] - } - if (!reachedEnd) { - ret.push({ - selected: undefined, - options: currentOptions, - }) + if (!reachedEnd) { + ret.push({ + selected: undefined, + options: currentOptions, + }) + } + return ret + } else { + const ret: { + selected: CascaderOption | undefined + options: CascaderOption[] + }[] = [] + + let currentOptions = props.options + let reachedEnd = false + for (const v of value) { + const target = currentOptions.find(option => option[valueName] === v) + ret.push({ + selected: target, + options: currentOptions, + }) + if (!target || !target[childrenName]) { + reachedEnd = true + break + } + currentOptions = target[childrenName] + } + if (!reachedEnd) { + ret.push({ + selected: undefined, + options: currentOptions, + }) + } + return ret } - return ret }, [value, props.options]) useUpdateEffect(() => { - props.onTabsChange?.(tabActiveIndex) + if (!props.multiple) { + props.onTabsChange?.(tabActiveIndex) + } }, [tabActiveIndex]) useEffect(() => { - setTabActiveIndex(levels.length - 1) + if (!props.multiple) { + setTabActiveIndex(levels.length - 1) + } }, [value]) useEffect(() => { const max = levels.length - 1 @@ -108,12 +155,37 @@ export const CascaderView: FC = p => { } }, [tabActiveIndex, levels]) - const onItemSelect = (selectValue: CascaderValue, depth: number) => { + const onItemSelect = ( + selectValue: CascaderValue | CascaderValue[], + depth: number + ) => { const next = value.slice(0, depth) if (selectValue !== undefined) { next[depth] = selectValue } - setValue(next) + if (props.multiple) { + const cloneValue = cloneDeep(value) + cloneValue[depth] = next[depth] + setValue(cloneValue) + } else { + setValue(next) + } + } + + const setPath = (selectValue: string, depth: number) => { + const currentPath = cloneDeep(value) + if (selectValue !== undefined) { + if (!currentPath[depth]) { + currentPath[depth] = [] + } + currentPath[depth] = (currentPath[depth] as string[]).filter( + (item: string) => { + return item !== selectValue + } + ) + currentPath[depth].push(selectValue) + } + setValue(currentPath) } const whetherLoading = (options: T) => @@ -143,8 +215,8 @@ export const CascaderView: FC = p => { {selected ? selected[labelName] : typeof placeholder === 'function' - ? placeholder(index) - : placeholder} + ? placeholder(index) + : placeholder} } forceRender @@ -171,14 +243,31 @@ export const CascaderView: FC = p => { ) : ( - onItemSelect(selectValue[0], index) - } + value={props.multiple ? value[index] : [value[index]]} + onChange={selectValue => { + if (props.multiple) { + onItemSelect(selectValue, index) + } else { + onItemSelect(selectValue[0], index) + } + }} + multiple={props.multiple} + activeSetPathMiddleware={{ + index, + activeIconSetPath: props.activeIconSetPath, + setPath, + }} activeIcon={props.activeIcon} > {level.options.map(option => { - const active = value[index] === option[valueName] + let active + if (props.multiple) { + active = (value[index] as Array)?.includes( + option[valueName] + ) + } else { + active = value[index] === option[valueName] + } return ( { + const option = currentOptions.find( + (opt: { value: any }) => opt.value === value + ) + if (option) { + isFinish = false + helper(option.children || [], remainingData, path.concat(option.value)) + } + }) + + if (isFinish) { + if (Array.isArray(path)) { + result.push(path) + return + } + } + } + + helper(options, data, []) + return result +} + export type CascaderProps = { + activeIconSetPath?: boolean options: CascaderOption[] - value?: CascaderValue[] + multiple?: boolean + value?: CascaderValue[] | CascaderValue[][] defaultValue?: CascaderValue[] placeholder?: string - onSelect?: (value: CascaderValue[], extend: CascaderValueExtend) => void - onConfirm?: (value: CascaderValue[], extend: CascaderValueExtend) => void + onSelect?: ( + value: CascaderValue[] | CascaderValue[][], + extend: CascaderValueExtend + ) => void + onConfirm?: ( + value: CascaderValue[] | CascaderValue[][], + extend: CascaderValueExtend + ) => void onCancel?: () => void onClose?: () => void visible?: boolean @@ -68,6 +121,23 @@ const defaultProps = { forceRender: false, } +/** + * @description 树转化成二维数组 + */ +const dataPreview = (data: Array) => { + const temp: any[] = [] + cloneDeep(data).forEach(item => { + item.forEach((e: any, index: number) => { + if (!temp[index]) { + temp[index] = [e] + } else { + temp[index].push(e) + } + }) + }) + return temp +} + export const Cascader = forwardRef((p, ref) => { const { locale } = useConfig() const props = mergeProps( @@ -107,7 +177,24 @@ export const Cascader = forwardRef((p, ref) => { const [value, setValue] = usePropsValue({ ...props, onChange: val => { - props.onConfirm?.(val, generateValueExtend(val)) + if (props.multiple) { + const hash: Record = {} + let result = generatePaths( + props.options, + val as Array + ) + result = result.filter(item => { + if (hash[JSON.stringify(item)]) { + return false + } else { + hash[JSON.stringify(item)] = true + return true + } + }) + props.onConfirm?.(result, {} as any) + } else { + props.onConfirm?.(val, generateValueExtend(val as CheckListValue[])) + } }, }) @@ -117,11 +204,23 @@ export const Cascader = forwardRef((p, ref) => { childrenName, }) - const [innerValue, setInnerValue] = useState(value) + let valueTemp + if (props.multiple) { + valueTemp = dataPreview(value as Array) + } else { + valueTemp = value as CascaderValue[] + } + const [innerValue, setInnerValue] = useState< + CascaderValue[] | CascaderValue[][] + >(valueTemp) useEffect(() => { if (!visible) { - setInnerValue(value) + if (props.multiple) { + setInnerValue(dataPreview(value as Array)) + } else { + setInnerValue(value) + } } }, [visible, value]) @@ -187,7 +286,10 @@ export const Cascader = forwardRef((p, ref) => { return ( <> {popupElement} - {props.children?.(generateValueExtend(value).items, actions)} + {props.children?.( + generateValueExtend(value as CheckListValue[]).items, + actions + )} ) }) diff --git a/src/components/cascader/prompt.tsx b/src/components/cascader/prompt.tsx index 6e31adb1ae..fa03e6c3b5 100644 --- a/src/components/cascader/prompt.tsx +++ b/src/components/cascader/prompt.tsx @@ -7,7 +7,7 @@ import type { CascaderProps, CascaderValue } from './index' export function prompt( props: Omit ) { - return new Promise(resolve => { + return new Promise(resolve => { const Wrapper: FC = () => { const [visible, setVisible] = useState(false) useEffect(() => { diff --git a/src/components/check-list/check-list-item.tsx b/src/components/check-list/check-list-item.tsx index dc17d9cca1..d9d6641208 100644 --- a/src/components/check-list/check-list-item.tsx +++ b/src/components/check-list/check-list-item.tsx @@ -1,3 +1,5 @@ +// eslint-disable-next-line @typescript-eslint/ban-ts-comment +// @ts-nocheck import React, { useContext } from 'react' import type { FC } from 'react' import List, { ListItemProps } from '../list' @@ -35,8 +37,26 @@ export const CheckListItem: FC = props => { const active = context.value.includes(props.value) const readOnly = props.readOnly || context.readOnly const defaultExtra = active ? context.activeIcon : null + + const { activeIconSetPath, index, setPath } = active + ? context.activeSetPathMiddleware + ? context.activeSetPathMiddleware + : {} + : {} const renderExtra = context.extra ? context.extra(active) : defaultExtra - const extra =
{renderExtra}
+ const extra = ( +
{ + if (activeIconSetPath) { + setPath(props.value, index) + event.stopPropagation() + } + }} + > + {renderExtra} +
+ ) return withNativeProps( props, diff --git a/src/components/check-list/check-list.tsx b/src/components/check-list/check-list.tsx index 74b6ec01d9..7c2256ca69 100644 --- a/src/components/check-list/check-list.tsx +++ b/src/components/check-list/check-list.tsx @@ -22,6 +22,13 @@ export type CheckListProps = Pick & { disabled?: boolean readOnly?: boolean children?: ReactNode + activeSetPathMiddleware: { + activeIconSetPath: any + value?: any + event?: any + setPath: any + index: any + } } & NativeProps const defaultProps = { @@ -38,7 +45,7 @@ export const CheckList: FC = props => { function check(val: CheckListValue) { if (mergedProps.multiple) { - setValue([...value, val]) + setValue([...(value as CheckListValue[]), val]) } else { setValue([val]) } @@ -48,7 +55,8 @@ export const CheckList: FC = props => { setValue(value.filter(item => item !== val)) } - const { activeIcon, extra, disabled, readOnly } = mergedProps + const { activeIcon, extra, disabled, readOnly, activeSetPathMiddleware } = + mergedProps return ( = props => { extra, disabled, readOnly, + activeSetPathMiddleware, }} > {withNativeProps( diff --git a/src/components/check-list/context.tsx b/src/components/check-list/context.tsx index 4af764ac20..0ffbcfaa1d 100644 --- a/src/components/check-list/context.tsx +++ b/src/components/check-list/context.tsx @@ -10,4 +10,11 @@ export const CheckListContext = createContext<{ extra?: (active: boolean) => ReactNode disabled?: boolean readOnly?: boolean + activeSetPathMiddleware: { + activeIconSetPath: any + value?: any + event?: any + setPath: any + index: any + } } | null>(null) From 1e1a5046068f76ba2b13de20bbe080f2c641dc5f Mon Sep 17 00:00:00 2001 From: Electrolux <59329360+electroluxcode@users.noreply.github.com> Date: Wed, 4 Sep 2024 11:34:14 +0800 Subject: [PATCH 2/4] Update src/components/cascader-view/cascader-view.tsx Co-authored-by: afc163 --- src/components/cascader-view/cascader-view.tsx | 6 +----- 1 file changed, 1 insertion(+), 5 deletions(-) diff --git a/src/components/cascader-view/cascader-view.tsx b/src/components/cascader-view/cascader-view.tsx index 55af2f9056..0fc1bfdd11 100644 --- a/src/components/cascader-view/cascader-view.tsx +++ b/src/components/cascader-view/cascader-view.tsx @@ -62,11 +62,7 @@ export const CascaderView: FC = p => { const [value, setValue] = usePropsValue({ ...props, onChange: val => { - if (props.multiple) { - props.onChange?.(val, {} as any) - } else { - props.onChange?.(val, generateValueExtend(val as string[])) - } + props.onChange?.(val, props.multiple ? generateValueExtend(val as string[]) : {}); }, }) const [tabActiveIndex, setTabActiveIndex] = useState(0) From dbc231d6f860076e16bbe1d91ab1770a9eee15db Mon Sep 17 00:00:00 2001 From: electroluxcode <3451613934@qq.com> Date: Wed, 4 Sep 2024 13:00:07 +0800 Subject: [PATCH 3/4] chore: cascaderview levels split multiple and single --- .../cascader-view/cascader-view.tsx | 154 +++++++++++------- 1 file changed, 95 insertions(+), 59 deletions(-) diff --git a/src/components/cascader-view/cascader-view.tsx b/src/components/cascader-view/cascader-view.tsx index 0fc1bfdd11..e7896be4b9 100644 --- a/src/components/cascader-view/cascader-view.tsx +++ b/src/components/cascader-view/cascader-view.tsx @@ -62,75 +62,111 @@ export const CascaderView: FC = p => { const [value, setValue] = usePropsValue({ ...props, onChange: val => { - props.onChange?.(val, props.multiple ? generateValueExtend(val as string[]) : {}); + props.onChange?.( + val, + props.multiple ? generateValueExtend(val as string[]) : {} + ) }, }) const [tabActiveIndex, setTabActiveIndex] = useState(0) - const levels = useMemo(() => { - if (props.multiple) { - const ret: { - selected: CascaderOption | undefined - options: CascaderOption[] - }[] = [] + const extractLevelsForMultiple = ( + value: any[], + options: CascaderOption[], + valueName: string, + childrenName: string + ) => { + const ret: { + selected: CascaderOption | undefined + options: CascaderOption[] + }[] = [] - let currentOptions = props.options - let reachedEnd = false + let currentOptions = options + let reachedEnd = false - for (const v of value) { - let target - ;(v as string[]).forEach(e => { - const temp = currentOptions.find(option => - e.includes(option[valueName]) - ) - if (temp) { - target = temp - } - }) - ret.push({ - selected: target, - options: currentOptions, - }) - if (!target || !target[childrenName]) { - reachedEnd = true - break + for (const v of value) { + let target: CascaderOption | undefined + ;(v as string[]).forEach(e => { + const temp = currentOptions.find(option => + e.includes(option[valueName]) + ) + if (temp) { + target = temp } - currentOptions = target[childrenName] - } - if (!reachedEnd) { - ret.push({ - selected: undefined, - options: currentOptions, - }) + }) + ret.push({ + selected: target, + options: currentOptions, + }) + if (!target || !target[childrenName]) { + reachedEnd = true + break } - return ret - } else { - const ret: { - selected: CascaderOption | undefined - options: CascaderOption[] - }[] = [] + currentOptions = target[childrenName] + } - let currentOptions = props.options - let reachedEnd = false - for (const v of value) { - const target = currentOptions.find(option => option[valueName] === v) - ret.push({ - selected: target, - options: currentOptions, - }) - if (!target || !target[childrenName]) { - reachedEnd = true - break - } - currentOptions = target[childrenName] - } - if (!reachedEnd) { - ret.push({ - selected: undefined, - options: currentOptions, - }) + if (!reachedEnd) { + ret.push({ + selected: undefined, + options: currentOptions, + }) + } + + return ret + } + + const extractLevelsForSingle = ( + value: any[], + options: CascaderOption[], + valueName: string, + childrenName: string + ) => { + const ret: { + selected: CascaderOption | undefined + options: CascaderOption[] + }[] = [] + + let currentOptions = options + let reachedEnd = false + + for (const v of value) { + const target = currentOptions.find(option => option[valueName] === v) + ret.push({ + selected: target, + options: currentOptions, + }) + if (!target || !target[childrenName]) { + reachedEnd = true + break } - return ret + currentOptions = target[childrenName] + } + + if (!reachedEnd) { + ret.push({ + selected: undefined, + options: currentOptions, + }) + } + + return ret + } + + const levels = useMemo(() => { + if (props.multiple) { + return extractLevelsForMultiple( + value, + props.options, + valueName, + childrenName + ) + } else { + return extractLevelsForSingle( + value, + props.options, + valueName, + childrenName + ) } }, [value, props.options]) From 69c90836aac2ca4f786df368945d1f09765619bf Mon Sep 17 00:00:00 2001 From: electroluxcode <3451613934@qq.com> Date: Fri, 6 Sep 2024 17:01:26 +0800 Subject: [PATCH 4/4] docs: add multiple description --- src/components/cascader/index.en.md | 1 + src/components/cascader/index.zh.md | 1 + 2 files changed, 2 insertions(+) diff --git a/src/components/cascader/index.en.md b/src/components/cascader/index.en.md index 78100ec70f..745bd14458 100644 --- a/src/components/cascader/index.en.md +++ b/src/components/cascader/index.en.md @@ -34,6 +34,7 @@ type CascaderValueExtend = { | Name | Description | Type | Default | | --- | --- | --- | --- | +| multiple | Whether to support selecting multiple options | `boolean` | false | | activeIcon | The icon displayed when selected | `ReactNode` | - | | cancelText | Text of the cancel button | `ReactNode` | `'取消'` | | children | Render function of the selected options | `(items: CascaderOption[], actions: CascaderActions) => ReactNode` | - | diff --git a/src/components/cascader/index.zh.md b/src/components/cascader/index.zh.md index 9310cb7bd1..6b7cdd330d 100644 --- a/src/components/cascader/index.zh.md +++ b/src/components/cascader/index.zh.md @@ -34,6 +34,7 @@ type CascaderValueExtend = { | 属性 | 说明 | 类型 | 默认值 | | --- | --- | --- | --- | +| multiple | 是否允许多选 | `boolean` | false | | activeIcon | 选中图标 | `ReactNode` | - | | cancelText | 取消按钮的文字 | `ReactNode` | `'取消'` | | children | 所选项的渲染函数 | `(items: CascaderOption[], actions: CascaderActions) => ReactNode` | - |