तुमच्या कॉन्ट्रॅक्टसाठी युजर इंटरफेस तयार करणे
तुम्हाला इथेरियम इकोसिस्टममध्ये आवश्यक असलेले एक वैशिष्ट्य सापडले आहे. त्याची अंमलबजावणी करण्यासाठी तुम्ही स्मार्ट कॉन्ट्रॅक्ट्स लिहिले आहेत आणि कदाचित साखळीबाह्य चालणारा काही संबंधित कोडही लिहिला असेल. हे उत्तम आहे! दुर्दैवाने, युजर इंटरफेसशिवाय तुम्हाला कोणतेही युजर्स मिळणार नाहीत आणि तुम्ही शेवटच्या वेळी जेव्हा वेबसाइट लिहिली होती तेव्हा लोक डायल-अप मॉडेम वापरत होते आणि JavaScript नवीन होते.
हा लेख तुमच्यासाठी आहे. मी असे गृहीत धरतो की तुम्हाला प्रोग्रामिंग येते आणि कदाचित थोडे JavaScript आणि HTML देखील येते, परंतु तुमचे युजर इंटरफेस कौशल्य जुने आणि कालबाह्य झाले आहे. आपण एकत्रितपणे एका साध्या आधुनिक ॲप्लिकेशनची माहिती घेऊ जेणेकरून आजकाल हे कसे केले जाते हे तुम्हाला दिसेल.
हे का महत्त्वाचे आहे
तात्विकदृष्ट्या, तुम्ही लोकांना तुमच्या कॉन्ट्रॅक्ट्सशी संवाद साधण्यासाठी फक्त Etherscan (opens in a new tab) किंवा Blockscout (opens in a new tab) वापरण्यास सांगू शकता. अनुभवी इथेरियम युजर्ससाठी हे उत्तम आहे. परंतु आपण आणखी एक अब्ज लोकांना (opens in a new tab) सेवा देण्याचा प्रयत्न करत आहोत. एका उत्तम युजर अनुभवाशिवाय हे शक्य होणार नाही आणि एक अनुकूल युजर इंटरफेस हा त्याचा एक मोठा भाग आहे.
Greeter ॲप्लिकेशन
आधुनिक UI कसे कार्य करते यामागे बरेच सिद्धांत आहेत आणि अनेक चांगल्या साइट्स (opens in a new tab) ते स्पष्ट करतात (opens in a new tab). त्या साइट्सनी केलेले उत्तम काम पुन्हा सांगण्याऐवजी, मी असे गृहीत धरणार आहे की तुम्हाला प्रत्यक्ष कृतीतून शिकायला आवडते आणि आपण अशा एका ॲप्लिकेशनपासून सुरुवात करू ज्यासोबत तुम्ही प्रयोग करू शकता. गोष्टी पूर्ण करण्यासाठी तुम्हाला अजूनही सिद्धांताची आवश्यकता आहे आणि आपण त्याकडे वळू - आपण फक्त सोर्स फाईलनुसार पुढे जाऊ आणि जसे जसे आपण त्यांच्यापर्यंत पोहोचू तसतसे गोष्टींवर चर्चा करू.
इन्स्टॉलेशन
-
हे ॲप्लिकेशन Sepolia (opens in a new tab) टेस्ट नेटवर्क वापरते. आवश्यक असल्यास, Sepolia टेस्ट ETH मिळवा आणि तुमच्या वॉलेटमध्ये Sepolia जोडा (opens in a new tab).
-
GitHub रिपॉझिटरी क्लोन करा आणि आवश्यक पॅकेजेस इन्स्टॉल करा.
git clone https://github.com/qbzzt/260301-modern-ui-web3.git cd 260301-modern-ui-web3 npm install -
हे ॲप्लिकेशन मोफत ॲक्सेस पॉईंट्स वापरते, ज्यांच्या कार्यक्षमतेवर मर्यादा आहेत. जर तुम्हाला नोड ॲज अ सर्व्हिस (Node as a service) प्रोव्हायडर वापरायचा असेल, तर
src/wagmi.tsमधील URLs बदला. -
ॲप्लिकेशन सुरू करा.
npm run dev -
ॲप्लिकेशनद्वारे दर्शविलेल्या URL वर ब्राउझ करा. बहुतांश प्रकरणांमध्ये, ती http://localhost:5173/ (opens in a new tab) असते.
-
तुम्ही कॉन्ट्रॅक्टचा सोर्स कोड, जो Hardhat च्या Greeter ची सुधारित आवृत्ती आहे, ब्लॉकचेन एक्सप्लोररवर (opens in a new tab) पाहू शकता.
फाईल वॉक थ्रू
index.html
ही फाईल एक स्टँडर्ड HTML बॉयलरप्लेट आहे, फक्त या ओळीचा अपवाद वगळता, जी स्क्रिप्ट फाईल इम्पोर्ट करते.
<script type="module" src="/src/main.tsx"></script>
src/main.tsx
फाईल एक्स्टेंशन दर्शविते की हा एक React घटक (opens in a new tab) आहे जो TypeScript (opens in a new tab) मध्ये लिहिला आहे, जे JavaScript चे एक एक्स्टेंशन आहे आणि टाईप चेकिंगला (opens in a new tab) सपोर्ट करते. TypeScript हे JavaScript मध्ये कंपाईल केले जाते, त्यामुळे आपण ते क्लायंट साईडवर वापरू शकतो.
जर तुम्हाला स्वारस्य असेल तरच ही फाईल प्रामुख्याने स्पष्ट केली आहे. सहसा तुम्ही ही फाईल बदलत नाही, तर src/App.tsx आणि ती इम्पोर्ट करत असलेल्या फाईल्स बदलता.
import { QueryClient, QueryClientProvider } from '@tanstack/react-query'
import React from 'react'
import ReactDOM from 'react-dom/client'
import { WagmiProvider } from 'wagmi'
आपल्याला आवश्यक असलेला लायब्ररी कोड इम्पोर्ट करा.
import App from './App.tsx'
ॲप्लिकेशनची अंमलबजावणी करणारा React घटक इम्पोर्ट करा (खाली पहा).
import { config } from './wagmi.ts'
Wagmi (opens in a new tab) कॉन्फिगरेशन इम्पोर्ट करा, ज्यामध्ये ब्लॉकचेन कॉन्फिगरेशन समाविष्ट आहे.
const queryClient = new QueryClient()
React Query च्या (opens in a new tab) कॅशे मॅनेजरचा एक नवीन इन्स्टन्स तयार करते. हा ऑब्जेक्ट खालील गोष्टी स्टोअर करेल:
- कॅशे केलेले RPC कॉल्स
- कॉन्ट्रॅक्ट रीड्स
- बॅकग्राउंड रिफेचिंग स्थिती
आपल्याला कॅशे मॅनेजरची आवश्यकता आहे कारण Wagmi v3 अंतर्गतपणे React Query वापरते.
ReactDOM.createRoot(document.getElementById('root')!).render(
रूट React घटक तयार करा. render साठी पॅरामीटर JSX (opens in a new tab) आहे, जी एक एक्स्टेंशन भाषा आहे जी HTML आणि JavaScript/TypeScript दोन्ही वापरते. येथील उद्गारवाचक चिन्ह TypeScript घटकाला सांगते: "तुम्हाला माहित नाही की document.getElementById('root') हे ReactDOM.createRoot साठी एक वैध पॅरामीटर असेल, पण काळजी करू नका - मी डेव्हलपर आहे आणि मी तुम्हाला सांगत आहे की ते असेल".
<React.StrictMode>
ॲप्लिकेशन एका React.StrictMode घटकाच्या (opens in a new tab) आत जात आहे. हा घटक React लायब्ररीला अतिरिक्त डीबगिंग चेक्स समाविष्ट करण्यास सांगतो, जे डेव्हलपमेंट दरम्यान उपयुक्त ठरते.
<WagmiProvider config={config}>
ॲप्लिकेशन एका WagmiProvider घटकाच्या (opens in a new tab) आत देखील आहे. Wagmi (आपण ते बनवणार आहोत) लायब्ररी (opens in a new tab) इथेरियम विकेंद्रित ॲप्लिकेशन (dapp) लिहिण्यासाठी React UI व्याख्यांना Viem लायब्ररीशी (opens in a new tab) जोडते.
<QueryClientProvider client={queryClient}>
आणि शेवटी, एक React Query प्रोव्हायडर जोडा जेणेकरून कोणताही ॲप्लिकेशन घटक कॅशे केलेल्या क्वेरीज वापरू शकेल.
<App />
आता आपल्याकडे ॲप्लिकेशनसाठी घटक असू शकतो, जो प्रत्यक्षात UI ची अंमलबजावणी करतो. घटकाच्या शेवटी असलेले /> React ला सांगते की XML मानकानुसार या घटकाच्या आत कोणत्याही व्याख्या नाहीत.
</QueryClientProvider>
</WagmiProvider>
</React.StrictMode>,
)
अर्थात, आपल्याला इतर घटक बंद करावे लागतील.
src/App.tsx
import {
useConnect,
useConnection,
useDisconnect,
useSwitchChain
} from 'wagmi'
import { useEffect } from 'react'
import { Greeter } from './Greeter'
आपल्याला आवश्यक असलेल्या लायब्ररीज, तसेच Greeter घटक इम्पोर्ट करा.
const SEPOLIA_CHAIN_ID = 11155111
Sepolia चेन आयडी.
function App() {
React घटक तयार करण्याचा हा स्टँडर्ड मार्ग आहे: एक फंक्शन परिभाषित करा जे जेव्हा रेंडर करण्याची आवश्यकता असते तेव्हा कॉल केले जाते. या फंक्शनमध्ये सामान्यतः TypeScript किंवा JavaScript कोड असतो, त्यानंतर एक return स्टेटमेंट असते जे JSX कोड परत करते.
const connection = useConnection()
सध्याच्या कनेक्शनशी संबंधित माहिती मिळवण्यासाठी useConnection (opens in a new tab) वापरा, जसे की पत्ता आणि chainId.
प्रथेनुसार, React मध्ये use... नावाच्या फंक्शन्सना हुक्स (hooks) (opens in a new tab) म्हणतात. ही फंक्शन्स केवळ घटकाला डेटा परत करत नाहीत; तर जेव्हा तो डेटा बदलतो तेव्हा ते पुन्हा रेंडर केले जाईल (घटक फंक्शन पुन्हा कार्यान्वित केले जाते आणि त्याचे आउटपुट HTML मधील मागील आउटपुटची जागा घेते) याची देखील खात्री करतात.
const { connectors, connect, status, error } = useConnect()
वॉलेट कनेक्शनबद्दल माहिती मिळवण्यासाठी useConnect (opens in a new tab) वापरा.
const { disconnect } = useDisconnect()
हा हुक (opens in a new tab) आपल्याला वॉलेटमधून डिस्कनेक्ट करण्यासाठी फंक्शन देतो.
const { switchChain } = useSwitchChain()
हा हुक (opens in a new tab) आपल्याला चेन्स स्विच करू देतो.
useEffect(() => {
React हुक useEffect (opens in a new tab) तुम्हाला बाह्य सिस्टीम सिंक्रोनाइझ करण्यासाठी जेव्हा जेव्हा व्हेरिएबलचे मूल्य बदलते तेव्हा फंक्शन चालवू देतो.
if (connection.status === 'connected' &&
connection.chainId !== SEPOLIA_CHAIN_ID
) {
switchChain({ chainId: SEPOLIA_CHAIN_ID })
}
जर आपण कनेक्टेड असू, परंतु Sepolia ब्लॉकचेनशी नाही, तर Sepolia वर स्विच करा.
}, [connection.status, connection.chainId])
प्रत्येक वेळी जेव्हा कनेक्शन स्थिती किंवा कनेक्शन chainId बदलते तेव्हा फंक्शन पुन्हा चालवा.
return (
<>
React घटकाचे JSX नेहमी एकच HTML घटक परत केले पाहिजे. जेव्हा आपल्याकडे अनेक घटक असतात आणि त्या सर्वांना रॅप करण्यासाठी कंटेनरची आवश्यकता नसते, तेव्हा आपण त्यांना एकाच घटकामध्ये एकत्र करण्यासाठी एक रिकामा घटक (<> ... </>) वापरतो.
<h2>Connection</h2>
<div>
status: {connection.status}
<br />
addresses: {JSON.stringify(connection.addresses)}
<br />
chainId: {connection.chainId}
</div>
सध्याच्या कनेक्शनबद्दल माहिती द्या. JSX मध्ये, {<expression>} चा अर्थ JavaScript म्हणून एक्सप्रेशनचे मूल्यांकन करणे असा होतो.
{connection.status === 'connected' && (
सिंटॅक्स {<condition> && <value>} means "if the condition is true, evaluate to the value; if it isn't, evaluate to false`".
JSX मध्ये if स्टेटमेंट्स टाकण्याचा हा स्टँडर्ड मार्ग आहे.
<div>
<Greeter />
<hr />
JSX हे XML मानकाचे पालन करते, जे HTML पेक्षा अधिक कठोर आहे. जर एखाद्या टॅगला संबंधित एंड टॅग नसेल, तर तो संपुष्टात आणण्यासाठी त्याच्या शेवटी स्लॅश (/) असलाच पाहिजे.
येथे आपल्याकडे असे दोन टॅग आहेत, <Greeter /> (ज्यामध्ये प्रत्यक्षात HTML कोड आहे जो कॉन्ट्रॅक्टशी संवाद साधतो) आणि आडव्या रेषेसाठी <hr /> (opens in a new tab).
<button type="button" onClick={disconnect}>
Disconnect
</button>
</div>
)}
जर युजरने या बटणावर क्लिक केले, तर disconnect फंक्शनला कॉल करा.
{connection.status !== 'connected' && (
जर आपण कनेक्टेड नसू, तर वॉलेटशी कनेक्ट करण्यासाठी आवश्यक पर्याय दाखवा.
<div>
<h2>Connect</h2>
{connectors.map((connector) => (
connectors मध्ये आपल्याकडे कनेक्टर्सची यादी आहे. आपण ते प्रदर्शित करण्यासाठी JSX बटणांच्या यादीत बदलण्यासाठी map (opens in a new tab) वापरतो.
<button
key={connector.uid}
JSX मध्ये "सिबलिंग" टॅग्स (एकाच पॅरेंटपासून आलेले टॅग्स) साठी वेगवेगळे आयडेंटिफायर्स असणे आवश्यक आहे.
onClick={() => connect({ connector })}
type="button"
>
{connector.name}
</button>
))}
कनेक्टर बटणे.
<div>{status}</div>
<div>{error?.message}</div>
</div>
)}
अतिरिक्त माहिती द्या. एक्सप्रेशन सिंटॅक्स <variable>?.<field> JavaScript ला सांगतो की जर व्हेरिएबल परिभाषित केले असेल, तर त्या फील्डचे मूल्यांकन करा. जर व्हेरिएबल परिभाषित केले नसेल, तर या एक्सप्रेशनचे मूल्यांकन undefined असे होते.
जेव्हा कोणतीही त्रुटी नसते, तेव्हा error.message हे एक्सप्रेशन अपवाद (exception) निर्माण करेल. error?.message वापरल्याने आपल्याला ही समस्या टाळता येते.
src/Greeter.tsx
या फाईलमध्ये बहुतांश UI कार्यक्षमता आहे. यात अशा व्याख्या समाविष्ट आहेत ज्या सामान्यतः अनेक फाईल्समध्ये असतात, परंतु हे एक ट्युटोरियल असल्याने, प्रोग्राम कार्यक्षमता किंवा देखभालीच्या सुलभतेपेक्षा पहिल्यांदा समजण्यास सोपा असावा यासाठी ऑप्टिमाइझ केला आहे.
import {
useState,
useEffect,
} from 'react'
import { useChainId,
useAccount,
useReadContract,
useWriteContract,
useWatchContractEvent,
useSimulateContract
} from 'wagmi'
आपण ही लायब्ररी फंक्शन्स वापरतो. पुन्हा, ते जिथे वापरले आहेत तिथे खाली स्पष्ट केले आहेत.
import { AddressType } from 'abitype'
abitype लायब्ररी (opens in a new tab) आपल्याला विविध इथेरियम डेटा प्रकारांसाठी TypeScript व्याख्या प्रदान करते, जसे की AddressType (opens in a new tab).
let greeterABI = [
{ "type": "function", "name": "greet", ... },
{ "type": "function", "name": "setGreeting", ... },
{ "type": "event", "name": "SetGreeting", ... },
] as const // greeterABI
Greeter कॉन्ट्रॅक्टसाठी ABI.
जर तुम्ही कॉन्ट्रॅक्ट्स आणि UI एकाच वेळी विकसित करत असाल, तर तुम्ही सामान्यतः त्यांना एकाच रिपॉझिटरीमध्ये ठेवाल आणि Solidity कंपायलरने जनरेट केलेला ABI तुमच्या ॲप्लिकेशनमध्ये फाईल म्हणून वापराल. तथापि, येथे हे आवश्यक नाही कारण कॉन्ट्रॅक्ट आधीच विकसित केले आहे आणि ते बदलणार नाही.
आपण TypeScript ला हे सांगण्यासाठी as const (opens in a new tab) वापरतो की हा एक खरा कॉन्स्टंट आहे. सामान्यतः, जेव्हा तुम्ही JavaScript मध्ये const x = {"a": 1} निर्दिष्ट करता, तेव्हा तुम्ही x मधील मूल्य बदलू शकता, तुम्ही फक्त त्याला असाईन करू शकत नाही.
type AddressPerBlockchainType = {
[key: number]: AddressType
}
TypeScript हे स्ट्रॉंगली टाईप केलेले आहे. आपण ही व्याख्या Greeter कॉन्ट्रॅक्ट वेगवेगळ्या चेन्सवर कुठे डिप्लॉय केले आहे तो पत्ता निर्दिष्ट करण्यासाठी वापरतो. की (key) एक संख्या (chainId) आहे आणि मूल्य एक AddressType (एक पत्ता) आहे.
const contractAddrs : AddressPerBlockchainType = {
// Sepolia
11155111: '0xC87506C66c7896366b9E988FE0aA5B6dDE77CFfA'
}
Sepolia (opens in a new tab) वरील कॉन्ट्रॅक्टचा पत्ता.
Timer घटक
Timer घटक दिलेल्या वेळेपासून किती सेकंद झाले हे दर्शवितो. युजेबिलिटीच्या उद्देशाने हे महत्त्वाचे आहे. जेव्हा युजर्स काहीतरी करतात, तेव्हा त्यांना त्वरित प्रतिक्रियेची अपेक्षा असते. ब्लॉकचेन्समध्ये, हे अनेकदा अशक्य असते कारण जोपर्यंत ट्रान्झॅक्शन ब्लॉकमध्ये ठेवले जात नाही तोपर्यंत काहीही होत नाही. यावर एक उपाय म्हणजे युजरने कृती केल्यापासून किती वेळ झाला आहे हे दाखवणे, जेणेकरून लागणारा वेळ योग्य आहे की नाही हे युजर ठरवू शकेल.
type TimerProps = {
lastUpdate: Date
}
Timer घटक एक पॅरामीटर घेतो, lastUpdate, जी शेवटच्या कृतीची वेळ आहे.
const Timer = ({ lastUpdate }: TimerProps) => {
const [_, setNow] = useState(new Date())
घटकाने योग्यरित्या कार्य करण्यासाठी आपल्याकडे स्थिती (घटकाशी जोडलेले व्हेरिएबल) असणे आणि ती अपडेट करणे आवश्यक आहे. परंतु आपल्याला ती कधीही वाचण्याची आवश्यकता नसते, त्यामुळे व्हेरिएबल तयार करण्याची तसदी घेऊ नका.
useEffect(() => {
const id = setInterval(() => setNow(new Date()), 1000)
return () => clearInterval(id)
}, [])
setInterval (opens in a new tab) फंक्शन आपल्याला वेळोवेळी चालवण्यासाठी फंक्शन शेड्युल करू देते. या प्रकरणात, दर सेकंदाला. फंक्शन स्थिती अपडेट करण्यासाठी setNow ला कॉल करते, त्यामुळे Timer घटक पुन्हा रेंडर केला जाईल. आपण हे useEffect (opens in a new tab) च्या आत एका रिकाम्या डिपेंडन्सी लिस्टसह रॅप करतो जेणेकरून घटक प्रत्येक वेळी रेंडर होण्याऐवजी ते फक्त एकदाच घडेल.
const secondsSinceUpdate = Math.floor(
(Date.now() - lastUpdate.getTime()) / 1000
)
return (
<span>{secondsSinceUpdate} seconds ago</span>
)
}
शेवटच्या अपडेटपासून किती सेकंद झाले याची गणना करा आणि ते परत करा.
Greeter घटक
const Greeter = () => {
शेवटी, आपण घटक परिभाषित करतो.
const chainId = useChainId()
const account = useAccount()
आपण वापरत असलेल्या चेन आणि अकाउंटबद्दलची माहिती, Wagmi (opens in a new tab) च्या सौजन्याने. कारण हा एक हुक (use...) आहे, जेव्हा जेव्हा ही माहिती बदलते तेव्हा घटक पुन्हा रेंडर केला जातो.
const greeterAddr = chainId && contractAddrs[chainId]
Greeter कॉन्ट्रॅक्टचा पत्ता, जो undefined असतो जर आपल्याकडे चेनची माहिती नसेल, किंवा आपण त्या कॉन्ट्रॅक्टशिवाय असलेल्या चेनवर असू.
const readResults = useReadContract({
address: greeterAddr,
abi: greeterABI,
functionName: "greet", // कोणतेही आर्ग्युमेंट्स नाहीत
})
useReadContract हुक (opens in a new tab) कॉन्ट्रॅक्टच्या (opens in a new tab) greet फंक्शनला कॉल करतो.
const [ currentGreeting, setCurrentGreeting ] =
useState("Please wait while we fetch the greeting from the blockchain...")
const [ newGreeting, setNewGreeting ] = useState("")
React चा useState हुक (opens in a new tab) आपल्याला एक स्थिती व्हेरिएबल निर्दिष्ट करू देतो, ज्याचे मूल्य घटकाच्या एका रेंडरिंगपासून दुसऱ्या रेंडरिंगपर्यंत टिकून राहते. प्रारंभिक मूल्य हे पॅरामीटर असते, या प्रकरणात रिकामी स्ट्रिंग.
useState हुक दोन मूल्यांसह एक यादी परत करतो:
- स्थिती व्हेरिएबलचे वर्तमान मूल्य.
- आवश्यकतेनुसार स्थिती व्हेरिएबल सुधारण्यासाठी एक फंक्शन. हा एक हुक असल्याने, प्रत्येक वेळी जेव्हा त्याला कॉल केले जाते तेव्हा घटक पुन्हा रेंडर केला जातो.
या प्रकरणात, युजरला जो नवीन ग्रीटिंग सेट करायचा आहे त्यासाठी आपण एक स्थिती व्हेरिएबल वापरत आहोत.
const [ lastSetterAddress, setLastSetterAddress ] = useState("")
जर अनेक युजर्स एकाच वेळी एकच कॉन्ट्रॅक्ट वापरत असतील, तर ते एकमेकांचे ग्रीटिंग्ज ओव्हरराईट करू शकतात. यामुळे युजर्सना असे वाटेल की ॲप्लिकेशनमध्ये बिघाड झाला आहे. जर ॲप्लिकेशनने शेवटचे ग्रीटिंग कोणी सेट केले हे दाखवले, तर युजरला समजेल की ते दुसरे कोणीतरी होते आणि ॲप्लिकेशन योग्यरित्या काम करत आहे.
const [ status, setStatus ] = useState("")
const [ statusTime, setStatusTime ] = useState(new Date())
युजर्सना त्यांच्या कृतींचा त्वरित परिणाम झालेला पाहायला आवडते. तथापि, ब्लॉकचेनवर असे होत नाही. हे स्थिती व्हेरिएबल्स आपल्याला युजर्सना किमान काहीतरी प्रदर्शित करू देतात जेणेकरून त्यांना समजेल की त्यांची कृती प्रगतीपथावर आहे.
useEffect(() => {
if (readResults.data) {
setCurrentGreeting(readResults.data)
setStatus("Greeting fetched from blockchain")
}
}, [readResults.data])
जर वरील readResults डेटा बदलत असेल आणि तो फॉल्स मूल्यावर सेट केलेला नसेल (उदाहरणार्थ, undefined), तर वर्तमान ग्रीटिंग ब्लॉकचेनवरून वाचलेल्या ग्रीटिंगवर अपडेट करा. तसेच, स्थिती अपडेट करा.
useWatchContractEvent({
address: greeterAddr,
abi: greeterABI,
eventName: 'SetGreeting',
chainId,
SetGreeting इव्हेंट्स ऐका.
enabled: !!greeterAddr,
!!<value> चा अर्थ असा आहे की जर मूल्य false असेल, किंवा फॉल्स म्हणून मूल्यांकन करणारे मूल्य असेल, जसे की undefined, 0, किंवा रिकामी स्ट्रिंग, तर एकूण एक्सप्रेशन false असते. इतर कोणत्याही मूल्यासाठी, ते true असते. हा मूल्यांना बुलियन्समध्ये रूपांतरित करण्याचा एक मार्ग आहे, कारण जर greeterAddr नसेल, तर आपल्याला इव्हेंट्स ऐकायचे नाहीत.
onLogs: logs => {
const greetingFromContract = logs[0].args.greeting
setCurrentGreeting(greetingFromContract)
setLastSetterAddress(logs[0].args.sender)
updateStatus("Greeting updated by event")
},
})
जेव्हा आपण लॉग्स पाहतो (जे नवीन इव्हेंट पाहिल्यावर घडते), तेव्हा त्याचा अर्थ असा होतो की ग्रीटिंग सुधारित केले गेले आहे. त्या प्रकरणात, आपण currentGreeting आणि lastSetterAddress नवीन मूल्यांवर अपडेट करू शकतो. तसेच, आपल्याला स्थिती डिस्प्ले अपडेट करायचा आहे.
const updateStatus = (newStatus: string) => {
setStatus(newStatus)
setStatusTime(new Date())
}
जेव्हा आपण स्थिती अपडेट करतो तेव्हा आपल्याला दोन गोष्टी करायच्या असतात:
- स्थिती स्ट्रिंग अपडेट करणे (
status) - शेवटच्या स्थिती अपडेटची वेळ (
statusTime) आताच्या वेळेवर अपडेट करणे.
const greetingChange = (evt) =>
setNewGreeting(evt.target.value)
नवीन ग्रीटिंग इनपुट फील्डमधील बदलांसाठी हा इव्हेंट हँडलर आहे. आपण evt पॅरामीटरचा प्रकार निर्दिष्ट करू शकलो असतो, परंतु TypeScript ही एक टाईप पर्यायी भाषा आहे. हे फंक्शन HTML इव्हेंट हँडलरमध्ये फक्त एकदाच कॉल केले जात असल्याने, मला ते आवश्यक वाटत नाही.
const { writeContractAsync } = useWriteContract()
कॉन्ट्रॅक्टवर लिहिण्यासाठी फंक्शन. हे writeContracts (opens in a new tab) सारखेच आहे, परंतु अधिक चांगल्या स्थिती अपडेट्स सक्षम करते.
const simulation = useSimulateContract({
address: greeterAddr,
abi: greeterABI,
functionName: 'setGreeting',
args: [newGreeting],
account: account.address
})
क्लायंटच्या दृष्टिकोनातून ब्लॉकचेन ट्रान्झॅक्शन सबमिट करण्याची ही प्रक्रिया आहे:
eth_estimateGas(opens in a new tab) वापरून ब्लॉकचेनमधील नोडला ट्रान्झॅक्शन पाठवा.- नोडकडून प्रतिसादाची प्रतीक्षा करा.
- जेव्हा प्रतिसाद प्राप्त होतो, तेव्हा युजरला वॉलेटद्वारे ट्रान्झॅक्शनवर स्वाक्षरी करण्यास सांगा. ही पायरी नोडचा प्रतिसाद प्राप्त झाल्यानंतरच झाली पाहिजे कारण स्वाक्षरी करण्यापूर्वी युजरला ट्रान्झॅक्शनचा गॅस खर्च दाखवला जातो.
- युजरने मंजुरी देण्याची प्रतीक्षा करा.
- ट्रान्झॅक्शन पुन्हा पाठवा, यावेळी
eth_sendRawTransaction(opens in a new tab) वापरून.
पायरी 2 ला लक्षणीय वेळ लागण्याची शक्यता आहे, ज्या दरम्यान युजर्सना प्रश्न पडू शकतो की त्यांची कमांड युजर इंटरफेसद्वारे प्राप्त झाली आहे की नाही आणि त्यांना अद्याप ट्रान्झॅक्शनवर स्वाक्षरी करण्यास का सांगितले जात नाही. यामुळे एक खराब युजर अनुभव (UX) तयार होतो.
यावर एक उपाय म्हणजे प्रत्येक वेळी जेव्हा पॅरामीटर बदलतो तेव्हा eth_estimateGas पाठवणे. त्यानंतर, जेव्हा युजरला प्रत्यक्षात ट्रान्झॅक्शन पाठवायचे असते (या प्रकरणात Update greeting दाबून), तेव्हा गॅस खर्च माहित असतो आणि युजर वॉलेट पेज त्वरित पाहू शकतो.
return (
आता आपण शेवटी परत करण्यासाठी प्रत्यक्ष HTML तयार करू शकतो.
<>
<h2>Greeter</h2>
{currentGreeting}
वर्तमान ग्रीटिंग दाखवा.
{lastSetterAddress && (
<p>Last updated by {
lastSetterAddress === account.address ? "you" : lastSetterAddress
}</p>
)}
जर आपल्याला माहित असेल की शेवटचे ग्रीटिंग कोणी सेट केले, तर ती माहिती प्रदर्शित करा. Greeter या माहितीचा मागोवा ठेवत नाही आणि आपल्याला SetGreeting इव्हेंट्ससाठी मागे वळून पाहायचे नाही, त्यामुळे आपण रन करत असताना ग्रीटिंग बदलल्यावरच आपल्याला ते मिळते.
<hr />
<input type="text"
value={newGreeting}
onChange={greetingChange}
/>
<br />
हे इनपुट टेक्स्ट फील्ड आहे जिथे युजर नवीन ग्रीटिंग सेट करू शकतो. प्रत्येक वेळी जेव्हा युजर की दाबतो, तेव्हा आपण greetingChange ला कॉल करतो, जे setNewGreeting ला कॉल करते. setNewGreeting हे useState मधून येत असल्याने, त्यामुळे Greeter घटक पुन्हा रेंडर होतो. याचा अर्थ असा की:
- नवीन ग्रीटिंगचे मूल्य ठेवण्यासाठी आपल्याला
valueनिर्दिष्ट करणे आवश्यक आहे, कारण अन्यथा ते परत डीफॉल्टमध्ये, म्हणजे रिकाम्या स्ट्रिंगमध्ये बदलेल. - प्रत्येक वेळी जेव्हा
newGreetingबदलते तेव्हाsimulationदेखील अपडेट केले जाते, ज्याचा अर्थ असा की आपल्याला योग्य ग्रीटिंगसह एक सिम्युलेशन मिळेल. हे संबंधित असू शकते कारण गॅस खर्च कॉल डेटाच्या आकारावर अवलंबून असतो, जो स्ट्रिंगच्या लांबीवर अवलंबून असतो.
<button disabled={!simulation.data}
ट्रान्झॅक्शन पाठवण्यासाठी आवश्यक असलेली माहिती मिळाल्यावरच बटण सक्षम करा.
onClick={async () => {
updateStatus("Please confirm in wallet...")
स्थिती अपडेट करा. या टप्प्यावर, युजरने वॉलेटमध्ये पुष्टी करणे आवश्यक आहे.
await writeContractAsync(simulation.data.request)
updateStatus("Transaction sent, waiting for greeting to change...")
}}
>
Update greeting
</button>
ट्रान्झॅक्शन प्रत्यक्षात पाठवल्यानंतरच writeContractAsync परत येते. यामुळे आपण युजरला दाखवू शकतो की ट्रान्झॅक्शन ब्लॉकचेनमध्ये समाविष्ट होण्यासाठी किती काळ वाट पाहत आहे.
<h4>Status: {status}</h4>
<p>Updated <Timer lastUpdate={statusTime} /> </p>
</>
)
}
स्थिती आणि ती अपडेट केल्यापासून किती वेळ झाला आहे हे दाखवा.
export {Greeter}
घटक एक्सपोर्ट करा.
src/wagmi.ts
शेवटी, Wagmi शी संबंधित विविध व्याख्या src/wagmi.ts मध्ये आहेत. मी येथे सर्वकाही स्पष्ट करणार नाही, कारण त्यातील बहुतांश बॉयलरप्लेट आहे जे तुम्हाला बदलण्याची शक्यता नाही.
import { http, webSocket, createConfig, fallback } from 'wagmi'
import { sepolia } from 'wagmi/chains'
import { injected } from 'wagmi/connectors'
export const config = createConfig({
chains: [sepolia],
Wagmi कॉन्फिगरेशनमध्ये या ॲप्लिकेशनद्वारे समर्थित चेन्स समाविष्ट आहेत. तुम्ही उपलब्ध चेन्सची यादी (opens in a new tab) पाहू शकता.
connectors: [
injected(),
],
हा कनेक्टर (opens in a new tab) आपल्याला ब्राउझरमध्ये इन्स्टॉल केलेल्या वॉलेटशी संवाद साधू देतो.
transports: {
[sepolia.id]: http()
Viem सोबत येणारा डीफॉल्ट HTTP एंडपॉईंट पुरेसा चांगला आहे. जर आपल्याला वेगळी URL हवी असेल, तर आपण http("https:// hostname ") किंवा webSocket("wss:// hostname ") वापरू शकतो.
},
multiInjectedProviderDiscovery: false,
})
दुसरी ब्लॉकचेन जोडणे
आजकाल अनेक L2 स्केलिंग सोल्युशन्स (opens in a new tab) आहेत आणि तुम्हाला अशा काहींना सपोर्ट करायचा असेल ज्यांना Viem अद्याप सपोर्ट करत नाही. हे करण्यासाठी, तुम्ही src/wagmi.ts मध्ये बदल करता. या सूचना Optimism Sepolia (opens in a new tab) कसे जोडायचे हे स्पष्ट करतात.
-
src/wagmi.tsसंपादित कराA. Viem मधून
defineChainप्रकार इम्पोर्ट करा.import { defineChain } from 'viem'B. नेटवर्क व्याख्या जोडा. तुम्हाला Optimism Sepolia साठी हे करण्याची खरोखर गरज नाही, ते आधीपासूनच
viemमध्ये आहे (opens in a new tab), परंतु या मार्गाने तुम्हीviemमध्ये नसलेली ब्लॉकचेन कशी जोडायची हे शिकता.const optimismSepolia = defineChain({ id: 11_155_420, name: 'OP Sepolia', nativeCurrency: { name: 'Sepolia Ether', symbol: 'ETH', decimals: 18 }, rpcUrls: { default: { http: ['https://sepolia.optimism.io'], webSocket: ['wss://optimism-sepolia.drpc.org'], }, }, blockExplorers: { default: { name: 'Blockscout', url: 'https://optimism-sepolia.blockscout.com', apiUrl: 'https://optimism-sepolia.blockscout.com/api', } }, })C.
createConfigकॉलमध्ये नवीन चेन जोडा.export const config = createConfig({ chains: [sepolia, optimismSepolia], connectors: [ injected(), ], transports: { [optimismSepolia.id]: http(), [sepolia.id]: http() }, multiInjectedProviderDiscovery: false, }) -
Sepolia वरील स्वयंचलित स्विच कमेंट आउट करण्यासाठी
src/App.tsxसंपादित करा. प्रोडक्शन सिस्टीमवर, तुम्ही कदाचित तुम्ही सपोर्ट करत असलेल्या प्रत्येक ब्लॉकचेनच्या लिंक्ससह बटणे दाखवाल./* useEffect(() => { if (connection.status === 'connected' && connection.chainId !== SEPOLIA_CHAIN_ID ) { switchChain({ chainId: SEPOLIA_CHAIN_ID }) } }, [connection.status, connection.chainId]) */ -
ॲप्लिकेशनला नवीन नेटवर्कवरील तुमच्या कॉन्ट्रॅक्ट्सचा पत्ता माहित आहे याची खात्री करण्यासाठी
src/Greeter.tsxसंपादित करा.const contractAddrs: AddressPerBlockchainType = { // Optimism Sepolia 11155420: "0x4dd85791923E9294E934271522f63875EAe5806f", // Sepolia 11155111: "0x7143d5c190F048C8d19fe325b748b081903E3BF0", } -
तुमच्या ब्राउझरमध्ये.
A. ChainList (opens in a new tab) वर ब्राउझ करा आणि तुमच्या वॉलेटमध्ये चेन जोडण्यासाठी टेबलच्या उजव्या बाजूला असलेल्या बटणांपैकी एकावर क्लिक करा.
B. ॲप्लिकेशनमध्ये, ब्लॉकचेन बदलण्यासाठी Disconnect करा आणि नंतर पुन्हा कनेक्ट करा. हे हाताळण्याचे अधिक चांगले मार्ग आहेत, परंतु त्यासाठी ॲप्लिकेशनमध्ये बदल करावे लागतील.
निष्कर्ष
अर्थात, तुम्हाला Greeter साठी युजर इंटरफेस प्रदान करण्याची खरोखर काळजी नाही. तुम्हाला तुमच्या स्वतःच्या कॉन्ट्रॅक्ट्ससाठी युजर इंटरफेस तयार करायचा आहे. तुमचे स्वतःचे ॲप्लिकेशन तयार करण्यासाठी, या पायऱ्या चालवा:
-
Wagmi ॲप्लिकेशन तयार करण्यासाठी निर्दिष्ट करा.
npm create wagmi -
पुढे जाण्यासाठी
yटाईप करा. -
ॲप्लिकेशनला नाव द्या.
-
React फ्रेमवर्क निवडा.
-
Vite व्हेरिएंट निवडा.
आता जा आणि तुमचे कॉन्ट्रॅक्ट्स संपूर्ण जगासाठी वापरण्यायोग्य बनवा.
माझ्या अधिक कामासाठी येथे पहा (opens in a new tab).
पृष्ठ शेवटचे अपडेट: 3 मार्च, 2026