Description State in runes is based on primitives, which provides a concise syntax. However, when sending these primitives across boundaries, e.g. in function arguments, the state is not reactive, and rather, the static value gets sent.
Box provides several utilities to make sending and receiving reactive values easier.
Usage box
Initializes the boxed state.
< script lang = " ts " >
import { box } from " runed " ;
let count = box ( 0 ) ;
</ script >
< button onclick = { () => count . value ++ } >
clicks: { count . value}
</ button >
Copy box.with
Allows you to use getters and setters to define a box. Useful to pass around state.
< script lang = " ts " >
import { type WritableBox , box } from " runed " ;
function useCounter ( count : WritableBox < number > ) {
return {
increment () {
count . value ++;
} ,
// We pass a box that doubles the count value
double : box . with (() => count . value * 2 ) ,
} ;
}
let count = $ state ( 0 ) ;
// We pass count with box.with
const { double , increment } = useCounter (
box . with (
() => count . value ,
( v ) => ( count = v )
)
) ;
</ script >
< button onclick = { increment } >
clicks: { count }
double: { double . value}
</ button >
Copy box.from
Creates a box from an existing box, a getter function, or a static value.
Used in functions to receive props that are optionally reactive.
< script lang = " ts " >
import { box } from " runed " ;
function useCounter ( _count : WritableBox < number > | number ) {
const count = box . from ( _count ) ;
return {
count ,
increment () {
count . value ++;
} ,
// We pass a box that doubles the count value
double : box . with (() => count . value * 2 ) ,
} ;
}
const counter1 = useCounter ( 1 ) ;
console . log ( counter1 . count . value ) ; // 1
console . log ( counter1 . double . value ) ; // 2
const counter2 = useCounter ( box ( 2 )) ;
console . log ( counter2 . count . value ) ; // 2
console . log ( counter2 . double . value ) ; // 4
function useDouble ( _count : number | (() => number ) | ReadableBox < number > ) {
const count = box . from ( _count ) ;
return box . with (() => count . value * 2 ) ;
}
const double1 = useDouble ( 1 ) ;
console . log ( double1 . value ) ; // 2
const double2 = useDouble ( box ( 2 )) ;
console . log ( double2 . value ) ; // 4
const double3 = useDouble (() => counter1 . count . value ) ;
console . log ( double3 . value ) ; // 2
</ script >
Copy box.flatten
Transforms any boxes inside an object to reactive properties, flattening the .value
onto them.
const count = box ( 1 ) ;
const flat = box . flatten ({
count ,
double : box . with (() => count . value * 2 ) ,
increment () {
count . value ++;
} ,
}) ;
console . log ( flat . count ) ; // 1
console . log ( flat . double ) ; // 2
flat . increment () ;
console . log ( flat . count ) ; // 2
Copy box.readonly
Creates a readonly box from a writable box that remains in sync with the original box.
const count = box ( 1 )
const readonlyCount = box . readonly ( count )
console . log ( readonlyCount . value ) // 1
count . value ++
console . log ( readonlyCount . value ) // 2
readonlyCount . value = 3 // Error: Cannot assign to read only property 'value' of object
Copy box.isBox
Checks if a value is a Box
.
const count = box ( 1 ) ;
console . log ( box . isBox ( count )) ; // true
console . log ( box . isBox ( 1 )) ; // false
Copy box.isWritableBox
Checks if a value is a WritableBox
.
const count = box ( 1 ) ;
const double = box . with (() => count . value * 2 ) ;
console . log ( box . isWritableBox ( count )) ; // true
console . log ( box . isWritableBox ( double )) ; // false
Copy