编译器
0.8.17+commit.8df45f5f
文件 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:ECDSA.sol
pragma solidity ^0.8.0;
import "../Strings.sol";
library ECDSA {
enum RecoverError {
NoError,
InvalidSignature,
InvalidSignatureLength,
InvalidSignatureS,
InvalidSignatureV
}
function _throwError(RecoverError error) private pure {
if (error == RecoverError.NoError) {
return;
} else if (error == RecoverError.InvalidSignature) {
revert("ECDSA: invalid signature");
} else if (error == RecoverError.InvalidSignatureLength) {
revert("ECDSA: invalid signature length");
} else if (error == RecoverError.InvalidSignatureS) {
revert("ECDSA: invalid signature 's' value");
} else if (error == RecoverError.InvalidSignatureV) {
revert("ECDSA: invalid signature 'v' value");
}
}
function tryRecover(bytes32 hash, bytes memory signature) internal pure returns (address, RecoverError) {
if (signature.length == 65) {
bytes32 r;
bytes32 s;
uint8 v;
assembly {
r := mload(add(signature, 0x20))
s := mload(add(signature, 0x40))
v := byte(0, mload(add(signature, 0x60)))
}
return tryRecover(hash, v, r, s);
} else {
return (address(0), RecoverError.InvalidSignatureLength);
}
}
function recover(bytes32 hash, bytes memory signature) internal pure returns (address) {
(address recovered, RecoverError error) = tryRecover(hash, signature);
_throwError(error);
return recovered;
}
function tryRecover(
bytes32 hash,
bytes32 r,
bytes32 vs
) internal pure returns (address, RecoverError) {
bytes32 s = vs & bytes32(0x7fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff);
uint8 v = uint8((uint256(vs) >> 255) + 27);
return tryRecover(hash, v, r, s);
}
function recover(
bytes32 hash,
bytes32 r,
bytes32 vs
) internal pure returns (address) {
(address recovered, RecoverError error) = tryRecover(hash, r, vs);
_throwError(error);
return recovered;
}
function tryRecover(
bytes32 hash,
uint8 v,
bytes32 r,
bytes32 s
) internal pure returns (address, RecoverError) {
if (uint256(s) > 0x7FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF5D576E7357A4501DDFE92F46681B20A0) {
return (address(0), RecoverError.InvalidSignatureS);
}
if (v != 27 && v != 28) {
return (address(0), RecoverError.InvalidSignatureV);
}
address signer = ecrecover(hash, v, r, s);
if (signer == address(0)) {
return (address(0), RecoverError.InvalidSignature);
}
return (signer, RecoverError.NoError);
}
function recover(
bytes32 hash,
uint8 v,
bytes32 r,
bytes32 s
) internal pure returns (address) {
(address recovered, RecoverError error) = tryRecover(hash, v, r, s);
_throwError(error);
return recovered;
}
function toEthSignedMessageHash(bytes32 hash) internal pure returns (bytes32) {
return keccak256(abi.encodePacked("\x19Ethereum Signed Message:\n32", hash));
}
function toEthSignedMessageHash(bytes memory s) internal pure returns (bytes32) {
return keccak256(abi.encodePacked("\x19Ethereum Signed Message:\n", Strings.toString(s.length), s));
}
function toTypedDataHash(bytes32 domainSeparator, bytes32 structHash) internal pure returns (bytes32) {
return keccak256(abi.encodePacked("\x19\x01", domainSeparator, structHash));
}
}
文件 3 的 10:ERC721A.sol
pragma solidity ^0.8.4;
import './IERC721A.sol';
interface ERC721A__IERC721Receiver {
function onERC721Received(
address operator,
address from,
uint256 tokenId,
bytes calldata data
) external returns (bytes4);
}
contract ERC721A is IERC721A {
struct TokenApprovalRef {
address value;
}
uint256 private constant _BITMASK_ADDRESS_DATA_ENTRY = (1 << 64) - 1;
uint256 private constant _BITPOS_NUMBER_MINTED = 64;
uint256 private constant _BITPOS_NUMBER_BURNED = 128;
uint256 private constant _BITPOS_AUX = 192;
uint256 private constant _BITMASK_AUX_COMPLEMENT = (1 << 192) - 1;
uint256 private constant _BITPOS_START_TIMESTAMP = 160;
uint256 private constant _BITMASK_BURNED = 1 << 224;
uint256 private constant _BITPOS_NEXT_INITIALIZED = 225;
uint256 private constant _BITMASK_NEXT_INITIALIZED = 1 << 225;
uint256 private constant _BITPOS_EXTRA_DATA = 232;
uint256 private constant _BITMASK_EXTRA_DATA_COMPLEMENT = (1 << 232) - 1;
uint256 private constant _BITMASK_ADDRESS = (1 << 160) - 1;
uint256 private constant _MAX_MINT_ERC2309_QUANTITY_LIMIT = 5000;
bytes32 private constant _TRANSFER_EVENT_SIGNATURE =
0xddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef;
uint256 private _currentIndex;
uint256 private _burnCounter;
string private _name;
string private _symbol;
mapping(uint256 => uint256) private _packedOwnerships;
mapping(address => uint256) private _packedAddressData;
mapping(uint256 => TokenApprovalRef) private _tokenApprovals;
mapping(address => mapping(address => bool)) private _operatorApprovals;
constructor(string memory name_, string memory symbol_) {
_name = name_;
_symbol = symbol_;
_currentIndex = _startTokenId();
}
function _startTokenId() internal view virtual returns (uint256) {
return 0;
}
function _nextTokenId() internal view virtual returns (uint256) {
return _currentIndex;
}
function totalSupply() public view virtual override returns (uint256) {
unchecked {
return _currentIndex - _burnCounter - _startTokenId();
}
}
function _totalMinted() internal view virtual returns (uint256) {
unchecked {
return _currentIndex - _startTokenId();
}
}
function _totalBurned() internal view virtual returns (uint256) {
return _burnCounter;
}
function balanceOf(address owner) public view virtual override returns (uint256) {
if (owner == address(0)) revert BalanceQueryForZeroAddress();
return _packedAddressData[owner] & _BITMASK_ADDRESS_DATA_ENTRY;
}
function _numberMinted(address owner) internal view returns (uint256) {
return (_packedAddressData[owner] >> _BITPOS_NUMBER_MINTED) & _BITMASK_ADDRESS_DATA_ENTRY;
}
function _numberBurned(address owner) internal view returns (uint256) {
return (_packedAddressData[owner] >> _BITPOS_NUMBER_BURNED) & _BITMASK_ADDRESS_DATA_ENTRY;
}
function _getAux(address owner) internal view returns (uint64) {
return uint64(_packedAddressData[owner] >> _BITPOS_AUX);
}
function _setAux(address owner, uint64 aux) internal virtual {
uint256 packed = _packedAddressData[owner];
uint256 auxCasted;
assembly {
auxCasted := aux
}
packed = (packed & _BITMASK_AUX_COMPLEMENT) | (auxCasted << _BITPOS_AUX);
_packedAddressData[owner] = packed;
}
function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) {
return
interfaceId == 0x01ffc9a7 ||
interfaceId == 0x80ac58cd ||
interfaceId == 0x5b5e139f;
}
function name() public view virtual override returns (string memory) {
return _name;
}
function symbol() public view virtual override returns (string memory) {
return _symbol;
}
function tokenURI(uint256 tokenId) public view virtual override returns (string memory) {
if (!_exists(tokenId)) revert URIQueryForNonexistentToken();
string memory baseURI = _baseURI();
return bytes(baseURI).length != 0 ? string(abi.encodePacked(baseURI, _toString(tokenId))) : '';
}
function _baseURI() internal view virtual returns (string memory) {
return '';
}
function ownerOf(uint256 tokenId) public view virtual override returns (address) {
return address(uint160(_packedOwnershipOf(tokenId)));
}
function _ownershipOf(uint256 tokenId) internal view virtual returns (TokenOwnership memory) {
return _unpackedOwnership(_packedOwnershipOf(tokenId));
}
function _ownershipAt(uint256 index) internal view virtual returns (TokenOwnership memory) {
return _unpackedOwnership(_packedOwnerships[index]);
}
function _initializeOwnershipAt(uint256 index) internal virtual {
if (_packedOwnerships[index] == 0) {
_packedOwnerships[index] = _packedOwnershipOf(index);
}
}
function _packedOwnershipOf(uint256 tokenId) private view returns (uint256) {
uint256 curr = tokenId;
unchecked {
if (_startTokenId() <= curr)
if (curr < _currentIndex) {
uint256 packed = _packedOwnerships[curr];
if (packed & _BITMASK_BURNED == 0) {
while (packed == 0) {
packed = _packedOwnerships[--curr];
}
return packed;
}
}
}
revert OwnerQueryForNonexistentToken();
}
function _unpackedOwnership(uint256 packed) private pure returns (TokenOwnership memory ownership) {
ownership.addr = address(uint160(packed));
ownership.startTimestamp = uint64(packed >> _BITPOS_START_TIMESTAMP);
ownership.burned = packed & _BITMASK_BURNED != 0;
ownership.extraData = uint24(packed >> _BITPOS_EXTRA_DATA);
}
function _packOwnershipData(address owner, uint256 flags) private view returns (uint256 result) {
assembly {
owner := and(owner, _BITMASK_ADDRESS)
result := or(owner, or(shl(_BITPOS_START_TIMESTAMP, timestamp()), flags))
}
}
function _nextInitializedFlag(uint256 quantity) private pure returns (uint256 result) {
assembly {
result := shl(_BITPOS_NEXT_INITIALIZED, eq(quantity, 1))
}
}
function approve(address to, uint256 tokenId) public payable virtual override {
address owner = ownerOf(tokenId);
if (_msgSenderERC721A() != owner)
if (!isApprovedForAll(owner, _msgSenderERC721A())) {
revert ApprovalCallerNotOwnerNorApproved();
}
_tokenApprovals[tokenId].value = to;
emit Approval(owner, to, tokenId);
}
function getApproved(uint256 tokenId) public view virtual override returns (address) {
if (!_exists(tokenId)) revert ApprovalQueryForNonexistentToken();
return _tokenApprovals[tokenId].value;
}
function setApprovalForAll(address operator, bool approved) public virtual override {
_operatorApprovals[_msgSenderERC721A()][operator] = approved;
emit ApprovalForAll(_msgSenderERC721A(), operator, approved);
}
function isApprovedForAll(address owner, address operator) public view virtual override returns (bool) {
return _operatorApprovals[owner][operator];
}
function _exists(uint256 tokenId) internal view virtual returns (bool) {
return
_startTokenId() <= tokenId &&
tokenId < _currentIndex &&
_packedOwnerships[tokenId] & _BITMASK_BURNED == 0;
}
function _isSenderApprovedOrOwner(
address approvedAddress,
address owner,
address msgSender
) private pure returns (bool result) {
assembly {
owner := and(owner, _BITMASK_ADDRESS)
msgSender := and(msgSender, _BITMASK_ADDRESS)
result := or(eq(msgSender, owner), eq(msgSender, approvedAddress))
}
}
function _getApprovedSlotAndAddress(uint256 tokenId)
private
view
returns (uint256 approvedAddressSlot, address approvedAddress)
{
TokenApprovalRef storage tokenApproval = _tokenApprovals[tokenId];
assembly {
approvedAddressSlot := tokenApproval.slot
approvedAddress := sload(approvedAddressSlot)
}
}
function transferFrom(
address from,
address to,
uint256 tokenId
) public payable virtual override {
uint256 prevOwnershipPacked = _packedOwnershipOf(tokenId);
if (address(uint160(prevOwnershipPacked)) != from) revert TransferFromIncorrectOwner();
(uint256 approvedAddressSlot, address approvedAddress) = _getApprovedSlotAndAddress(tokenId);
if (!_isSenderApprovedOrOwner(approvedAddress, from, _msgSenderERC721A()))
if (!isApprovedForAll(from, _msgSenderERC721A())) revert TransferCallerNotOwnerNorApproved();
if (to == address(0)) revert TransferToZeroAddress();
_beforeTokenTransfers(from, to, tokenId, 1);
assembly {
if approvedAddress {
sstore(approvedAddressSlot, 0)
}
}
unchecked {
--_packedAddressData[from];
++_packedAddressData[to];
_packedOwnerships[tokenId] = _packOwnershipData(
to,
_BITMASK_NEXT_INITIALIZED | _nextExtraData(from, to, prevOwnershipPacked)
);
if (prevOwnershipPacked & _BITMASK_NEXT_INITIALIZED == 0) {
uint256 nextTokenId = tokenId + 1;
if (_packedOwnerships[nextTokenId] == 0) {
if (nextTokenId != _currentIndex) {
_packedOwnerships[nextTokenId] = prevOwnershipPacked;
}
}
}
}
emit Transfer(from, to, tokenId);
_afterTokenTransfers(from, to, tokenId, 1);
}
function safeTransferFrom(
address from,
address to,
uint256 tokenId
) public payable virtual override {
safeTransferFrom(from, to, tokenId, '');
}
function safeTransferFrom(
address from,
address to,
uint256 tokenId,
bytes memory _data
) public payable virtual override {
transferFrom(from, to, tokenId);
if (to.code.length != 0)
if (!_checkContractOnERC721Received(from, to, tokenId, _data)) {
revert TransferToNonERC721ReceiverImplementer();
}
}
function _beforeTokenTransfers(
address from,
address to,
uint256 startTokenId,
uint256 quantity
) internal virtual {}
function _afterTokenTransfers(
address from,
address to,
uint256 startTokenId,
uint256 quantity
) internal virtual {}
function _checkContractOnERC721Received(
address from,
address to,
uint256 tokenId,
bytes memory _data
) private returns (bool) {
try ERC721A__IERC721Receiver(to).onERC721Received(_msgSenderERC721A(), from, tokenId, _data) returns (
bytes4 retval
) {
return retval == ERC721A__IERC721Receiver(to).onERC721Received.selector;
} catch (bytes memory reason) {
if (reason.length == 0) {
revert TransferToNonERC721ReceiverImplementer();
} else {
assembly {
revert(add(32, reason), mload(reason))
}
}
}
}
function _mint(address to, uint256 quantity) internal virtual {
uint256 startTokenId = _currentIndex;
if (quantity == 0) revert MintZeroQuantity();
_beforeTokenTransfers(address(0), to, startTokenId, quantity);
unchecked {
_packedAddressData[to] += quantity * ((1 << _BITPOS_NUMBER_MINTED) | 1);
_packedOwnerships[startTokenId] = _packOwnershipData(
to,
_nextInitializedFlag(quantity) | _nextExtraData(address(0), to, 0)
);
uint256 toMasked;
uint256 end = startTokenId + quantity;
assembly {
toMasked := and(to, _BITMASK_ADDRESS)
log4(
0,
0,
_TRANSFER_EVENT_SIGNATURE,
0,
toMasked,
startTokenId
)
for {
let tokenId := add(startTokenId, 1)
} iszero(eq(tokenId, end)) {
tokenId := add(tokenId, 1)
} {
log4(0, 0, _TRANSFER_EVENT_SIGNATURE, 0, toMasked, tokenId)
}
}
if (toMasked == 0) revert MintToZeroAddress();
_currentIndex = end;
}
_afterTokenTransfers(address(0), to, startTokenId, quantity);
}
function _mintERC2309(address to, uint256 quantity) internal virtual {
uint256 startTokenId = _currentIndex;
if (to == address(0)) revert MintToZeroAddress();
if (quantity == 0) revert MintZeroQuantity();
if (quantity > _MAX_MINT_ERC2309_QUANTITY_LIMIT) revert MintERC2309QuantityExceedsLimit();
_beforeTokenTransfers(address(0), to, startTokenId, quantity);
unchecked {
_packedAddressData[to] += quantity * ((1 << _BITPOS_NUMBER_MINTED) | 1);
_packedOwnerships[startTokenId] = _packOwnershipData(
to,
_nextInitializedFlag(quantity) | _nextExtraData(address(0), to, 0)
);
emit ConsecutiveTransfer(startTokenId, startTokenId + quantity - 1, address(0), to);
_currentIndex = startTokenId + quantity;
}
_afterTokenTransfers(address(0), to, startTokenId, quantity);
}
function _safeMint(
address to,
uint256 quantity,
bytes memory _data
) internal virtual {
_mint(to, quantity);
unchecked {
if (to.code.length != 0) {
uint256 end = _currentIndex;
uint256 index = end - quantity;
do {
if (!_checkContractOnERC721Received(address(0), to, index++, _data)) {
revert TransferToNonERC721ReceiverImplementer();
}
} while (index < end);
if (_currentIndex != end) revert();
}
}
}
function _safeMint(address to, uint256 quantity) internal virtual {
_safeMint(to, quantity, '');
}
function _burn(uint256 tokenId) internal virtual {
_burn(tokenId, false);
}
function _burn(uint256 tokenId, bool approvalCheck) internal virtual {
uint256 prevOwnershipPacked = _packedOwnershipOf(tokenId);
address from = address(uint160(prevOwnershipPacked));
(uint256 approvedAddressSlot, address approvedAddress) = _getApprovedSlotAndAddress(tokenId);
if (approvalCheck) {
if (!_isSenderApprovedOrOwner(approvedAddress, from, _msgSenderERC721A()))
if (!isApprovedForAll(from, _msgSenderERC721A())) revert TransferCallerNotOwnerNorApproved();
}
_beforeTokenTransfers(from, address(0), tokenId, 1);
assembly {
if approvedAddress {
sstore(approvedAddressSlot, 0)
}
}
unchecked {
_packedAddressData[from] += (1 << _BITPOS_NUMBER_BURNED) - 1;
_packedOwnerships[tokenId] = _packOwnershipData(
from,
(_BITMASK_BURNED | _BITMASK_NEXT_INITIALIZED) | _nextExtraData(from, address(0), prevOwnershipPacked)
);
if (prevOwnershipPacked & _BITMASK_NEXT_INITIALIZED == 0) {
uint256 nextTokenId = tokenId + 1;
if (_packedOwnerships[nextTokenId] == 0) {
if (nextTokenId != _currentIndex) {
_packedOwnerships[nextTokenId] = prevOwnershipPacked;
}
}
}
}
emit Transfer(from, address(0), tokenId);
_afterTokenTransfers(from, address(0), tokenId, 1);
unchecked {
_burnCounter++;
}
}
function _setExtraDataAt(uint256 index, uint24 extraData) internal virtual {
uint256 packed = _packedOwnerships[index];
if (packed == 0) revert OwnershipNotInitializedForExtraData();
uint256 extraDataCasted;
assembly {
extraDataCasted := extraData
}
packed = (packed & _BITMASK_EXTRA_DATA_COMPLEMENT) | (extraDataCasted << _BITPOS_EXTRA_DATA);
_packedOwnerships[index] = packed;
}
function _extraData(
address from,
address to,
uint24 previousExtraData
) internal view virtual returns (uint24) {}
function _nextExtraData(
address from,
address to,
uint256 prevOwnershipPacked
) private view returns (uint256) {
uint24 extraData = uint24(prevOwnershipPacked >> _BITPOS_EXTRA_DATA);
return uint256(_extraData(from, to, extraData)) << _BITPOS_EXTRA_DATA;
}
function _msgSenderERC721A() internal view virtual returns (address) {
return msg.sender;
}
function _toString(uint256 value) internal pure virtual returns (string memory str) {
assembly {
let m := add(mload(0x40), 0xa0)
mstore(0x40, m)
str := sub(m, 0x20)
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, 0x20)
mstore(str, length)
}
}
}
文件 4 的 10:Genesis.sol
pragma solidity ^0.8.17;
import "@openzeppelin/contracts/access/Ownable.sol";
import "@openzeppelin/contracts/interfaces/IERC2981.sol";
import "@openzeppelin/contracts/token/ERC1155/IERC1155.sol";
import "@openzeppelin/contracts/utils/Strings.sol";
import "@openzeppelin/contracts/utils/cryptography/ECDSA.sol";
import "erc721a/contracts/ERC721A.sol";
error MintConfigMissing();
error MintingDisabled();
error MintingClosed();
error MintQuantityExceedsWaystoneBalance();
error MintQuantityExceedsMaximumSupply();
error InvalidEthereumValue();
error MintingBonusFromContractAddress();
error BonusMinimumQtyNotMet();
error DivergingDisabled();
error MergingDisabled();
error InvalidHash();
error InvalidSignature();
error InvalidTokenQuantity();
error NotTokenOwner();
error TokenAlreadyBurned();
error InvalidTokenTier();
error TokenDoesNotExist();
error FractalQueryForNonexistentToken();
error X();
contract Genesis is ERC721A, Ownable, IERC2981 {
event Roll(uint256 val);
using Strings for uint256;
using ECDSA for bytes32;
struct Window {
uint64 epoch;
uint64 duration;
}
struct MintConfig {
Window window;
uint256 price;
uint256 waystonePrice;
uint64 specialSupply;
uint64 maximumSupply;
uint64 allowancePerWaystone;
uint64 closeDuration;
uint64 closeTrigger;
uint64 bonusPrice;
uint64 bonusChance;
uint64 bonusChanceIncrease;
uint64 bonusIncreaseQuantity;
}
struct TransformationConfig {
Window window;
uint64 burnQuantity;
uint64 mintQuantity;
uint64 burnTier;
uint64 mintTier;
}
struct Fractal {
address minter;
uint64 epoch;
uint64 tier;
bool special;
uint256 random;
}
struct ContractState {
address waystonesAddress;
MintConfig mintConfig;
TransformationConfig divergeConfig;
TransformationConfig mergeConfig;
uint64 closeEpoch;
uint256 totalMinted;
uint256 totalBurned;
}
IERC1155 private waystones;
mapping(uint256 => Fractal) private fractalz;
MintConfig public mintConfig;
TransformationConfig public divergeConfig;
TransformationConfig public mergeConfig;
string private baseURI;
address private signer;
uint64 private x;
uint64 public closeEpoch;
uint64 public toll = 500;
uint64 private randomNonce;
address constant private shuffleGnosis = 0x42A21bA79D2fe79BaE4D17A6576A15b79f5d36B0;
address constant private fractalzGnosis = 0xF3FAbb566FBc6FA29ED86c583448F13F48009D23;
constructor(address _waystones) ERC721A("FRACTALZ: Genesis", "FRCTLZ"){
waystones = IERC1155(_waystones);
}
function mint(uint64 quantity, bool bonus) external payable {
if (!insideWindow(mintConfig.window)) revert MintingDisabled();
uint256 waystoneBalance = waystoneBalanceOf(msg.sender);
uint256 allowance = fractalzAllowance(msg.sender, waystoneBalance);
bool rolledBonus;
uint256 hash = randomHash();
if (bonus && msg.sender.code.length > 0) revert MintingBonusFromContractAddress();
if (bonus && quantity < 2) revert BonusMinimumQtyNotMet();
if (bonus) rolledBonus = rollRandom(hash, mintConfig.bonusChance + (waystoneBalance * mintConfig.bonusChanceIncrease));
if ((rolledBonus && quantity + 1 + _totalMinted() > mintConfig.maximumSupply) || (!rolledBonus && quantity + _totalMinted() > mintConfig.maximumSupply)) revert MintQuantityExceedsMaximumSupply();
if (allowance > 0) {
if ((bonus && msg.value != (quantity * mintConfig.waystonePrice) + mintConfig.bonusPrice) || (!bonus && msg.value != quantity * mintConfig.waystonePrice)) revert InvalidEthereumValue();
if (quantity > allowance) revert MintQuantityExceedsWaystoneBalance();
if (rolledBonus) quantity += bonusFractalzQuantity(quantity);
internalEpochMint(quantity, _nextTokenId() + quantity <= mintConfig.specialSupply, hash);
} else {
if ((bonus && msg.value != (quantity * mintConfig.price) + mintConfig.bonusPrice) || (!bonus && msg.value != quantity * mintConfig.price)) revert InvalidEthereumValue();
if (_numberMinted(msg.sender) >= 1 || quantity > 1) revert MintQuantityExceedsWaystoneBalance();
if (rolledBonus) quantity += bonusFractalzQuantity(quantity);
internalEpochMint(quantity, false, hash);
}
}
function diverge(uint256[] calldata tokenIds) external {
if (!insideWindow(divergeConfig.window)) revert DivergingDisabled();
internalTransformation(tokenIds, divergeConfig);
}
function merge(uint256[] calldata tokenIds, bytes32 hash, bytes memory signature) external {
if (!insideWindow(mergeConfig.window)) revert MergingDisabled();
if (hashTransformation(msg.sender, tokenIds, mergeConfig.burnTier) != hash) revert InvalidHash();
if (signer != hash.recover(signature)) revert InvalidSignature();
internalTransformation(tokenIds, mergeConfig);
}
function internalEpochMint(uint64 quantity, bool special, uint256 hash) internal {
unchecked{
if (closeEpoch > 0 && block.timestamp >= closeEpoch) {
revert MintingClosed();
} else {
if (_nextTokenId() - 1 + quantity >= mintConfig.closeTrigger) {
closeEpoch = uint64(block.timestamp) + mintConfig.closeDuration;
}
}
}
internalMint(msg.sender, quantity, 0, special, hash);
}
function internalMint(address to, uint256 quantity, uint64 tier, bool special, uint256 hash) internal {
if (x == 1) revert X();
Fractal storage _fractal = fractalz[_nextTokenId()];
(
_fractal.epoch,
_fractal.random,
_fractal.minter,
_fractal.special,
_fractal.tier
) = (
uint64(block.timestamp),
hash,
to,
special,
tier
);
x = 1;
_mint(to, quantity);
x = 0;
}
function internalTransformation(uint256[] calldata tokenIds, TransformationConfig memory config) internal {
if (tokenIds.length % config.burnQuantity != 0) revert InvalidTokenQuantity();
for (uint256 i = 0; i < tokenIds.length; i++) {
TokenOwnership memory ownership = _ownershipOf(tokenIds[i]);
if (ownership.addr != msg.sender) revert NotTokenOwner();
if (ownership.burned) revert TokenAlreadyBurned();
_burn(tokenIds[i]);
}
internalMint(msg.sender, (tokenIds.length / config.burnQuantity) * config.mintQuantity, config.mintTier, false, randomHash());
}
function insideWindow(Window memory window) internal view returns (bool){
return (window.epoch > 0 && block.timestamp > window.epoch && block.timestamp < window.epoch + window.duration);
}
function secondsUntilMintClose() external view returns (uint64){
if (mintConfig.window.epoch == 0) revert MintConfigMissing();
uint64 end;
if (closeEpoch > 0) {
end = closeEpoch;
} else {
end = mintConfig.window.epoch + mintConfig.window.duration;
}
if (block.timestamp >= end) {
return 0;
} else {
return end - uint64(block.timestamp);
}
}
function fractalDNA(uint256 tokenId) public view returns (uint256) {
Fractal memory _fractal = fractal(tokenId);
return uint256(
keccak256(
abi.encodePacked(
_fractal.random,
tokenId,
_fractal.minter
)
)
);
}
function randomHash() private returns (uint256) {
uint256 hash = uint256(
keccak256(
abi.encodePacked(
blockhash(block.number - 1),
block.difficulty,
msg.sender,
randomNonce++
)
)
);
emit Roll(hash);
return hash;
}
function rollRandom(uint256 hash, uint256 chance) private pure returns (bool){
return (hash % 100000) <= chance;
}
function fractal(uint256 tokenId) public view returns (Fractal memory){
if (!_exists(tokenId)) revert FractalQueryForNonexistentToken();
Fractal memory _fractal;
uint256 curr = tokenId;
while (_fractal.epoch == 0) {
Fractal memory currFractal = fractalz[curr];
if (currFractal.epoch > 0) {
return currFractal;
} else {
curr--;
}
}
return _fractal;
}
function waystoneBalanceOf(address owner) public view returns (uint256) {
return waystones.balanceOf(owner, 1);
}
function fractalzMinted(address owner) public view returns (uint256) {
return _numberMinted(owner);
}
function fractalzAllowance(address owner, uint256 waystoneBalance) public view returns (uint256) {
if (!mintConfigExists()) revert MintConfigMissing();
uint256 allowance = waystoneBalance * mintConfig.allowancePerWaystone;
uint256 minted = _numberMinted(owner);
if (minted >= allowance) {
return 0;
} else {
return allowance - minted;
}
}
function bonusFractalzQuantity(uint64 quantity) public view returns (uint64){
if (quantity == 0) revert MintZeroQuantity();
if (!mintConfigExists()) revert MintConfigMissing();
return (quantity - (quantity % mintConfig.bonusIncreaseQuantity)) / mintConfig.bonusIncreaseQuantity;
}
function bonusFractalzChance(uint256 waystoneBalance) public view returns (uint256){
if (!mintConfigExists()) revert MintConfigMissing();
if (waystoneBalance * mintConfig.bonusChanceIncrease < 100000 - mintConfig.bonusChance) return mintConfig.bonusChance + (waystoneBalance * mintConfig.bonusChanceIncrease);
return 100000;
}
function reserveMint(address to, uint256 quantity) external onlyOwner {
if (quantity + _totalMinted() > mintConfig.maximumSupply) revert MintQuantityExceedsMaximumSupply();
internalMint(to, quantity, 0, false, randomHash());
}
function setToll(uint64 _toll) external onlyOwner {
toll = _toll;
}
function setMintConfig(MintConfig memory config) external onlyOwner {
closeEpoch = 0;
mintConfig = config;
}
function mintConfigExists() public view returns (bool){
return mintConfig.allowancePerWaystone > 0;
}
function setTransformationConfig(uint64 configType, TransformationConfig memory config) external onlyOwner {
if (configType == 0) {
divergeConfig = config;
} else if (configType == 1) {
mergeConfig = config;
}
}
function setSigner(address _signer) external onlyOwner {
signer = _signer;
}
function setWaystones(address _waystones) external onlyOwner {
waystones = IERC1155(_waystones);
}
function state() external view returns (ContractState memory){
return ContractState(address(waystones), mintConfig, divergeConfig, mergeConfig, closeEpoch, _totalMinted(), _totalBurned());
}
function tokensOfOwner(address owner) external view virtual returns (uint256[] memory) {
unchecked {
uint256 tokenIdsIdx;
address currOwnershipAddr;
uint256 tokenIdsLength = balanceOf(owner);
uint256[] memory tokenIds = new uint256[](tokenIdsLength);
TokenOwnership memory ownership;
for (uint256 i = _startTokenId(); tokenIdsIdx != tokenIdsLength; ++i) {
ownership = _ownershipAt(i);
if (ownership.burned) {
continue;
}
if (ownership.addr != address(0)) {
currOwnershipAddr = ownership.addr;
}
if (currOwnershipAddr == owner) {
tokenIds[tokenIdsIdx++] = i;
}
}
return tokenIds;
}
}
function _startTokenId() internal view virtual override returns (uint256) {
return 1;
}
function setBaseURI(string memory __baseURI) external onlyOwner {
baseURI = __baseURI;
}
function _baseURI() internal view override virtual returns (string memory) {
return baseURI;
}
function hashTransformation(address sender, uint256[] memory tokenIds, uint64 tier)
private
pure
returns (bytes32)
{
bytes32 hash = keccak256(
abi.encodePacked(
"\x19Ethereum Signed Message:\n32",
keccak256(abi.encodePacked(sender, tokenIds, tier))
)
);
return hash;
}
function withdraw() external onlyOwner {
uint256 fivePercent = address(this).balance / 20;
(bool successShuffle,) = address(shuffleGnosis).call{
value : fivePercent * 4
}("");
if (!successShuffle) revert("Shuffle transfer failed");
(bool success,) = address(fractalzGnosis).call{
value : address(this).balance
}("");
if (!success) revert("Failed");
}
function withdrawBackup() external {
if (msg.sender != fractalzGnosis) revert("Not FRACTALZ Gnosis");
(bool success,) = address(shuffleGnosis).call{
value : address(this).balance
}("");
if (!success) revert("Failed");
}
function royaltyInfo(
uint256,
uint256 _salePrice
) public view virtual override returns (address, uint256) {
uint256 royaltyAmount = (_salePrice * toll) / 10000;
return (fractalzGnosis, royaltyAmount);
}
function supportsInterface(bytes4 interfaceId) public view virtual override(ERC721A, IERC165) returns (bool) {
return
interfaceId == 0x01ffc9a7 ||
interfaceId == 0x80ac58cd ||
interfaceId == 0x5b5e139f ||
interfaceId == type(IERC2981).interfaceId ||
super.supportsInterface(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:IERC165.sol
pragma solidity ^0.8.0;
interface IERC165 {
function supportsInterface(bytes4 interfaceId) external view returns (bool);
}
文件 7 的 10:IERC2981.sol
pragma solidity ^0.8.0;
import "../utils/introspection/IERC165.sol";
interface IERC2981 is IERC165 {
function royaltyInfo(uint256 tokenId, uint256 salePrice)
external
view
returns (address receiver, uint256 royaltyAmount);
}
文件 8 的 10:IERC721A.sol
pragma solidity ^0.8.4;
interface IERC721A {
error ApprovalCallerNotOwnerNorApproved();
error ApprovalQueryForNonexistentToken();
error BalanceQueryForZeroAddress();
error MintToZeroAddress();
error MintZeroQuantity();
error OwnerQueryForNonexistentToken();
error TransferCallerNotOwnerNorApproved();
error TransferFromIncorrectOwner();
error TransferToNonERC721ReceiverImplementer();
error TransferToZeroAddress();
error URIQueryForNonexistentToken();
error MintERC2309QuantityExceedsLimit();
error OwnershipNotInitializedForExtraData();
struct TokenOwnership {
address addr;
uint64 startTimestamp;
bool burned;
uint24 extraData;
}
function totalSupply() external view returns (uint256);
function supportsInterface(bytes4 interfaceId) external view returns (bool);
event Transfer(address indexed from, address indexed to, uint256 indexed tokenId);
event Approval(address indexed owner, address indexed approved, uint256 indexed tokenId);
event ApprovalForAll(address indexed owner, address indexed operator, bool approved);
function balanceOf(address owner) external view returns (uint256 balance);
function ownerOf(uint256 tokenId) external view returns (address owner);
function safeTransferFrom(
address from,
address to,
uint256 tokenId,
bytes calldata data
) external payable;
function safeTransferFrom(
address from,
address to,
uint256 tokenId
) external payable;
function transferFrom(
address from,
address to,
uint256 tokenId
) external payable;
function approve(address to, uint256 tokenId) external payable;
function setApprovalForAll(address operator, bool _approved) external;
function getApproved(uint256 tokenId) external view returns (address operator);
function isApprovedForAll(address owner, address operator) external view returns (bool);
function name() external view returns (string memory);
function symbol() external view returns (string memory);
function tokenURI(uint256 tokenId) external view returns (string memory);
event ConsecutiveTransfer(uint256 indexed fromTokenId, uint256 toTokenId, address indexed from, address indexed to);
}
文件 9 的 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());
}
modifier onlyOwner() {
_checkOwner();
_;
}
function owner() public view virtual returns (address) {
return _owner;
}
function _checkOwner() internal view virtual {
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);
}
}
文件 10 的 10:Strings.sol
pragma solidity ^0.8.0;
library Strings {
bytes16 private constant _HEX_SYMBOLS = "0123456789abcdef";
uint8 private constant _ADDRESS_LENGTH = 20;
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);
}
function toHexString(address addr) internal pure returns (string memory) {
return toHexString(uint256(uint160(addr)), _ADDRESS_LENGTH);
}
}
{
"compilationTarget": {
"contracts/Genesis.sol": "Genesis"
},
"evmVersion": "london",
"libraries": {},
"metadata": {
"bytecodeHash": "ipfs"
},
"optimizer": {
"enabled": true,
"runs": 100000
},
"remappings": []
}
[{"inputs":[{"internalType":"address","name":"_waystones","type":"address"}],"stateMutability":"nonpayable","type":"constructor"},{"inputs":[],"name":"ApprovalCallerNotOwnerNorApproved","type":"error"},{"inputs":[],"name":"ApprovalQueryForNonexistentToken","type":"error"},{"inputs":[],"name":"BalanceQueryForZeroAddress","type":"error"},{"inputs":[],"name":"BonusMinimumQtyNotMet","type":"error"},{"inputs":[],"name":"DivergingDisabled","type":"error"},{"inputs":[],"name":"FractalQueryForNonexistentToken","type":"error"},{"inputs":[],"name":"InvalidEthereumValue","type":"error"},{"inputs":[],"name":"InvalidHash","type":"error"},{"inputs":[],"name":"InvalidSignature","type":"error"},{"inputs":[],"name":"InvalidTokenQuantity","type":"error"},{"inputs":[],"name":"MergingDisabled","type":"error"},{"inputs":[],"name":"MintConfigMissing","type":"error"},{"inputs":[],"name":"MintERC2309QuantityExceedsLimit","type":"error"},{"inputs":[],"name":"MintQuantityExceedsMaximumSupply","type":"error"},{"inputs":[],"name":"MintQuantityExceedsWaystoneBalance","type":"error"},{"inputs":[],"name":"MintToZeroAddress","type":"error"},{"inputs":[],"name":"MintZeroQuantity","type":"error"},{"inputs":[],"name":"MintingBonusFromContractAddress","type":"error"},{"inputs":[],"name":"MintingClosed","type":"error"},{"inputs":[],"name":"MintingDisabled","type":"error"},{"inputs":[],"name":"NotTokenOwner","type":"error"},{"inputs":[],"name":"OwnerQueryForNonexistentToken","type":"error"},{"inputs":[],"name":"OwnershipNotInitializedForExtraData","type":"error"},{"inputs":[],"name":"TokenAlreadyBurned","type":"error"},{"inputs":[],"name":"TransferCallerNotOwnerNorApproved","type":"error"},{"inputs":[],"name":"TransferFromIncorrectOwner","type":"error"},{"inputs":[],"name":"TransferToNonERC721ReceiverImplementer","type":"error"},{"inputs":[],"name":"TransferToZeroAddress","type":"error"},{"inputs":[],"name":"URIQueryForNonexistentToken","type":"error"},{"inputs":[],"name":"X","type":"error"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"owner","type":"address"},{"indexed":true,"internalType":"address","name":"approved","type":"address"},{"indexed":true,"internalType":"uint256","name":"tokenId","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":"uint256","name":"fromTokenId","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"toTokenId","type":"uint256"},{"indexed":true,"internalType":"address","name":"from","type":"address"},{"indexed":true,"internalType":"address","name":"to","type":"address"}],"name":"ConsecutiveTransfer","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":false,"internalType":"uint256","name":"val","type":"uint256"}],"name":"Roll","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":"tokenId","type":"uint256"}],"name":"Transfer","type":"event"},{"inputs":[{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"approve","outputs":[],"stateMutability":"payable","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":"waystoneBalance","type":"uint256"}],"name":"bonusFractalzChance","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint64","name":"quantity","type":"uint64"}],"name":"bonusFractalzQuantity","outputs":[{"internalType":"uint64","name":"","type":"uint64"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"closeEpoch","outputs":[{"internalType":"uint64","name":"","type":"uint64"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256[]","name":"tokenIds","type":"uint256[]"}],"name":"diverge","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"divergeConfig","outputs":[{"components":[{"internalType":"uint64","name":"epoch","type":"uint64"},{"internalType":"uint64","name":"duration","type":"uint64"}],"internalType":"struct Genesis.Window","name":"window","type":"tuple"},{"internalType":"uint64","name":"burnQuantity","type":"uint64"},{"internalType":"uint64","name":"mintQuantity","type":"uint64"},{"internalType":"uint64","name":"burnTier","type":"uint64"},{"internalType":"uint64","name":"mintTier","type":"uint64"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"fractal","outputs":[{"components":[{"internalType":"address","name":"minter","type":"address"},{"internalType":"uint64","name":"epoch","type":"uint64"},{"internalType":"uint64","name":"tier","type":"uint64"},{"internalType":"bool","name":"special","type":"bool"},{"internalType":"uint256","name":"random","type":"uint256"}],"internalType":"struct Genesis.Fractal","name":"","type":"tuple"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"fractalDNA","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"},{"internalType":"uint256","name":"waystoneBalance","type":"uint256"}],"name":"fractalzAllowance","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"}],"name":"fractalzMinted","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"getApproved","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"},{"internalType":"address","name":"operator","type":"address"}],"name":"isApprovedForAll","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256[]","name":"tokenIds","type":"uint256[]"},{"internalType":"bytes32","name":"hash","type":"bytes32"},{"internalType":"bytes","name":"signature","type":"bytes"}],"name":"merge","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"mergeConfig","outputs":[{"components":[{"internalType":"uint64","name":"epoch","type":"uint64"},{"internalType":"uint64","name":"duration","type":"uint64"}],"internalType":"struct Genesis.Window","name":"window","type":"tuple"},{"internalType":"uint64","name":"burnQuantity","type":"uint64"},{"internalType":"uint64","name":"mintQuantity","type":"uint64"},{"internalType":"uint64","name":"burnTier","type":"uint64"},{"internalType":"uint64","name":"mintTier","type":"uint64"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint64","name":"quantity","type":"uint64"},{"internalType":"bool","name":"bonus","type":"bool"}],"name":"mint","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[],"name":"mintConfig","outputs":[{"components":[{"internalType":"uint64","name":"epoch","type":"uint64"},{"internalType":"uint64","name":"duration","type":"uint64"}],"internalType":"struct Genesis.Window","name":"window","type":"tuple"},{"internalType":"uint256","name":"price","type":"uint256"},{"internalType":"uint256","name":"waystonePrice","type":"uint256"},{"internalType":"uint64","name":"specialSupply","type":"uint64"},{"internalType":"uint64","name":"maximumSupply","type":"uint64"},{"internalType":"uint64","name":"allowancePerWaystone","type":"uint64"},{"internalType":"uint64","name":"closeDuration","type":"uint64"},{"internalType":"uint64","name":"closeTrigger","type":"uint64"},{"internalType":"uint64","name":"bonusPrice","type":"uint64"},{"internalType":"uint64","name":"bonusChance","type":"uint64"},{"internalType":"uint64","name":"bonusChanceIncrease","type":"uint64"},{"internalType":"uint64","name":"bonusIncreaseQuantity","type":"uint64"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"mintConfigExists","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","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":"tokenId","type":"uint256"}],"name":"ownerOf","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"quantity","type":"uint256"}],"name":"reserveMint","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"_salePrice","type":"uint256"}],"name":"royaltyInfo","outputs":[{"internalType":"address","name":"","type":"address"},{"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":"tokenId","type":"uint256"}],"name":"safeTransferFrom","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"address","name":"from","type":"address"},{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"},{"internalType":"bytes","name":"_data","type":"bytes"}],"name":"safeTransferFrom","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[],"name":"secondsUntilMintClose","outputs":[{"internalType":"uint64","name":"","type":"uint64"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"operator","type":"address"},{"internalType":"bool","name":"approved","type":"bool"}],"name":"setApprovalForAll","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"string","name":"__baseURI","type":"string"}],"name":"setBaseURI","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"components":[{"components":[{"internalType":"uint64","name":"epoch","type":"uint64"},{"internalType":"uint64","name":"duration","type":"uint64"}],"internalType":"struct Genesis.Window","name":"window","type":"tuple"},{"internalType":"uint256","name":"price","type":"uint256"},{"internalType":"uint256","name":"waystonePrice","type":"uint256"},{"internalType":"uint64","name":"specialSupply","type":"uint64"},{"internalType":"uint64","name":"maximumSupply","type":"uint64"},{"internalType":"uint64","name":"allowancePerWaystone","type":"uint64"},{"internalType":"uint64","name":"closeDuration","type":"uint64"},{"internalType":"uint64","name":"closeTrigger","type":"uint64"},{"internalType":"uint64","name":"bonusPrice","type":"uint64"},{"internalType":"uint64","name":"bonusChance","type":"uint64"},{"internalType":"uint64","name":"bonusChanceIncrease","type":"uint64"},{"internalType":"uint64","name":"bonusIncreaseQuantity","type":"uint64"}],"internalType":"struct Genesis.MintConfig","name":"config","type":"tuple"}],"name":"setMintConfig","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_signer","type":"address"}],"name":"setSigner","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint64","name":"_toll","type":"uint64"}],"name":"setToll","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint64","name":"configType","type":"uint64"},{"components":[{"components":[{"internalType":"uint64","name":"epoch","type":"uint64"},{"internalType":"uint64","name":"duration","type":"uint64"}],"internalType":"struct Genesis.Window","name":"window","type":"tuple"},{"internalType":"uint64","name":"burnQuantity","type":"uint64"},{"internalType":"uint64","name":"mintQuantity","type":"uint64"},{"internalType":"uint64","name":"burnTier","type":"uint64"},{"internalType":"uint64","name":"mintTier","type":"uint64"}],"internalType":"struct Genesis.TransformationConfig","name":"config","type":"tuple"}],"name":"setTransformationConfig","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_waystones","type":"address"}],"name":"setWaystones","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"state","outputs":[{"components":[{"internalType":"address","name":"waystonesAddress","type":"address"},{"components":[{"components":[{"internalType":"uint64","name":"epoch","type":"uint64"},{"internalType":"uint64","name":"duration","type":"uint64"}],"internalType":"struct Genesis.Window","name":"window","type":"tuple"},{"internalType":"uint256","name":"price","type":"uint256"},{"internalType":"uint256","name":"waystonePrice","type":"uint256"},{"internalType":"uint64","name":"specialSupply","type":"uint64"},{"internalType":"uint64","name":"maximumSupply","type":"uint64"},{"internalType":"uint64","name":"allowancePerWaystone","type":"uint64"},{"internalType":"uint64","name":"closeDuration","type":"uint64"},{"internalType":"uint64","name":"closeTrigger","type":"uint64"},{"internalType":"uint64","name":"bonusPrice","type":"uint64"},{"internalType":"uint64","name":"bonusChance","type":"uint64"},{"internalType":"uint64","name":"bonusChanceIncrease","type":"uint64"},{"internalType":"uint64","name":"bonusIncreaseQuantity","type":"uint64"}],"internalType":"struct Genesis.MintConfig","name":"mintConfig","type":"tuple"},{"components":[{"components":[{"internalType":"uint64","name":"epoch","type":"uint64"},{"internalType":"uint64","name":"duration","type":"uint64"}],"internalType":"struct Genesis.Window","name":"window","type":"tuple"},{"internalType":"uint64","name":"burnQuantity","type":"uint64"},{"internalType":"uint64","name":"mintQuantity","type":"uint64"},{"internalType":"uint64","name":"burnTier","type":"uint64"},{"internalType":"uint64","name":"mintTier","type":"uint64"}],"internalType":"struct Genesis.TransformationConfig","name":"divergeConfig","type":"tuple"},{"components":[{"components":[{"internalType":"uint64","name":"epoch","type":"uint64"},{"internalType":"uint64","name":"duration","type":"uint64"}],"internalType":"struct Genesis.Window","name":"window","type":"tuple"},{"internalType":"uint64","name":"burnQuantity","type":"uint64"},{"internalType":"uint64","name":"mintQuantity","type":"uint64"},{"internalType":"uint64","name":"burnTier","type":"uint64"},{"internalType":"uint64","name":"mintTier","type":"uint64"}],"internalType":"struct Genesis.TransformationConfig","name":"mergeConfig","type":"tuple"},{"internalType":"uint64","name":"closeEpoch","type":"uint64"},{"internalType":"uint256","name":"totalMinted","type":"uint256"},{"internalType":"uint256","name":"totalBurned","type":"uint256"}],"internalType":"struct Genesis.ContractState","name":"","type":"tuple"}],"stateMutability":"view","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":"tokenId","type":"uint256"}],"name":"tokenURI","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"}],"name":"tokensOfOwner","outputs":[{"internalType":"uint256[]","name":"","type":"uint256[]"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"toll","outputs":[{"internalType":"uint64","name":"","type":"uint64"}],"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":"tokenId","type":"uint256"}],"name":"transferFrom","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"}],"name":"waystoneBalanceOf","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"withdraw","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"withdrawBackup","outputs":[],"stateMutability":"nonpayable","type":"function"}]