Add a system
In this tutorial you add a system to decrement the counter and update the application to use it.
Setup
Create a new MUD application from the template. Use the vanilla template.
pnpm create mud@latest --name tutorial --template vanilla
cd tutorial
pnpm install
pnpm devAdd a contract for the new system
Create a file packages/contracts/src/systems/DecrementSystem.sol.
Note that we could have just added a function to the existing system, IncrementSystem.sol.
The only reason we are adding a new system here is to see how to do it.
// SPDX-License-Identifier: MIT
pragma solidity >=0.8.0;
 
import { System } from "@latticexyz/world/src/System.sol";
import { Counter } from "../codegen/index.sol";
 
contract DecrementSystem is System {
  function decrement() public returns (uint32) {
    uint32 counter = Counter.get();
    uint32 newValue = counter - 1;
    Counter.set(newValue);
    return newValue;
  }
}Explanation
import { System } from "@latticexyz/world/src/System.sol";
import { Counter } from "../codegen/index.sol";The two things the system needs to know: how to be a System and how to access the Counter.
uint32 counter = Counter.get();Get the counter value.
Counter.set(newValue);Set the counter to a new value.
Add decrement to the user interface
Having a system be able to do something doesn't help anybody unless it is called from somewhere. In this case, the vanilla getting started front end.
- 
Edit packages/client/src/mud/createSystemCalls.tsto includedecrement. This is the file after the changes:createSystemCalls.ts/* * Create the system calls that the client can use to ask * for changes in the World state (using the System contracts). */ import { getComponentValue } from "@latticexyz/recs"; import { ClientComponents } from "./createClientComponents"; import { SetupNetworkResult } from "./setupNetwork"; import { singletonEntity } from "@latticexyz/store-sync/recs"; export type SystemCalls = ReturnType<typeof createSystemCalls>; export function createSystemCalls( /* * The parameter list informs TypeScript that: * * - The first parameter is expected to be a * SetupNetworkResult, as defined in setupNetwork.ts * * Out of this parameter, we only care about two fields: * - worldContract (which comes from getContract, see * https://github.com/latticexyz/mud/blob/main/templates/vanilla/packages/client/src/mud/setupNetwork.ts#L63-L69). * * - waitForTransaction (which comes from syncToRecs, see * https://github.com/latticexyz/mud/blob/main/templates/vanilla/packages/client/src/mud/setupNetwork.ts#L77-L83). * * - From the second parameter, which is a ClientComponent, * we only care about Counter. This parameter comes to use * through createClientComponents.ts, but it originates in * syncToRecs * (https://github.com/latticexyz/mud/blob/main/templates/vanilla/packages/client/src/mud/setupNetwork.ts#L77-L83). */ { worldContract, waitForTransaction }: SetupNetworkResult, { Counter }: ClientComponents, ) { const increment = async () => { /* * Because IncrementSystem * (https://mud.dev/templates/typescript/contracts#incrementsystemsol) * is in the root namespace, `.increment` can be called directly * on the World contract. */ const tx = await worldContract.write.app__increment(); await waitForTransaction(tx); return getComponentValue(Counter, singletonEntity); }; const decrement = async () => { const tx = await worldContract.write.app__decrement(); await waitForTransaction(tx); return getComponentValue(Counter, singletonEntity); }; return { increment, decrement, }; }ExplanationThe new function is decrement.const decrement = async () => {This function involves sending a transaction, which is a slow process, so it needs to be asynchronous (opens in a new tab). const tx = await worldContract.write.decrement();This is the way we call functions in systems in the root namespace of the world. await waitForTransaction(tx);Await until the transaction has been included in a block and the corresponding state has been synchronized with the client. return getComponentValue(Counter, singletonEntity) };Get the value of Counterto return it. It should already be the updated value.return { increment, decrement, };Of course, we also need to return decrementso it can be used elsewhere.
- 
Update packages/client/src/index.tsto includedecrement. This is the file after the changes:index.tsimport { setup } from "./mud/setup"; import mudConfig from "contracts/mud.config"; const { components, systemCalls: { increment, decrement }, network, } = await setup(); // Components expose a stream that triggers when the component is updated. components.Counter.update$.subscribe((update) => { const [nextValue, prevValue] = update.value; console.log("Counter updated", update, { nextValue, prevValue }); document.getElementById("counter")!.innerHTML = String(nextValue?.value ?? "unset"); }); // Attach the increment function to the html element with ID `incrementButton` (if it exists) document.querySelector("#incrementButton")?.addEventListener("click", increment); document.querySelector("#decrementButton").addEventListener("click", decrement); // https://vitejs.dev/guide/env-and-mode.html if (import.meta.env.DEV) { const { mount: mountDevTools } = await import("@latticexyz/dev-tools"); mountDevTools({ config: mudConfig, publicClient: network.publicClient, walletClient: network.walletClient, latestBlock$: network.latestBlock$, storedBlockLogs$: network.storedBlockLogs$, worldAddress: network.worldContract.address, worldAbi: network.worldContract.abi, write$: network.write$, recsWorld: network.world, }); }Explanationconst { components, systemCalls: { decrement, increment }, network, } = await setup();This syntax means we call setup()(opens in a new tab) and then set variables to portions of the result.- components←- (await setup()).components
- increment←- (await setup()).systemCalls.increment
- decrement←- (await setup()).systemCalls.decrement
- network←- (await setup()).network
 systemCallscomes fromcreateSystemCalls(), which we modified in the previous step.document.querySelector("#decrementButton").addEventListener("click", decrement);We need to make decrementavailable to our application code. Most frameworks have a standard mechanism to do this, but we are usingvanilla, which doesn't. So we usedocument.querySelector(opens in a new tab) to find the appropriate button and thenaddEventListener(opens in a new tab) to registerdecrement.
- 
Modify packages/client/index.htmlto add a decrement button. This is the file after the changes:index.html<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8" /> <meta name="viewport" content="width=device-width, initial-scale=1.0" /> <title>a minimal MUD client</title> </head> <body> <script type="module" src="/src/index.ts"></script> <div>Counter: <span id="counter">0</span></div> <button id="incrementButton">Increment</button> <button id="decrementButton">Decrement</button> </body> </html>
- 
See that there is a decrement button and that you can use it.