Expand description
§Hooks
Hooks are special functions to be used inside of Components that lets you handle different things like the state or lifecycle of your component. They are usually prefixed with use
, e.g use_signal
, use_effect
, use_memo
, etc.
§Rules of Hooks
Even though hooks appear to be normal functions they are in fact special so you cannot just call them however you want.
§1. They cannot be called conditionally
You cannot do the following because hooks need to maintain their order. So, if one component is calling 3 different hooks in one render, and then in another render if just calls 2, it would be breaking this rule.
❌:
#[component]
fn MyComponent(value: bool) -> Element {
let is_enabled = if value {
// This should be moved out of the conditional
let state = use_signal(|| value);
state()
} else {
true
};
Ok(VNode::placeholder())
}
✅:
#[component]
fn MyComponent(initial_value: bool) -> Element {
let is_enabled = use_signal(move || initial_value);
Ok(VNode::placeholder())
}
§2. They can only be called inside of Component functions
You cannot call them inside of event handlers, futures, etc.
❌:
#[component]
fn MyComponent() -> Element {
let onclick = |_| {
let state = use_signal(|| false);
};
rsx!(
label {
onclick,
"Hello, World!"
}
)
}
✅:
#[component]
fn MyComponent() -> Element {
let mut state = use_signal(|| false);
let onclick = move |_| {
state.set(true);
};
rsx!(
label {
onclick,
"Hello, World!"
}
)
}
§3. They cannot be called in loops
Hooks cannot be called in loops as the numbers of iterations might change between renders.
❌:
#[component]
fn MyComponent() -> Element {
for i in 0..5 {
let state = use_signal(|| i);
}
Ok(VNode::placeholder())
}
✅:
#[component]
fn MyComponent() -> Element {
let state = use_signal(|| (0..5).into_iter().collect::<Vec<_>>());
Ok(VNode::placeholder())
}