Разработка лендинга токенсейла
Лендинг токенсейла — это не просто маркетинговая страница. Это web3-приложение, которое взаимодействует со смарт-контрактом продажи токенов, обрабатывает платежи в криптовалюте, управляет whitelist, и должно работать надёжно в момент высокой нагрузки (когда тысячи пользователей заходят одновременно при открытии раунда).
Смарт-контракт токенсейла
Лендинг — это frontend к контракту. Сначала проектируем контракт.
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.24;
import "@openzeppelin/contracts/token/ERC20/IERC20.sol";
import "@openzeppelin/contracts/token/ERC20/utils/SafeERC20.sol";
import "@openzeppelin/contracts/access/Ownable2Step.sol";
import "@openzeppelin/contracts/utils/ReentrancyGuard.sol";
import "@openzeppelin/contracts/utils/Pausable.sol";
import "@openzeppelin/contracts/utils/cryptography/MerkleProof.sol";
contract TokenSale is Ownable2Step, ReentrancyGuard, Pausable {
using SafeERC20 for IERC20;
IERC20 public immutable saleToken;
IERC20 public immutable paymentToken; // USDC
uint256 public immutable tokenPrice; // USDC per token, 6 decimals
uint256 public immutable hardCap; // total tokens for sale
uint256 public immutable minPurchase; // per wallet min
uint256 public immutable maxPurchase; // per wallet max
uint256 public saleStart;
uint256 public saleEnd;
bytes32 public whitelistMerkleRoot;
bool public whitelistRequired;
uint256 public totalSold;
mapping(address => uint256) public purchased;
event TokensPurchased(address indexed buyer, uint256 usdcAmount, uint256 tokenAmount);
event SaleConfigured(uint256 start, uint256 end, bool whitelistRequired);
constructor(
address _saleToken,
address _paymentToken,
uint256 _tokenPrice,
uint256 _hardCap,
uint256 _minPurchase,
uint256 _maxPurchase,
address _owner
) Ownable2Step() {
saleToken = IERC20(_saleToken);
paymentToken = IERC20(_paymentToken);
tokenPrice = _tokenPrice;
hardCap = _hardCap;
minPurchase = _minPurchase;
maxPurchase = _maxPurchase;
_transferOwnership(_owner);
}
function configureSale(
uint256 _start,
uint256 _end,
bytes32 _merkleRoot,
bool _whitelistRequired
) external onlyOwner {
saleStart = _start;
saleEnd = _end;
whitelistMerkleRoot = _merkleRoot;
whitelistRequired = _whitelistRequired;
emit SaleConfigured(_start, _end, _whitelistRequired);
}
function buy(
uint256 usdcAmount,
bytes32[] calldata merkleProof
) external nonReentrant whenNotPaused {
require(block.timestamp >= saleStart, "Sale not started");
require(block.timestamp <= saleEnd, "Sale ended");
if (whitelistRequired) {
bytes32 leaf = keccak256(abi.encodePacked(msg.sender));
require(
MerkleProof.verify(merkleProof, whitelistMerkleRoot, leaf),
"Not whitelisted"
);
}
uint256 tokenAmount = usdcAmount * 10**18 / tokenPrice;
require(usdcAmount >= minPurchase, "Below min purchase");
require(purchased[msg.sender] + tokenAmount <= maxPurchase, "Exceeds max per wallet");
require(totalSold + tokenAmount <= hardCap, "Hard cap reached");
purchased[msg.sender] += tokenAmount;
totalSold += tokenAmount;
paymentToken.safeTransferFrom(msg.sender, address(this), usdcAmount);
saleToken.safeTransfer(msg.sender, tokenAmount);
emit TokensPurchased(msg.sender, usdcAmount, tokenAmount);
}
function withdrawFunds(address to) external onlyOwner {
uint256 balance = paymentToken.balanceOf(address(this));
paymentToken.safeTransfer(to, balance);
}
function withdrawUnsoldTokens(address to) external onlyOwner {
require(block.timestamp > saleEnd, "Sale not ended");
uint256 balance = saleToken.balanceOf(address(this));
saleToken.safeTransfer(to, balance);
}
}
Merkle Tree whitelist: вместо хранения каждого адреса on-chain (дорого), храним только merkle root. Proof генерируется off-chain и передаётся при покупке. Для 10,000 адресов в whitelist — экономия ~$1000+ на gas при деплое.
Frontend: Web3 интеграция
Подключение кошелька и состояние продажи
import { useReadContracts, useWriteContract, useWaitForTransactionReceipt } from "wagmi";
import { formatUnits, parseUnits } from "viem";
const SALE_ABI = [...] as const;
function useSaleData(saleAddress: `0x${string}`) {
const result = useReadContracts({
contracts: [
{ address: saleAddress, abi: SALE_ABI, functionName: "saleStart" },
{ address: saleAddress, abi: SALE_ABI, functionName: "saleEnd" },
{ address: saleAddress, abi: SALE_ABI, functionName: "totalSold" },
{ address: saleAddress, abi: SALE_ABI, functionName: "hardCap" },
{ address: saleAddress, abi: SALE_ABI, functionName: "tokenPrice" },
{ address: saleAddress, abi: SALE_ABI, functionName: "whitelistRequired" },
],
query: { refetchInterval: 10_000 }, // обновляем каждые 10 сек
});
const [start, end, totalSold, hardCap, tokenPrice, whitelistRequired] =
result.data ?? [];
const now = Date.now() / 1000;
const saleStatus = !start?.result ? "loading" :
now < Number(start.result) ? "upcoming" :
now > Number(end?.result) ? "ended" :
"active";
const progress = totalSold?.result && hardCap?.result
? Number(totalSold.result * 100n / hardCap.result)
: 0;
return { saleStatus, progress, tokenPrice: tokenPrice?.result, whitelistRequired: whitelistRequired?.result };
}
Покупка с апрувом USDC
USDC требует approve перед buy. Паттерн: сначала проверяем allowance, если недостаточно — первый шаг approve, второй шаг — buy:
function BuyFlow({ saleAddress, usdcAddress, amount }) {
const { address } = useAccount();
const [step, setStep] = useState<"approve" | "buy" | "done">("approve");
const { data: allowance } = useReadContract({
address: usdcAddress,
abi: ERC20_ABI,
functionName: "allowance",
args: [address, saleAddress],
query: { enabled: !!address },
});
const needsApprove = !allowance || allowance < parseUnits(amount, 6);
const { writeContract: approve, data: approveTx } = useWriteContract();
const { writeContract: buy, data: buyTx } = useWriteContract();
const { isSuccess: approveSuccess } = useWaitForTransactionReceipt({ hash: approveTx });
useEffect(() => {
if (approveSuccess) setStep("buy");
}, [approveSuccess]);
// Merkle proof для whitelist (если нужен)
const merkleProof = useMerkleProof(address);
const handleApprove = () => {
approve({
address: usdcAddress,
abi: ERC20_ABI,
functionName: "approve",
args: [saleAddress, parseUnits(amount, 6)],
});
};
const handleBuy = () => {
buy({
address: saleAddress,
abi: SALE_ABI,
functionName: "buy",
args: [parseUnits(amount, 6), merkleProof ?? []],
});
};
if (needsApprove && step === "approve") {
return <Button onClick={handleApprove}>Разрешить USDC (шаг 1/2)</Button>;
}
return <Button onClick={handleBuy}>Купить токены (шаг 2/2)</Button>;
}
Merkle Tree whitelist: генерация и управление
import { MerkleTree } from "merkletreejs";
import { keccak256, encodePacked } from "viem";
function generateMerkleTree(addresses: string[]) {
const leaves = addresses.map((addr) =>
keccak256(encodePacked(["address"], [addr as `0x${string}`]))
);
const tree = new MerkleTree(leaves, keccak256, { sortPairs: true });
const root = tree.getHexRoot();
return { tree, root };
}
function getMerkleProof(tree: MerkleTree, address: string): `0x${string}`[] {
const leaf = keccak256(encodePacked(["address"], [address as `0x${string}`]));
return tree.getHexProof(leaf) as `0x${string}`[];
}
// API эндпоинт: GET /api/whitelist/proof?address=0x...
async function getProofForAddress(req, res) {
const { address } = req.query;
const whitelist = await loadWhitelistFromDB(); // ваша логика
const { tree } = generateMerkleTree(whitelist);
const proof = getMerkleProof(tree, address);
if (proof.length === 0) {
return res.status(403).json({ error: "Not whitelisted" });
}
res.json({ proof, address });
}
Real-time обновления и очередь
При открытии раунда — нагрузочный спайк. Frontend не должен делать запрос к ноде каждую секунду для тысяч пользователей.
Решение: WebSocket или SSE от backend → клиенты. Backend подписывается на события контракта, при TokensPurchased пушит обновлённые данные всем подключённым клиентам.
// Backend: pusher или собственный WebSocket
import { WebSocket } from "ws";
const wss = new WebSocket.Server({ port: 3001 });
const clients = new Set<WebSocket>();
// Слушаем события контракта
publicClient.watchContractEvent({
address: SALE_ADDRESS,
abi: SALE_ABI,
eventName: "TokensPurchased",
onLogs: async (logs) => {
const totalSold = await publicClient.readContract({
address: SALE_ADDRESS,
abi: SALE_ABI,
functionName: "totalSold",
});
const update = JSON.stringify({ type: "sale_update", totalSold: totalSold.toString() });
clients.forEach((client) => {
if (client.readyState === WebSocket.OPEN) client.send(update);
});
},
});
Важные UX детали
Таймер до старта: обратный отсчёт до saleStart. Не используйте серверное время — синхронизируйте с block.timestamp через контракт.
Progress bar: totalSold / hardCap * 100%. Обновляется в реальном времени через WebSocket.
Расчёт суммы: пользователь вводит USDC — показываем сколько токенов получит, и наоборот. Реальная цена из контракта, не захардкоженная.
Gasless approve через Permit: если токен поддерживает EIP-2612, можно объединить approve + buy в одну транзакцию через permit + buy паттерн — улучшает UX.
Mobile responsive: большинство крипто-пользователей покупают с телефона. Кнопки крупные, MetaMask Mobile deep link.
Стек
| Компонент | Технология |
|---|---|
| Frontend | Next.js 14 + TypeScript |
| Web3 | wagmi v2 + viem + RainbowKit |
| Whitelist | Merkle Tree + API endpoint |
| Real-time | WebSocket или Pusher |
| Analytics | собственные events + Dune Dashboard |
| Hosting | Vercel + Cloudflare |
Срок разработки: 2–3 недели для full-stack лендинга со смарт-контрактом, whitelist, real-time updates. Дизайн и маркетинговый контент — отдельно.







