编译器
0.8.11+commit.d7f03943
文件 1 的 10: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 的 10:ERC1155Holder.sol
pragma solidity ^0.8.0;
import "./ERC1155Receiver.sol";
contract ERC1155Holder is ERC1155Receiver {
function onERC1155Received(
address,
address,
uint256,
uint256,
bytes memory
) public virtual override returns (bytes4) {
return this.onERC1155Received.selector;
}
function onERC1155BatchReceived(
address,
address,
uint256[] memory,
uint256[] memory,
bytes memory
) public virtual override returns (bytes4) {
return this.onERC1155BatchReceived.selector;
}
}
文件 3 的 10:ERC1155Receiver.sol
pragma solidity ^0.8.0;
import "../IERC1155Receiver.sol";
import "../../../utils/introspection/ERC165.sol";
abstract contract ERC1155Receiver is ERC165, IERC1155Receiver {
function supportsInterface(bytes4 interfaceId) public view virtual override(ERC165, IERC165) returns (bool) {
return interfaceId == type(IERC1155Receiver).interfaceId || super.supportsInterface(interfaceId);
}
}
文件 4 的 10:ERC165.sol
pragma solidity ^0.8.0;
import "./IERC165.sol";
abstract contract ERC165 is IERC165 {
function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) {
return interfaceId == type(IERC165).interfaceId;
}
}
文件 5 的 10:IERC1155.sol
pragma solidity ^0.8.0;
import "../../utils/introspection/IERC165.sol";
interface IERC1155 is IERC165 {
event TransferSingle(address indexed operator, address indexed from, address indexed to, uint256 id, uint256 value);
event TransferBatch(
address indexed operator,
address indexed from,
address indexed to,
uint256[] ids,
uint256[] values
);
event ApprovalForAll(address indexed account, address indexed operator, bool approved);
event URI(string value, uint256 indexed id);
function balanceOf(address account, uint256 id) external view returns (uint256);
function balanceOfBatch(address[] calldata accounts, uint256[] calldata ids)
external
view
returns (uint256[] memory);
function setApprovalForAll(address operator, bool approved) external;
function isApprovedForAll(address account, address operator) external view returns (bool);
function safeTransferFrom(
address from,
address to,
uint256 id,
uint256 amount,
bytes calldata data
) external;
function safeBatchTransferFrom(
address from,
address to,
uint256[] calldata ids,
uint256[] calldata amounts,
bytes calldata data
) external;
}
文件 6 的 10:IERC1155Receiver.sol
pragma solidity ^0.8.0;
import "../../utils/introspection/IERC165.sol";
interface IERC1155Receiver is IERC165 {
function onERC1155Received(
address operator,
address from,
uint256 id,
uint256 value,
bytes calldata data
) external returns (bytes4);
function onERC1155BatchReceived(
address operator,
address from,
uint256[] calldata ids,
uint256[] calldata values,
bytes calldata data
) external returns (bytes4);
}
文件 7 的 10:IERC165.sol
pragma solidity ^0.8.0;
interface IERC165 {
function supportsInterface(bytes4 interfaceId) external view returns (bool);
}
文件 8 的 10:IOrbsNFT.sol
pragma solidity 0.8.11;
interface IOrbsNFT {
function requestMint(address beneficiary, uint256 amount)
external
returns (bytes32);
}
文件 9 的 10:OrbsSale.sol
pragma solidity 0.8.11;
import "@openzeppelin/contracts/access/Ownable.sol";
import "@openzeppelin/contracts/token/ERC1155/IERC1155.sol";
import "@openzeppelin/contracts/token/ERC1155/utils/ERC1155Holder.sol";
import "./interfaces/IOrbsNFT.sol";
contract OrbsSale is Ownable, ERC1155Holder {
event MaxSaleAmountUpdated(uint64 maxSaleByEther, uint256 maxSaleBy888);
event SaleRequested(address indexed user, uint256 amount, bool with888);
event FlushETH(uint256 amount);
event Flush888(address indexed to, uint256 amount);
uint256 public constant ETH_MAX_PRICE = 1111 * 1e16;
uint256 public constant ETH_MIN_PRICE = 1;
uint64 public constant ETH_SALE_COUNT = 2445;
uint64 public constant ETH_PRICE_FREEZE_HR = 47;
uint256 public constant EIGHT_TOKEN_ID = 888;
uint256 public constant PRICE_IN_888_NFT = 3;
IOrbsNFT public immutable orbsNft;
address public immutable eigthEightEightNft;
uint64 public immutable startTime;
uint64 public immutable ethStartTime;
uint64 public saledByEtherCount;
uint64 public maxSaleByEther;
uint64 public maxSaleBy888;
constructor(
address _orbsNft,
address _eigthEightEightNft,
uint64 _startTime,
uint64 _maxSaleByEther,
uint64 _maxSaleBy888
) {
require(_orbsNft != address(0), "OrbsSale: orbsNft is address(0)");
require(
_eigthEightEightNft != address(0),
"OrbsSale: 888NFT is address(0)"
);
require(
_startTime >= block.timestamp,
"OrbsSale: start time must be greater than now"
);
orbsNft = IOrbsNFT(_orbsNft);
eigthEightEightNft = _eigthEightEightNft;
startTime = _startTime;
ethStartTime = _startTime + 1 days;
setMaxSaleAmount(_maxSaleByEther, _maxSaleBy888);
}
function purchaseWithEther(uint64 amount) external payable {
require(
block.timestamp >= ethStartTime,
"OrbsSale: eth sale not started"
);
require(amount <= maxSaleByEther, "OrbsSale: limit max sale amount");
require(
amount * currentPrice() == msg.value,
"OrbsSale: invalid price"
);
saledByEtherCount += amount;
require(
saledByEtherCount <= ETH_SALE_COUNT,
"OrbsSale: cannot sale with Ether"
);
orbsNft.requestMint(msg.sender, amount);
emit SaleRequested(msg.sender, amount, false);
}
function onERC1155Received(
address,
address from,
uint256 id,
uint256 value,
bytes calldata
) public override returns (bytes4) {
require(block.timestamp >= startTime, "OrbsSale: not started");
require(msg.sender == eigthEightEightNft, "OrbsSale: not 888 NFT");
require(id == EIGHT_TOKEN_ID, "OrbsSale: not 888 Token");
require(value % PRICE_IN_888_NFT == 0, "OrbsSale: invalid value");
require(from != address(0), "OrbsSale: from is address(0)");
uint256 amount = value / PRICE_IN_888_NFT;
require(amount <= maxSaleBy888, "OrbsSale: limit max sale amount");
orbsNft.requestMint(from, amount);
emit SaleRequested(from, amount, true);
return this.onERC1155Received.selector;
}
function onERC1155BatchReceived(
address,
address,
uint256[] calldata,
uint256[] calldata,
bytes calldata
) public override returns (bytes4) {
revert("OrbsSale: Batch transfer is not allowed");
}
function currentPrice() public view returns (uint256) {
if (ethStartTime >= block.timestamp) {
return ETH_MAX_PRICE;
}
uint64 hrsSinceStart = (uint64(block.timestamp) - ethStartTime) /
1 hours;
if (hrsSinceStart > ETH_PRICE_FREEZE_HR) {
return ETH_MIN_PRICE;
}
uint256 price = ETH_MAX_PRICE -
(((ETH_MAX_PRICE - ETH_MIN_PRICE) * hrsSinceStart) /
ETH_PRICE_FREEZE_HR);
return ((price + 5e15) / 1e16) * 1e16;
}
function setMaxSaleAmount(uint64 _maxSaleByEther, uint64 _maxSaleBy888)
public
onlyOwner
{
maxSaleByEther = _maxSaleByEther;
maxSaleBy888 = _maxSaleBy888;
emit MaxSaleAmountUpdated(_maxSaleByEther, _maxSaleBy888);
}
function flushETH() external onlyOwner {
uint256 balance = address(this).balance;
require(balance > 0, "OrbsSale: nothing to flush");
payable(msg.sender).transfer(balance);
emit FlushETH(balance);
}
function flush888(
address _to,
uint256 _amount,
bytes calldata _data
) external onlyOwner {
uint256 balance = IERC1155(eigthEightEightNft).balanceOf(
address(this),
EIGHT_TOKEN_ID
);
require(
balance != 0 && balance >= _amount,
"OrbsSale: nothing to flush"
);
uint256 amount = _amount == 0 ? balance : _amount;
IERC1155(eigthEightEightNft).safeTransferFrom(
address(this),
_to,
EIGHT_TOKEN_ID,
amount,
_data
);
emit Flush888(_to, amount);
}
}
文件 10 的 10: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() {
_transferOwnership(_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 {
_transferOwnership(address(0));
}
function transferOwnership(address newOwner) public virtual onlyOwner {
require(newOwner != address(0), "Ownable: new owner is the zero address");
_transferOwnership(newOwner);
}
function _transferOwnership(address newOwner) internal virtual {
address oldOwner = _owner;
_owner = newOwner;
emit OwnershipTransferred(oldOwner, newOwner);
}
}
{
"compilationTarget": {
"contracts/OrbsSale.sol": "OrbsSale"
},
"evmVersion": "london",
"libraries": {},
"metadata": {
"bytecodeHash": "ipfs",
"useLiteralContent": true
},
"optimizer": {
"enabled": true,
"runs": 200
},
"remappings": []
}
[{"inputs":[{"internalType":"address","name":"_orbsNft","type":"address"},{"internalType":"address","name":"_eigthEightEightNft","type":"address"},{"internalType":"uint64","name":"_startTime","type":"uint64"},{"internalType":"uint64","name":"_maxSaleByEther","type":"uint64"},{"internalType":"uint64","name":"_maxSaleBy888","type":"uint64"}],"stateMutability":"nonpayable","type":"constructor"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"to","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"Flush888","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"FlushETH","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint64","name":"maxSaleByEther","type":"uint64"},{"indexed":false,"internalType":"uint256","name":"maxSaleBy888","type":"uint256"}],"name":"MaxSaleAmountUpdated","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"previousOwner","type":"address"},{"indexed":true,"internalType":"address","name":"newOwner","type":"address"}],"name":"OwnershipTransferred","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"user","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"},{"indexed":false,"internalType":"bool","name":"with888","type":"bool"}],"name":"SaleRequested","type":"event"},{"inputs":[],"name":"EIGHT_TOKEN_ID","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"ETH_MAX_PRICE","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"ETH_MIN_PRICE","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"ETH_PRICE_FREEZE_HR","outputs":[{"internalType":"uint64","name":"","type":"uint64"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"ETH_SALE_COUNT","outputs":[{"internalType":"uint64","name":"","type":"uint64"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"PRICE_IN_888_NFT","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"currentPrice","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"eigthEightEightNft","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"ethStartTime","outputs":[{"internalType":"uint64","name":"","type":"uint64"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_to","type":"address"},{"internalType":"uint256","name":"_amount","type":"uint256"},{"internalType":"bytes","name":"_data","type":"bytes"}],"name":"flush888","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"flushETH","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"maxSaleBy888","outputs":[{"internalType":"uint64","name":"","type":"uint64"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"maxSaleByEther","outputs":[{"internalType":"uint64","name":"","type":"uint64"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"},{"internalType":"address","name":"","type":"address"},{"internalType":"uint256[]","name":"","type":"uint256[]"},{"internalType":"uint256[]","name":"","type":"uint256[]"},{"internalType":"bytes","name":"","type":"bytes"}],"name":"onERC1155BatchReceived","outputs":[{"internalType":"bytes4","name":"","type":"bytes4"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"},{"internalType":"address","name":"from","type":"address"},{"internalType":"uint256","name":"id","type":"uint256"},{"internalType":"uint256","name":"value","type":"uint256"},{"internalType":"bytes","name":"","type":"bytes"}],"name":"onERC1155Received","outputs":[{"internalType":"bytes4","name":"","type":"bytes4"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"orbsNft","outputs":[{"internalType":"contract IOrbsNFT","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint64","name":"amount","type":"uint64"}],"name":"purchaseWithEther","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[],"name":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"saledByEtherCount","outputs":[{"internalType":"uint64","name":"","type":"uint64"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint64","name":"_maxSaleByEther","type":"uint64"},{"internalType":"uint64","name":"_maxSaleBy888","type":"uint64"}],"name":"setMaxSaleAmount","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"startTime","outputs":[{"internalType":"uint64","name":"","type":"uint64"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"bytes4","name":"interfaceId","type":"bytes4"}],"name":"supportsInterface","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"}]