文件 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:IRiskToken.sol
pragma solidity ^0.8.6;
abstract contract IRiskToken {
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:RiskMinter.sol
pragma solidity ^0.8.6;
import "@openzeppelin/contracts/access/Ownable.sol";
import "@openzeppelin/contracts/security/ReentrancyGuard.sol";
import "./IRiskToken.sol";
contract RiskMinter is Ownable, ReentrancyGuard {
uint256 public constant MAX_MINTS_PER_TX = 20;
uint256 public maxMintsPerAddress;
uint256 public maxTokens;
uint256 public constant TOKEN_COST = 0.07 ether;
bool public saleIsActive = false;
uint256 public preSaleStart;
uint256 public publicSaleStart;
mapping(address => uint256) private addressToMintCount;
mapping(address => uint256) private saleWhitelist;
IRiskToken public riskToken;
constructor(address riskNftAddress,
uint256 preSaleStartTimestamp,
uint256 publicSaleStartTimestamp,
uint256 riskTokenSupply,
uint256 maxMintsAddress) {
riskToken = IRiskToken(riskNftAddress);
preSaleStart = preSaleStartTimestamp;
publicSaleStart = publicSaleStartTimestamp;
maxTokens = riskTokenSupply;
maxMintsPerAddress = maxMintsAddress;
}
function mintPublic(uint amount) public payable nonReentrant {
uint256 supply = riskToken.tokenCount();
require(saleIsActive, "Sale must be active to claim!");
require(block.timestamp >= preSaleStart, "Sale not started!");
require(amount <= MAX_MINTS_PER_TX, "Exceeds max mint per tx!");
require(addressToMintCount[msg.sender] + amount <= maxMintsPerAddress, "Exceeded wallet mint limit!");
if (block.timestamp < publicSaleStart) {
require(saleWhitelist[msg.sender] > 0, "Must be on the sale whitelist");
}
require(supply + amount <= maxTokens, "Exceeds max hero supply!");
require(msg.value >= TOKEN_COST * amount, "Invalid ETH value sent!");
riskToken.mint(amount, msg.sender);
addressToMintCount[msg.sender] += amount;
if (block.timestamp < publicSaleStart) {
saleWhitelist[msg.sender] -= amount;
}
}
function reserveTeamTokens(address _to, uint256 _reserveAmount) public onlyOwner {
uint256 supply = riskToken.tokenCount();
require(supply + _reserveAmount <= maxTokens, "Exceeds max hero supply!");
riskToken.mint(_reserveAmount, _to);
}
function addWhitelisted(address[] memory _addresses, uint[] memory _tokensToClaim) external onlyOwner {
require(_addresses.length == _tokensToClaim.length, "Invalid whitelist data");
for (uint256 i = 0; i < _addresses.length; i++) {
saleWhitelist[_addresses[i]] = _tokensToClaim[i];
}
}
function setMaxMintPerAddress(uint _max) public onlyOwner {
maxMintsPerAddress = _max;
}
function remainingWhitelistTokensForClaim(address _address) external view returns (uint) {
return saleWhitelist[_address];
}
function mintCount(address _address) external view returns (uint) {
return addressToMintCount[_address];
}
function isPreSaleActive() external view returns (bool) {
return block.timestamp >= preSaleStart && block.timestamp < publicSaleStart && saleIsActive;
}
function isPublicSaleActive() external view returns (bool) {
return block.timestamp >= publicSaleStart && saleIsActive;
}
function flipSaleStatus() public onlyOwner {
saleIsActive = !saleIsActive;
}
function withdraw() public payable onlyOwner {
uint balance = address(this).balance;
require(payable(msg.sender).send(balance));
}
}
{
"compilationTarget": {
"contracts/RiskMinter.sol": "RiskMinter"
},
"evmVersion": "berlin",
"libraries": {},
"metadata": {
"bytecodeHash": "ipfs"
},
"optimizer": {
"enabled": false,
"runs": 200
},
"remappings": []
}
[{"inputs":[{"internalType":"address","name":"riskNftAddress","type":"address"},{"internalType":"uint256","name":"preSaleStartTimestamp","type":"uint256"},{"internalType":"uint256","name":"publicSaleStartTimestamp","type":"uint256"},{"internalType":"uint256","name":"riskTokenSupply","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":[{"internalType":"address[]","name":"_addresses","type":"address[]"},{"internalType":"uint256[]","name":"_tokensToClaim","type":"uint256[]"}],"name":"addWhitelisted","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"flipSaleStatus","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"isPreSaleActive","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","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":"preSaleStart","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"publicSaleStart","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_address","type":"address"}],"name":"remainingWhitelistTokensForClaim","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_to","type":"address"},{"internalType":"uint256","name":"_reserveAmount","type":"uint256"}],"name":"reserveTeamTokens","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"riskToken","outputs":[{"internalType":"contract IRiskToken","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"saleIsActive","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_max","type":"uint256"}],"name":"setMaxMintPerAddress","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"withdraw","outputs":[],"stateMutability":"payable","type":"function"}]