编译器
0.8.23+commit.f704f362
文件 1 的 7:ERC721.sol
pragma solidity >=0.8.0;
abstract contract ERC721 {
event Transfer(address indexed from, address indexed to, uint256 indexed id);
event Approval(address indexed owner, address indexed spender, uint256 indexed id);
event ApprovalForAll(address indexed owner, address indexed operator, bool approved);
string public name;
string public symbol;
function tokenURI(uint256 id) public view virtual returns (string memory);
mapping(uint256 => address) internal _ownerOf;
mapping(address => uint256) internal _balanceOf;
function ownerOf(uint256 id) public view virtual returns (address owner) {
require((owner = _ownerOf[id]) != address(0), "NOT_MINTED");
}
function balanceOf(address owner) public view virtual returns (uint256) {
require(owner != address(0), "ZERO_ADDRESS");
return _balanceOf[owner];
}
mapping(uint256 => address) public getApproved;
mapping(address => mapping(address => bool)) public isApprovedForAll;
constructor(string memory _name, string memory _symbol) {
name = _name;
symbol = _symbol;
}
function approve(address spender, uint256 id) public virtual {
address owner = _ownerOf[id];
require(msg.sender == owner || isApprovedForAll[owner][msg.sender], "NOT_AUTHORIZED");
getApproved[id] = spender;
emit Approval(owner, spender, id);
}
function setApprovalForAll(address operator, bool approved) public virtual {
isApprovedForAll[msg.sender][operator] = approved;
emit ApprovalForAll(msg.sender, operator, approved);
}
function transferFrom(
address from,
address to,
uint256 id
) public virtual {
require(from == _ownerOf[id], "WRONG_FROM");
require(to != address(0), "INVALID_RECIPIENT");
require(
msg.sender == from || isApprovedForAll[from][msg.sender] || msg.sender == getApproved[id],
"NOT_AUTHORIZED"
);
unchecked {
_balanceOf[from]--;
_balanceOf[to]++;
}
_ownerOf[id] = to;
delete getApproved[id];
emit Transfer(from, to, id);
}
function safeTransferFrom(
address from,
address to,
uint256 id
) public virtual {
transferFrom(from, to, id);
require(
to.code.length == 0 ||
ERC721TokenReceiver(to).onERC721Received(msg.sender, from, id, "") ==
ERC721TokenReceiver.onERC721Received.selector,
"UNSAFE_RECIPIENT"
);
}
function safeTransferFrom(
address from,
address to,
uint256 id,
bytes calldata data
) public virtual {
transferFrom(from, to, id);
require(
to.code.length == 0 ||
ERC721TokenReceiver(to).onERC721Received(msg.sender, from, id, data) ==
ERC721TokenReceiver.onERC721Received.selector,
"UNSAFE_RECIPIENT"
);
}
function supportsInterface(bytes4 interfaceId) public view virtual returns (bool) {
return
interfaceId == 0x01ffc9a7 ||
interfaceId == 0x80ac58cd ||
interfaceId == 0x5b5e139f;
}
function _mint(address to, uint256 id) internal virtual {
require(to != address(0), "INVALID_RECIPIENT");
require(_ownerOf[id] == address(0), "ALREADY_MINTED");
unchecked {
_balanceOf[to]++;
}
_ownerOf[id] = to;
emit Transfer(address(0), to, id);
}
function _burn(uint256 id) internal virtual {
address owner = _ownerOf[id];
require(owner != address(0), "NOT_MINTED");
unchecked {
_balanceOf[owner]--;
}
delete _ownerOf[id];
delete getApproved[id];
emit Transfer(owner, address(0), id);
}
function _safeMint(address to, uint256 id) internal virtual {
_mint(to, id);
require(
to.code.length == 0 ||
ERC721TokenReceiver(to).onERC721Received(msg.sender, address(0), id, "") ==
ERC721TokenReceiver.onERC721Received.selector,
"UNSAFE_RECIPIENT"
);
}
function _safeMint(
address to,
uint256 id,
bytes memory data
) internal virtual {
_mint(to, id);
require(
to.code.length == 0 ||
ERC721TokenReceiver(to).onERC721Received(msg.sender, address(0), id, data) ==
ERC721TokenReceiver.onERC721Received.selector,
"UNSAFE_RECIPIENT"
);
}
}
abstract contract ERC721TokenReceiver {
function onERC721Received(
address,
address,
uint256,
bytes calldata
) external virtual returns (bytes4) {
return ERC721TokenReceiver.onERC721Received.selector;
}
}
文件 2 的 7:IANGAC.sol
pragma solidity ^0.8.23;
import "@solmate/tokens/ERC721.sol";
import "@solmate/auth/Owned.sol";
import "@solmate/utils/ReentrancyGuard.sol";
import "@solmate/utils/LibString.sol";
import "@openzeppelin/contracts/interfaces/IERC2981.sol";
interface IHypercards {
function getUserTokens(
address user
) external view returns (uint256[] memory);
function balanceOf(
address account,
uint256 id
) external view returns (uint256);
}
contract IANGAC is ERC721, Owned, ReentrancyGuard {
IHypercards public hyperCardsContract;
address public payoutAddress;
uint256 public constant FULL_PRICE = 0.015 ether;
uint256 public constant DISCOUNTED_PRICE = 0.01 ether;
uint256 public constant CARDS_FOR_DISCOUNT = 4;
uint256 public constant MAX_SUPPLY = 10000;
uint256 public constant ROYALTY_PERCENTAGE = 500;
string private _baseURI;
uint256 public totalSupply;
mapping(address => mapping(uint256 => bool))
public usedHyperCardForDiscount;
event IANGACMinted(address indexed to, uint256 tokenId, bool discounted);
event HyperCardsUsedForDiscount(address indexed user, uint256[] cardIds);
error InsufficientPayment();
error UserDoesNotOwnHyperCard();
error NotEnoughHyperCardsForDiscount(uint256 required, uint256 available);
error NotEligibleForDiscountedIANGAC();
error HyperCardAlreadyUsedForDiscount();
error MaxSupplyReached();
error InvalidTokenId();
modifier withinMaxSupply() {
if (totalSupply >= MAX_SUPPLY) revert MaxSupplyReached();
_;
}
constructor(
address _hyperCardsAddress,
string memory baseURI,
address _payoutAddress
) ERC721("IANGAC", "IANGAC") Owned(msg.sender) {
hyperCardsContract = IHypercards(_hyperCardsAddress);
_baseURI = baseURI;
payoutAddress = _payoutAddress;
}
function setHyperCardsAddress(
address _hyperCardsAddress
) external onlyOwner {
hyperCardsContract = IHypercards(_hyperCardsAddress);
}
function setPayoutAddress(address _newPayoutAddress) external onlyOwner {
payoutAddress = _newPayoutAddress;
}
function getPotentialDiscountEligibility(
address user,
uint256[] memory ownedHyperCardIds
) public view returns (bool) {
uint256 eligibleCards = 0;
for (uint256 i = 0; i < ownedHyperCardIds.length; i++) {
uint256 cardId = ownedHyperCardIds[i];
if (hyperCardsContract.balanceOf(user, cardId) == 0) {
return false;
}
if (!usedHyperCardForDiscount[user][cardId]) {
eligibleCards++;
}
}
return eligibleCards >= CARDS_FOR_DISCOUNT;
}
function mintIANGAC(
uint256 tokenId
) external payable withinMaxSupply nonReentrant {
if (tokenId < 1 || tokenId > MAX_SUPPLY) revert InvalidTokenId();
if (msg.value < FULL_PRICE) revert InsufficientPayment();
_mint(msg.sender, tokenId);
totalSupply++;
emit IANGACMinted(msg.sender, tokenId, false);
if (msg.value > FULL_PRICE) {
payable(msg.sender).transfer(msg.value - FULL_PRICE);
}
}
function mintDiscountedIANGAC(
uint256 tokenId
) external payable withinMaxSupply nonReentrant {
if (tokenId < 1 || tokenId > MAX_SUPPLY) revert InvalidTokenId();
uint256[] memory hyperCardIds = getEligibleTokenIds(msg.sender);
if (hyperCardIds.length < CARDS_FOR_DISCOUNT)
revert NotEnoughHyperCardsForDiscount(
CARDS_FOR_DISCOUNT,
hyperCardIds.length
);
if (msg.value < DISCOUNTED_PRICE) revert InsufficientPayment();
for (uint256 i = 0; i < CARDS_FOR_DISCOUNT; i++) {
uint256 cardId = hyperCardIds[i];
if (usedHyperCardForDiscount[msg.sender][cardId])
revert HyperCardAlreadyUsedForDiscount();
}
bool isEligible = getPotentialDiscountEligibility(
msg.sender,
hyperCardIds
);
if (!isEligible) revert NotEligibleForDiscountedIANGAC();
for (uint256 i = 0; i < CARDS_FOR_DISCOUNT; i++) {
uint256 cardId = hyperCardIds[i];
usedHyperCardForDiscount[msg.sender][cardId] = true;
}
_mint(msg.sender, tokenId);
totalSupply++;
emit IANGACMinted(msg.sender, tokenId, true);
emit HyperCardsUsedForDiscount(msg.sender, hyperCardIds);
if (msg.value > DISCOUNTED_PRICE) {
payable(msg.sender).transfer(msg.value - DISCOUNTED_PRICE);
}
}
function withdraw() external onlyOwner {
uint256 balance = address(this).balance;
payable(payoutAddress).transfer(balance);
}
function tokenURI(uint256 id) public view override returns (string memory) {
return string(abi.encodePacked(_baseURI, LibString.toString(id)));
}
function royaltyInfo(
uint256 tokenId,
uint256 salePrice
) external view returns (address receiver, uint256 royaltyAmount) {
uint256 royalty = (salePrice * ROYALTY_PERCENTAGE) / 10000;
return (payoutAddress, royalty);
}
function supportsInterface(
bytes4 interfaceId
) public view virtual override returns (bool) {
return
interfaceId == 0x01ffc9a7 ||
interfaceId == 0x80ac58cd ||
interfaceId == 0x5b5e139f ||
interfaceId == type(IERC2981).interfaceId;
}
function getEligibleTokenIds(
address user
) public view returns (uint256[] memory) {
uint256[] memory userTokens = hyperCardsContract.getUserTokens(user);
uint256[] memory tempEligibleTokens = new uint256[](userTokens.length);
uint256 eligibleCount = 0;
for (uint256 i = 0; i < userTokens.length; i++) {
uint256 cardId = userTokens[i];
if (
hyperCardsContract.balanceOf(user, cardId) > 0 &&
!usedHyperCardForDiscount[user][cardId]
) {
tempEligibleTokens[eligibleCount] = cardId;
eligibleCount++;
}
}
uint256[] memory eligibleTokens = new uint256[](eligibleCount);
for (uint256 i = 0; i < eligibleCount; i++) {
eligibleTokens[i] = tempEligibleTokens[i];
}
return eligibleTokens;
}
function getAvailableDiscountedMints(
address user
) public view returns (uint256) {
uint256[] memory eligibleTokens = getEligibleTokenIds(user);
return eligibleTokens.length / CARDS_FOR_DISCOUNT;
}
function isTokenMinted(uint256 tokenId) public view returns (bool) {
return _ownerOf[tokenId] != address(0);
}
}
文件 3 的 7:IERC165.sol
pragma solidity ^0.8.20;
interface IERC165 {
function supportsInterface(bytes4 interfaceId) external view returns (bool);
}
文件 4 的 7:IERC2981.sol
pragma solidity ^0.8.20;
import {IERC165} from "../utils/introspection/IERC165.sol";
interface IERC2981 is IERC165 {
function royaltyInfo(
uint256 tokenId,
uint256 salePrice
) external view returns (address receiver, uint256 royaltyAmount);
}
文件 5 的 7:LibString.sol
pragma solidity >=0.8.0;
library LibString {
function toString(int256 value) internal pure returns (string memory str) {
if (value >= 0) return toString(uint256(value));
unchecked {
str = toString(uint256(-value));
assembly {
let length := mload(str)
mstore(str, 45)
str := sub(str, 1)
mstore(str, add(length, 1))
}
}
}
function toString(uint256 value) internal pure returns (string memory str) {
assembly {
let newFreeMemoryPointer := add(mload(0x40), 160)
mstore(0x40, newFreeMemoryPointer)
str := sub(newFreeMemoryPointer, 32)
mstore(str, 0)
let end := str
for { let temp := value } 1 {} {
str := sub(str, 1)
mstore8(str, add(48, mod(temp, 10)))
temp := div(temp, 10)
if iszero(temp) { break }
}
let length := sub(end, str)
str := sub(str, 32)
mstore(str, length)
}
}
}
文件 6 的 7:Owned.sol
pragma solidity >=0.8.0;
abstract contract Owned {
event OwnershipTransferred(address indexed user, address indexed newOwner);
address public owner;
modifier onlyOwner() virtual {
require(msg.sender == owner, "UNAUTHORIZED");
_;
}
constructor(address _owner) {
owner = _owner;
emit OwnershipTransferred(address(0), _owner);
}
function transferOwnership(address newOwner) public virtual onlyOwner {
owner = newOwner;
emit OwnershipTransferred(msg.sender, newOwner);
}
}
文件 7 的 7:ReentrancyGuard.sol
pragma solidity >=0.8.0;
abstract contract ReentrancyGuard {
uint256 private locked = 1;
modifier nonReentrant() virtual {
require(locked == 1, "REENTRANCY");
locked = 2;
_;
locked = 1;
}
}
{
"compilationTarget": {
"src/IANGAC.sol": "IANGAC"
},
"evmVersion": "paris",
"libraries": {},
"metadata": {
"bytecodeHash": "ipfs"
},
"optimizer": {
"enabled": true,
"runs": 200
},
"remappings": [
":@openzeppelin/=lib/openzeppelin-contracts/",
":@openzeppelin/contracts/=lib/openzeppelin-contracts/contracts/",
":@solmate/=lib/solmate/src/",
":ds-test/=lib/solmate/lib/ds-test/src/",
":erc4626-tests/=lib/openzeppelin-contracts/lib/erc4626-tests/",
":forge-std/=lib/forge-std/src/",
":openzeppelin-contracts/=lib/openzeppelin-contracts/",
":solmate/=lib/solmate/src/"
]
}
[{"inputs":[{"internalType":"address","name":"_hyperCardsAddress","type":"address"},{"internalType":"string","name":"baseURI","type":"string"},{"internalType":"address","name":"_payoutAddress","type":"address"}],"stateMutability":"nonpayable","type":"constructor"},{"inputs":[],"name":"HyperCardAlreadyUsedForDiscount","type":"error"},{"inputs":[],"name":"InsufficientPayment","type":"error"},{"inputs":[],"name":"InvalidTokenId","type":"error"},{"inputs":[],"name":"MaxSupplyReached","type":"error"},{"inputs":[],"name":"NotEligibleForDiscountedIANGAC","type":"error"},{"inputs":[{"internalType":"uint256","name":"required","type":"uint256"},{"internalType":"uint256","name":"available","type":"uint256"}],"name":"NotEnoughHyperCardsForDiscount","type":"error"},{"inputs":[],"name":"UserDoesNotOwnHyperCard","type":"error"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"owner","type":"address"},{"indexed":true,"internalType":"address","name":"spender","type":"address"},{"indexed":true,"internalType":"uint256","name":"id","type":"uint256"}],"name":"Approval","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"owner","type":"address"},{"indexed":true,"internalType":"address","name":"operator","type":"address"},{"indexed":false,"internalType":"bool","name":"approved","type":"bool"}],"name":"ApprovalForAll","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"user","type":"address"},{"indexed":false,"internalType":"uint256[]","name":"cardIds","type":"uint256[]"}],"name":"HyperCardsUsedForDiscount","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"to","type":"address"},{"indexed":false,"internalType":"uint256","name":"tokenId","type":"uint256"},{"indexed":false,"internalType":"bool","name":"discounted","type":"bool"}],"name":"IANGACMinted","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"user","type":"address"},{"indexed":true,"internalType":"address","name":"newOwner","type":"address"}],"name":"OwnershipTransferred","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"from","type":"address"},{"indexed":true,"internalType":"address","name":"to","type":"address"},{"indexed":true,"internalType":"uint256","name":"id","type":"uint256"}],"name":"Transfer","type":"event"},{"inputs":[],"name":"CARDS_FOR_DISCOUNT","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"DISCOUNTED_PRICE","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"FULL_PRICE","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"MAX_SUPPLY","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"ROYALTY_PERCENTAGE","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"spender","type":"address"},{"internalType":"uint256","name":"id","type":"uint256"}],"name":"approve","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"}],"name":"balanceOf","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"getApproved","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"user","type":"address"}],"name":"getAvailableDiscountedMints","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"user","type":"address"}],"name":"getEligibleTokenIds","outputs":[{"internalType":"uint256[]","name":"","type":"uint256[]"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"user","type":"address"},{"internalType":"uint256[]","name":"ownedHyperCardIds","type":"uint256[]"}],"name":"getPotentialDiscountEligibility","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"hyperCardsContract","outputs":[{"internalType":"contract IHypercards","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"},{"internalType":"address","name":"","type":"address"}],"name":"isApprovedForAll","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"isTokenMinted","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"mintDiscountedIANGAC","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"mintIANGAC","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[],"name":"name","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"id","type":"uint256"}],"name":"ownerOf","outputs":[{"internalType":"address","name":"owner","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"payoutAddress","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"},{"internalType":"uint256","name":"salePrice","type":"uint256"}],"name":"royaltyInfo","outputs":[{"internalType":"address","name":"receiver","type":"address"},{"internalType":"uint256","name":"royaltyAmount","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"from","type":"address"},{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"id","type":"uint256"}],"name":"safeTransferFrom","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"from","type":"address"},{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"id","type":"uint256"},{"internalType":"bytes","name":"data","type":"bytes"}],"name":"safeTransferFrom","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"operator","type":"address"},{"internalType":"bool","name":"approved","type":"bool"}],"name":"setApprovalForAll","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_hyperCardsAddress","type":"address"}],"name":"setHyperCardsAddress","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_newPayoutAddress","type":"address"}],"name":"setPayoutAddress","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bytes4","name":"interfaceId","type":"bytes4"}],"name":"supportsInterface","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"symbol","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"id","type":"uint256"}],"name":"tokenURI","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":"from","type":"address"},{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"id","type":"uint256"}],"name":"transferFrom","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"},{"internalType":"uint256","name":"","type":"uint256"}],"name":"usedHyperCardForDiscount","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"withdraw","outputs":[],"stateMutability":"nonpayable","type":"function"}]