Complex state management
Edit this pageAs applications grow and start to involve many components, more intricate user interactions, and possibly communication with backend services, you may find that staying organized with more basic state management methods can become difficult to maintain.
Consider this example:
import { For, createSignal, Show, createMemo } from "solid-js"
const App = () => {  const [tasks, setTasks] = createSignal([])  const [numberOfTasks, setNumberOfTasks] = createSignal(tasks.length)  const completedTasks = createMemo(() => tasks().filter((task) => task.completed))  let input
  const addTask = (text) => {    setTasks([...tasks(), { id: tasks().length, text, completed: false }])    setNumberOfTasks(numberOfTasks() + 1)  }  const toggleTask = (id) => {    setTasks(      tasks().map((task) =>        task.id !== id ? task : { ...task, completed: !task.completed }      )    )  }
  return (    <>      <h1>My list</h1>      <span>You have {numberOfTasks()} task(s) today!</span>      <div>        <input ref={input} />        <button          onClick={(e) => {            if (!input.value.trim()) return            addTask(input.value)            input.value = ""          }}        >          Add Task        </button>      </div>      <For each={tasks()}>        {(task) => {          const { id, text } = task          console.log(`Creating ${text}`)          return (            <div>              <input                type="checkbox"                checked={task.completed}                onChange={[toggleTask, id]}              />              <span                style={{                  "text-decoration": task.completed ? "line-through" : "none",                }}              >                {text}              </span>            </div>          )        }}      </For>    </>  )}
export default AppThere are several challenges to managing state in this way:
- 
Increased verbosity with the multiple createSignalcalls fortasks,numberOfTasks, as well as acreateMemofunction forcompletedTasks. Additionally, with each state update, there requires manual updates to other related states which risks the application becoming out of sync.
- 
While Solid is optimized, this components design leads to frequent recalculations, such as updating completedTaskswith every toggle action, which can negatively impact performance. In addition, the dependence on the component's logic on the current state fornumberOfTasksandcompletedTaskscan complicate code understanding.
As an application like this scales, managing state in this manner becomes even more complex. Introducing other dependent state variables would require updates across the entire component which would likely introduce more errors. This would likely make it more difficult to separate specific functionalities into distinct, reusable components without transferring a substantial portion of state management logic, as well.
Introducing stores
Through recreating this list using Stores, you will see how stores can improve the readability and management of your code.
If you're new to the concept of stores, see the stores section.
Creating a store
To reduce the amount of signals that were used in the original example, you can do the following using a store:
import { createStore } from "solid-js/store"
const App = () => {  const [state, setState] = createStore({    tasks: [],    numberOfTasks: 0,  })}
export default AppThrough using a store, you no longer need to keep track of separate signals for tasks, numberOfTasks, and completedTasks.
Accessing state values
Once you have created your store, the values can be accessed directly through the first value returned by the createStore function:
import { createStore } from "solid-js/store"
const App = () => {  const [state, setState] = createStore({    tasks: [],    numberOfTasks: 0,  })  return (    <>      <h1>My Task List for Today</h1>      <span>You have {state.numberOfTasks} task(s) for today!</span>    </>  )}
export default AppThrough state.numberOfTasks, the display will now show the store's value held in the numberOfTasks property.
Making changes to the store
When you want to modify your store, you use the second element returned by the createStore function.
This element allows you to make modifications to the store, letting you both add new properties and update existing ones.
However, because properties within a store are created lazily, setting a property in the component function body without creating a tracking scope will not update the value.
To create the signal so it reactively updates, you have to access the property within a tracking scope, such as using a createEffect:
// not reactivesetState("numberOfTasks", state.tasks.length)
// reactivecreateEffect(() => {  setState("numberOfTasks", state.tasks.length)})Adding to an array
To add an element to an array, in this case the new task, you can append to the next index of the array through state.tasks.length.
By pinpointing the tasks key in combination with the upcoming position, the new task is added to the end of the array.
const addTask = (text) => {  setState("tasks", state.tasks.length, {    id: state.tasks.length,    text,    completed: false,  })}The setter in stores follow path syntax: setStore("key", value).
In the addTask function the tasks array is appended through setState("tasks", state.tasks.length, { id: state.tasks.length, text, completed: false }), an example of this in action.
Mutating state with produce
In situations where you need to make multiple setState calls and target multiple properties, you can simplify your code and improve readability by using Solid's produce utility function.
Something such as toggle function:
const toggleTask = (id) => {  const currentCompletedStatus = state.tasks[id].completed  setState(    "tasks",    (task) => task.id === id,    "completed",    !currentCompletedStatus  )}Can be simplified using produce:
import { produce } from "solid-js/store"
const toggleTask = (id) => {  setState(    "tasks",    (task) => task.id === id,    produce((task) => {      task.completed = !task.completed    })  )}
// You can also rewrite the `addTask` function through produceconst addTask = (text) => {  setState(    "tasks",    produce((task) => {      task.push({ id: state.tasks.length, text, completed: false })    })  )}Read about some of the other advantages to using produce.
Another benefit to working with produce is that it offers a way to modify a store without having to make multiple setStore calls.
// without producebatch(() => {setState(0, "text", "I'm updated text")setState(0, "completed", true)})
// with producesetState(0,produce((task) => {  task.text = "I'm updated text";  task.completed = true;}))The updated example:
import { For, createEffect, Show } from "solid-js"import { createStore, produce } from "solid-js/store"
const App = () => {  let input // lets you target the input value  const [state, setState] = createStore({    tasks: [],    numberOfTasks: 0,  })
  const addTask = (text) => {    setState("tasks", state.tasks.length, {      id: state.tasks.length,      text,      completed: false,    })  }
  const toggleTask = (id) => {    setState(      "tasks",      (task) => task.id === id,      produce((task) => {        task.completed = !task.completed      })    )  }
  createEffect(() => {    setState("numberOfTasks", state.tasks.length)  })
  return (    <>      <div>        <h1>My Task List for Today</h1>        <span>You have {state.numberOfTasks} task(s) for today!</span>      </div>      <input ref={input} />      <button        onClick={(e) => {          if (!input.value.trim()) return          addTask(input.value)          input.value = ""        }}      >        Add Task      </button>      <For each={state.tasks}>        {(task) => {          const { id, text } = task          return (            <div>              <input                type="checkbox"                checked={task.completed}                onChange={() => toggleTask(task.id)}              />              <span>{text}</span>            </div>          )        }}      </For>    </>  )}
export default AppState sharing
As applications grow and become more complex, sharing state between components can become a challenge. Passing state and functions from parent to child components, especially across multiple levels, is commonly referred to as "prop drilling". Prop drilling can lead to verbose, hard-to-maintain code, and can make the data flow in an application more difficult to follow. To solve this problem and allow for a more scalable and maintainable codebase, Solid provides context.
To use this, you need to create a context. This context will have a default value and can be consumed by any descendant component.
import { createContext } from "solid-js"
const TaskContext = createContext()Your components will be wrapped with the Provider from the context, and passed with the values that you wish to share.
import { createStore } from "solid-js/store"
const TaskApp = () => {  const [state, setState] = createStore({    tasks: [],    numberOfTasks: 0,  })
  return (    <TaskContext.Provider value={{ state, setState }}>      {/* Your components */}    </TaskContext.Provider>  )}In any descendent component, you can consume the context values using useContext:
import { useContext } from "solid-js"
const TaskList = () => {  const { state, setState } = useContext(TaskContext)
  // Now you can use the shared state and functions}For a deeper dive, please refer to our dedicated page on context.