ప్రధాన కంటెంట్‌కి స్కిప్ చేయండి

మీ కాంట్రాక్ట్ కోసం యూజర్ ఇంటర్‌ఫేస్‌ను నిర్మించడం

TypeScript
react
vite
wagmi
frontend
ప్రారంభ
Ori Pomerantz
1 నవంబర్, 2023
15 నిమిషం పఠనం

Ethereum పర్యావరణ వ్యవస్థలో మాకు అవసరమైన ఒక ఫీచర్‌ను మీరు కనుగొన్నారు. దానిని అమలు చేయడానికి మీరు స్మార్ట్ కాంట్రాక్టులను వ్రాశారు, మరియు బహుశా ఆఫ్‌చెయిన్‌లో నడిచే కొన్ని సంబంధిత కోడ్‌ను కూడా వ్రాశారు. ఇది చాలా బాగుంది! దురదృష్టవశాత్తు, యూజర్ ఇంటర్‌ఫేస్ లేకుండా మీకు ఏ వినియోగదారులు ఉండరు, మరియు మీరు చివరిసారిగా వెబ్‌సైట్‌ను వ్రాసినప్పుడు ప్రజలు డయల్-అప్ మోడెమ్‌లను ఉపయోగించారు మరియు జావాస్క్రిప్ట్ కొత్తది.

ఈ వ్యాసం మీ కోసం. మీకు ప్రోగ్రామింగ్ తెలుసని, మరియు బహుశా కొద్దిగా జావాస్క్రిప్ట్ మరియు HTML కూడా తెలుసని నేను భావిస్తున్నాను, కానీ మీ యూజర్ ఇంటర్‌ఫేస్ నైపుణ్యాలు తుప్పుపట్టినవి మరియు కాలం చెల్లినవి. మనం కలిసి ఒక సాధారణ ఆధునిక అప్లికేషన్‌ను పరిశీలిద్దాం, తద్వారా ఈ రోజుల్లో ఇది ఎలా జరుగుతుందో మీరు చూస్తారు.

ఇది ఎందుకు ముఖ్యం

సిద్ధాంతపరంగా, మీ కాంట్రాక్టులతో పరస్పర చర్య చేయడానికి మీరు ప్రజలను Etherscan (opens in a new tab) లేదా Blockscout (opens in a new tab) ఉపయోగించేలా చేయవచ్చు. అది అనుభవజ్ఞులైన Ethereans కోసం చాలా బాగుంటుంది. కానీ మేము మరొక బిలియన్ ప్రజలకు (opens in a new tab) సేవ చేయడానికి ప్రయత్నిస్తున్నాము. గొప్ప వినియోగదారు అనుభవం లేకుండా ఇది జరగదు, మరియు స్నేహపూర్వక యూజర్ ఇంటర్‌ఫేస్ అందులో ఒక పెద్ద భాగం.

గ్రీటర్ అప్లికేషన్

ఆధునిక UI ఎలా పనిచేస్తుందనే దాని వెనుక చాలా సిద్ధాంతం ఉంది, మరియు చాలా మంచి సైట్లు (opens in a new tab) దానిని వివరిస్తాయి (opens in a new tab). ఆ సైట్‌లు చేసిన మంచి పనిని పునరావృతం చేయడానికి బదులుగా, మీరు చేయడం ద్వారా నేర్చుకోవడానికి ఇష్టపడతారని నేను భావిస్తున్నాను మరియు మీరు ఆడగల అప్లికేషన్‌తో ప్రారంభిస్తున్నాను. విషయాలను పూర్తి చేయడానికి మీకు ఇంకా సిద్ధాంతం అవసరం, మరియు మేము దానిని చేరుకుంటాము - మేము సోర్స్ ఫైల్ ద్వారా సోర్స్ ఫైల్‌కు వెళ్తాము మరియు మేము వాటిని చేరుకున్నప్పుడు విషయాలను చర్చిస్తాము.

సంస్థాపన

  1. అప్లికేషన్ Sepolia (opens in a new tab) టెస్ట్ నెట్‌వర్క్‌ను ఉపయోగిస్తుంది. అవసరమైతే, Sepolia టెస్ట్ ETH పొందండి మరియు మీ వాలెట్‌కు Sepoliaను జోడించండి (opens in a new tab).

  2. GitHub రిపోజిటరీని క్లోన్ చేసి అవసరమైన ప్యాకేజీలను ఇన్‌స్టాల్ చేయండి.

    1git clone https://github.com/qbzzt/260301-modern-ui-web3.git
    2cd 260301-modern-ui-web3
    3npm install
  3. అప్లికేషన్ ఉచిత యాక్సెస్ పాయింట్లను ఉపయోగిస్తుంది, వాటికి పనితీరు పరిమితులు ఉంటాయి. మీరు నోడ్ యాజ్ ఎ సర్వీస్ ప్రొవైడర్‌ను ఉపయోగించాలనుకుంటే, src/wagmi.ts లో URLలను భర్తీ చేయండి.

  4. అప్లికేషన్‌ను ప్రారంభించండి.

    1npm run dev
  5. అప్లికేషన్ చూపిన URLకు బ్రౌజ్ చేయండి. చాలా సందర్భాలలో, అది http://localhost:5173/ (opens in a new tab).

  6. మీరు ఒక బ్లాక్‌చైన్ ఎక్స్‌ప్లోరర్‌లో (opens in a new tab) కాంట్రాక్ట్ సోర్స్ కోడ్‌ను, Hardhat యొక్క గ్రీటర్ యొక్క కొద్దిగా సవరించిన సంస్కరణను చూడవచ్చు.

ఫైల్ వాక్ త్రూ

index.html

ఈ ఫైల్ ప్రామాణిక HTML బాయిలర్‌ప్లేట్, ఈ పంక్తి మినహా, ఇది స్క్రిప్ట్ ఫైల్‌ను దిగుమతి చేస్తుంది.

1<script type="module" src="/src/main.tsx"></script>

src/main.tsx

ఫైల్ ఎక్స్‌టెన్షన్ ఈ ఫైల్ TypeScript (opens in a new tab) లో వ్రాయబడిన React కాంపోనెంట్ (opens in a new tab) అని మనకు చెబుతుంది, ఇది టైప్ చెకింగ్ (opens in a new tab)కు మద్దతు ఇచ్చే జావాస్క్రిప్ట్ యొక్క పొడిగింపు. TypeScript జావాస్క్రిప్ట్‌లోకి కంపైల్ చేయబడుతుంది, కాబట్టి మనం క్లయింట్ వైపు దీనిని ఉపయోగించవచ్చు.

ఈ ఫైల్ మీకు ఆసక్తి ఉంటే ఎక్కువగా వివరించబడింది. సాధారణంగా మీరు ఈ ఫైల్‌ను కాకుండా src/App.tsx మరియు అది దిగుమతి చేసే ఫైల్‌లను సవరిస్తారు.

1import { QueryClient, QueryClientProvider } from '@tanstack/react-query'
2import React from 'react'
3import ReactDOM from 'react-dom/client'
4import { WagmiProvider } from 'wagmi'

మనకు అవసరమైన లైబ్రరీ కోడ్‌ను దిగుమతి చేసుకోండి.

1import App from './App.tsx'

అప్లికేషన్‌ను అమలు చేసే React కాంపోనెంట్‌ను దిగుమతి చేయండి (క్రింద చూడండి).

1import { config } from './wagmi.ts'

బ్లాక్‌చైన్ కాన్ఫిగరేషన్‌తో సహా wagmi (opens in a new tab) కాన్ఫిగరేషన్‌ను దిగుమతి చేయండి.

1const queryClient = new QueryClient()

React Query యొక్క (opens in a new tab) కాష్ మేనేజర్ యొక్క కొత్త ఉదాహరణను సృష్టిస్తుంది. ఈ ఆబ్జెక్ట్ ఇవి నిల్వ చేస్తుంది:

  • కాష్ చేయబడిన RPC కాల్‌లు
  • కాంట్రాక్ట్ రీడ్‌లు
  • బ్యాక్‌గ్రౌండ్ రీఫెచింగ్ స్థితి

wagmi v3 అంతర్గతంగా React Queryని ఉపయోగిస్తుంది కాబట్టి మనకు కాష్ మేనేజర్ అవసరం.

1ReactDOM.createRoot(document.getElementById('root')!).render(

రూట్ React కాంపోనెంట్‌ను సృష్టించండి. render కు పరామితి JSX (opens in a new tab), ఇది HTML మరియు జావాస్క్రిప్ట్/TypeScript రెండింటినీ ఉపయోగించే ఒక పొడిగింపు భాష. ఇక్కడ ఆశ్చర్యార్థకం గుర్తు TypeScript కాంపోనెంట్‌కు చెబుతుంది: "document.getElementById('root') ReactDOM.createRootకు చెల్లుబాటు అయ్యే పరామితి అవుతుందని మీకు తెలియదు, కానీ చింతించకండి - నేను డెవలపర్‌ని మరియు అది అక్కడ ఉంటుందని నేను మీకు చెబుతున్నాను".

1 <React.StrictMode>

అప్లికేషన్ React.StrictMode కాంపోనెంట్ (opens in a new tab) లోపల వెళ్తోంది. ఈ కాంపోనెంట్ అదనపు డీబగ్గింగ్ తనిఖీలను చేర్చమని React లైబ్రరీకి చెబుతుంది, ఇది అభివృద్ధి సమయంలో ఉపయోగపడుతుంది.

1 <WagmiProvider config={config}>

అప్లికేషన్ WagmiProvider కాంపోనెంట్ (opens in a new tab) లోపల కూడా ఉంది. wagmi (we are going to make it) లైబ్రరీ (opens in a new tab) React UI నిర్వచనాలను Ethereum వికేంద్రీకృత అప్లికేషన్ వ్రాయడం కోసం viem లైబ్రరీ (opens in a new tab)తో కలుపుతుంది.

1 <QueryClientProvider client={queryClient}>

చివరగా, ఏ అప్లికేషన్ కాంపోనెంట్ అయినా కాష్ చేసిన క్వెరీలను ఉపయోగించగలిగేలా React Query ప్రొవైడర్‌ను జోడించండి.

1 <App />

ఇప్పుడు మనం అప్లికేషన్ కోసం కాంపోనెంట్‌ను కలిగి ఉండవచ్చు, ఇది వాస్తవానికి UIని అమలు చేస్తుంది. కాంపోనెంట్ చివరిలో ఉన్న /> XML ప్రమాణం ప్రకారం, ఈ కాంపోనెంట్‌లో ఎటువంటి నిర్వచనాలు లేవని Reactకు చెబుతుంది.

1 </QueryClientProvider>
2 </WagmiProvider>
3 </React.StrictMode>,
4)

అయితే, మనం ఇతర కాంపోనెంట్లను కూడా మూసివేయాలి.

src/App.tsx

1import {
2 useConnect,
3 useConnection,
4 useDisconnect,
5 useSwitchChain
6} from 'wagmi'
7
8import { useEffect } from 'react'
9import { Greeter } from './Greeter'
అన్నీ చూపించు

మనకు అవసరమైన లైబ్రరీలను, అలాగే Greeter కాంపోనెంట్‌ను దిగుమతి చేయండి.

1const SEPOLIA_CHAIN_ID = 11155111

Sepolia చైన్ ID.

1function App() {

ఇది React కాంపోనెంట్‌ను సృష్టించడానికి ప్రామాణిక మార్గం: రెండర్ చేయవలసిన ప్రతిసారీ పిలువబడే ఒక ఫంక్షన్‌ను నిర్వచించడం. ఈ ఫంక్షన్ సాధారణంగా TypeScript లేదా జావాస్క్రిప్ట్ కోడ్‌ను కలిగి ఉంటుంది, దాని తర్వాత JSX కోడ్‌ను తిరిగి ఇచ్చే return స్టేట్‌మెంట్ ఉంటుంది.

1 const connection = useConnection()

ప్రస్తుత కనెక్షన్‌కు సంబంధించిన సమాచారం, అడ్రసు మరియు chainId వంటివి పొందడానికి useConnection (opens in a new tab) ను ఉపయోగించండి.

సంప్రదాయం ప్రకారం, use... అని పిలువబడే React ఫంక్షన్లు హుక్స్ (opens in a new tab). ఈ ఫంక్షన్లు కాంపోనెంట్‌కు డేటాను తిరిగి ఇవ్వడమే కాకుండా; ఆ డేటా మారినప్పుడు అది రీ-రెండర్ చేయబడేలా (కాంపోనెంట్ ఫంక్షన్ మళ్లీ అమలు చేయబడుతుంది, మరియు దాని అవుట్‌పుట్ HTMLలో మునుపటిదాన్ని భర్తీ చేస్తుంది) నిర్ధారిస్తాయి.

1 const { connectors, connect, status, error } = useConnect()

వాలెట్ కనెక్షన్ గురించి సమాచారం పొందడానికి useConnect (opens in a new tab) ను ఉపయోగించండి.

1 const { disconnect } = useDisconnect()

ఈ హుక్ (opens in a new tab) వాలెట్ నుండి డిస్‌కనెక్ట్ చేయడానికి ఫంక్షన్‌ను మనకు ఇస్తుంది.

1 const { switchChain } = useSwitchChain()

ఈ హుక్ (opens in a new tab) చైన్‌లను మార్చడానికి అనుమతిస్తుంది.

1 useEffect(() => {

React హుక్ useEffect (opens in a new tab) ఒక వేరియబుల్ విలువ మారినప్పుడు బాహ్య వ్యవస్థను సమకాలీకరించడానికి ఒక ఫంక్షన్‌ను అమలు చేయడానికి అనుమతిస్తుంది.

1 if (connection.status === 'connected' &&
2 connection.chainId !== SEPOLIA_CHAIN_ID
3 ) {
4 switchChain({ chainId: SEPOLIA_CHAIN_ID })
5 }

మనం కనెక్ట్ అయి ఉంటే, కానీ Sepolia బ్లాక్‌చైన్‌కు కాకపోతే, Sepoliaకు మారండి.

1 }, [connection.status, connection.chainId])

కనెక్షన్ స్థితి లేదా కనెక్షన్ chainId మారిన ప్రతిసారీ ఫంక్షన్‌ను మళ్లీ అమలు చేయండి.

1 return (
2 <>

React కాంపోనెంట్ యొక్క JSX ఒక HTML కాంపోనెంట్‌ను తిరిగి ఇవ్వాలి. మనకు బహుళ కాంపోనెంట్లు ఉన్నప్పుడు మరియు వాటన్నింటినీ చుట్టడానికి కంటైనర్ అవసరం లేనప్పుడు, మనం ఖాళీ కాంపోనెంట్‌ను (<> ... </>) ఒకే కాంపోనెంట్‌గా కలపడానికి ఉపయోగిస్తాము.

1 <h2>Connection</h2>
2 <div>
3 status: {connection.status}
4 <br />
5 addresses: {JSON.stringify(connection.addresses)}
6 <br />
7 chainId: {connection.chainId}
8</div>

ప్రస్తుత కనెక్షన్ గురించి సమాచారాన్ని అందించండి. JSX లోపల, {<expression>} అంటే ఎక్స్‌ప్రెషన్‌ను జావాస్క్రిప్ట్‌గా మూల్యాంకనం చేయడం.

1 {connection.status === 'connected' && (

{<condition> && <value>} సింటాక్స్ అంటే "షరతు true అయితే, విలువకు మూల్యాంకనం చేయండి; కాకపోతే, falseకు మూల్యాంకనం చేయండి".

JSX లోపల if స్టేట్‌మెంట్లను ఉంచడానికి ఇది ప్రామాణిక మార్గం.

1 <div>
2 <Greeter />
3 <hr />

JSX XML ప్రమాణాన్ని అనుసరిస్తుంది, ఇది HTML కంటే కఠినమైనది. ఒక ట్యాగ్‌కు సంబంధిత ముగింపు ట్యాగ్ లేకపోతే, అది ముగించడానికి చివరలో తప్పనిసరిగా స్లాష్ (/) కలిగి ఉండాలి.

ఇక్కడ మనకు రెండు అలాంటి ట్యాగ్‌లు ఉన్నాయి, <Greeter /> (ఇది వాస్తవానికి కాంట్రాక్ట్‌తో మాట్లాడే HTML కోడ్‌ను కలిగి ఉంటుంది) మరియు అడ్డు గీత కోసం <hr /> (opens in a new tab).

1 <button type="button" onClick={disconnect}>
2 Disconnect
3 </button>
4</div>
5 )}

వినియోగదారుడు ఈ బటన్‌ను క్లిక్ చేస్తే, disconnect ఫంక్షన్‌ను కాల్ చేయండి.

1 {connection.status !== 'connected' && (

మనం కనెక్ట్ కాకపోతే, వాలెట్‌కు కనెక్ట్ అవడానికి అవసరమైన ఎంపికలను చూపించండి.

1 <div>
2 <h2>Connect</h2>
3 {connectors.map((connector) => (

connectorsలో మనకు కనెక్టర్ల జాబితా ఉంది. ప్రదర్శించడానికి JSX బటన్‌ల జాబితాగా మార్చడానికి మనం map (opens in a new tab)ను ఉపయోగిస్తాము.

1 <button
2 key={connector.uid}

JSXలో "సోదర" ట్యాగ్‌లకు (ఒకే పేరెంట్ నుండి వచ్చిన ట్యాగ్‌లు) వేర్వేరు ఐడెంటిఫయర్‌లు ఉండటం అవసరం.

1 onClick={() => connect({ connector })}
2 type="button"
3 >
4 {connector.name}
5 </button>
6 ))}

కనెక్టర్ బటన్‌లు.

1 <div>{status}</div>
2 <div>{error?.message}</div>
3
4</div>
5 )}

అదనపు సమాచారాన్ని అందించండి. <variable>?.<field> ఎక్స్‌ప్రెషన్ సింటాక్స్ జావాస్క్రిప్ట్‌కు చెబుతుంది, వేరియబుల్ నిర్వచించబడి ఉంటే, ఆ ఫీల్డ్‌కు మూల్యాంకనం చేయండి. వేరియబుల్ నిర్వచించబడకపోతే, ఈ ఎక్స్‌ప్రెషన్ undefinedకు మూల్యాంకనం అవుతుంది.

ఎర్రర్ లేనప్పుడు error.message ఎక్స్‌ప్రెషన్ ఒక ఎక్సెప్షన్‌ను ఉత్పన్నం చేస్తుంది. error?.message ఉపయోగించడం ఈ సమస్యను నివారించడానికి అనుమతిస్తుంది.

src/Greeter.tsx

ఈ ఫైల్ చాలా UI కార్యాచరణను కలిగి ఉంది. ఇందులో సాధారణంగా బహుళ ఫైల్‌లలో ఉండే నిర్వచనాలు ఉన్నాయి, కానీ ఇది ఒక ట్యుటోరియల్ కాబట్టి ప్రోగ్రామ్ మొదటిసారి సులభంగా అర్థం చేసుకోవడానికి ఆప్టిమైజ్ చేయబడింది, పనితీరు లేదా నిర్వహణ సౌలభ్యం కోసం కాదు.

1import {
2 useState,
3 useEffect,
4 } from 'react'
5import { useChainId,
6 useAccount,
7 useReadContract,
8 useWriteContract,
9 useWatchContractEvent,
10 useSimulateContract
11 } from 'wagmi'
అన్నీ చూపించు

మనం ఈ లైబ్రరీ ఫంక్షన్‌లను ఉపయోగిస్తాము. మళ్ళీ, అవి ఉపయోగించే చోట క్రింద వివరించబడ్డాయి.

1import { AddressType } from 'abitype'

abitype లైబ్రరీ (opens in a new tab) AddressType (opens in a new tab) వంటి వివిధ Ethereum డేటా రకాల కోసం TypeScript నిర్వచనాలను అందిస్తుంది.

1let greeterABI = [
2 { "type": "function", "name": "greet", ... },
3 { "type": "function", "name": "setGreeting", ... },
4 { "type": "event", "name": "SetGreeting", ... },
5] as const // greeterABI

Greeter కాంట్రాక్ట్ కోసం ABI. మీరు కాంట్రాక్టులను మరియు UIని ఒకే సమయంలో అభివృద్ధి చేస్తుంటే, సాధారణంగా వాటిని ఒకే రిపోజిటరీలో ఉంచి Solidity కంపైలర్ రూపొందించిన ABIని మీ అప్లికేషన్‌లో ఫైల్‌గా ఉపయోగిస్తారు. అయితే, కాంట్రాక్ట్ ఇప్పటికే అభివృద్ధి చేయబడింది మరియు మారదు కాబట్టి ఇక్కడ ఇది అవసరం లేదు.

TypeScriptకు ఇది నిజమైన స్థిరాంకం అని చెప్పడానికి మనం as const (opens in a new tab)ను ఉపయోగిస్తాము. సాధారణంగా, జావాస్క్రిప్ట్‌లో const x = {"a": 1} అని పేర్కొన్నప్పుడు, మీరు xలోని విలువను మార్చవచ్చు, మీరు దానికి అసైన్ చేయలేరు మాత్రమే.

1type AddressPerBlockchainType = {
2 [key: number]: AddressType
3}

TypeScript బలంగా టైప్ చేయబడింది. వేర్వేరు చైన్‌లలో Greeter కాంట్రాక్ట్ డిప్లాయ్ చేయబడిన అడ్రసును పేర్కొనడానికి మనం ఈ నిర్వచనాన్ని ఉపయోగిస్తాము. కీ ఒక సంఖ్య (chainId), మరియు విలువ ఒక AddressType (అడ్రసు).

1const contractAddrs : AddressPerBlockchainType = {
2 // Sepolia
3 11155111: '0xC87506C66c7896366b9E988FE0aA5B6dDE77CFfA'
4}

Sepolia (opens in a new tab) లో కాంట్రాక్ట్ అడ్రసు.

Timer కాంపోనెంట్

Timer కాంపోనెంట్ ఇచ్చిన సమయం నుండి సెకన్ల సంఖ్యను చూపిస్తుంది. వినియోగ ప్రయోజనాల కోసం ఇది ముఖ్యమైనది. వినియోగదారులు ఏదైనా చేసినప్పుడు, వారు తక్షణ స్పందనను ఆశిస్తారు. బ్లాక్‌చైన్‌లలో, లావాదేవీ ఒక బ్లాక్‌లో ఉంచబడే వరకు ఏమీ జరగదు కాబట్టి ఇది తరచుగా అసాధ్యం. వినియోగదారుడు చర్య చేసి ఎంత సమయం అయిందో చూపించడం ఒక పరిష్కారం, తద్వారా అవసరమైన సమయం సహేతుకమా కాదా అని వినియోగదారుడు నిర్ణయించగలరు.

1type TimerProps = {
2 lastUpdate: Date
3}

Timer కాంపోనెంట్ ఒక పరామితిని తీసుకుంటుంది, lastUpdate, ఇది చివరి చర్య సమయం.

1const Timer = ({ lastUpdate }: TimerProps) => {
2 const [_, setNow] = useState(new Date())

కాంపోనెంట్ సరిగ్గా పనిచేయడానికి మనకు స్టేట్ (కాంపోనెంట్‌తో ముడిపడిన వేరియబుల్) అవసరం మరియు దానిని అప్‌డేట్ చేయాలి. కానీ మనం దానిని చదవాల్సిన అవసరం ఎన్నడూ లేదు, కాబట్టి వేరియబుల్ కోసం ఇబ్బంది పడవద్దు.

1 useEffect(() => {
2 const id = setInterval(() => setNow(new Date()), 1000)
3 return () => clearInterval(id)
4 }, [])

setInterval (opens in a new tab) ఫంక్షన్ ఒక ఫంక్షన్‌ను ఆవర్తనంగా అమలు చేయడానికి షెడ్యూల్ చేయడానికి అనుమతిస్తుంది. ఈ సందర్భంలో, ప్రతి సెకను. ఫంక్షన్ స్టేట్‌ను అప్‌డేట్ చేయడానికి setNowను కాల్ చేస్తుంది, కాబట్టి Timer కాంపోనెంట్ రీ-రెండర్ చేయబడుతుంది. ఇది ఒక్కసారి మాత్రమే జరిగేలా ఖాళీ డిపెండెన్సీ జాబితాతో useEffect (opens in a new tab) లోపల దీనిని చుట్టాము.

1 const secondsSinceUpdate = Math.floor(
2 (Date.now() - lastUpdate.getTime()) / 1000
3 )
4
5 return (
6 <span>{secondsSinceUpdate} seconds ago</span>
7 )
8}

చివరి అప్‌డేట్ నుండి సెకన్ల సంఖ్యను లెక్కించి తిరిగి ఇవ్వండి.

Greeter కాంపోనెంట్
1const Greeter = () => {

చివరగా, మనం కాంపోనెంట్‌ను నిర్వచించగలుగుతున్నాము.

1 const chainId = useChainId()
2 const account = useAccount()

wagmi (opens in a new tab) సౌజన్యంతో, మనం ఉపయోగిస్తున్న చైన్ మరియు ఖాతా గురించి సమాచారం. ఇది హుక్ (use...) కాబట్టి, ఈ సమాచారం మారినప్పుడు కాంపోనెంట్ రీ-రెండర్ చేయబడుతుంది.

1 const greeterAddr = chainId && contractAddrs[chainId]

Greeter కాంట్రాక్ట్ అడ్రసు, మనకు చైన్ సమాచారం లేకపోతే లేదా ఆ కాంట్రాక్ట్ లేని చైన్‌లో ఉంటే undefined అవుతుంది.

1 const readResults = useReadContract({
2 address: greeterAddr,
3 abi: greeterABI,
4 functionName: "greet", // No arguments
5 })

useReadContract హుక్ (opens in a new tab) కాంట్రాక్ట్ (opens in a new tab) యొక్క greet ఫంక్షన్‌ను కాల్ చేస్తుంది.

1 const [ currentGreeting, setCurrentGreeting ] =
2 useState("Please wait while we fetch the greeting from the blockchain...")
3 const [ newGreeting, setNewGreeting ] = useState("")

React యొక్క useState హుక్ (opens in a new tab) ఒక స్టేట్ వేరియబుల్‌ను పేర్కొనడానికి అనుమతిస్తుంది, దీని విలువ కాంపోనెంట్ యొక్క ఒక రెండరింగ్ నుండి మరొకదానికి కొనసాగుతుంది. ప్రారంభ విలువ పరామితి, ఈ సందర్భంలో ఖాళీ స్ట్రింగ్.

useState హుక్ రెండు విలువలతో జాబితాను తిరిగి ఇస్తుంది:

  1. స్టేట్ వేరియబుల్ యొక్క ప్రస్తుత విలువ.
  2. అవసరమైనప్పుడు స్టేట్ వేరియబుల్‌ను సవరించడానికి ఫంక్షన్. ఇది హుక్ కాబట్టి, ఇది కాల్ చేసిన ప్రతిసారీ కాంపోనెంట్ మళ్లీ రెండర్ చేయబడుతుంది.

ఈ సందర్భంలో, వినియోగదారుడు సెట్ చేయాలనుకునే కొత్త గ్రీటింగ్ కోసం మనం స్టేట్ వేరియబుల్‌ను ఉపయోగిస్తున్నాము.

1 const [ lastSetterAddress, setLastSetterAddress ] = useState("")

బహుళ వినియోగదారులు ఒకే సమయంలో ఒకే కాంట్రాక్ట్‌ను ఉపయోగిస్తుంటే, వారు ఒకరి గ్రీటింగ్‌లను మరొకరు ఓవర్‌రైట్ చేయవచ్చు. ఇది వినియోగదారులకు అప్లికేషన్ సరిగ్గా పనిచేయడం లేదని అనిపించేలా చేస్తుంది. గ్రీటింగ్‌ను చివరిగా ఎవరు సెట్ చేశారో అప్లికేషన్ చూపిస్తే, వినియోగదారుడికి అది ఎవరో ఇతరులు చేశారని తెలుస్తుంది.

1 const [ status, setStatus ] = useState("")
2 const [ statusTime, setStatusTime ] = useState(new Date())

వినియోగదారులు తమ చర్యలకు తక్షణ ప్రభావం చూడాలనుకుంటారు. అయితే, బ్లాక్‌చైన్‌లో ఇది అలా జరగదు. ఈ స్టేట్ వేరియబుల్‌లు వినియోగదారులకు వారి చర్య ప్రగతిలో ఉందని తెలియజేయడానికి కనీసం ఏదైనా ప్రదర్శించడానికి అనుమతిస్తాయి.

1 useEffect(() => {
2 if (readResults.data) {
3 setCurrentGreeting(readResults.data)
4 setStatus("Greeting fetched from blockchain")
5 }
6 }, [readResults.data])

పైన ఉన్న readResults డేటాను మారిస్తే మరియు అది తప్పుడు విలువకు (undefined, ఉదాహరణకు) సెట్ చేయబడకపోతే, ప్రస్తుత గ్రీటింగ్‌ను బ్లాక్‌చైన్ నుండి చదివిన దానికి అప్‌డేట్ చేయండి. అలాగే, స్థితిని అప్‌డేట్ చేయండి.

1 useWatchContractEvent({
2 address: greeterAddr,
3 abi: greeterABI,
4 eventName: 'SetGreeting',
5 chainId,

SetGreeting ఈవెంట్‌లను వినండి.

1 enabled: !!greeterAddr,

!!<value> అంటే విలువ false అయితే, లేదా undefined, 0, లేదా ఖాళీ స్ట్రింగ్ వంటి తప్పుగా మూల్యాంకనం చేయబడే విలువ అయితే, మొత్తం ఎక్స్‌ప్రెషన్ false. ఏ ఇతర విలువకైనా, ఇది true. విలువలను బూలియన్‌లుగా మార్చడానికి ఇది ఒక మార్గం, ఎందుకంటే greeterAddr లేకపోతే, మనం ఈవెంట్‌లను వినాలనుకోము.

1 onLogs: logs => {
2 const greetingFromContract = logs[0].args.greeting
3 setCurrentGreeting(greetingFromContract)
4 setLastSetterAddress(logs[0].args.sender)
5 updateStatus("Greeting updated by event")
6 },
7 })

మనం లాగ్‌లను చూసినప్పుడు (ఇది కొత్త ఈవెంట్‌ను చూసినప్పుడు జరుగుతుంది), గ్రీటింగ్ సవరించబడిందని అర్థం. ఆ సందర్భంలో, currentGreeting మరియు lastSetterAddressను కొత్త విలువలకు అప్‌డేట్ చేయవచ్చు. అలాగే, స్థితి ప్రదర్శనను అప్‌డేట్ చేయాలనుకుంటున్నాము.

1 const updateStatus = (newStatus: string) => {
2 setStatus(newStatus)
3 setStatusTime(new Date())
4 }

మనం స్థితిని అప్‌డేట్ చేసినప్పుడు రెండు పనులు చేయాలనుకుంటున్నాము:

  1. స్థితి స్ట్రింగ్‌ను (status) అప్‌డేట్ చేయడం
  2. చివరి స్థితి అప్‌డేట్ సమయాన్ని (statusTime) ఇప్పుడుకు అప్‌డేట్ చేయడం.
1 const greetingChange = (evt) =>
2 setNewGreeting(evt.target.value)

కొత్త గ్రీటింగ్ ఇన్‌పుట్ ఫీల్డ్‌లో మార్పులకు ఈవెంట్ హ్యాండ్లర్. ఈ ఫంక్షన్ ఒక్కసారి మాత్రమే కాల్ చేయబడుతుంది కాబట్టి evt పరామితి రకాన్ని పేర్కొనడం అవసరం లేదని నేను భావిస్తున్నాను.

1 const { writeContractAsync } = useWriteContract()

కాంట్రాక్ట్‌కు వ్రాయడానికి ఫంక్షన్. ఇది writeContracts (opens in a new tab)కు సమానం, కానీ మెరుగైన స్థితి అప్‌డేట్‌లను అనుమతిస్తుంది.

1 const simulation = useSimulateContract({
2 address: greeterAddr,
3 abi: greeterABI,
4 functionName: 'setGreeting',
5 args: [newGreeting],
6 account: account.address
7 })

క్లయింట్ కోణం నుండి బ్లాక్‌చైన్ లావాదేవీని సమర్పించే ప్రక్రియ ఇది:

  1. eth_estimateGas (opens in a new tab) ఉపయోగించి బ్లాక్‌చైన్‌లోని నోడ్‌కు లావాదేవీని పంపండి.
  2. నోడ్ నుండి ప్రతిస్పందన కోసం వేచి ఉండండి.
  3. ప్రతిస్పందన అందినప్పుడు, వాలెట్ ద్వారా లావాదేవీపై సంతకం చేయమని వినియోగదారుడిని అడగండి. వినియోగదారుడికి సంతకం చేయడానికి ముందు లావాదేవీ యొక్క గ్యాస్ ఖర్చు చూపించబడుతుంది కాబట్టి ఈ దశ తప్పనిసరిగా నోడ్ ప్రతిస్పందన అందిన తర్వాత జరగాలి.
  4. వినియోగదారుడు ఆమోదించడానికి వేచి ఉండండి.
  5. ఈసారి eth_sendRawTransaction (opens in a new tab) ఉపయోగించి లావాదేవీని మళ్లీ పంపండి.

దశ 2 గుర్తించగల సమయం తీసుకునే అవకాశం ఉంది, ఈ సమయంలో వినియోగదారులు తమ ఆదేశం యూజర్ ఇంటర్‌ఫేస్ ద్వారా అందుకోబడిందా మరియు లావాదేవీపై సంతకం చేయమని ఎందుకు అడగడం లేదో ఆశ్చర్యపోవచ్చు. ఇది పేలవమైన వినియోగదారు అనుభవాన్ని (UX) సృష్టిస్తుంది.

ఒక పరిష్కారం ఏమిటంటే పరామితి మారిన ప్రతిసారీ eth_estimateGasను పంపడం. అప్పుడు, వినియోగదారుడు వాస్తవంగా లావాదేవీని పంపాలనుకున్నప్పుడు (ఈ సందర్భంలో Update greeting నొక్కడం ద్వారా), గ్యాస్ ఖర్చు ఇప్పటికే తెలుసు, మరియు వినియోగదారుడు వెంటనే వాలెట్ పేజీని చూడగలరు.

1 return (

ఇప్పుడు మనం తిరిగి ఇవ్వడానికి వాస్తవ HTMLను సృష్టించవచ్చు.

1 <>
2 <h2>Greeter</h2>
3 {currentGreeting}

ప్రస్తుత గ్రీటింగ్‌ను చూపించండి.

1 {lastSetterAddress && (
2 <p>Last updated by {
3 lastSetterAddress === account.address ? "you" : lastSetterAddress
4 }</p>
5 )}

గ్రీటింగ్‌ను చివరిగా ఎవరు సెట్ చేశారో మనకు తెలిస్తే, ఆ సమాచారాన్ని ప్రదర్శించండి. Greeter ఈ సమాచారాన్ని ట్రాక్ చేయదు, మరియు SetGreeting ఈవెంట్‌లను వెనక్కి చూడాలనుకోము, కాబట్టి మనం నడుస్తున్నప్పుడు గ్రీటింగ్ మారిన తర్వాత మాత్రమే దానిని పొందుతాము.

1 <hr />
2 <input type="text"
3 value={newGreeting}
4 onChange={greetingChange}
5 />
6 <br />

ఇది వినియోగదారుడు కొత్త గ్రీటింగ్‌ను సెట్ చేయగల ఇన్‌పుట్ టెక్స్ట్ ఫీల్డ్. వినియోగదారుడు ఒక కీని నొక్కిన ప్రతిసారీ, మనం greetingChangeను కాల్ చేస్తాము, ఇది setNewGreetingను కాల్ చేస్తుంది. setNewGreeting useState నుండి వచ్చినది కాబట్టి, ఇది Greeter కాంపోనెంట్‌ను రీ-రెండర్ చేస్తుంది. దీని అర్థం:

  • కొత్త గ్రీటింగ్ విలువను ఉంచడానికి మనం valueను పేర్కొనాలి, లేకపోతే అది డిఫాల్ట్ ఖాళీ స్ట్రింగ్‌కు తిరిగి వెళ్ళిపోతుంది.
  • newGreeting మారిన ప్రతిసారీ simulation కూడా అప్‌డేట్ చేయబడుతుంది, అంటే సరైన గ్రీటింగ్‌తో సిమ్యులేషన్ మనకు వస్తుంది.
1 <button disabled={!simulation.data}

లావాదేవీని పంపడానికి అవసరమైన సమాచారం మనకు వచ్చిన తర్వాత మాత్రమే బటన్‌ను ఎనేబుల్ చేయండి.

1 onClick={async () => {
2 updateStatus("Please confirm in wallet...")

స్థితిని అప్‌డేట్ చేయండి. ఈ సమయంలో, వినియోగదారుడు వాలెట్‌లో నిర్ధారించాలి.

1 await writeContractAsync(simulation.data.request)
2 updateStatus("Transaction sent, waiting for greeting to change...")
3 }}
4 >
5 Update greeting
6 </button>
7

writeContractAsync లావాదేవీ వాస్తవంగా పంపబడిన తర్వాత మాత్రమే తిరిగి వస్తుంది. ఇది బ్లాక్‌చైన్‌లో చేర్చబడటానికి లావాదేవీ ఎంతసేపు వేచి ఉందో వినియోగదారుడికి చూపించడానికి అనుమతిస్తుంది.

1 <h4>Status: {status}</h4>
2 <p>Updated <Timer lastUpdate={statusTime} /> </p>
3 </>
4 )
5}

స్థితిని మరియు చివరిగా అప్‌డేట్ చేసి ఎంత సమయం అయిందో చూపించండి.

1export {Greeter}

కాంపోనెంట్‌ను ఎగుమతి చేయండి.

src/wagmi.ts

చివరగా, wagmiకి సంబంధించిన వివిధ నిర్వచనాలు src/wagmi.tsలో ఉన్నాయి. ఇక్కడ నేను ప్రతిదీ వివరించబోవడం లేదు, ఎందుకంటే చాలా వరకు మీరు మార్చవలసిన అవసరం లేని బాయిలర్‌ప్లేట్.

1import { http, webSocket, createConfig, fallback } from 'wagmi'
2import { sepolia } from 'wagmi/chains'
3import { injected } from 'wagmi/connectors'
4
5export const config = createConfig({
6 chains: [sepolia],

wagmi కాన్ఫిగరేషన్‌లో ఈ అప్లికేషన్ మద్దతు ఇచ్చే చైన్‌లు ఉంటాయి. మీరు అందుబాటులో ఉన్న చైన్‌ల జాబితా (opens in a new tab)ను చూడవచ్చు.

1 connectors: [
2 injected(),
3 ],

ఈ కనెక్టర్ (opens in a new tab) బ్రౌజర్‌లో ఇన్‌స్టాల్ చేసిన వాలెట్‌తో మాట్లాడటానికి అనుమతిస్తుంది.

1 transports: {
2 [sepolia.id]: http()

viemతో వచ్చే డిఫాల్ట్ HTTP ఎండ్‌పాయింట్ సరిపోతుంది. మీకు వేరే URL కావాలంటే, http("https:// hostname ") లేదా webSocket("wss:// hostname ") ఉపయోగించవచ్చు.

1 },
2 multiInjectedProviderDiscovery: false,
3})

మరొక బ్లాక్‌చైన్‌ను జోడించడం

ఈ రోజుల్లో చాలా L2 స్కేలింగ్ సొల్యూషన్స్ (opens in a new tab) ఉన్నాయి, మరియు viem ఇంకా మద్దతు ఇవ్వని కొన్నింటికి మీరు మద్దతు ఇవ్వాలనుకోవచ్చు. అది చేయడానికి, మీరు src/wagmi.tsను సవరించండి. ఈ సూచనలు Optimism Sepolia (opens in a new tab)ని ఎలా జోడించాలో వివరిస్తాయి.

  1. src/wagmi.tsను సవరించండి

    A. viem నుండి defineChain రకాన్ని దిగుమతి చేయండి.

    1import { defineChain } from 'viem'

    B. నెట్‌వర్క్ నిర్వచనాన్ని జోడించండి. Optimism Sepolia కోసం మీరు దీనిని చేయవలసిన అవసరం లేదు, ఇది ఇప్పటికే viemలో ఉంది (opens in a new tab), కానీ ఈ విధంగా viemలో లేని బ్లాక్‌చైన్‌ను ఎలా జోడించాలో మీరు నేర్చుకుంటారు.

    1const optimismSepolia = defineChain({
    2 id: 11_155_420,
    3 name: 'OP Sepolia',
    4 nativeCurrency: { name: 'Sepolia Ether', symbol: 'ETH', decimals: 18 },
    5 rpcUrls: {
    6 default: {
    7 http: ['https://sepolia.optimism.io'],
    8 webSocket: ['wss://optimism-sepolia.drpc.org'],
    9 },
    10 },
    11 blockExplorers: {
    12 default: {
    13 name: 'Blockscout',
    14 url: 'https://optimism-sepolia.blockscout.com',
    15 apiUrl: 'https://optimism-sepolia.blockscout.com/api',
    16 }
    17 },
    18})
    అన్నీ చూపించు

    C. createConfig కాల్‌కు కొత్త చైన్‌ను జోడించండి.

    1export const config = createConfig({
    2 chains: [sepolia, optimismSepolia],
    3 connectors: [
    4 injected(),
    5 ],
    6 transports: {
    7 [optimismSepolia.id]: http(),
    8 [sepolia.id]: http()
    9 },
    10 multiInjectedProviderDiscovery: false,
    11})
    అన్నీ చూపించు
  2. ఆటోమేటిక్‌గా Sepoliaకు మారడాన్ని కామెంట్ అవుట్ చేయడానికి src/App.tsxను సవరించండి. ఒక ఉత్పత్తి వ్యవస్థలో, మీరు బహుశా మీరు మద్దతు ఇచ్చే ప్రతి బ్లాక్‌చైన్‌కు లింకులతో బటన్‌లను చూపిస్తారు.

    1/*
    2useEffect(() => {
    3 if (connection.status === 'connected' &&
    4 connection.chainId !== SEPOLIA_CHAIN_ID
    5 ) {
    6 switchChain({ chainId: SEPOLIA_CHAIN_ID })
    7 }
    8}, [connection.status, connection.chainId])
    9*/
    అన్నీ చూపించు
  3. కొత్త నెట్‌వర్క్‌లో మీ కాంట్రాక్టుల కోసం చిరునామా అప్లికేషన్‌కు తెలుసని నిర్ధారించడానికి src/Greeter.tsxను సవరించండి.

    1const contractAddrs: AddressPerBlockchainType = {
    2 // Optimism Sepolia
    3 11155420: "0x4dd85791923E9294E934271522f63875EAe5806f",
    4
    5 // Sepolia
    6 11155111: "0x7143d5c190F048C8d19fe325b748b081903E3BF0",
    7}
  4. మీ బ్రౌజర్‌లో.

    A. ChainList (opens in a new tab)కు బ్రౌజ్ చేసి, మీ వాలెట్‌కు చైన్‌ను జోడించడానికి టేబుల్ కుడి వైపు ఉన్న బటన్‌లలో ఒకదాన్ని క్లిక్ చేయండి.

    B. అప్లికేషన్‌లో, Disconnect చేసి, బ్లాక్‌చైన్‌ను మార్చడానికి మళ్లీ కనెక్ట్ అవ్వండి. దీన్ని నిర్వహించడానికి మెరుగైన మార్గాలు ఉన్నాయి, కానీ అవి అప్లికేషన్ మార్పులు అవసరం.

ముగింపు

అయితే, మీరు నిజంగా Greeter కోసం యూజర్ ఇంటర్‌ఫేస్ అందించడం గురించి పట్టించుకోరు. మీరు మీ స్వంత కాంట్రాక్టుల కోసం యూజర్ ఇంటర్‌ఫేస్‌ను సృష్టించాలనుకుంటున్నారు. మీ స్వంత అప్లికేషన్‌ను సృష్టించడానికి, ఈ దశలను అమలు చేయండి:

  1. wagmi అప్లికేషన్‌ను సృష్టించమని పేర్కొనండి.

    1npm create wagmi
  2. కొనసాగడానికి y టైప్ చేయండి.

  3. అప్లికేషన్‌కు పేరు పెట్టండి.

  4. React ఫ్రేమ్‌వర్క్‌ను ఎంచుకోండి.

  5. Vite వేరియంట్‌ను ఎంచుకోండి.

ఇప్పుడు వెళ్ళి మీ కాంట్రాక్టులను విస్తృత ప్రపంచానికి ఉపయోగపడేలా చేయండి.

నా మరిన్ని పనుల కోసం ఇక్కడ చూడండి (opens in a new tab).

పేజీ చివరి అప్‌డేట్: 3 మార్చి, 2026

ఈ ట్యుటోరియల్ ఉపయోగపడిందా?