文件 1 的 5:Context.sol
pragma solidity ^0.8.0;
abstract contract Context {
function _msgSender() internal view virtual returns (address) {
return msg.sender;
}
function _msgData() internal view virtual returns (bytes calldata) {
return msg.data;
}
}
文件 2 的 5:IRueArcanaToken.sol
pragma solidity ^0.8.6;
abstract contract IRueArcanaToken {
function setProvenanceHash(string memory _provenanceHash) virtual external;
function mint(uint256 _count, address _recipient) virtual external;
function setBaseURI(string memory baseURI) virtual external;
function updateMinter(address _minter) virtual external;
function lockMinter() virtual external;
function tokenCount() virtual external returns (uint256);
}
文件 3 的 5:Ownable.sol
pragma solidity ^0.8.0;
import "../utils/Context.sol";
abstract contract Ownable is Context {
address private _owner;
event OwnershipTransferred(address indexed previousOwner, address indexed newOwner);
constructor() {
_setOwner(_msgSender());
}
function owner() public view virtual returns (address) {
return _owner;
}
modifier onlyOwner() {
require(owner() == _msgSender(), "Ownable: caller is not the owner");
_;
}
function renounceOwnership() public virtual onlyOwner {
_setOwner(address(0));
}
function transferOwnership(address newOwner) public virtual onlyOwner {
require(newOwner != address(0), "Ownable: new owner is the zero address");
_setOwner(newOwner);
}
function _setOwner(address newOwner) private {
address oldOwner = _owner;
_owner = newOwner;
emit OwnershipTransferred(oldOwner, newOwner);
}
}
文件 4 的 5:ReentrancyGuard.sol
pragma solidity ^0.8.0;
abstract contract ReentrancyGuard {
uint256 private constant _NOT_ENTERED = 1;
uint256 private constant _ENTERED = 2;
uint256 private _status;
constructor() {
_status = _NOT_ENTERED;
}
modifier nonReentrant() {
require(_status != _ENTERED, "ReentrancyGuard: reentrant call");
_status = _ENTERED;
_;
_status = _NOT_ENTERED;
}
}
文件 5 的 5:RueArcanaMinter.sol
pragma solidity ^0.8.6;
import "@openzeppelin/contracts/access/Ownable.sol";
import "@openzeppelin/contracts/security/ReentrancyGuard.sol";
import "./IRueArcanaToken.sol";
contract RueArcanaMinter is Ownable, ReentrancyGuard {
uint256 public constant MAX_MINTS_PER_TX = 20;
uint256 public maxMintsPerAddress;
uint256 public maxTokens;
uint256 public constant TOKEN_COST = 0.05 ether;
bool public saleIsActive = false;
uint256 public publicSaleStart;
mapping(address => uint256) private addressToMintCount;
IRueArcanaToken public token;
constructor(address nftAddress,
uint256 publicSaleStartTimestamp,
uint256 tokenSupply,
uint256 maxMintsAddress) {
token = IRueArcanaToken(nftAddress);
publicSaleStart = publicSaleStartTimestamp;
maxTokens = tokenSupply;
maxMintsPerAddress = maxMintsAddress;
}
function mintCount(address _address) external view returns (uint) {
return addressToMintCount[_address];
}
function isPublicSaleActive() external view returns (bool) {
return block.timestamp >= publicSaleStart && saleIsActive;
}
function flipSaleStatus() public onlyOwner {
saleIsActive = !saleIsActive;
}
function mintPublic(uint amount) public payable nonReentrant {
uint256 supply = token.tokenCount();
require(saleIsActive, "Sale is not active");
require(block.timestamp >= publicSaleStart, "Sale has not commenced");
require(amount <= MAX_MINTS_PER_TX, "Exceeds max mint per tx!");
require(addressToMintCount[msg.sender] + amount <= maxMintsPerAddress, "Cannont exceed this many tokens per tx");
require(msg.value >= TOKEN_COST * amount, "ETH amount sent is not correct");
require(supply + amount <= maxTokens, "More than max token supply");
token.mint(amount, msg.sender);
addressToMintCount[msg.sender] += amount;
}
function withdraw() public payable onlyOwner {
uint balance = address(this).balance;
require(payable(msg.sender).send(balance));
}
}
{
"compilationTarget": {
"contracts/RueArcanaMinter.sol": "RueArcanaMinter"
},
"evmVersion": "berlin",
"libraries": {},
"metadata": {
"bytecodeHash": "ipfs"
},
"optimizer": {
"enabled": false,
"runs": 200
},
"remappings": []
}
[{"inputs":[{"internalType":"address","name":"nftAddress","type":"address"},{"internalType":"uint256","name":"publicSaleStartTimestamp","type":"uint256"},{"internalType":"uint256","name":"tokenSupply","type":"uint256"},{"internalType":"uint256","name":"maxMintsAddress","type":"uint256"}],"stateMutability":"nonpayable","type":"constructor"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"previousOwner","type":"address"},{"indexed":true,"internalType":"address","name":"newOwner","type":"address"}],"name":"OwnershipTransferred","type":"event"},{"inputs":[],"name":"MAX_MINTS_PER_TX","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"TOKEN_COST","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"flipSaleStatus","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"isPublicSaleActive","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"maxMintsPerAddress","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"maxTokens","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_address","type":"address"}],"name":"mintCount","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"mintPublic","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"publicSaleStart","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"saleIsActive","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"token","outputs":[{"internalType":"contract IRueArcanaToken","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"withdraw","outputs":[],"stateMutability":"payable","type":"function"}]