To change values in Leva store from the outside, use the function API by passing () => schema to useControls.
const [{ text }, set] = useControls(() => ({ text: 'my string' }))
return <input type="text" value={text} onChange={(e) => set({ text: e.target.value })} />Use set to update values from outside of Leva GUI.
See an example in Storybook.
onChange callback in the schema is called for all changes to the value. Values with onChange will no longer cause React to rerender (unless passing an additional transient flag set to true, see below), so you can use it to efficiently update frequently changing values.
const divRef = React.useRef(null)
const data = useControls({
color: {
value: '#f00',
onChange: (v) => {
// imperatively update the world after Leva input changes
divRef.current.style.color = v
},
},
})
// `data.color` is undefinedSee an example in Storybook.
If you need the onChange callback while still wanting to retrieve the input value, you can set transient: false.
const divRef = React.useRef(null)
const data = useControls({
color: { value: '#f00', onChange: (v) => {}, transient: false },
})
// `data.color` will be definedWith set and onChange we can bind to any imperative API. Whenever external
const [, set] = useControls(() => ({
position: {
value: { x: 0, y: 0 },
onChange: (value) => {
// imperatively update the world after Leva input changes
},
},
}))
const targetRef = useRef()
useDrag(({ offset: [x, y] }) => set({ position: { x, y } }), { domTarget: targetRef })
const targetRef = useRef()
useDrag(({ offset: [x, y] }) => set({ position: { x, y } }), { domTarget: targetRef })See this CodeSandbox for an example.
@TODO This page will go in depth on how to replicate use-cases like https://twitter.com/simonghales/status/1357630773323436033 where the values are being changed outside of the GUI
This page should also probably talk about multi-panes and other non dat.GUI situations
