// SPDX-License-Identifier: MIT pragma solidity ^0.8.15; import "@openzeppelin/contracts/utils/cryptography/MerkleProof.sol"; import "@openzeppelin/contracts/token/ERC20/IERC20.sol"; import "./ITieredSalesInternal.sol"; import "./TieredSalesStorage.sol"; import "../../access/ownable/OwnableInternal.sol"; /** * @title Sales mechanism for NFTs with multiple tiered pricing, allowlist and allocation plans */ abstract contract TieredSalesInternal is ITieredSalesInternal, Context, OwnableInternal { using TieredSalesStorage for TieredSalesStorage.Layout; function _configureTiering(uint256 tierId, Tier calldata tier) internal virtual { TieredSalesStorage.Layout storage l = TieredSalesStorage.layout(); require(tier.maxAllocation >= l.tierMints[tierId], "LOWER_THAN_MINTED"); if (l.tiers[tierId].reserved > 0) { require(tier.reserved >= l.tierMints[tierId], "LOW_RESERVE_AMOUNT"); } if (l.tierMints[tierId] > 0) { require(tier.maxPerWallet >= l.tiers[tierId].maxPerWallet, "LOW_MAX_PER_WALLET"); } l.totalReserved -= l.tiers[tierId].reserved; l.tiers[tierId] = tier; l.totalReserved += tier.reserved; } function _configureTiering(uint256[] calldata _tierIds, Tier[] calldata _tiers) internal virtual { for (uint256 i = 0; i < _tierIds.length; i++) { _configureTiering(_tierIds[i], _tiers[i]); } } function _onTierAllowlist( uint256 tierId, address minter, uint256 maxAllowance, bytes32[] calldata proof ) internal view virtual returns (bool) { return MerkleProof.verify( proof, TieredSalesStorage.layout().tiers[tierId].merkleRoot, _generateMerkleLeaf(minter, maxAllowance) ); } function _eligibleForTier( uint256 tierId, address minter, uint256 maxAllowance, bytes32[] calldata proof ) internal view virtual returns (uint256 maxMintable) { TieredSalesStorage.Layout storage l = TieredSalesStorage.layout(); require(l.tiers[tierId].maxPerWallet > 0, "NOT_EXISTS"); require(block.timestamp >= l.tiers[tierId].start, "NOT_STARTED"); require(block.timestamp <= l.tiers[tierId].end, "ALREADY_ENDED"); maxMintable = l.tiers[tierId].maxPerWallet - l.walletMinted[tierId][minter]; if (l.tiers[tierId].merkleRoot != bytes32(0)) { require(l.walletMinted[tierId][minter] < maxAllowance, "MAXED_ALLOWANCE"); require(_onTierAllowlist(tierId, minter, maxAllowance, proof), "NOT_ALLOWLISTED"); uint256 remainingAllowance = maxAllowance - l.walletMinted[tierId][minter]; if (maxMintable > remainingAllowance) { maxMintable = remainingAllowance; } } } function _availableSupplyForTier(uint256 tierId) internal view virtual returns (uint256 remaining) { TieredSalesStorage.Layout storage l = TieredSalesStorage.layout(); // Substract all the remaining reserved spots from the total remaining supply... remaining = _remainingSupply(tierId) - (l.totalReserved - l.reservedMints); // If this tier has reserved spots, add remaining spots back to result... if (l.tiers[tierId].reserved > 0) { remaining += (l.tiers[tierId].reserved - l.tierMints[tierId]); } } function _executeSale( uint256 tierId, uint256 count, uint256 maxAllowance, bytes32[] calldata proof ) internal virtual { address minter = _msgSender(); uint256 maxMintable = _eligibleForTier(tierId, minter, maxAllowance, proof); TieredSalesStorage.Layout storage l = TieredSalesStorage.layout(); require(count <= maxMintable, "EXCEEDS_MAX"); require(count <= _availableSupplyForTier(tierId), "EXCEEDS_SUPPLY"); require(count + l.tierMints[tierId] <= l.tiers[tierId].maxAllocation, "EXCEEDS_ALLOCATION"); if (l.tiers[tierId].currency == address(0)) { require(l.tiers[tierId].price * count <= msg.value, "INSUFFICIENT_AMOUNT"); } else { IERC20(l.tiers[tierId].currency).transferFrom(minter, address(this), l.tiers[tierId].price * count); } l.walletMinted[tierId][minter] += count; l.tierMints[tierId] += count; if (l.tiers[tierId].reserved > 0) { l.reservedMints += count; } } function _remainingSupply( uint256 /*tierId*/ ) internal view virtual returns (uint256) { // By default assume supply is unlimited (that means reserving allocation for tiers is irrelevant) return type(uint256).max; } /* PRIVATE */ function _generateMerkleLeaf(address account, uint256 maxAllowance) private pure returns (bytes32) { return keccak256(abi.encodePacked(account, maxAllowance)); } }