文件 1 的 18:AccessControl.sol
pragma solidity ^0.8.0;
import "./IAccessControl.sol";
import "../utils/Context.sol";
import "../utils/Strings.sol";
import "../utils/introspection/ERC165.sol";
abstract contract AccessControl is Context, IAccessControl, ERC165 {
struct RoleData {
mapping(address => bool) members;
bytes32 adminRole;
}
mapping(bytes32 => RoleData) private _roles;
bytes32 public constant DEFAULT_ADMIN_ROLE = 0x00;
modifier onlyRole(bytes32 role) {
_checkRole(role, _msgSender());
_;
}
function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) {
return interfaceId == type(IAccessControl).interfaceId || super.supportsInterface(interfaceId);
}
function hasRole(bytes32 role, address account) public view override returns (bool) {
return _roles[role].members[account];
}
function _checkRole(bytes32 role, address account) internal view {
if (!hasRole(role, account)) {
revert(
string(
abi.encodePacked(
"AccessControl: account ",
Strings.toHexString(uint160(account), 20),
" is missing role ",
Strings.toHexString(uint256(role), 32)
)
)
);
}
}
function getRoleAdmin(bytes32 role) public view override returns (bytes32) {
return _roles[role].adminRole;
}
function grantRole(bytes32 role, address account) public virtual override onlyRole(getRoleAdmin(role)) {
_grantRole(role, account);
}
function revokeRole(bytes32 role, address account) public virtual override onlyRole(getRoleAdmin(role)) {
_revokeRole(role, account);
}
function renounceRole(bytes32 role, address account) public virtual override {
require(account == _msgSender(), "AccessControl: can only renounce roles for self");
_revokeRole(role, account);
}
function _setupRole(bytes32 role, address account) internal virtual {
_grantRole(role, account);
}
function _setRoleAdmin(bytes32 role, bytes32 adminRole) internal virtual {
bytes32 previousAdminRole = getRoleAdmin(role);
_roles[role].adminRole = adminRole;
emit RoleAdminChanged(role, previousAdminRole, adminRole);
}
function _grantRole(bytes32 role, address account) internal virtual {
if (!hasRole(role, account)) {
_roles[role].members[account] = true;
emit RoleGranted(role, account, _msgSender());
}
}
function _revokeRole(bytes32 role, address account) internal virtual {
if (hasRole(role, account)) {
_roles[role].members[account] = false;
emit RoleRevoked(role, account, _msgSender());
}
}
}
文件 2 的 18:AccessControlEnumerable.sol
pragma solidity ^0.8.0;
import "./IAccessControlEnumerable.sol";
import "./AccessControl.sol";
import "../utils/structs/EnumerableSet.sol";
abstract contract AccessControlEnumerable is IAccessControlEnumerable, AccessControl {
using EnumerableSet for EnumerableSet.AddressSet;
mapping(bytes32 => EnumerableSet.AddressSet) private _roleMembers;
function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) {
return interfaceId == type(IAccessControlEnumerable).interfaceId || super.supportsInterface(interfaceId);
}
function getRoleMember(bytes32 role, uint256 index) public view override returns (address) {
return _roleMembers[role].at(index);
}
function getRoleMemberCount(bytes32 role) public view override returns (uint256) {
return _roleMembers[role].length();
}
function _grantRole(bytes32 role, address account) internal virtual override {
super._grantRole(role, account);
_roleMembers[role].add(account);
}
function _revokeRole(bytes32 role, address account) internal virtual override {
super._revokeRole(role, account);
_roleMembers[role].remove(account);
}
}
文件 3 的 18:Address.sol
pragma solidity ^0.8.0;
library Address {
function isContract(address account) internal view returns (bool) {
uint256 size;
assembly {
size := extcodesize(account)
}
return size > 0;
}
function sendValue(address payable recipient, uint256 amount) internal {
require(address(this).balance >= amount, "Address: insufficient balance");
(bool success, ) = recipient.call{value: amount}("");
require(success, "Address: unable to send value, recipient may have reverted");
}
function functionCall(address target, bytes memory data) internal returns (bytes memory) {
return functionCall(target, data, "Address: low-level call failed");
}
function functionCall(
address target,
bytes memory data,
string memory errorMessage
) internal returns (bytes memory) {
return functionCallWithValue(target, data, 0, errorMessage);
}
function functionCallWithValue(
address target,
bytes memory data,
uint256 value
) internal returns (bytes memory) {
return functionCallWithValue(target, data, value, "Address: low-level call with value failed");
}
function functionCallWithValue(
address target,
bytes memory data,
uint256 value,
string memory errorMessage
) internal returns (bytes memory) {
require(address(this).balance >= value, "Address: insufficient balance for call");
require(isContract(target), "Address: call to non-contract");
(bool success, bytes memory returndata) = target.call{value: value}(data);
return verifyCallResult(success, returndata, errorMessage);
}
function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) {
return functionStaticCall(target, data, "Address: low-level static call failed");
}
function functionStaticCall(
address target,
bytes memory data,
string memory errorMessage
) internal view returns (bytes memory) {
require(isContract(target), "Address: static call to non-contract");
(bool success, bytes memory returndata) = target.staticcall(data);
return verifyCallResult(success, returndata, errorMessage);
}
function functionDelegateCall(address target, bytes memory data) internal returns (bytes memory) {
return functionDelegateCall(target, data, "Address: low-level delegate call failed");
}
function functionDelegateCall(
address target,
bytes memory data,
string memory errorMessage
) internal returns (bytes memory) {
require(isContract(target), "Address: delegate call to non-contract");
(bool success, bytes memory returndata) = target.delegatecall(data);
return verifyCallResult(success, returndata, errorMessage);
}
function verifyCallResult(
bool success,
bytes memory returndata,
string memory errorMessage
) internal pure returns (bytes memory) {
if (success) {
return returndata;
} else {
if (returndata.length > 0) {
assembly {
let returndata_size := mload(returndata)
revert(add(32, returndata), returndata_size)
}
} else {
revert(errorMessage);
}
}
}
}
文件 4 的 18:BitMaps.sol
pragma solidity ^0.8.0;
library BitMaps {
struct BitMap {
mapping(uint256 => uint256) _data;
}
function get(BitMap storage bitmap, uint256 index) internal view returns (bool) {
uint256 bucket = index >> 8;
uint256 mask = 1 << (index & 0xff);
return bitmap._data[bucket] & mask != 0;
}
function setTo(
BitMap storage bitmap,
uint256 index,
bool value
) internal {
if (value) {
set(bitmap, index);
} else {
unset(bitmap, index);
}
}
function set(BitMap storage bitmap, uint256 index) internal {
uint256 bucket = index >> 8;
uint256 mask = 1 << (index & 0xff);
bitmap._data[bucket] |= mask;
}
function unset(BitMap storage bitmap, uint256 index) internal {
uint256 bucket = index >> 8;
uint256 mask = 1 << (index & 0xff);
bitmap._data[bucket] &= ~mask;
}
}
文件 5 的 18: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;
}
}
文件 6 的 18:ContinuousDutchAuction.sol
pragma solidity ^0.8.0;
import "./Manageable.sol";
import "@openzeppelin/contracts/utils/Address.sol";
abstract contract ContinuousDutchAuction is Manageable {
struct Auction {
uint256 startingPrice;
uint128 decreasingConstant;
uint64 startingTime;
uint64 period;
}
mapping (uint => Auction) private _auctions;
function auctions(uint256 auctionId) public view returns (
uint256 startingPrice,
uint128 decreasingConstant,
uint64 startingTime,
uint64 period,
bool active
) {
Auction memory auction = _auctions[auctionId];
startingPrice = auction.startingPrice;
decreasingConstant = auction.decreasingConstant;
startingTime = auction.startingTime;
period = auction.period;
active = startingTime > 0 && block.timestamp >= startingTime;
}
function setAuction(
uint256 auctionId,
uint256 startingPrice,
uint128 decreasingConstant,
uint64 startingBlock,
uint64 period
) virtual public onlyRole(MANAGER_ROLE) {
unchecked {
require(startingPrice - decreasingConstant * period <= startingPrice, "setAuction: floor price underflow");
}
_auctions[auctionId] = Auction(startingPrice, decreasingConstant, startingBlock, period);
}
function getPrice(uint256 auctionId) virtual public view returns (uint256 price) {
Auction memory auction = _auctions[auctionId];
if (block.timestamp < auction.startingTime) price = auction.startingPrice;
else if (block.timestamp >= auction.startingTime + auction.period) price = auction.startingPrice - auction.period * auction.decreasingConstant;
else price = auction.startingPrice - (auction.decreasingConstant * (block.timestamp - auction.startingTime));
}
function verifyBid(uint256 auctionId) internal returns (uint256) {
Auction memory auction = _auctions[auctionId];
require(auction.startingTime > 0, "AUCTION:NOT CREATED");
require(block.timestamp >= auction.startingTime, "PURCHASE:AUCTION NOT STARTED");
uint256 pricePaid = getPrice(auctionId);
require(msg.value >= pricePaid, "PURCHASE:INCORRECT MSG.VALUE");
if (msg.value - pricePaid > 0) Address.sendValue(payable(msg.sender), msg.value-pricePaid);
return pricePaid;
}
}
文件 7 的 18: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;
}
}
文件 8 的 18:EnumerableSet.sol
pragma solidity ^0.8.0;
library EnumerableSet {
struct Set {
bytes32[] _values;
mapping(bytes32 => uint256) _indexes;
}
function _add(Set storage set, bytes32 value) private returns (bool) {
if (!_contains(set, value)) {
set._values.push(value);
set._indexes[value] = set._values.length;
return true;
} else {
return false;
}
}
function _remove(Set storage set, bytes32 value) private returns (bool) {
uint256 valueIndex = set._indexes[value];
if (valueIndex != 0) {
uint256 toDeleteIndex = valueIndex - 1;
uint256 lastIndex = set._values.length - 1;
if (lastIndex != toDeleteIndex) {
bytes32 lastvalue = set._values[lastIndex];
set._values[toDeleteIndex] = lastvalue;
set._indexes[lastvalue] = valueIndex;
}
set._values.pop();
delete set._indexes[value];
return true;
} else {
return false;
}
}
function _contains(Set storage set, bytes32 value) private view returns (bool) {
return set._indexes[value] != 0;
}
function _length(Set storage set) private view returns (uint256) {
return set._values.length;
}
function _at(Set storage set, uint256 index) private view returns (bytes32) {
return set._values[index];
}
function _values(Set storage set) private view returns (bytes32[] memory) {
return set._values;
}
struct Bytes32Set {
Set _inner;
}
function add(Bytes32Set storage set, bytes32 value) internal returns (bool) {
return _add(set._inner, value);
}
function remove(Bytes32Set storage set, bytes32 value) internal returns (bool) {
return _remove(set._inner, value);
}
function contains(Bytes32Set storage set, bytes32 value) internal view returns (bool) {
return _contains(set._inner, value);
}
function length(Bytes32Set storage set) internal view returns (uint256) {
return _length(set._inner);
}
function at(Bytes32Set storage set, uint256 index) internal view returns (bytes32) {
return _at(set._inner, index);
}
function values(Bytes32Set storage set) internal view returns (bytes32[] memory) {
return _values(set._inner);
}
struct AddressSet {
Set _inner;
}
function add(AddressSet storage set, address value) internal returns (bool) {
return _add(set._inner, bytes32(uint256(uint160(value))));
}
function remove(AddressSet storage set, address value) internal returns (bool) {
return _remove(set._inner, bytes32(uint256(uint160(value))));
}
function contains(AddressSet storage set, address value) internal view returns (bool) {
return _contains(set._inner, bytes32(uint256(uint160(value))));
}
function length(AddressSet storage set) internal view returns (uint256) {
return _length(set._inner);
}
function at(AddressSet storage set, uint256 index) internal view returns (address) {
return address(uint160(uint256(_at(set._inner, index))));
}
function values(AddressSet storage set) internal view returns (address[] memory) {
bytes32[] memory store = _values(set._inner);
address[] memory result;
assembly {
result := store
}
return result;
}
struct UintSet {
Set _inner;
}
function add(UintSet storage set, uint256 value) internal returns (bool) {
return _add(set._inner, bytes32(value));
}
function remove(UintSet storage set, uint256 value) internal returns (bool) {
return _remove(set._inner, bytes32(value));
}
function contains(UintSet storage set, uint256 value) internal view returns (bool) {
return _contains(set._inner, bytes32(value));
}
function length(UintSet storage set) internal view returns (uint256) {
return _length(set._inner);
}
function at(UintSet storage set, uint256 index) internal view returns (uint256) {
return uint256(_at(set._inner, index));
}
function values(UintSet storage set) internal view returns (uint256[] memory) {
bytes32[] memory store = _values(set._inner);
uint256[] memory result;
assembly {
result := store
}
return result;
}
}
文件 9 的 18:IAccessControl.sol
pragma solidity ^0.8.0;
interface IAccessControl {
event RoleAdminChanged(bytes32 indexed role, bytes32 indexed previousAdminRole, bytes32 indexed newAdminRole);
event RoleGranted(bytes32 indexed role, address indexed account, address indexed sender);
event RoleRevoked(bytes32 indexed role, address indexed account, address indexed sender);
function hasRole(bytes32 role, address account) external view returns (bool);
function getRoleAdmin(bytes32 role) external view returns (bytes32);
function grantRole(bytes32 role, address account) external;
function revokeRole(bytes32 role, address account) external;
function renounceRole(bytes32 role, address account) external;
}
文件 10 的 18:IAccessControlEnumerable.sol
pragma solidity ^0.8.0;
import "./IAccessControl.sol";
interface IAccessControlEnumerable is IAccessControl {
function getRoleMember(bytes32 role, uint256 index) external view returns (address);
function getRoleMemberCount(bytes32 role) external view returns (uint256);
}
文件 11 的 18:IERC165.sol
pragma solidity ^0.8.0;
interface IERC165 {
function supportsInterface(bytes4 interfaceId) external view returns (bool);
}
文件 12 的 18:IQuantumArt.sol
pragma solidity ^0.8.0;
interface IQuantumArt {
function mintTo(uint256 dropId, address artist) external returns (uint256);
function burn(uint256 tokenId) external;
function getArtist(uint256 dropId) external view returns (address);
}
文件 13 的 18:IQuantumMintPass.sol
pragma solidity ^0.8.0;
interface IQuantumMintPass {
function burnFromRedeem(address user, uint256 mpId, uint256 amount) external;
}
文件 14 的 18:Manageable.sol
pragma solidity ^0.8.0;
import "@openzeppelin/contracts/access/AccessControlEnumerable.sol";
abstract contract Manageable is AccessControlEnumerable {
bytes32 public constant MANAGER_ROLE = keccak256("MANAGER_ROLE");
}
文件 15 的 18:MerkleProof.sol
pragma solidity ^0.8.0;
library MerkleProof {
function verify(
bytes32[] memory proof,
bytes32 root,
bytes32 leaf
) internal pure returns (bool) {
return processProof(proof, leaf) == root;
}
function processProof(bytes32[] memory proof, bytes32 leaf) internal pure returns (bytes32) {
bytes32 computedHash = leaf;
for (uint256 i = 0; i < proof.length; i++) {
bytes32 proofElement = proof[i];
if (computedHash <= proofElement) {
computedHash = keccak256(abi.encodePacked(computedHash, proofElement));
} else {
computedHash = keccak256(abi.encodePacked(proofElement, computedHash));
}
}
return computedHash;
}
}
文件 16 的 18: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;
}
}
文件 17 的 18:SalePlatform.sol
pragma solidity ^0.8.0;
import "./interfaces/IQuantumArt.sol";
import "./interfaces/IQuantumMintPass.sol";
import "./ContinuousDutchAuction.sol";
import "@openzeppelin/contracts/security/ReentrancyGuard.sol";
import "@openzeppelin/contracts/utils/Address.sol";
import "@openzeppelin/contracts/utils/Strings.sol";
import "@openzeppelin/contracts/utils/structs/BitMaps.sol";
import "@openzeppelin/contracts/utils/cryptography/MerkleProof.sol";
contract SalePlatform is ContinuousDutchAuction, ReentrancyGuard {
using BitMaps for BitMaps.BitMap;
using Strings for uint256;
struct Sale {
uint128 price;
uint64 start;
uint64 limit;
}
struct MPClaim {
uint64 mpId;
uint64 start;
uint128 price;
}
struct Whitelist {
uint192 price;
uint64 start;
bytes32 merkleRoot;
}
event Purchased(uint256 indexed dropId, uint256 tokenId, address to);
mapping (uint256 => Sale) public sales;
mapping (uint256 => MPClaim) public mpClaims;
mapping (uint256 => Whitelist) public whitelists;
mapping (uint256 => BitMaps.BitMap) private _claimedWL;
mapping (address => BitMaps.BitMap) private _alreadyBought;
uint256 public defaultArtistCut;
IQuantumArt public quantum;
IQuantumMintPass public mintpass;
mapping (uint256 => uint256) private _overridedArtistCut;
address payable private _quantumTreasury;
constructor(
address deployedQuantum,
address deployedMP,
address admin,
address payable treasury) {
quantum = IQuantumArt(deployedQuantum);
mintpass = IQuantumMintPass(deployedMP);
_setupRole(DEFAULT_ADMIN_ROLE, admin);
_quantumTreasury = treasury;
defaultArtistCut = 8000;
}
modifier checkCaller {
require(!Address.isContract(msg.sender), "Contract forbidden from buying");
_;
}
modifier isFirstTime(uint256 dropId) {
require(!_alreadyBought[msg.sender].get(dropId), string(abi.encodePacked("Already bought drop ", dropId.toString())));
_alreadyBought[msg.sender].set(dropId);
_;
}
function withdraw(address payable to) onlyRole(DEFAULT_ADMIN_ROLE) public {
Address.sendValue(to, address(this).balance);
}
function setManager(address manager) onlyRole(DEFAULT_ADMIN_ROLE) public {
grantRole(MANAGER_ROLE, manager);
}
function unsetManager(address manager) onlyRole(DEFAULT_ADMIN_ROLE) public {
revokeRole(MANAGER_ROLE, manager);
}
function premint(uint256 dropId, address[] calldata recipients) onlyRole(DEFAULT_ADMIN_ROLE) public {
for(uint256 i = 0; i < recipients.length; i++) {
uint256 tokenId = quantum.mintTo(dropId, recipients[i]);
emit Purchased(dropId, tokenId, recipients[i]);
}
}
function setMintpass(address deployedMP) onlyRole(MANAGER_ROLE) public {
mintpass = IQuantumMintPass(deployedMP);
}
function createSale(uint256 dropId, uint128 price, uint64 start, uint64 limit) onlyRole(MANAGER_ROLE) public {
sales[dropId] = Sale(price, start, limit);
}
function createAuctionNow(
uint256 auctionId,
uint256 startingPrice,
uint128 decreasingConstant,
uint64 period
) public {
super.setAuction(
auctionId,
startingPrice,
decreasingConstant,
uint64(block.timestamp),
period
);
}
function createMPClaim(uint256 dropId, uint64 mpId, uint64 start, uint128 price) onlyRole(MANAGER_ROLE) public {
mpClaims[dropId] = MPClaim(mpId, start, price);
}
function createWLClaim(uint256 dropId, uint192 price, uint64 start, bytes32 root) onlyRole(MANAGER_ROLE) public {
whitelists[dropId] = Whitelist(price, start, root);
}
function setDefaultArtistCut(uint256 cut) onlyRole(MANAGER_ROLE) public {
defaultArtistCut = cut;
}
function flipUint64(uint64 x) internal pure returns (uint64) {
return x > 0 ? 0 : type(uint64).max;
}
function flipSaleState(uint256 dropId) onlyRole(MANAGER_ROLE) public {
sales[dropId].start = flipUint64(sales[dropId].start);
}
function flipMPClaimState(uint256 dropId) onlyRole(MANAGER_ROLE) public {
mpClaims[dropId].start = flipUint64(mpClaims[dropId].start);
}
function flipWLState(uint256 dropId) onlyRole(MANAGER_ROLE) public {
whitelists[dropId].start = flipUint64(whitelists[dropId].start);
}
function overrideArtistcut(uint256 dropId, uint256 cut) onlyRole(MANAGER_ROLE) public {
_overridedArtistCut[dropId] = cut;
}
function payout(address artist, uint256 dropId, uint256 amount) internal {
uint256 artistCut = _overridedArtistCut[dropId] == 0 ? defaultArtistCut : _overridedArtistCut[dropId];
uint256 payout_ = (amount*artistCut)/10000;
Address.sendValue(payable(artist), payout_);
Address.sendValue(_quantumTreasury, amount - payout_);
}
function purchase(uint256 dropId, uint256 amount) nonReentrant checkCaller isFirstTime(dropId) payable public {
Sale memory sale = sales[dropId];
require(block.timestamp >= sale.start, "PURCHASE:SALE INACTIVE");
require(amount <= sale.limit, "PURCHASE:OVER LIMIT");
require(msg.value == amount * sale.price, "PURCHASE:INCORRECT MSG.VALUE");
for(uint256 i = 0; i < amount; i++) {
uint256 tokenId = quantum.mintTo(dropId, msg.sender);
emit Purchased(dropId, tokenId, msg.sender);
}
payout(quantum.getArtist(dropId), dropId, msg.value);
}
function purchaseThroughAuction(uint256 dropId) nonReentrant checkCaller isFirstTime(dropId) payable public {
uint256 userPaid = verifyBid(dropId);
uint256 tokenId = quantum.mintTo(dropId, msg.sender);
emit Purchased(dropId, tokenId, msg.sender);
payout(quantum.getArtist(dropId), dropId, userPaid);
}
function claimWithMintPass(uint256 dropId, uint256 amount) nonReentrant payable public {
MPClaim memory mpClaim = mpClaims[dropId];
require(block.timestamp >= mpClaim.start, "MP: CLAIMING INACTIVE");
require(msg.value == amount * mpClaim.price, "MP:WRONG MSG.VALUE");
mintpass.burnFromRedeem(msg.sender, mpClaim.mpId, amount);
for(uint256 i = 0; i < amount; i++) {
uint256 tokenId = quantum.mintTo(dropId, msg.sender);
emit Purchased(dropId, tokenId, msg.sender);
}
if (msg.value > 0) payout(quantum.getArtist(dropId), dropId, msg.value);
}
function purchaseThroughWhitelist(uint256 dropId, uint256 amount, uint256 index, bytes32[] calldata merkleProof) nonReentrant external payable {
Whitelist memory whitelist = whitelists[dropId];
require(block.timestamp >= whitelist.start, "WL:INACTIVE");
require(msg.value == whitelist.price * amount, "WL: INVALID MSG.VALUE");
require(!_claimedWL[dropId].get(index), "WL:ALREADY CLAIMED");
bytes32 node = keccak256(abi.encodePacked(msg.sender, amount, index));
require(MerkleProof.verify(merkleProof, whitelist.merkleRoot, node),"WL:INVALID PROOF");
_claimedWL[dropId].set(index);
uint256 tokenId = quantum.mintTo(dropId, msg.sender);
emit Purchased(dropId, tokenId, msg.sender);
payout(quantum.getArtist(dropId), dropId, msg.value);
}
function isWLClaimed(uint256 dropId, uint256 index) public view returns (bool) {
return _claimedWL[dropId].get(index);
}
}
文件 18 的 18:Strings.sol
pragma solidity ^0.8.0;
library Strings {
bytes16 private constant _HEX_SYMBOLS = "0123456789abcdef";
function toString(uint256 value) internal pure returns (string memory) {
if (value == 0) {
return "0";
}
uint256 temp = value;
uint256 digits;
while (temp != 0) {
digits++;
temp /= 10;
}
bytes memory buffer = new bytes(digits);
while (value != 0) {
digits -= 1;
buffer[digits] = bytes1(uint8(48 + uint256(value % 10)));
value /= 10;
}
return string(buffer);
}
function toHexString(uint256 value) internal pure returns (string memory) {
if (value == 0) {
return "0x00";
}
uint256 temp = value;
uint256 length = 0;
while (temp != 0) {
length++;
temp >>= 8;
}
return toHexString(value, length);
}
function toHexString(uint256 value, uint256 length) internal pure returns (string memory) {
bytes memory buffer = new bytes(2 * length + 2);
buffer[0] = "0";
buffer[1] = "x";
for (uint256 i = 2 * length + 1; i > 1; --i) {
buffer[i] = _HEX_SYMBOLS[value & 0xf];
value >>= 4;
}
require(value == 0, "Strings: hex length insufficient");
return string(buffer);
}
}
{
"compilationTarget": {
"contracts/SalePlatform.sol": "SalePlatform"
},
"evmVersion": "london",
"libraries": {},
"metadata": {
"bytecodeHash": "ipfs"
},
"optimizer": {
"enabled": false,
"runs": 200
},
"remappings": []
}
[{"inputs":[{"internalType":"address","name":"deployedQuantum","type":"address"},{"internalType":"address","name":"deployedMP","type":"address"},{"internalType":"address","name":"admin","type":"address"},{"internalType":"address payable","name":"treasury","type":"address"}],"stateMutability":"nonpayable","type":"constructor"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"uint256","name":"dropId","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"tokenId","type":"uint256"},{"indexed":false,"internalType":"address","name":"to","type":"address"}],"name":"Purchased","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"bytes32","name":"role","type":"bytes32"},{"indexed":true,"internalType":"bytes32","name":"previousAdminRole","type":"bytes32"},{"indexed":true,"internalType":"bytes32","name":"newAdminRole","type":"bytes32"}],"name":"RoleAdminChanged","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"bytes32","name":"role","type":"bytes32"},{"indexed":true,"internalType":"address","name":"account","type":"address"},{"indexed":true,"internalType":"address","name":"sender","type":"address"}],"name":"RoleGranted","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"bytes32","name":"role","type":"bytes32"},{"indexed":true,"internalType":"address","name":"account","type":"address"},{"indexed":true,"internalType":"address","name":"sender","type":"address"}],"name":"RoleRevoked","type":"event"},{"inputs":[],"name":"DEFAULT_ADMIN_ROLE","outputs":[{"internalType":"bytes32","name":"","type":"bytes32"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"MANAGER_ROLE","outputs":[{"internalType":"bytes32","name":"","type":"bytes32"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"auctionId","type":"uint256"}],"name":"auctions","outputs":[{"internalType":"uint256","name":"startingPrice","type":"uint256"},{"internalType":"uint128","name":"decreasingConstant","type":"uint128"},{"internalType":"uint64","name":"startingTime","type":"uint64"},{"internalType":"uint64","name":"period","type":"uint64"},{"internalType":"bool","name":"active","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"dropId","type":"uint256"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"claimWithMintPass","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"uint256","name":"auctionId","type":"uint256"},{"internalType":"uint256","name":"startingPrice","type":"uint256"},{"internalType":"uint128","name":"decreasingConstant","type":"uint128"},{"internalType":"uint64","name":"period","type":"uint64"}],"name":"createAuctionNow","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"dropId","type":"uint256"},{"internalType":"uint64","name":"mpId","type":"uint64"},{"internalType":"uint64","name":"start","type":"uint64"},{"internalType":"uint128","name":"price","type":"uint128"}],"name":"createMPClaim","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"dropId","type":"uint256"},{"internalType":"uint128","name":"price","type":"uint128"},{"internalType":"uint64","name":"start","type":"uint64"},{"internalType":"uint64","name":"limit","type":"uint64"}],"name":"createSale","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"dropId","type":"uint256"},{"internalType":"uint192","name":"price","type":"uint192"},{"internalType":"uint64","name":"start","type":"uint64"},{"internalType":"bytes32","name":"root","type":"bytes32"}],"name":"createWLClaim","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"defaultArtistCut","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"dropId","type":"uint256"}],"name":"flipMPClaimState","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"dropId","type":"uint256"}],"name":"flipSaleState","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"dropId","type":"uint256"}],"name":"flipWLState","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"auctionId","type":"uint256"}],"name":"getPrice","outputs":[{"internalType":"uint256","name":"price","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"bytes32","name":"role","type":"bytes32"}],"name":"getRoleAdmin","outputs":[{"internalType":"bytes32","name":"","type":"bytes32"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"bytes32","name":"role","type":"bytes32"},{"internalType":"uint256","name":"index","type":"uint256"}],"name":"getRoleMember","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"bytes32","name":"role","type":"bytes32"}],"name":"getRoleMemberCount","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"bytes32","name":"role","type":"bytes32"},{"internalType":"address","name":"account","type":"address"}],"name":"grantRole","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bytes32","name":"role","type":"bytes32"},{"internalType":"address","name":"account","type":"address"}],"name":"hasRole","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"dropId","type":"uint256"},{"internalType":"uint256","name":"index","type":"uint256"}],"name":"isWLClaimed","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"mintpass","outputs":[{"internalType":"contract IQuantumMintPass","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"mpClaims","outputs":[{"internalType":"uint64","name":"mpId","type":"uint64"},{"internalType":"uint64","name":"start","type":"uint64"},{"internalType":"uint128","name":"price","type":"uint128"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"dropId","type":"uint256"},{"internalType":"uint256","name":"cut","type":"uint256"}],"name":"overrideArtistcut","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"dropId","type":"uint256"},{"internalType":"address[]","name":"recipients","type":"address[]"}],"name":"premint","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"dropId","type":"uint256"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"purchase","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"uint256","name":"dropId","type":"uint256"}],"name":"purchaseThroughAuction","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"uint256","name":"dropId","type":"uint256"},{"internalType":"uint256","name":"amount","type":"uint256"},{"internalType":"uint256","name":"index","type":"uint256"},{"internalType":"bytes32[]","name":"merkleProof","type":"bytes32[]"}],"name":"purchaseThroughWhitelist","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[],"name":"quantum","outputs":[{"internalType":"contract IQuantumArt","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"bytes32","name":"role","type":"bytes32"},{"internalType":"address","name":"account","type":"address"}],"name":"renounceRole","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bytes32","name":"role","type":"bytes32"},{"internalType":"address","name":"account","type":"address"}],"name":"revokeRole","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"sales","outputs":[{"internalType":"uint128","name":"price","type":"uint128"},{"internalType":"uint64","name":"start","type":"uint64"},{"internalType":"uint64","name":"limit","type":"uint64"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"auctionId","type":"uint256"},{"internalType":"uint256","name":"startingPrice","type":"uint256"},{"internalType":"uint128","name":"decreasingConstant","type":"uint128"},{"internalType":"uint64","name":"startingBlock","type":"uint64"},{"internalType":"uint64","name":"period","type":"uint64"}],"name":"setAuction","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"cut","type":"uint256"}],"name":"setDefaultArtistCut","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"manager","type":"address"}],"name":"setManager","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"deployedMP","type":"address"}],"name":"setMintpass","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":[{"internalType":"address","name":"manager","type":"address"}],"name":"unsetManager","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"whitelists","outputs":[{"internalType":"uint192","name":"price","type":"uint192"},{"internalType":"uint64","name":"start","type":"uint64"},{"internalType":"bytes32","name":"merkleRoot","type":"bytes32"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address payable","name":"to","type":"address"}],"name":"withdraw","outputs":[],"stateMutability":"nonpayable","type":"function"}]