Listen Contract Events

Author of this section: @Fish

this session describes how to monitor contract events in DApp and update the DApp interface in real time.

Introduction

the events of blockchain smart contracts are somewhat different from the concept of events that we understand in traditional application development, and the blockchain itself does not have a messaging mechanism to send events to applications. It is essentially an abstraction of the logs on the EVM.

This type of log is cheaper than the state change of a smart contract and is a more economical data storage method. Each event consumes about 2000 gas, while storing a new variable on the chain requires at least 20000 gas. So in smart contracts, we usually use events to record some important state changes. In addition, based on the RPC interface provided by the Node Service, the DApp front-end page can monitor contract events to achieve relatively real-time updates.

How to add events to a smart contract

in smart contracts, events are generated event statement emit trigger, the following is an example, here only to do a simple demonstration, specific we will later in the contract development course to explain:

// SPDX-License-Identifier: MIT
pragma solidity ^0.8.20;

import "@openzeppelin/contracts/token/ERC721/ERC721.sol";
import "@openzeppelin/contracts/access/Ownable.sol";

contract MyToken is ERC721, Ownable {
  uint256 private _nextTokenId = 0;
+  event Minted(address minter, uint256 amount);
  constructor()
    ERC721("MyToken", "MTK")
    Ownable(msg.sender)
  {}
  function mint(uint256 quantity) public payable {
    require(quantity == 1, "quantity must be 1");
    require(msg.value == 0.01 ether, "must pay 0.01 ether");
    uint256 tokenId = _nextTokenId++;
    _mint(msg.sender, tokenId);
+    emit Minted(msg.sender, quantity);
  }
}

how to listen for events in a DApp

on the DApp front-end page, we can listen to contract events by calling the RPC interface provided by the Node Service. Here we continue to use wagmi to develop.

First introduced useWatchContractEvent .

import {
  createConfig,
  http,
  useReadContract,
  useWriteContract,
+  useWatchContractEvent,
} from "wagmi";

Then use useWatchContractEvent listening for contract events.

useWatchContractEvent({
  address: "0xEcd0D12E21805803f70de03B72B1C162dB0898d9",
  abi: [
    {
      anonymous: false,
      inputs: [
        {
          indexed: false,
          internalType: "address",
          name: "minter",
          type: "address",
        },
        {
          indexed: false,
          internalType: "uint256",
          name: "amount",
          type: "uint256",
        },
      ],
      name: "Minted",
      type: "event",
    },
  ],
  eventName: "Minted",
  onLogs() {
    message.success("new minted!");
  },
});

In fact, the specific usage method is not much different from the traditional front-end development under the package of SDK like wagmi. After you pass in the contract address and the event to be monitored, you can onLogs handle the event in the callback.

Of course, the above code is just for demonstration. When actually coding, you may need onLogs to determine the specific content of the event, and then update the page.

In addition, abi parameters are usually automatically generated when the contract is compiled, and a complete copy of the content can be maintained in the project as a whole. There is no need to manually write it at each call place as in the example.

Because the event will only be triggered after the contract is called, we can't debug it yet. You can continue the course and wait until we can deploy a test contract in the Test environment.

The complete code can be referenced web3.tsx .