useProxyModel
A composable for syncing refs bidirectionally with selection contexts, enabling seamless v-model integration with selection state.
Usage
The useProxyModel composable syncs an existing ref (like from defineModel()) with a selection context bidirectionally. Changes in either direction automatically propagate.
import { ref } from 'vue'
import { createSelection, useProxyModel } from '@vuetify/v0'
const model = ref<string>()
const selection = createSelection({ events: true })
selection.onboard([
{ id: 'apple', value: 'Apple' },
{ id: 'banana', value: 'Banana' },
])
// Sync model with selection
const stop = useProxyModel(selection, model)
model.value = 'Apple'
console.log(selection.selectedIds) // Set { 'apple' }
selection.select('banana')
console.log(model.value) // 'Banana'Multiple mode
Pass multiple: true to sync an array model with a multi-select context. This must be explicit — multiple is never inferred from the context:
const model = ref<string[]>([])
const selection = createGroup({ events: true })
selection.onboard([
{ id: 'red', value: 'Red' },
{ id: 'green', value: 'Green' },
{ id: 'blue', value: 'Blue' },
])
const stop = useProxyModel(selection, model, { multiple: true })
selection.toggle('red')
selection.toggle('blue')
console.log(model.value) // ['Red', 'Blue']The multiple option accepts MaybeRefOrGetter<boolean>, so you can drive it from a prop or computed:
const props = defineProps<{ multiple?: boolean }>()
const stop = useProxyModel(selection, model, { multiple: () => props.multiple ?? false })Disabled items
When a new item registers with the selection, useProxyModel checks whether the model already holds that item’s value and selects it if so — this is how items that were selected before they mounted get their initial state. The check is skipped if the item is disabled. The disabled property is automatically unwrapped if it is a Ref<boolean>:
selection.onboard([
{ id: 'a', value: 'A' },
{ id: 'b', value: 'B', disabled: ref(true) }, // reactive disabled
])
model.value = 'B'
// 'b' is NOT auto-selected because disabled is true at registration timeArchitecture
useProxyModel creates bidirectional sync between v-model refs and selection state:
Reactivity
useProxyModel creates bidirectional reactive sync between a ref and a selection registry. It uses Vue watchers internally for automatic propagation.
| Behavior | Reactive | Notes |
|---|---|---|
| Model → Selection | Changes to model auto-select items | |
| Selection → Model | Selection changes update model |
Automatic cleanup The watchers are disposed automatically via onScopeDispose. The returned stop() function allows early manual cleanup.
Examples
Color Picker
A color swatch selector wired to a v-model-style ref via useProxyModel, keeping selection state and external model in sync bidirectionally.
Functions
useProxyModel
(context: ProxyModelTarget, model: Ref<unknown, unknown>, options?: ProxyModelOptions | undefined) => () => voidSyncs a ref with a model context bidirectionally.
Options
multiple
MaybeRefOrGetter<boolean> | undefinedtransformIn
((val: unknown) => unknown) | undefinedtransformOut
((val: unknown) => unknown) | undefined