Este proyecto es una introducción paso a paso a varios aspectos importantes del desarrollo frontend en Web3, usando React y Vite. Cubriremos:
- Creación y configuración de un proyecto React usando Vite
- Integración de Tailwind CSS para un diseño eficiente
- Incorporación de bibliotecas Web3 para interactuar con la blockchain
- Estructura básica para un proyecto Web3
- Desarrollo de un ejemplo interactuando con un contrato ERC20
- Despliegue en Vercel para llevar tu proyecto a la web
Sigue estos pasos para configurar un nuevo proyecto de React con Vite:
- Creación del Proyecto
-
Ejecuta el siguiente comando en tu terminal para crear un nuevo proyecto:
npm create vite@latest . -- --template react
-
Instalamos las dependencias.
npm install
-
Ejecutamos la app de muestra para comprobar que funciona correctamente.
npm run dev
-
Comprueba que se ha ejecutado correctamente en el servidor local. Por defecto, Vite utiliza el puerto 5173
- Abre tu navegador y visita http://localhost:5173/
- Ahora que hemos comprobado que todo funciona correctamente, podemos parar el servidor y seguir preparando nuestra dapp.
-
Para personalizar tu aplicación, edita el archivo index.html en tu proyecto:
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8" /> <!-- Ponemos el icono de la carpeta public --> <link rel="icon" type="image/svg+xml" href="/blockmaker-small-logo.png" /> <meta name="viewport" content="width=device-width, initial-scale=1.0" /> <!-- Ponemos el título de la App --> <title>Blockmaker Token</title> </head> <body> <div id="root"></div> <script type="module" src="/src/main.jsx"></script> </body> </html>
-
Instalamos y configuramos nuestro framework de estilos para nuestra aplicación:
npm install -D tailwindcss postcss autoprefixer npx tailwindcss init -p
-
Configuramos las rutas que contendrán estilos tailwind en el fichero tailwind.config.js:
/** @type {import('tailwindcss').Config} */ export default { content: ['./index.html', './src/**/*.{js,ts,jsx,tsx}'], theme: { extend: {} }, plugins: [] }
-
Añadimos las directivas de tailwind en nuestro fichero index.css
@tailwind base; @tailwind components; @tailwind utilities;
-
Cambiamos la fuente por defecto en nuestra aplicación y el color de fondo:
@tailwind base; @tailwind components; @tailwind utilities; @layer base { html { font-family: 'Roboto', system-ui, sans-serif; background-color: #f9fafb; } }
-
Eliminamos el fichero App.css (si no es necesario)
-
Añadimos el archivo imagen blockmaker a nuestra carpeta raíz
public
. -
Creamos el componente Header añadiéndole nuestro logo y le damos estilos usando tailwind.
export default function Header() { return ( <header className="py-2 px-3 sm:py-4 sm:px-8 flex justify-between items-center bg-white border-b shadow-xs"> {/* Logo para version mobile */} <img src="/blockmaker-small-logo.png" alt="blockmaker-logo" width={47} className="sm:hidden" /> {/* Logo para version desktop */} <img src="/blockmaker-full-logo.png" alt="blockmaker-logo" width={300} className="hidden sm:flex" /> </header> ) }
-
Creamos el componente Footer utilizando
new Date().getFullYear()
para que se actualice el año.export default function Footer() { return ( <footer className="py-4 px-3 sm:p-8 flex justify-center bg-white border-t shadow-xs"> <p>Derechos de autor © {new Date().getFullYear()} Blockmaker Academy</p> </footer> ) }
-
Dentro de la carpeta
componentes
, creamos una carpetaui
, y dentro deui
creamos una carpetalayouts
(diseños). -
Creamos un componente
AppLayout
que contiene el diseño principal de nuestra app, el Header y el Footer.import PropTypes from 'prop-types' import React from 'react' import Footer from '../../Footer' import Header from '../../Header' export default function AppLayout({ children }) { return ( <React.Fragment> <Header /> <main className="mx-auto min-h-[85vh]">{children}</main> <Footer /> </React.Fragment> ) } AppLayout.propTypes = { children: PropTypes.node.isRequired }
Añadimos un fichero index.js para exportar directamente el componente layout.
export { default as AppLayout } from './AppLayout'
-
Creamos una carpeta
pages
en la raíz de src (src/pages) -
Creamos un componente Home
export default function Home() { return ( <div> <h1>Esta es la página principal</h1> </div> ) }
-
Creamos un fichero index.js para exportar todos los componentes página directamente, en este caso, solo Home:
export { default as Home } from './Home'
-
Actualizamos nuestro componente
App
añadiendo nuestro componenteAppLayout
yHome
.import { AppLayout } from './components/ui/layouts' import { Home } from './pages' function App() { return ( <AppLayout> <Home /> </AppLayout> ) } export default App
-
Comprobamos que todo funciona correctamente, que la app funciona bien y se aplican correctamente los estilos de tailwind.
Seguimos las instrucciones de connectkit para la instalación y configuración de connectkit y wagmi en nuestra aplicación para poder interactuar con la blockchain de manera más sencilla y práctica.
Documentación oficial: https://docs.family.co/connectkit/getting-started
Nota: Debido a una reciente y significativa actualización que modifica sustancialmente la configuración inicial, se deberá consultar la documentación de versiones anteriores:
- Documentación oficial version 1.5 ConnectKit: https://web.archive.org/web/20230929142557/https://docs.family.co/connectkit/getting-started
- Documentación oficial version anterior Wagmi: https://0.6.x.wagmi.sh/react/getting-started
-
Instalación de connectkit y sus dependencias
npm install connectkit@1.5.3 viem@1.20.0 wagmi@1.4.12
-
Configuración de las API Keys (si aún no tienes ninguna).
-
ConnectKit utiliza el SDK de WalletConnect para facilitar la conexión de carteras. WalletConnect 2.0 requiere un projectId, que puedes crear de manera rápida y gratuita en WalletConnect Cloud.
-
ConnectKit necesita una clave API de Infura o Alchemy, dependiendo de las redes que tu dApp necesite. Para aprender más sobre las diferencias entre Infura y Alchemy, puedes visitar la sección de Proveedores.
-
Creamos un fichero .env y ponemos las siguientes variables de entorno:
# Alchemy VITE_ALCHEMY_ID=<TU_ALCHEMY_ID> # WalletConnect VITE_WALLETCONNECT_PROJECT_ID=<TU_WALLETCONNECT_PROJECT_ID> # Contract address VITE_TOKEN_CONTRACT_ADDRESS=0xA529Dfd9CB8ea733613021999B2d2D55DD376528
-
Creación del archivo de configuración de Wagmi
- Creamos una carpeta
config
dentro de la carpetasrc
. - Dentro de la carpeta
config
, creamos el siguiente archivo con la configuración de Wagmi:
// Archivo wagmi.js import { getDefaultConfig } from 'connectkit' import { createConfig } from 'wagmi' import { sepolia } from 'wagmi/chains' export const config = createConfig( getDefaultConfig({ // Importamos la variable de entorno con import.meta.env alchemyId: import.meta.env.VITE_ALCHEMY_ID, walletConnectProjectId: import.meta.env.VITE_WALLETCONNECT_PROJECT_ID, appName: 'Blockmaker ERC20 DApp', // Configuramos la chain chains: [sepolia] }) ) export default config
- Creamos una carpeta
-
Importamos la configuración en nuestro componente APP y lo envolvemos con los componentes proveedores de Wagmi y ConnectKit.
import { WagmiConfig } from 'wagmi' import { ConnectKitProvider } from 'connectkit' import { config } from './config/wagmi' function App() { return ( <WagmiConfig config={config}> // Lo ponemos en modo light ya que construiremos la aplicación en modo claro <ConnectKitProvider mode="light"> <AppLayout> <Home /> </AppLayout> </ConnectKitProvider> </WagmiConfig> ) } export default App
-
Añadimos el botón de ConnectKit para conectar la wallet.
import { ConnectKitButton } from 'connectkit' export default function Header() { return ( <header className="py-2 px-3 sm:py-4 sm:px-8 flex justify-between items-center bg-white border-b shadow-xs"> {/* Logo para version mobile */} <img src="/blockmaker-small-logo.png" alt="blockmaker-logo" width={47} className="sm:hidden" /> {/* Logo para version desktop */} <img src="/blockmaker-full-logo.png" alt="blockmaker-logo" width={300} className="hidden sm:flex" /> {/* Botton de conexion wallet con prop showBalance para mostrar balance cuenta */} <ConnectKitButton showBalance /> </header> ) }
-
Ejecutamos la aplicación y comprobamos que el botón de conectar funciona correctamente y nos conecta a la red de Sepolia.
-
Instalamos la librería
react-icons
para poder utilizar múltiples íconos de manera sencilla en nuestra app.npm install react-icons --save
-
Creamos el componente
LoadingSpinner
para mostrarlo durante las cargas:import PropTypes from 'prop-types' export default function LoadingSpinner({ className }) { return ( <div role="status"> <svg aria-hidden="true" className={`w-8 h-8 text-gray-200 animate-spin dark:text-gray-600 fill-blue-600 ${className}`} viewBox="0 0 100 101" fill="none" xmlns="http://www.w3.org/2000/svg" > <path d="M100 50.5908C100 78.2051 77.6142 100.591 50 100.591C22.3858 100.591 0 78.2051 0 50.5908C0 22.9766 22.3858 0.59082 50 0.59082C77.6142 0.59082 100 22.9766 100 50.5908ZM9.08144 50.5908C9.08144 73.1895 27.4013 91.5094 50 91.5094C72.5987 91.5094 90.9186 73.1895 90.9186 50.5908C90.9186 27.9921 72.5987 9.67226 50 9.67226C27.4013 9.67226 9.08144 27.9921 9.08144 50.5908Z" fill="currentColor" /> <path d="M93.9676 39.0409C96.393 38.4038 97.8624 35.9116 97.0079 33.5539C95.2932 28.8227 92.871 24.3692 89.8167 20.348C85.8452 15.1192 80.8826 10.7238 75.2124 7.41289C69.5422 4.10194 63.2754 1.94025 56.7698 1.05124C51.7666 0.367541 46.6976 0.446843 41.7345 1.27873C39.2613 1.69328 37.813 4.19778 38.4501 6.62326C39.0873 9.04874 41.5694 10.4717 44.0505 10.1071C47.8511 9.54855 51.7191 9.52689 55.5402 10.0491C60.8642 10.7766 65.9928 12.5457 70.6331 15.2552C75.2735 17.9648 79.3347 21.5619 82.5849 25.841C84.9175 28.9121 86.7997 32.2913 88.1811 35.8758C89.083 38.2158 91.5421 39.6781 93.9676 39.0409Z" fill="currentFill" /> </svg> <span className="sr-only">Loading...</span> </div> ) } LoadingSpinner.propTypes = { className: PropTypes.string }
-
Creamos el componente
Botón
:
-
Añadimos estilos para los distintos estados del botón (hover, disabled).
-
Añadimos la propiedad
isLoading
y el componenteLoadingSpinner
para mostrar el estado de carga en el botón.import PropTypes from 'prop-types' import LoadingSpinner from './LoadingSpinner' export default function Button({ type = 'button', onClick, disabled, isLoading, children }) { return ( <button type={type} className="bg-indigo-500 hover:bg-indigo-700 text-white font-bold py-2 px-4 rounded-lg disabled:opacity-75 disabled:cursor-not-allowed" onClick={onClick} disabled={disabled} > <span className="flex justify-center items-center gap-2"> {isLoading && <LoadingSpinner className="h-6 w-6" />} {children} </span> </button> ) } Button.propTypes = { type: PropTypes.oneOf(['button', 'submit', 'reset']), onClick: PropTypes.func, disabled: PropTypes.bool, isLoading: PropTypes.bool, children: PropTypes.node.isRequired }
-
Creamos el componente
ErrorInfo
para mostrar mensajes de error con un ícono.import { MdInfoOutline } from 'react-icons/md' import PropTypes from 'prop-types' export default function ErrorInfo({ message }) { return ( <div className="flex items-center gap-2 bg-red-200 p-2 rounded"> <MdInfoOutline className="text-red-500" /> <p className="text-xs text-red-500">{message}</p> </div> ) } ErrorInfo.propTypes = { message: PropTypes.string.isRequired }
-
Creamos el componente
TextInput
:import PropTypes from 'prop-types' export default function TextInput({ type, onChange, value, placeholder, disabled }) { return ( <input type={type} onChange={onChange} value={value} placeholder={placeholder} disabled={disabled} className="border border-gray-300 rounded-md w-full px-3 py-2 outline-none focus:ring-2 focus:ring-indigo-500 focus:border-transparent disabled:cursor-not-allowed" /> ) } TextInput.propTypes = { type: PropTypes.oneOf(['text', 'number', 'password']), onChange: PropTypes.func, value: PropTypes.oneOfType([PropTypes.string, PropTypes.number]), placeholder: PropTypes.string, disabled: PropTypes.bool }
-
Creamos el componente
Title
:import PropTypes from 'prop-types' export default function Title({ children }) { return <h3 className="font-bold text-lg mb-2 w-fit text-slate-700">{children}</h3> } Title.propTypes = { children: PropTypes.node.isRequired }
-
Creamos un fichero
index.js
dentro de la carpetacomponents
para exportar todos los componentes directamente:export { default as Button } from './Button' export { default as ErrorInfo } from './ErrorInfo' export { default as LoadingSpinner } from './LoadingSpinner' export { default as TextInput } from './TextInput' export { default as Title } from './Title'
-
Esto nos permitirá mostrar notificaciones de eventos a través de nuestra app y dar un mejor feedback al usuario.
-
Instalación de la librería:
npm install react-hot-toast
-
Configuración de la librería.
-
Añadimos el componente
Toaster
en nuestro componente App. -
Más adelante veremos cómo usar los
toast
para mostrar las notificaciones.import { ConnectKitProvider } from 'connectkit' import { Toaster } from 'react-hot-toast' import { WagmiConfig } from 'wagmi' import { AppLayout } from './components/ui/layouts' import { config } from './config/wagmi' import { Home } from './pages' function App() { return ( <WagmiConfig config={config}> <Toaster position="bottom-right" /> <ConnectKitProvider mode="light"> <AppLayout> <Home /> </AppLayout> </ConnectKitProvider> </WagmiConfig> ) } export default App
En esta sección crearemos los siguientes componentes con sus componentes auxiliares:
- Token Balance
- Owner
- Total Supply
- TokenInfo
- Transfer
- Mint
- Burn
- Como usaremos métodos del contrato en algunos de los componentes, primero tenemos que integrar el ABI del contrato para poder leer los métodos del contrato sin error.
-
Nos dirigimos al contrato en
etherscan
para descargar suABI
: -
Exportamos el ABI en formato texto y lo copiamos.
-
Creamos una carpeta contracts/ABIs.
-
Creamos un archivo blockmakerTokenABI.js y exportamos la variable con el ABI que hemos copiado anteriormente:
export const blockmakerTokenABI = [ { inputs: [{ internalType: 'address', name: 'initialOwner', type: 'address' }], stateMutability: 'nonpayable', type: 'constructor' }, { inputs: [], name: 'ECDSAInvalidSignature', type: 'error' }, { inputs: [{ internalType: 'uint256', name: 'length', type: 'uint256' }], name: 'ECDSAInvalidSignatureLength', type: 'error' }, { inputs: [{ internalType: 'bytes32', name: 's', type: 'bytes32' }], name: 'ECDSAInvalidSignatureS', type: 'error' }, { inputs: [ { internalType: 'address', name: 'spender', type: 'address' }, { internalType: 'uint256', name: 'allowance', type: 'uint256' }, { internalType: 'uint256', name: 'needed', type: 'uint256' } ], name: 'ERC20InsufficientAllowance', type: 'error' }, { inputs: [ { internalType: 'address', name: 'sender', type: 'address' }, { internalType: 'uint256', name: 'balance', type: 'uint256' }, { internalType: 'uint256', name: 'needed', type: 'uint256' } ], name: 'ERC20InsufficientBalance', type: 'error' }, { inputs: [{ internalType: 'address', name: 'approver', type: 'address' }], name: 'ERC20InvalidApprover', type: 'error' }, { inputs: [{ internalType: 'address', name: 'receiver', type: 'address' }], name: 'ERC20InvalidReceiver', type: 'error' }, { inputs: [{ internalType: 'address', name: 'sender', type: 'address' }], name: 'ERC20InvalidSender', type: 'error' }, { inputs: [{ internalType: 'address', name: 'spender', type: 'address' }], name: 'ERC20InvalidSpender', type: 'error' }, { inputs: [{ internalType: 'uint256', name: 'deadline', type: 'uint256' }], name: 'ERC2612ExpiredSignature', type: 'error' }, { inputs: [ { internalType: 'address', name: 'signer', type: 'address' }, { internalType: 'address', name: 'owner', type: 'address' } ], name: 'ERC2612InvalidSigner', type: 'error' }, { inputs: [], name: 'EnforcedPause', type: 'error' }, { inputs: [], name: 'ExpectedPause', type: 'error' }, { inputs: [ { internalType: 'address', name: 'account', type: 'address' }, { internalType: 'uint256', name: 'currentNonce', type: 'uint256' } ], name: 'InvalidAccountNonce', type: 'error' }, { inputs: [], name: 'InvalidShortString', type: 'error' }, { inputs: [{ internalType: 'address', name: 'owner', type: 'address' }], name: 'OwnableInvalidOwner', type: 'error' }, { inputs: [{ internalType: 'address', name: 'account', type: 'address' }], name: 'OwnableUnauthorizedAccount', type: 'error' }, { inputs: [{ internalType: 'string', name: 'str', type: 'string' }], name: 'StringTooLong', type: 'error' }, { anonymous: false, inputs: [ { indexed: true, internalType: 'address', name: 'owner', type: 'address' }, { indexed: true, internalType: 'address', name: 'spender', type: 'address' }, { indexed: false, internalType: 'uint256', name: 'value', type: 'uint256' } ], name: 'Approval', type: 'event' }, { anonymous: false, inputs: [], name: 'EIP712DomainChanged', type: 'event' }, { anonymous: false, inputs: [ { indexed: true, internalType: 'address', name: 'previousOwner', type: 'address' }, { indexed: true, internalType: 'address', name: 'newOwner', type: 'address' } ], name: 'OwnershipTransferred', type: 'event' }, { anonymous: false, inputs: [{ indexed: false, internalType: 'address', name: 'account', type: 'address' }], name: 'Paused', type: 'event' }, { anonymous: false, inputs: [ { indexed: true, internalType: 'address', name: 'from', type: 'address' }, { indexed: true, internalType: 'address', name: 'to', type: 'address' }, { indexed: false, internalType: 'uint256', name: 'value', type: 'uint256' } ], name: 'Transfer', type: 'event' }, { anonymous: false, inputs: [{ indexed: false, internalType: 'address', name: 'account', type: 'address' }], name: 'Unpaused', type: 'event' }, { inputs: [], name: 'DOMAIN_SEPARATOR', outputs: [{ internalType: 'bytes32', name: '', type: 'bytes32' }], stateMutability: 'view', type: 'function' }, { inputs: [ { internalType: 'address', name: 'owner', type: 'address' }, { internalType: 'address', name: 'spender', type: 'address' } ], name: 'allowance', outputs: [{ internalType: 'uint256', name: '', type: 'uint256' }], stateMutability: 'view', type: 'function' }, { inputs: [ { internalType: 'address', name: 'spender', type: 'address' }, { internalType: 'uint256', name: 'value', type: 'uint256' } ], name: 'approve', outputs: [{ internalType: 'bool', name: '', type: 'bool' }], stateMutability: 'nonpayable', type: 'function' }, { inputs: [{ internalType: 'address', name: 'account', type: 'address' }], name: 'balanceOf', outputs: [{ internalType: 'uint256', name: '', type: 'uint256' }], stateMutability: 'view', type: 'function' }, { inputs: [{ internalType: 'uint256', name: 'value', type: 'uint256' }], name: 'burn', outputs: [], stateMutability: 'nonpayable', type: 'function' }, { inputs: [ { internalType: 'address', name: 'account', type: 'address' }, { internalType: 'uint256', name: 'value', type: 'uint256' } ], name: 'burnFrom', outputs: [], stateMutability: 'nonpayable', type: 'function' }, { inputs: [], name: 'decimals', outputs: [{ internalType: 'uint8', name: '', type: 'uint8' }], stateMutability: 'view', type: 'function' }, { inputs: [], name: 'eip712Domain', outputs: [ { internalType: 'bytes1', name: 'fields', type: 'bytes1' }, { internalType: 'string', name: 'name', type: 'string' }, { internalType: 'string', name: 'version', type: 'string' }, { internalType: 'uint256', name: 'chainId', type: 'uint256' }, { internalType: 'address', name: 'verifyingContract', type: 'address' }, { internalType: 'bytes32', name: 'salt', type: 'bytes32' }, { internalType: 'uint256[]', name: 'extensions', type: 'uint256[]' } ], stateMutability: 'view', type: 'function' }, { inputs: [ { internalType: 'address', name: 'to', type: 'address' }, { internalType: 'uint256', name: 'amount', type: 'uint256' } ], name: 'mint', outputs: [], stateMutability: 'nonpayable', type: 'function' }, { inputs: [], name: 'name', outputs: [{ internalType: 'string', name: '', type: 'string' }], stateMutability: 'view', type: 'function' }, { inputs: [{ internalType: 'address', name: 'owner', type: 'address' }], name: 'nonces', outputs: [{ internalType: 'uint256', name: '', type: 'uint256' }], stateMutability: 'view', type: 'function' }, { inputs: [], name: 'owner', outputs: [{ internalType: 'address', name: '', type: 'address' }], stateMutability: 'view', type: 'function' }, { inputs: [], name: 'pause', outputs: [], stateMutability: 'nonpayable', type: 'function' }, { inputs: [], name: 'paused', outputs: [{ internalType: 'bool', name: '', type: 'bool' }], stateMutability: 'view', type: 'function' }, { inputs: [ { internalType: 'address', name: 'owner', type: 'address' }, { internalType: 'address', name: 'spender', type: 'address' }, { internalType: 'uint256', name: 'value', type: 'uint256' }, { internalType: 'uint256', name: 'deadline', type: 'uint256' }, { internalType: 'uint8', name: 'v', type: 'uint8' }, { internalType: 'bytes32', name: 'r', type: 'bytes32' }, { internalType: 'bytes32', name: 's', type: 'bytes32' } ], name: 'permit', outputs: [], stateMutability: 'nonpayable', type: 'function' }, { inputs: [], name: 'renounceOwnership', outputs: [], stateMutability: 'nonpayable', type: 'function' }, { inputs: [], name: 'symbol', outputs: [{ internalType: 'string', name: '', type: 'string' }], stateMutability: 'view', type: 'function' }, { inputs: [], name: 'totalSupply', outputs: [{ internalType: 'uint256', name: '', type: 'uint256' }], stateMutability: 'view', type: 'function' }, { inputs: [ { internalType: 'address', name: 'to', type: 'address' }, { internalType: 'uint256', name: 'value', type: 'uint256' } ], name: 'transfer', outputs: [{ internalType: 'bool', name: '', type: 'bool' }], stateMutability: 'nonpayable', type: 'function' }, { inputs: [ { internalType: 'address', name: 'from', type: 'address' }, { internalType: 'address', name: 'to', type: 'address' }, { internalType: 'uint256', name: 'value', type: 'uint256' } ], name: 'transferFrom', outputs: [{ internalType: 'bool', name: '', type: 'bool' }], stateMutability: 'nonpayable', type: 'function' }, { inputs: [{ internalType: 'address', name: 'newOwner', type: 'address' }], name: 'transferOwnership', outputs: [], stateMutability: 'nonpayable', type: 'function' }, { inputs: [], name: 'unpause', outputs: [], stateMutability: 'nonpayable', type: 'function' } ]
-
Creamos un fichero
index.js
para exportar directamente los ABIs:export * from './blockmakerTokenABI'
-
Creamos el componente Token Balance que mostrará cuántos tokens tiene el usuario que haya conectado la wallet.
-
Usamos los hooks useAccount y useBalance de Wagmi para mostrar el balance de manera eficaz y sencilla.
-
Importamos el componente en nuestra página
Home
para ver que funciona correctamente.import { useAccount, useBalance } from 'wagmi' import { ErrorInfo } from './ui' // Componente Skeleton para mostrar mientras se carga el balance function TokenBalanceSkeleton() { return ( <div className="w-48 sm:w-60 bg-white border shadow p-2.5 rounded-md"> <div className="h-5 bg-gray-300 rounded animate-pulse" /> </div> ) } export default function TokenBalance() { const { address, isConnected } = useAccount() const { data, isLoading } = useBalance({ address, token: import.meta.env.VITE_TOKEN_CONTRACT_ADDRESS, // Ponemos watch para que se actualice el balance automáticamente watch: true }) // Ponemos optional chaining con "?" para evitar errores si aún no tenemos el token en nuestra wallet const tokenBalance = Number(data?.formatted) return isLoading ? ( <TokenBalanceSkeleton /> ) : ( <div className="bg-white border shadow w-fit p-1 gap-2 rounded-md flex items-center text-xs md:text-base"> <p className="bg-zinc-700 text-white p-2 sm:py-1 rounded-l-md">Token Balance:</p> {/* Comprobamos si está conectado para mostrar balance, si no, mostramos el componente ErrorInfo con mensaje */} {isConnected ? ( <p className="p-2 sm:py-1"> {/* Si el número no es entero, fijamos los decimales en 3 */} {Number.isInteger(tokenBalance) ? tokenBalance : tokenBalance.toFixed(3)} {data?.symbol} </p> ) : ( <ErrorInfo message="Conecta tu wallet para ver tu balance de BM" /> )} </div> ) }
-
Creamos el componente Owner que mostrará la dirección que creó el contrato ERC20 del token.
-
Como usaremos un método del contrato, tendremos que importar el ABI que hemos añadido anteriormente.
-
Utilizamos el hook useContractRead para leer el método owner del contrato.
import { useContractRead } from 'wagmi' import { blockmakerTokenABI } from '../contracts/ABIs' import { Title } from './ui' function OwnerSkeleton() { return ( <div className="w-full bg-white border shadow px-3.5 py-5 rounded-md grid gap-2"> <div className="h-5 bg-gray-300 rounded animate-pulse w-20" /> <div className="h-7 bg-gray-300 rounded animate-pulse" /> </div> ) } export default function Owner() { const { data, isLoading } = useContractRead({ address: import.meta.env.VITE_TOKEN_CONTRACT_ADDRESS, // Indicamos el ABI del contrato abi: blockmakerTokenABI, // Indicamos el método del contrato que queremos usar functionName: 'owner' }) return isLoading ? ( <OwnerSkeleton /> ) : ( <section className="flex flex-col p-4 bg-white border shadow rounded-lg w-[360px] sm:w-[469px]"> <Title>Owner</Title> <p className="text-zinc-600 text-xs sm:text-sm bg-gray-100 p-2 rounded-md">{data}</p> </section> ) }
-
Creamos un componente TotalSupply para mostrar el suministro total del token.
-
Como también es un método del contrato, importamos el ABI y se lo pasamos al hook useContractRead.
-
Indicamos que queremos leer el método
totalSupply
. -
Añadimos la propiedad
watch
con valortrue
, ya que el total supply puede ir cambiando y de esta manera nuestro componente se actualizará con el nuevo valor. -
Importamos la función formatEther para formatear el valor que nos devuelve el método a ether.
import { formatEther } from 'viem/utils' import { useContractRead } from 'wagmi' import { blockmakerTokenABI } from '../contracts/ABIs' import { Title } from './ui' function TotalSupplySkeleton() { return ( <div className="w-full bg-white border shadow px-3.5 py-5 rounded-md grid gap-2"> <div className="h-5 bg-gray-300 rounded animate-pulse w-32" /> <div className="h-7 bg-gray-300 rounded animate-pulse" /> </div> ) } export default function TotalSupply() { const { data, isLoading } = useContractRead({ address: import.meta.env.VITE_TOKEN_CONTRACT_ADDRESS, abi: blockmakerTokenABI, functionName: 'totalSupply', watch: true }) return isLoading ? ( <TotalSupplySkeleton /> ) : ( <section className="flex flex-col p-4 bg-white border shadow rounded-lg w-[360px] sm:w-[469px]"> <Title>Total Supply</Title> <p className="text-zinc-600 text-xs sm:text-sm bg-gray-100 p-2 rounded-md ">{formatEther(Number(data))}</p> </section> ) }
-
Para este componente, que contendrá una lista, crearemos primero el componente
TokenInfoItem
. -
Podemos crear los componentes auxiliares en el mismo fichero o tenerlos separados en otros ficheros.
-
Para leer la información del token usaremos el hook de wagmi useToken.
import PropTypes from 'prop-types' export default function TokenInfoItem({ label, value }) { return ( <li className="bg-gray-100 p-2 rounded-md flex flex-col sm:flex-row gap-1"> <span className="text-gray-700 font-semibold">{label}:</span> <p className="text-xs sm:text-sm">{value}</p> </li> ) } TokenInfoItem.propTypes = { label: PropTypes.string.isRequired, value: PropTypes.oneOfType([PropTypes.string, PropTypes.number]) }
import { useToken } from 'wagmi' import { ErrorInfo, Title } from './ui' import TokenInfoItem from './TokenInfoItem' function TokenInfoSkeleton() { return ( <section className="p-4 bg-white border shadow rounded-lg w-[360px] sm:w-[469px]"> <div className="h-6 bg-gray-300 rounded mb-4 w-[156px] animate-pulse" /> <ul className="grid gap-4 animate-pulse"> <li className="h-14 sm:h-9 bg-gray-300 rounded" /> <li className="h-14 sm:h-9 bg-gray-300 rounded" /> <li className="h-14 sm:h-9 bg-gray-300 rounded" /> <li className="h-14 sm:h-9 bg-gray-300 rounded" /> <li className="h-14 sm:h-9 bg-gray-300 rounded" /> </ul> </section> ) } export default function TokenInfo() { const { data, isLoading, isError } = useToken({ address: import.meta.env.VITE_TOKEN_CONTRACT_ADDRESS, watch: true }) if (isLoading) return <TokenInfoSkeleton /> return ( <section className="p-4 bg-white border shadow w-fit rounded-lg text-sm"> <Title>Token Information</Title> {isError ? ( <ErrorInfo message="Error cargando la información del token. Prueba de nuevo más tarde." /> ) : ( <ul className="grid gap-4"> <TokenInfoItem label="Name" value={data.name} /> <TokenInfoItem label="Symbol" value={data.symbol} /> <TokenInfoItem label="Address" value={data.address} /> <TokenInfoItem label="Decimals" value={data.decimals} /> </ul> )} </section> ) }
Este componente se crea con el propósito de permitir la transferencia de tokens. Para lograrlo, se utilizan métodos de escritura del contrato, y el proceso se detalla a continuación:
-
usePrepareContractWrite
: En primer lugar, se hace uso del hook usePrepareContractWrite de Wagmi. Este hook se emplea para preparar los detalles de la transacción, incluyendo la función del contrato, los parámetros necesarios y la dirección del contrato. -
useContractWrite
: En segundo lugar, se utiliza el hook useContractWrite, el cual aprovecha la configuración previamente preparada con usePrepareContractWrite para ejecutar la transacción en la cadena de bloques. -
useWaitForTransaction
: Finalmente, se utiliza el hook useWaitForTransaction para esperar la confirmación de la transacción en la cadena de bloques antes de proceder con otras acciones.import { useEffect, useState } from 'react' import { toast } from 'react-hot-toast' import { useContractWrite, usePrepareContractWrite, useWaitForTransaction } from 'wagmi' import { blockmakerTokenABI } from '../contracts/ABIs' import { Button, TextInput, Title } from './ui' export default function TransferTokenForm() { const [to, setTo] = useState('') const [amount, setAmount] = useState('') const { config } = usePrepareContractWrite({ address: import.meta.env.VITE_TOKEN_CONTRACT_ADDRESS, abi: blockmakerTokenABI, functionName: 'transfer', enabled: to && amount > 0, args: [to, BigInt(amount * 10 ** 18)] }) const { data: writeData, write } = useContractWrite(config) const { isLoading: isTransactionLoading, isSuccess: isTransactionSuccess, isError: isTransactionError } = useWaitForTransaction({ hash: writeData?.hash }) const handleToInputChange = (e) => { setTo(e.target.value) } const handleAmountInputChange = (e) => { setAmount(e.target.value) } useEffect(() => { if (isTransactionSuccess) { toast.success('Se han transferido los tokens con éxito.') setTo('') setAmount('') } if (isTransactionError) { toast.error('No se ha podido realizar la transacción. Prueba de nuevo más tarde.') } }, [isTransactionSuccess, isTransactionError]) return ( <section className="p-4 bg-white border shadow rounded-lg text-sm w-[360px] sm:w-[469px]"> <div className="flex gap-1"> <Title>Transfer</Title> </div> <form className="grid gap-4"> <TextInput type="text" placeholder="To" value={to} disabled={isTransactionLoading} onChange={handleToInputChange} /> <TextInput type="number" placeholder="Amount" value={amount} disabled={isTransactionLoading} onChange={handleAmountInputChange} /> <Button disabled={!to || !amount || isTransactionLoading} isLoading={isTransactionLoading} onClick={() => write?.()} > {isTransactionLoading ? 'Transfiriendo BM Tokens...' : 'Transferir BM Tokens'} </Button> </form> </section> ) }
Este componente se crea con el propósito de permitir la quema de tokens para el creador del contrato. Para lograrlo, se utilizan métodos de escritura del contrato, y el proceso se detalla a continuación:
-
usePrepareContractWrite
: En primer lugar, se hace uso del hook usePrepareContractWrite de Wagmi. Este hook se emplea para preparar los detalles de la transacción, incluyendo la función del contrato, los parámetros necesarios y la dirección del contrato. -
useContractWrite
: En segundo lugar, se utiliza el hook useContractWrite, el cual aprovecha la configuración previamente preparada con usePrepareContractWrite para ejecutar la transacción en la cadena de bloques. -
useWaitForTransaction
: Finalmente, se utiliza el hook useWaitForTransaction para esperar la confirmación de la transacción en la cadena de bloques antes de proceder con otras acciones.import { useEffect, useState } from 'react' import { toast } from 'react-hot-toast' import { useAccount, useContractRead, useContractWrite, usePrepareContractWrite, useWaitForTransaction } from 'wagmi' import { blockmakerTokenABI } from '../contracts/ABIs' import { Button, TextInput, Title } from './ui' export default function BurnTokensForm() { const [amount, setAmount] = useState('') const { address } = useAccount() const { data } = useContractRead({ address: import.meta.env.VITE_TOKEN_CONTRACT_ADDRESS, abi: blockmakerTokenABI, functionName: 'owner' }) // Comprobamos si la cuenta que está conectada es la misma que la del owner const isOwner = address === data const { config } = usePrepareContractWrite({ address: import.meta.env.VITE_TOKEN_CONTRACT_ADDRESS, abi: blockmakerTokenABI, functionName: 'burn', enabled: amount > 0, args: [BigInt(amount * 10 ** 18)] }) const { data: writeData, write } = useContractWrite(config) const { isLoading: isTransactionLoading, isSuccess: isTransactionSuccess, isError: isTransactionError } = useWaitForTransaction({ hash: writeData?.hash }) const handleAmountInputChange = (e) => { setAmount(e.target.value) } useEffect(() => { if (isTransactionSuccess) { toast.success('Quema de tokens realizada con éxito!🔥') setAmount('') } if (isTransactionError) { toast.error('No se ha podido realizar la quema de tokens. Prueba de nuevo más tarde.') } }, [isTransactionSuccess, isTransactionError]) return ( <section className="p-4 bg-white border shadow rounded-lg text-sm w-[360px] sm:w-[469px]"> <div className="flex gap-1"> <Title>Burn</Title> <p className="text-zinc-400 p-1">(Only Owner)</p> </div> <form className="grid gap-4"> <TextInput type="number" placeholder="Amount" value={amount} onChange={handleAmountInputChange} disabled={!isOwner || isTransactionLoading} /> <Button disabled={!amount || isTransactionLoading || !isOwner} isLoading={isTransactionLoading} onClick={() => write?.()} > {isOwner ? isTransactionLoading ? 'Burning BM Tokens...' : 'Burn BM Tokens' : 'Only Owner Can Burn Tokens'} </Button> </form> </section> ) }
-
Creamos un índice para exportar todos los componentes de la carpeta components
export { default as BurnTokensForm } from './BurnTokensForm' export { default as Footer } from './Footer' export { default as Header } from './Header' export { default as MintTokensForm } from './MintTokensForm' export { default as Owner } from './Owner' export { default as TokenBalance } from './TokenBalance' export { default as TokenInfo } from './TokenInfo' export { default as TokenInfoItem } from './TokenInfoItem' export { default as TokenInfoSkeleton } from './TokenInfoSkeleton' export { default as TotalSupply } from './TotalSupply' export { default as TransferTokensForm } from './TransferTokensForm'
-
Una vez ya tenemos todos nuestros componentes necesarios podemos terminar de diseñar nuestra página principal en el componente Home.
import { ConnectKitButton } from 'connectkit' import { useAccount } from 'wagmi' import { BurnTokensForm, MintTokensForm, Owner, TokenBalance, TokenInfo, TotalSupply, TransferTokensForm } from '../components' export default function Home() { const { isConnected } = useAccount() return ( <div className="flex flex-col self-center sm:grid place-items-center px-3 py-16 md:px-5 gap-12"> <h1 className="font-bold text-3xl sm:text-5xl md:text-6xl mb-2 bg-gradient-to-r from-pink-600 to-indigo-500 text-transparent bg-clip-text"> Blockmaker ERC20 Token </h1> {isConnected ? ( <> <TokenBalance /> <div className="flex flex-col xl:flex-row gap-4"> <div className="grid gap-4 h-fit"> <Owner /> <TotalSupply /> <TokenInfo /> </div> <div className="flex flex-col gap-4 justify-center"> <TransferTokensForm /> <MintTokensForm /> <BurnTokensForm /> </div> </div> </> ) : ( <> <p className="text-gray-500 md:text-xl text-center "> Un token revolucionario en la blockchain, diseñado para empoderar a los usuarios y desarrolladores. <br /> Blockmaker ERC20 ofrece transacciones rápidas, bajas tarifas y una plataforma segura. <br /> Ideal para proyectos de DeFi, juegos en blockchain y mucho más. </p> <p className="text-xl sm:text-2xl">🔒 Conecta tu wallet para comenzar.</p> <ConnectKitButton /> </> )} </div> ) }
Para desplegar un proyecto de React en Vercel, sigue estos pasos:
-
Preparación del Proyecto de React:
- Asegúrate de que tu proyecto de React esté funcionando correctamente en tu entorno local.
- Si aún no lo has hecho, inicia un repositorio Git en tu proyecto (
git init
) y haz un commit de tus archivos (git add .
seguido degit commit -m "Primer commit"
).
-
Creación de una cuenta en Vercel:
- Si aún no tienes una cuenta en Vercel, ve a vercel.com y regístrate. Puedes usar tu cuenta de GitHub, GitLab o Bitbucket para un acceso más fácil.
-
Conexión del Repositorio con Vercel:
- Una vez que hayas iniciado sesión en Vercel, selecciona "New Project".
- Vercel te pedirá que conectes con tu proveedor de repositorios (GitHub, GitLab, Bitbucket). Sigue las instrucciones para conectar tu cuenta.
- Después de conectar tu cuenta, selecciona el repositorio que contiene tu proyecto de React.
-
Configuración del Proyecto en Vercel:
- Vercel detectará automáticamente que es un proyecto de React y establecerá algunas configuraciones predeterminadas.
- Si necesitas configuraciones específicas (variables de entorno, por ejemplo), puedes añadirlas en este paso.
-
Despliegue del Proyecto:
- Una vez que hayas configurado tu proyecto, haz clic en "Deploy".
- Vercel comenzará el proceso de despliegue. Este proceso incluye la instalación de dependencias, la construcción del proyecto y finalmente la publicación del sitio.
- Puedes seguir el progreso del despliegue en el dashboard de Vercel.
-
Acceso al Proyecto Desplegado:
- Una vez completado el despliegue, Vercel proporcionará una URL única para acceder a tu proyecto.
- Puedes visitar esta URL para ver tu aplicación de React en vivo.
-
Actualizaciones Futuras:
- Para actualizar tu proyecto en Vercel, simplemente haz cambios en tu código, haz un commit y luego un push a tu repositorio.
- Vercel detectará automáticamente los cambios y desplegará la nueva versión de tu proyecto.
- URL de la Aplicación: https://erc20-web3-dapp.vercel.app/