将 Web3Auth 集成到 dApp 中
导言
Web3Auth 是一种钱包基础设施,可插入 dApp 或钱包。 它是 Web3 钱包和应用程序的可插拔认证基础设施。 凭借 Web3Auth 卓越的用户体验,无论是主流用户还是加密货币原住民,都可以在几分钟内完成注册。
作为一种钱包基础设施,它为所有社交登录、网络和移动原生平台、钱包和其他密钥管理方法提供开箱即用的支持。 本指南结束时,您将把 Web3Auth 集成到基于 Kaia 网络构建的去中心化网络应用程序中。 要将 Web3Auth 集成到其他平台(Android、iOS、React Native、Flutter 和 Unity),请参考本 指南。
为了快速入门,本教程的完整代码可在 GitHub 上获取。 您可以克隆或下载该版本库,以便跟随学习。
先决条件
- 一个正在运行的 react 项目(通过执行
npm create vite@latest project-name -- --template react-ts
) - 安装必要的钱包(Coinbase Wallet、Metamask)。
- RPC 端点:您可以从支持的端点提供者中获取。
- 从 水龙头测试 KAIA:为账户注入足够的 KAIA。
- 从 Web3Auth 控制面板 获取您的客户 ID。
安装
要在 dApp 中使用 Web3Auth,必须先安装所需的库和 SDK。 因此,您需要设置 ethers.js 和 Web3Auth Web SDK。 您可以将 Web3Auth 与 etherthers.js、web3.js 或 kaia sdk 库一起使用,以便与 Kaia 区块链通信。 本指南将使用 ethers.js。
npm install --save @web3auth/modal @web3auth/base @web3auth/ethereum-provider @web3auth/default-evm-adapternpm install --save ethers
初始化 Web3Auth 和提供程序实例
成功安装所需的库后,接下来要初始化 Web3Auth 实例,在 "useState() "钩子中设置 Web3Auth provider 实例,并在 "useEffect() "钩子中设置 "init() "函数。
导入 Web3Auth 和其他依赖软件包。
导入 React 挂钩(useState 和 useEffect)和实用功能:
useState
和useEffect
:用于状态管理和副作用的 React 挂钩。RPC
:来自etherRPC.ts
的自定义实用功能,用于使用 ethers.js 进行以太坊兼容的区块链交互。
从 Web3Auth 控制面板粘贴您的 ** 客户 ID**。
设置 ** 链配置**:要使用 Web3Auth,您需要为选定的链 - Kaia 设置链配置。
通过使用构造函数初始化 Web3Auth,在构造函数中,您可以传入您想要的 Web3Auth 的所有配置。
在 "useState() "钩子中设置 Web3Auth provider 实例、userInfo 和 "useEffect() "钩子中的 "init() "函数。
连接钱包
在 App.tsx
文件的 App 函数中,调用 web3Auth 实例的 connect() 方法启动钱包连接。
function App() { const [provider, setProvider] = (useState < IProvider) | (null > null) const [loggedIn, setLoggedIn] = useState < boolean > false const login = async () => { if (!web3auth) { console.log('web3auth not initialized yet') return } const web3authProvider = await web3auth.connect() setProvider(web3authProvider) if (web3auth.connected) { setLoggedIn(true) await updateUserInfo() } } return ( <div className="App"> <button onClick={login}>Login</button> </div> )}
设置实用工具功能
在本指南中,我们将使用实用工具函数:truncateAddress()
。 truncateAddress() 函数接收一个有效地址,并返回一个更可读的地址格式。 以下步骤展示了如何在项目中设置和使用 utils 函数。
第 1 步在 src
根文件夹中创建一个 utils.ts
文件。
将以下代码粘贴到新创建的 utils.ts 文件中:
export const truncateAddress = (address) => { if (!address) return 'No Account' const match = address.match( /^(0x[a-zA-Z0-9]{2})[a-zA-Z0-9]+([a-zA-Z0-9]{4})$/ ) if (!match) return address return `${match[1]}…${match[2]}`}
第 2 步在您的 App.tsx
文件中导入该函数。
从'./utils'导入 { truncateAddress }
获取账户和余额
调用 Web3Auth 实例上的 connect()
方法成功连接钱包后,就可以使用提供者和签名者对象获取用户账户及其余额。
function App() { const [provider, setProvider] = useState<IProvider | null>(null); const [address, setAddress] = useState<string>(""); const [balance, setBalance] = useState<string>(""); const getAccounts = async () => { if (!provider) { console.log('provider not initialized yet') return } const address = await RPC.getAccounts(provider) setAddress(address) console.log('Address:', address) } const getBalance = async () => { if (!provider) { console.log('provider not initialized yet') return } const balance = await RPC.getBalance(provider) setBalance(balance) console.log('Balance:', balance) } return ( <div className="App"> <div> <button onClick={getAccounts}>Get Account</button> <h2> {' '} Address: <span> {' '} {`${truncateAddress(address)}` || 'Not available'} </span> </h2> </div> <div> <button onClick={getBalance}>Get Balance</button> <h2> Balance: <span> {balance || 'Not available'}</span> </h2> </div> </div> )}
断开钱包连接
使用 Web3Auth 实例上的 logout() 方法可以断开与钱包的连接。 此外,一个好的做法是刷新状态,清除之前存储的连接数据。
function App() { const logout = async () => { if (!web3auth) { console.log('web3auth not initialized yet') return } await web3auth.logout() setProvider(null) setLoggedIn(false) setAddress('') setBalance('') setUserInfo(null) console.log('Logged out') } return ( <div className="App"> <button onClick={logout}>Logout</button> </div> )}
获取用户信息
Web3Auth 的一个独特功能是社交登录。 用户使用社交平台登录后,Web3Auth 实例会返回登录用户的一些信息。 在 Web3Auth 实例上调用 getUserInfo()
方法即可简单地获取登录用户信息。
const [userInfo, setUserInfo] = useState<UserInfo | null>(null);const getUserInfo = async () => { if (!web3auth) { console.log('web3auth not initialized yet') return } const user = await web3auth.getUserInfo() // @ts-ignore setUserInfo(user) console.log('User Info:', user)}return ( <div className="App"> <button onClick={getUserInfo}>Get User Info</button> <div> <h2>User Info:</h2> <pre> {userInfo ? JSON.stringify(userInfo, null, 2) : 'Not available'} </pre> </div> </div>)
签署信息
初始化提供者和签名者对象后,用户就可以签署任意字符串。
// add to the existing useState hook.const [signedMessage, setSignedMessage] = useState<string>("");const signMessage = async () => { if (!provider) { console.log("provider not initialized yet"); return; } const originalMessage = "YOUR_MESSAGE"; const signedMessage = await RPC.signMessage(provider, originalMessage); setSignedMessage(signedMessage); console.log("Signed Message:", signedMessage);};return ( <div className="App"> <button onClick={signMessage}>Sign Message</button> {signedMessage && ( <div> <h2 className="text-wrap text-center text-sm font-semibold"> Signed Message: <span className="font-normal">{signedMessage}</span> </h2> </div> )} </div>)
发送本地事务
您可以执行本地事务,如将 KAIA 从一个用户发 送到另一个用户。
// add to the existing useState hook.const [txHash, setTxHash] = useState<string>("");const sendKaiaTx = async () => { if (!provider) { console.log("provider not initialized yet"); return; } console.log("Sending Transaction..."); const destination = "0x75Bc50a5664657c869Edc0E058d192EeEfD570eb"; const amount = "0.1"; const receipt = await RPC.sendKaiaTx(provider, destination, amount); setTxHash(receipt.hash); console.log("Transaction Receipt:", receipt);};return ( <div className="App"> <button onClick={sendKaiaTx}>Send Kaia</button> {txHash && ( <div> <h2> Transaction Hash:{' '} <a href={`${chainConfig.blockExplorerUrl}/tx/${txHash}`} target="_blank" rel="noopener noreferrer" > {txHash} </a> </h2> </div> )} </div>)
使用智能合约
您可以根据智能合约的应用程序二进制接口(ABI)和合约地址与已部署的智能合约进行交互。 以下步骤说明了如何在项目中设置和使用合同地址和 ABI。
第 1 步在 src
根文件夹中创建一个 constants.ts
文件。
将以下代码粘贴到新创建的 constants.ts 文件中:
export const contractABI = [ { "inputs": [ { "internalType": "uint256", "name": "_initNum", "type": "uint256" } ], "stateMutability": "nonpayable", "type": "constructor" }, { "inputs": [], "name": "retrieve", "outputs": [ { "internalType": "uint256", "name": "", "type": "uint256" } ], "stateMutability": "view", "type": "function" }, { "inputs": [ { "internalType": "uint256", "name": "num", "type": "uint256" } ], "name": "store", "outputs": [], "stateMutability": "nonpayable", "type": "function" } ] as const; export const contractAddress = "0x3b01E4025B428fFad9481a500BAc36396719092C";
第 2 步:在 etherRPC.ts
文件中导入contractABI和contractAddress。
从"./constants "导入 { contractAddress, contractABI };
1. 撰写合同
// add to existing useState hookconst [contractTxHash, setContractTxHash] = useState<string>("");const setContractValue = async () => { if (!provider) { console.log("provider not initialized yet"); return; } console.log("setting contract value..."); const value = "100"; const tx = await RPC.setContractValue(provider, value); setContractTxHash(tx.hash); console.log("Transaction Receipt:", tx);}return ( <div className="App"> <button onClick={setContractValue}>Write to Contract</button> {contractTxHash && ( <div> <h2> Contract Tx Hash: <span>{contractTxHash}</span> </h2> </div> )} </div>)
2. 阅读合同
// add to existing useState hook const [contractMessage, setContractMessage] = useState<string>(""); const getContractValue = async () => { if (!provider) { console.log("provider not initialized yet"); return; } console.log("getting value from contract..."); const message = await RPC.getContractValue(provider); setContractMessage(message); console.log("contract message:", message); } return ( <button onClick={getContractValue}>Read From Contract</button> {contractMessage && ( <div> <h2 className="text-wrap text-center text-sm font-semibold">Read Message: <span className="font-normal">{contractMessage}</span></h2> </div> )} )
故障排除
您可以访问故障排除页面,从不同的捆绑程序中探索常见挑战和问题的解决方案。
下一步
有关 Web3Auth 的更多深入指南,请参阅 Web3Auth 文档 和 Web3Auth Github 存储库。