ColPicker
Use multi-column picker for cascading selection, providing better interaction experience. The multi-column picker supports unlimited level selection.
Note
Multi-column picker is commonly used for selecting provinces, cities, and districts. We use Vant's China province, city, and district data as the data source. You can install the @vant/area-data npm package to import it:
# via npm
npm i @vant/area-data
# via yarn
yarn add @vant/area-data
# via pnpm
pnpm add @vant/area-data
# via Bun
bun add @vant/area-data
To facilitate developers using @vant/area-data
for development and debugging, we have encapsulated useColPickerData
. You can directly use useColPickerData
to get the data source.
useColPickerData
wrapper based on @vant/area-data
// You can place this code in src/hooks/useColPickerData.ts in your project
import { useCascaderAreaData } from '@vant/area-data'
export type CascaderOption = {
text: string
value: string
children?: CascaderOption[]
}
/**
* Use '@vant/area-data' as data source to construct ColPicker component data
* @returns
*/
export function useColPickerData() {
// '@vant/area-data' data source
const colPickerData: CascaderOption[] = useCascaderAreaData()
// Find child nodes by code, return all nodes if no code provided
function findChildrenByCode(data: CascaderOption[], code?: string): CascaderOption[] | null {
if (!code) {
return data
}
for (const item of data) {
if (item.value === code) {
return item.children || null
}
if (item.children) {
const childrenResult = findChildrenByCode(item.children, code)
if (childrenResult) {
return childrenResult
}
}
}
return null
}
return { colPickerData, findChildrenByCode }
}
Basic Usage
label
sets the left text content;
columns
sets the data source, which is a two-dimensional array. Each column is a one-dimensional array, and each option includes value
(option value) and label
(option name).
v-model
sets the selected value, which is an array;
You can also listen to the change
event to get the selected value. The event
is an object containing two properties: value
(selected value array) and selectedItems
(selected item object array).
Pass in the column-change
property, which is of type function
and receives an options object parameter. The structure of options is as follows:
Parameter | Type | Description | Version |
---|---|---|---|
selectedItem | object | Currently selected item in the column, data structure is consistent with the options in columns | - |
index | number | Current column index | - |
rowIndex | number | Current column selected item index | - |
resolve | function | Receives the option array for the next column | - |
finish | function | End picker selection, if cannot close normally like data fetch failure, execute finish(false) | - |
<wd-col-picker label="Select Address" v-model="value" :columns="area" :column-change="columnChange" @confirm="handleConfirm"></wd-col-picker>
// useColPickerData can be referenced from the introduction at the top of this section
// Adjust the import path according to your actual situation, don't just copy and paste
import { useColPickerData } from '@/hooks/useColPickerData'
const { colPickerData, findChildrenByCode } = useColPickerData()
const value = ref<string[]>([])
const area = ref<any[]>([
colPickerData.map((item) => {
return {
value: item.value,
label: item.text
}
})
])
const columnChange = ({ selectedItem, resolve, finish }) => {
const areaData = findChildrenByCode(colPickerData, selectedItem.value)
if (areaData && areaData.length) {
resolve(
areaData.map((item) => {
return {
value: item.value,
label: item.text
}
})
)
} else {
finish()
}
}
function handleConfirm({ value }) {
console.log(value)
}
Asynchronous Loading
Generally, column-change is an asynchronous data fetching operation. When column-change is triggered, the component will have a default loading state, which is closed after the data responds.
If the asynchronous data request fails, call finish(false)
.
<wd-col-picker label="Select Address" v-model="value" :columns="area" :column-change="columnChange" @confirm="handleConfirm"></wd-col-picker>
// useColPickerData can be referenced from the introduction at the top of this section
// Adjust the import path according to your actual situation, don't just copy and paste
import { useColPickerData } from '@/hooks/useColPickerData'
const { colPickerData, findChildrenByCode } = useColPickerData()
const value = ref<string[]>([])
const area = ref<any[]>([
colPickerData.map((item) => {
return {
value: item.value,
label: item.text
}
})
])
const columnChange = ({ selectedItem, resolve, finish }) => {
// Simulate asynchronous request
setTimeout(() => {
// Simulate request failure
if (Math.random() > 0.7) {
finish(false)
toast.error.error('Data request failed, please try again')
return
}
// Why use selectedItem.value as code? Because when constructing area, we put the identifier in the value field, similarly you can change it to other fields as long as they correspond to area's fields
const areaData = findChildrenByCode(colPickerData, selectedItem.value)
if (areaData && areaData.length) {
resolve(
areaData.map((item) => {
return {
value: item.value,
label: item.text
}
})
)
} else {
// When there are no more items, complete the operation
finish()
}
}, 300)
}
function handleConfirm({ value }) {
console.log(value)
}
Initial Options
There are two ways to set initial options:
- When setting initial options, the length of the
columns
array should match the length of thevalue
array, and each value invalue
must be found incolumns
.
<wd-col-picker label="Select Address" v-model="value" :columns="area" :column-change="columnChange"></wd-col-picker>