ETH Price: $3,415.74 (+3.96%)

Token

PERFUN (PERFUN)
 

Overview

Max Total Supply

300 PERFUN

Holders

274

Market

Volume (24H)

N/A

Min Price (24H)

N/A

Max Price (24H)

N/A
Balance
1 PERFUN
0x90F952E7222b3C5FD6A429230c131B2189762Da2
Loading...
Loading
Loading...
Loading
Loading...
Loading

Click here to update the token information / general information
# Exchange Pair Price  24H Volume % Volume

Contract Source Code Verified (Exact Match)

Contract Name:
PERFUN

Compiler Version
v0.8.7+commit.e28d00a7

Optimization Enabled:
Yes with 200 runs

Other Settings:
default evmVersion, Unlicense license

Contract Source Code (Solidity)

/**
 *Submitted for verification at Etherscan.io on 2022-07-13
*/

// SPDX-License-Identifier: UNLICENSED
// File: @openzeppelin/contracts/utils/Strings.sol


// OpenZeppelin Contracts v4.4.1 (utils/Strings.sol)

pragma solidity ^0.8.0;

/**
 * @dev String operations.
 */
library Strings {
    bytes16 private constant _HEX_SYMBOLS = "0123456789abcdef";

    /**
     * @dev Converts a `uint256` to its ASCII `string` decimal representation.
     */
    function toString(uint256 value) internal pure returns (string memory) {
        // Inspired by OraclizeAPI's implementation - MIT licence
        // https://github.com/oraclize/ethereum-api/blob/b42146b063c7d6ee1358846c198246239e9360e8/oraclizeAPI_0.4.25.sol

        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);
    }

    /**
     * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation.
     */
    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);
    }

    /**
     * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation with fixed 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);
    }
}

// File: @openzeppelin/contracts/utils/cryptography/ECDSA.sol


// OpenZeppelin Contracts (last updated v4.5.0) (utils/cryptography/ECDSA.sol)

pragma solidity ^0.8.0;


/**
 * @dev Elliptic Curve Digital Signature Algorithm (ECDSA) operations.
 *
 * These functions can be used to verify that a message was signed by the holder
 * of the private keys of a given address.
 */
library ECDSA {
    enum RecoverError {
        NoError,
        InvalidSignature,
        InvalidSignatureLength,
        InvalidSignatureS,
        InvalidSignatureV
    }

    function _throwError(RecoverError error) private pure {
        if (error == RecoverError.NoError) {
            return; // no error: do nothing
        } 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");
        }
    }

    /**
     * @dev Returns the address that signed a hashed message (`hash`) with
     * `signature` or error string. This address can then be used for verification purposes.
     *
     * The `ecrecover` EVM opcode allows for malleable (non-unique) signatures:
     * this function rejects them by requiring the `s` value to be in the lower
     * half order, and the `v` value to be either 27 or 28.
     *
     * IMPORTANT: `hash` _must_ be the result of a hash operation for the
     * verification to be secure: it is possible to craft signatures that
     * recover to arbitrary addresses for non-hashed data. A safe way to ensure
     * this is by receiving a hash of the original message (which may otherwise
     * be too long), and then calling {toEthSignedMessageHash} on it.
     *
     * Documentation for signature generation:
     * - with https://web3js.readthedocs.io/en/v1.3.4/web3-eth-accounts.html#sign[Web3.js]
     * - with https://docs.ethers.io/v5/api/signer/#Signer-signMessage[ethers]
     *
     * _Available since v4.3._
     */
    function tryRecover(bytes32 hash, bytes memory signature) internal pure returns (address, RecoverError) {
        // Check the signature length
        // - case 65: r,s,v signature (standard)
        // - case 64: r,vs signature (cf https://eips.ethereum.org/EIPS/eip-2098) _Available since v4.1._
        if (signature.length == 65) {
            bytes32 r;
            bytes32 s;
            uint8 v;
            // ecrecover takes the signature parameters, and the only way to get them
            // currently is to use assembly.
            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 if (signature.length == 64) {
            bytes32 r;
            bytes32 vs;
            // ecrecover takes the signature parameters, and the only way to get them
            // currently is to use assembly.
            assembly {
                r := mload(add(signature, 0x20))
                vs := mload(add(signature, 0x40))
            }
            return tryRecover(hash, r, vs);
        } else {
            return (address(0), RecoverError.InvalidSignatureLength);
        }
    }

    /**
     * @dev Returns the address that signed a hashed message (`hash`) with
     * `signature`. This address can then be used for verification purposes.
     *
     * The `ecrecover` EVM opcode allows for malleable (non-unique) signatures:
     * this function rejects them by requiring the `s` value to be in the lower
     * half order, and the `v` value to be either 27 or 28.
     *
     * IMPORTANT: `hash` _must_ be the result of a hash operation for the
     * verification to be secure: it is possible to craft signatures that
     * recover to arbitrary addresses for non-hashed data. A safe way to ensure
     * this is by receiving a hash of the original message (which may otherwise
     * be too long), and then calling {toEthSignedMessageHash} on it.
     */
    function recover(bytes32 hash, bytes memory signature) internal pure returns (address) {
        (address recovered, RecoverError error) = tryRecover(hash, signature);
        _throwError(error);
        return recovered;
    }

    /**
     * @dev Overload of {ECDSA-tryRecover} that receives the `r` and `vs` short-signature fields separately.
     *
     * See https://eips.ethereum.org/EIPS/eip-2098[EIP-2098 short signatures]
     *
     * _Available since v4.3._
     */
    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);
    }

    /**
     * @dev Overload of {ECDSA-recover} that receives the `r and `vs` short-signature fields separately.
     *
     * _Available since v4.2._
     */
    function recover(
        bytes32 hash,
        bytes32 r,
        bytes32 vs
    ) internal pure returns (address) {
        (address recovered, RecoverError error) = tryRecover(hash, r, vs);
        _throwError(error);
        return recovered;
    }

    /**
     * @dev Overload of {ECDSA-tryRecover} that receives the `v`,
     * `r` and `s` signature fields separately.
     *
     * _Available since v4.3._
     */
    function tryRecover(
        bytes32 hash,
        uint8 v,
        bytes32 r,
        bytes32 s
    ) internal pure returns (address, RecoverError) {
        // EIP-2 still allows signature malleability for ecrecover(). Remove this possibility and make the signature
        // unique. Appendix F in the Ethereum Yellow paper (https://ethereum.github.io/yellowpaper/paper.pdf), defines
        // the valid range for s in (301): 0 < s < secp256k1n ÷ 2 + 1, and for v in (302): v ∈ {27, 28}. Most
        // signatures from current libraries generate a unique signature with an s-value in the lower half order.
        //
        // If your library generates malleable signatures, such as s-values in the upper range, calculate a new s-value
        // with 0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEBAAEDCE6AF48A03BBFD25E8CD0364141 - s1 and flip v from 27 to 28 or
        // vice versa. If your library also generates signatures with 0/1 for v instead 27/28, add 27 to v to accept
        // these malleable signatures as well.
        if (uint256(s) > 0x7FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF5D576E7357A4501DDFE92F46681B20A0) {
            return (address(0), RecoverError.InvalidSignatureS);
        }
        if (v != 27 && v != 28) {
            return (address(0), RecoverError.InvalidSignatureV);
        }

        // If the signature is valid (and not malleable), return the signer address
        address signer = ecrecover(hash, v, r, s);
        if (signer == address(0)) {
            return (address(0), RecoverError.InvalidSignature);
        }

        return (signer, RecoverError.NoError);
    }

    /**
     * @dev Overload of {ECDSA-recover} that receives the `v`,
     * `r` and `s` signature fields separately.
     */
    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;
    }

    /**
     * @dev Returns an Ethereum Signed Message, created from a `hash`. This
     * produces hash corresponding to the one signed with the
     * https://eth.wiki/json-rpc/API#eth_sign[`eth_sign`]
     * JSON-RPC method as part of EIP-191.
     *
     * See {recover}.
     */
    function toEthSignedMessageHash(bytes32 hash) internal pure returns (bytes32) {
        // 32 is the length in bytes of hash,
        // enforced by the type signature above
        return keccak256(abi.encodePacked("\x19Ethereum Signed Message:\n32", hash));
    }

    /**
     * @dev Returns an Ethereum Signed Message, created from `s`. This
     * produces hash corresponding to the one signed with the
     * https://eth.wiki/json-rpc/API#eth_sign[`eth_sign`]
     * JSON-RPC method as part of EIP-191.
     *
     * See {recover}.
     */
    function toEthSignedMessageHash(bytes memory s) internal pure returns (bytes32) {
        return keccak256(abi.encodePacked("\x19Ethereum Signed Message:\n", Strings.toString(s.length), s));
    }

    /**
     * @dev Returns an Ethereum Signed Typed Data, created from a
     * `domainSeparator` and a `structHash`. This produces hash corresponding
     * to the one signed with the
     * https://eips.ethereum.org/EIPS/eip-712[`eth_signTypedData`]
     * JSON-RPC method as part of EIP-712.
     *
     * See {recover}.
     */
    function toTypedDataHash(bytes32 domainSeparator, bytes32 structHash) internal pure returns (bytes32) {
        return keccak256(abi.encodePacked("\x19\x01", domainSeparator, structHash));
    }
}

// File: @openzeppelin/contracts/utils/Context.sol


// OpenZeppelin Contracts v4.4.1 (utils/Context.sol)

pragma solidity ^0.8.0;

/**
 * @dev Provides information about the current execution context, including the
 * sender of the transaction and its data. While these are generally available
 * via msg.sender and msg.data, they should not be accessed in such a direct
 * manner, since when dealing with meta-transactions the account sending and
 * paying for execution may not be the actual sender (as far as an application
 * is concerned).
 *
 * This contract is only required for intermediate, library-like contracts.
 */
abstract contract Context {
    function _msgSender() internal view virtual returns (address) {
        return msg.sender;
    }

    function _msgData() internal view virtual returns (bytes calldata) {
        return msg.data;
    }
}

// File: @openzeppelin/contracts/access/Ownable.sol


// OpenZeppelin Contracts v4.4.1 (access/Ownable.sol)

pragma solidity ^0.8.0;


/**
 * @dev Contract module which provides a basic access control mechanism, where
 * there is an account (an owner) that can be granted exclusive access to
 * specific functions.
 *
 * By default, the owner account will be the one that deploys the contract. This
 * can later be changed with {transferOwnership}.
 *
 * This module is used through inheritance. It will make available the modifier
 * `onlyOwner`, which can be applied to your functions to restrict their use to
 * the owner.
 */
abstract contract Ownable is Context {
    address private _owner;

    event OwnershipTransferred(address indexed previousOwner, address indexed newOwner);

    /**
     * @dev Initializes the contract setting the deployer as the initial owner.
     */
    constructor() {
        _transferOwnership(_msgSender());
    }

    /**
     * @dev Returns the address of the current owner.
     */
    function owner() public view virtual returns (address) {
        return _owner;
    }

    /**
     * @dev Throws if called by any account other than the owner.
     */
    modifier onlyOwner() {
        require(owner() == _msgSender(), "Ownable: caller is not the owner");
        _;
    }

    /**
     * @dev Leaves the contract without owner. It will not be possible to call
     * `onlyOwner` functions anymore. Can only be called by the current owner.
     *
     * NOTE: Renouncing ownership will leave the contract without an owner,
     * thereby removing any functionality that is only available to the owner.
     */
    function renounceOwnership() public virtual onlyOwner {
        _transferOwnership(address(0));
    }

    /**
     * @dev Transfers ownership of the contract to a new account (`newOwner`).
     * Can only be called by the current owner.
     */
    function transferOwnership(address newOwner) public virtual onlyOwner {
        require(newOwner != address(0), "Ownable: new owner is the zero address");
        _transferOwnership(newOwner);
    }

    /**
     * @dev Transfers ownership of the contract to a new account (`newOwner`).
     * Internal function without access restriction.
     */
    function _transferOwnership(address newOwner) internal virtual {
        address oldOwner = _owner;
        _owner = newOwner;
        emit OwnershipTransferred(oldOwner, newOwner);
    }
}

// File: erc721a/contracts/IERC721A.sol


// ERC721A Contracts v4.1.0
// Creator: Chiru Labs

pragma solidity ^0.8.4;

/**
 * @dev Interface of an ERC721A compliant contract.
 */
interface IERC721A {
    /**
     * The caller must own the token or be an approved operator.
     */
    error ApprovalCallerNotOwnerNorApproved();

    /**
     * The token does not exist.
     */
    error ApprovalQueryForNonexistentToken();

    /**
     * The caller cannot approve to their own address.
     */
    error ApproveToCaller();

    /**
     * Cannot query the balance for the zero address.
     */
    error BalanceQueryForZeroAddress();

    /**
     * Cannot mint to the zero address.
     */
    error MintToZeroAddress();

    /**
     * The quantity of tokens minted must be more than zero.
     */
    error MintZeroQuantity();

    /**
     * The token does not exist.
     */
    error OwnerQueryForNonexistentToken();

    /**
     * The caller must own the token or be an approved operator.
     */
    error TransferCallerNotOwnerNorApproved();

    /**
     * The token must be owned by `from`.
     */
    error TransferFromIncorrectOwner();

    /**
     * Cannot safely transfer to a contract that does not implement the ERC721Receiver interface.
     */
    error TransferToNonERC721ReceiverImplementer();

    /**
     * Cannot transfer to the zero address.
     */
    error TransferToZeroAddress();

    /**
     * The token does not exist.
     */
    error URIQueryForNonexistentToken();

    /**
     * The `quantity` minted with ERC2309 exceeds the safety limit.
     */
    error MintERC2309QuantityExceedsLimit();

    /**
     * The `extraData` cannot be set on an unintialized ownership slot.
     */
    error OwnershipNotInitializedForExtraData();

    struct TokenOwnership {
        // The address of the owner.
        address addr;
        // Keeps track of the start time of ownership with minimal overhead for tokenomics.
        uint64 startTimestamp;
        // Whether the token has been burned.
        bool burned;
        // Arbitrary data similar to `startTimestamp` that can be set through `_extraData`.
        uint24 extraData;
    }

    /**
     * @dev Returns the total amount of tokens stored by the contract.
     *
     * Burned tokens are calculated here, use `_totalMinted()` if you want to count just minted tokens.
     */
    function totalSupply() external view returns (uint256);

    // ==============================
    //            IERC165
    // ==============================

    /**
     * @dev Returns true if this contract implements the interface defined by
     * `interfaceId`. See the corresponding
     * https://eips.ethereum.org/EIPS/eip-165#how-interfaces-are-identified[EIP section]
     * to learn more about how these ids are created.
     *
     * This function call must use less than 30 000 gas.
     */
    function supportsInterface(bytes4 interfaceId) external view returns (bool);

    // ==============================
    //            IERC721
    // ==============================

    /**
     * @dev Emitted when `tokenId` token is transferred from `from` to `to`.
     */
    event Transfer(address indexed from, address indexed to, uint256 indexed tokenId);

    /**
     * @dev Emitted when `owner` enables `approved` to manage the `tokenId` token.
     */
    event Approval(address indexed owner, address indexed approved, uint256 indexed tokenId);

    /**
     * @dev Emitted when `owner` enables or disables (`approved`) `operator` to manage all of its assets.
     */
    event ApprovalForAll(address indexed owner, address indexed operator, bool approved);

    /**
     * @dev Returns the number of tokens in ``owner``'s account.
     */
    function balanceOf(address owner) external view returns (uint256 balance);

    /**
     * @dev Returns the owner of the `tokenId` token.
     *
     * Requirements:
     *
     * - `tokenId` must exist.
     */
    function ownerOf(uint256 tokenId) external view returns (address owner);

    /**
     * @dev Safely transfers `tokenId` token from `from` to `to`.
     *
     * Requirements:
     *
     * - `from` cannot be the zero address.
     * - `to` cannot be the zero address.
     * - `tokenId` token must exist and be owned by `from`.
     * - If the caller is not `from`, it must be approved to move this token by either {approve} or {setApprovalForAll}.
     * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer.
     *
     * Emits a {Transfer} event.
     */
    function safeTransferFrom(
        address from,
        address to,
        uint256 tokenId,
        bytes calldata data
    ) external;

    /**
     * @dev Safely transfers `tokenId` token from `from` to `to`, checking first that contract recipients
     * are aware of the ERC721 protocol to prevent tokens from being forever locked.
     *
     * Requirements:
     *
     * - `from` cannot be the zero address.
     * - `to` cannot be the zero address.
     * - `tokenId` token must exist and be owned by `from`.
     * - If the caller is not `from`, it must be have been allowed to move this token by either {approve} or {setApprovalForAll}.
     * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer.
     *
     * Emits a {Transfer} event.
     */
    function safeTransferFrom(
        address from,
        address to,
        uint256 tokenId
    ) external;

    /**
     * @dev Transfers `tokenId` token from `from` to `to`.
     *
     * WARNING: Usage of this method is discouraged, use {safeTransferFrom} whenever possible.
     *
     * Requirements:
     *
     * - `from` cannot be the zero address.
     * - `to` cannot be the zero address.
     * - `tokenId` token must be owned by `from`.
     * - If the caller is not `from`, it must be approved to move this token by either {approve} or {setApprovalForAll}.
     *
     * Emits a {Transfer} event.
     */
    function transferFrom(
        address from,
        address to,
        uint256 tokenId
    ) external;

    /**
     * @dev Gives permission to `to` to transfer `tokenId` token to another account.
     * The approval is cleared when the token is transferred.
     *
     * Only a single account can be approved at a time, so approving the zero address clears previous approvals.
     *
     * Requirements:
     *
     * - The caller must own the token or be an approved operator.
     * - `tokenId` must exist.
     *
     * Emits an {Approval} event.
     */
    function approve(address to, uint256 tokenId) external;

    /**
     * @dev Approve or remove `operator` as an operator for the caller.
     * Operators can call {transferFrom} or {safeTransferFrom} for any token owned by the caller.
     *
     * Requirements:
     *
     * - The `operator` cannot be the caller.
     *
     * Emits an {ApprovalForAll} event.
     */
    function setApprovalForAll(address operator, bool _approved) external;

    /**
     * @dev Returns the account approved for `tokenId` token.
     *
     * Requirements:
     *
     * - `tokenId` must exist.
     */
    function getApproved(uint256 tokenId) external view returns (address operator);

    /**
     * @dev Returns if the `operator` is allowed to manage all of the assets of `owner`.
     *
     * See {setApprovalForAll}
     */
    function isApprovedForAll(address owner, address operator) external view returns (bool);

    // ==============================
    //        IERC721Metadata
    // ==============================

    /**
     * @dev Returns the token collection name.
     */
    function name() external view returns (string memory);

    /**
     * @dev Returns the token collection symbol.
     */
    function symbol() external view returns (string memory);

    /**
     * @dev Returns the Uniform Resource Identifier (URI) for `tokenId` token.
     */
    function tokenURI(uint256 tokenId) external view returns (string memory);

    // ==============================
    //            IERC2309
    // ==============================

    /**
     * @dev Emitted when tokens in `fromTokenId` to `toTokenId` (inclusive) is transferred from `from` to `to`,
     * as defined in the ERC2309 standard. See `_mintERC2309` for more details.
     */
    event ConsecutiveTransfer(uint256 indexed fromTokenId, uint256 toTokenId, address indexed from, address indexed to);
}

// File: erc721a/contracts/ERC721A.sol


// ERC721A Contracts v4.1.0
// Creator: Chiru Labs

pragma solidity ^0.8.4;


/**
 * @dev ERC721 token receiver interface.
 */
interface ERC721A__IERC721Receiver {
    function onERC721Received(
        address operator,
        address from,
        uint256 tokenId,
        bytes calldata data
    ) external returns (bytes4);
}

/**
 * @dev Implementation of https://eips.ethereum.org/EIPS/eip-721[ERC721] Non-Fungible Token Standard,
 * including the Metadata extension. Built to optimize for lower gas during batch mints.
 *
 * Assumes serials are sequentially minted starting at `_startTokenId()`
 * (defaults to 0, e.g. 0, 1, 2, 3..).
 *
 * Assumes that an owner cannot have more than 2**64 - 1 (max value of uint64) of supply.
 *
 * Assumes that the maximum token id cannot exceed 2**256 - 1 (max value of uint256).
 */
contract ERC721A is IERC721A {
    // Mask of an entry in packed address data.
    uint256 private constant BITMASK_ADDRESS_DATA_ENTRY = (1 << 64) - 1;

    // The bit position of `numberMinted` in packed address data.
    uint256 private constant BITPOS_NUMBER_MINTED = 64;

    // The bit position of `numberBurned` in packed address data.
    uint256 private constant BITPOS_NUMBER_BURNED = 128;

    // The bit position of `aux` in packed address data.
    uint256 private constant BITPOS_AUX = 192;

    // Mask of all 256 bits in packed address data except the 64 bits for `aux`.
    uint256 private constant BITMASK_AUX_COMPLEMENT = (1 << 192) - 1;

    // The bit position of `startTimestamp` in packed ownership.
    uint256 private constant BITPOS_START_TIMESTAMP = 160;

    // The bit mask of the `burned` bit in packed ownership.
    uint256 private constant BITMASK_BURNED = 1 << 224;

    // The bit position of the `nextInitialized` bit in packed ownership.
    uint256 private constant BITPOS_NEXT_INITIALIZED = 225;

    // The bit mask of the `nextInitialized` bit in packed ownership.
    uint256 private constant BITMASK_NEXT_INITIALIZED = 1 << 225;

    // The bit position of `extraData` in packed ownership.
    uint256 private constant BITPOS_EXTRA_DATA = 232;

    // Mask of all 256 bits in a packed ownership except the 24 bits for `extraData`.
    uint256 private constant BITMASK_EXTRA_DATA_COMPLEMENT = (1 << 232) - 1;

    // The mask of the lower 160 bits for addresses.
    uint256 private constant BITMASK_ADDRESS = (1 << 160) - 1;

    // The maximum `quantity` that can be minted with `_mintERC2309`.
    // This limit is to prevent overflows on the address data entries.
    // For a limit of 5000, a total of 3.689e15 calls to `_mintERC2309`
    // is required to cause an overflow, which is unrealistic.
    uint256 private constant MAX_MINT_ERC2309_QUANTITY_LIMIT = 5000;

    // The tokenId of the next token to be minted.
    uint256 private _currentIndex;

    // The number of tokens burned.
    uint256 private _burnCounter;

    // Token name
    string private _name;

    // Token symbol
    string private _symbol;

    // Mapping from token ID to ownership details
    // An empty struct value does not necessarily mean the token is unowned.
    // See `_packedOwnershipOf` implementation for details.
    //
    // Bits Layout:
    // - [0..159]   `addr`
    // - [160..223] `startTimestamp`
    // - [224]      `burned`
    // - [225]      `nextInitialized`
    // - [232..255] `extraData`
    mapping(uint256 => uint256) private _packedOwnerships;

    // Mapping owner address to address data.
    //
    // Bits Layout:
    // - [0..63]    `balance`
    // - [64..127]  `numberMinted`
    // - [128..191] `numberBurned`
    // - [192..255] `aux`
    mapping(address => uint256) private _packedAddressData;

    // Mapping from token ID to approved address.
    mapping(uint256 => address) private _tokenApprovals;

    // Mapping from owner to operator approvals
    mapping(address => mapping(address => bool)) private _operatorApprovals;

    constructor(string memory name_, string memory symbol_) {
        _name = name_;
        _symbol = symbol_;
        _currentIndex = _startTokenId();
    }

    /**
     * @dev Returns the starting token ID.
     * To change the starting token ID, please override this function.
     */
    function _startTokenId() internal view virtual returns (uint256) {
        return 0;
    }

    /**
     * @dev Returns the next token ID to be minted.
     */
    function _nextTokenId() internal view returns (uint256) {
        return _currentIndex;
    }

    /**
     * @dev Returns the total number of tokens in existence.
     * Burned tokens will reduce the count.
     * To get the total number of tokens minted, please see `_totalMinted`.
     */
    function totalSupply() public view override returns (uint256) {
        // Counter underflow is impossible as _burnCounter cannot be incremented
        // more than `_currentIndex - _startTokenId()` times.
        unchecked {
            return _currentIndex - _burnCounter - _startTokenId();
        }
    }

    /**
     * @dev Returns the total amount of tokens minted in the contract.
     */
    function _totalMinted() internal view returns (uint256) {
        // Counter underflow is impossible as _currentIndex does not decrement,
        // and it is initialized to `_startTokenId()`
        unchecked {
            return _currentIndex - _startTokenId();
        }
    }

    /**
     * @dev Returns the total number of tokens burned.
     */
    function _totalBurned() internal view returns (uint256) {
        return _burnCounter;
    }

    /**
     * @dev See {IERC165-supportsInterface}.
     */
    function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) {
        // The interface IDs are constants representing the first 4 bytes of the XOR of
        // all function selectors in the interface. See: https://eips.ethereum.org/EIPS/eip-165
        // e.g. `bytes4(i.functionA.selector ^ i.functionB.selector ^ ...)`
        return
            interfaceId == 0x01ffc9a7 || // ERC165 interface ID for ERC165.
            interfaceId == 0x80ac58cd || // ERC165 interface ID for ERC721.
            interfaceId == 0x5b5e139f; // ERC165 interface ID for ERC721Metadata.
    }

    /**
     * @dev See {IERC721-balanceOf}.
     */
    function balanceOf(address owner) public view override returns (uint256) {
        if (owner == address(0)) revert BalanceQueryForZeroAddress();
        return _packedAddressData[owner] & BITMASK_ADDRESS_DATA_ENTRY;
    }

    /**
     * Returns the number of tokens minted by `owner`.
     */
    function _numberMinted(address owner) internal view returns (uint256) {
        return (_packedAddressData[owner] >> BITPOS_NUMBER_MINTED) & BITMASK_ADDRESS_DATA_ENTRY;
    }

    /**
     * Returns the number of tokens burned by or on behalf of `owner`.
     */
    function _numberBurned(address owner) internal view returns (uint256) {
        return (_packedAddressData[owner] >> BITPOS_NUMBER_BURNED) & BITMASK_ADDRESS_DATA_ENTRY;
    }

    /**
     * Returns the auxiliary data for `owner`. (e.g. number of whitelist mint slots used).
     */
    function _getAux(address owner) internal view returns (uint64) {
        return uint64(_packedAddressData[owner] >> BITPOS_AUX);
    }

    /**
     * Sets the auxiliary data for `owner`. (e.g. number of whitelist mint slots used).
     * If there are multiple variables, please pack them into a uint64.
     */
    function _setAux(address owner, uint64 aux) internal {
        uint256 packed = _packedAddressData[owner];
        uint256 auxCasted;
        // Cast `aux` with assembly to avoid redundant masking.
        assembly {
            auxCasted := aux
        }
        packed = (packed & BITMASK_AUX_COMPLEMENT) | (auxCasted << BITPOS_AUX);
        _packedAddressData[owner] = packed;
    }

    /**
     * Returns the packed ownership data of `tokenId`.
     */
    function _packedOwnershipOf(uint256 tokenId) private view returns (uint256) {
        uint256 curr = tokenId;

        unchecked {
            if (_startTokenId() <= curr)
                if (curr < _currentIndex) {
                    uint256 packed = _packedOwnerships[curr];
                    // If not burned.
                    if (packed & BITMASK_BURNED == 0) {
                        // Invariant:
                        // There will always be an ownership that has an address and is not burned
                        // before an ownership that does not have an address and is not burned.
                        // Hence, curr will not underflow.
                        //
                        // We can directly compare the packed value.
                        // If the address is zero, packed is zero.
                        while (packed == 0) {
                            packed = _packedOwnerships[--curr];
                        }
                        return packed;
                    }
                }
        }
        revert OwnerQueryForNonexistentToken();
    }

    /**
     * Returns the unpacked `TokenOwnership` struct from `packed`.
     */
    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);
    }

    /**
     * Returns the unpacked `TokenOwnership` struct at `index`.
     */
    function _ownershipAt(uint256 index) internal view returns (TokenOwnership memory) {
        return _unpackedOwnership(_packedOwnerships[index]);
    }

    /**
     * @dev Initializes the ownership slot minted at `index` for efficiency purposes.
     */
    function _initializeOwnershipAt(uint256 index) internal {
        if (_packedOwnerships[index] == 0) {
            _packedOwnerships[index] = _packedOwnershipOf(index);
        }
    }

    /**
     * Gas spent here starts off proportional to the maximum mint batch size.
     * It gradually moves to O(1) as tokens get transferred around in the collection over time.
     */
    function _ownershipOf(uint256 tokenId) internal view returns (TokenOwnership memory) {
        return _unpackedOwnership(_packedOwnershipOf(tokenId));
    }

    /**
     * @dev Packs ownership data into a single uint256.
     */
    function _packOwnershipData(address owner, uint256 flags) private view returns (uint256 result) {
        assembly {
            // Mask `owner` to the lower 160 bits, in case the upper bits somehow aren't clean.
            owner := and(owner, BITMASK_ADDRESS)
            // `owner | (block.timestamp << BITPOS_START_TIMESTAMP) | flags`.
            result := or(owner, or(shl(BITPOS_START_TIMESTAMP, timestamp()), flags))
        }
    }

    /**
     * @dev See {IERC721-ownerOf}.
     */
    function ownerOf(uint256 tokenId) public view override returns (address) {
        return address(uint160(_packedOwnershipOf(tokenId)));
    }

    /**
     * @dev See {IERC721Metadata-name}.
     */
    function name() public view virtual override returns (string memory) {
        return _name;
    }

    /**
     * @dev See {IERC721Metadata-symbol}.
     */
    function symbol() public view virtual override returns (string memory) {
        return _symbol;
    }

    /**
     * @dev See {IERC721Metadata-tokenURI}.
     */
    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))) : '';
    }

    /**
     * @dev Base URI for computing {tokenURI}. If set, the resulting URI for each
     * token will be the concatenation of the `baseURI` and the `tokenId`. Empty
     * by default, it can be overridden in child contracts.
     */
    function _baseURI() internal view virtual returns (string memory) {
        return '';
    }

    /**
     * @dev Returns the `nextInitialized` flag set if `quantity` equals 1.
     */
    function _nextInitializedFlag(uint256 quantity) private pure returns (uint256 result) {
        // For branchless setting of the `nextInitialized` flag.
        assembly {
            // `(quantity == 1) << BITPOS_NEXT_INITIALIZED`.
            result := shl(BITPOS_NEXT_INITIALIZED, eq(quantity, 1))
        }
    }

    /**
     * @dev See {IERC721-approve}.
     */
    function approve(address to, uint256 tokenId) public override {
        address owner = ownerOf(tokenId);

        if (_msgSenderERC721A() != owner)
            if (!isApprovedForAll(owner, _msgSenderERC721A())) {
                revert ApprovalCallerNotOwnerNorApproved();
            }

        _tokenApprovals[tokenId] = to;
        emit Approval(owner, to, tokenId);
    }

    /**
     * @dev See {IERC721-getApproved}.
     */
    function getApproved(uint256 tokenId) public view override returns (address) {
        if (!_exists(tokenId)) revert ApprovalQueryForNonexistentToken();

        return _tokenApprovals[tokenId];
    }

    /**
     * @dev See {IERC721-setApprovalForAll}.
     */
    function setApprovalForAll(address operator, bool approved) public virtual override {
        if (operator == _msgSenderERC721A()) revert ApproveToCaller();

        _operatorApprovals[_msgSenderERC721A()][operator] = approved;
        emit ApprovalForAll(_msgSenderERC721A(), operator, approved);
    }

    /**
     * @dev See {IERC721-isApprovedForAll}.
     */
    function isApprovedForAll(address owner, address operator) public view virtual override returns (bool) {
        return _operatorApprovals[owner][operator];
    }

    /**
     * @dev See {IERC721-safeTransferFrom}.
     */
    function safeTransferFrom(
        address from,
        address to,
        uint256 tokenId
    ) public virtual override {
        safeTransferFrom(from, to, tokenId, '');
    }

    /**
     * @dev See {IERC721-safeTransferFrom}.
     */
    function safeTransferFrom(
        address from,
        address to,
        uint256 tokenId,
        bytes memory _data
    ) public virtual override {
        transferFrom(from, to, tokenId);
        if (to.code.length != 0)
            if (!_checkContractOnERC721Received(from, to, tokenId, _data)) {
                revert TransferToNonERC721ReceiverImplementer();
            }
    }

    /**
     * @dev Returns whether `tokenId` exists.
     *
     * Tokens can be managed by their owner or approved accounts via {approve} or {setApprovalForAll}.
     *
     * Tokens start existing when they are minted (`_mint`),
     */
    function _exists(uint256 tokenId) internal view returns (bool) {
        return
            _startTokenId() <= tokenId &&
            tokenId < _currentIndex && // If within bounds,
            _packedOwnerships[tokenId] & BITMASK_BURNED == 0; // and not burned.
    }

    /**
     * @dev Equivalent to `_safeMint(to, quantity, '')`.
     */
    function _safeMint(address to, uint256 quantity) internal {
        _safeMint(to, quantity, '');
    }

    /**
     * @dev Safely mints `quantity` tokens and transfers them to `to`.
     *
     * Requirements:
     *
     * - If `to` refers to a smart contract, it must implement
     *   {IERC721Receiver-onERC721Received}, which is called for each safe transfer.
     * - `quantity` must be greater than 0.
     *
     * See {_mint}.
     *
     * Emits a {Transfer} event for each mint.
     */
    function _safeMint(
        address to,
        uint256 quantity,
        bytes memory _data
    ) internal {
        _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);
                // Reentrancy protection.
                if (_currentIndex != end) revert();
            }
        }
    }

    /**
     * @dev Mints `quantity` tokens and transfers them to `to`.
     *
     * Requirements:
     *
     * - `to` cannot be the zero address.
     * - `quantity` must be greater than 0.
     *
     * Emits a {Transfer} event for each mint.
     */
    function _mint(address to, uint256 quantity) internal {
        uint256 startTokenId = _currentIndex;
        if (to == address(0)) revert MintToZeroAddress();
        if (quantity == 0) revert MintZeroQuantity();

        _beforeTokenTransfers(address(0), to, startTokenId, quantity);

        // Overflows are incredibly unrealistic.
        // `balance` and `numberMinted` have a maximum limit of 2**64.
        // `tokenId` has a maximum limit of 2**256.
        unchecked {
            // Updates:
            // - `balance += quantity`.
            // - `numberMinted += quantity`.
            //
            // We can directly add to the `balance` and `numberMinted`.
            _packedAddressData[to] += quantity * ((1 << BITPOS_NUMBER_MINTED) | 1);

            // Updates:
            // - `address` to the owner.
            // - `startTimestamp` to the timestamp of minting.
            // - `burned` to `false`.
            // - `nextInitialized` to `quantity == 1`.
            _packedOwnerships[startTokenId] = _packOwnershipData(
                to,
                _nextInitializedFlag(quantity) | _nextExtraData(address(0), to, 0)
            );

            uint256 tokenId = startTokenId;
            uint256 end = startTokenId + quantity;
            do {
                emit Transfer(address(0), to, tokenId++);
            } while (tokenId < end);

            _currentIndex = end;
        }
        _afterTokenTransfers(address(0), to, startTokenId, quantity);
    }

    /**
     * @dev Mints `quantity` tokens and transfers them to `to`.
     *
     * This function is intended for efficient minting only during contract creation.
     *
     * It emits only one {ConsecutiveTransfer} as defined in
     * [ERC2309](https://eips.ethereum.org/EIPS/eip-2309),
     * instead of a sequence of {Transfer} event(s).
     *
     * Calling this function outside of contract creation WILL make your contract
     * non-compliant with the ERC721 standard.
     * For full ERC721 compliance, substituting ERC721 {Transfer} event(s) with the ERC2309
     * {ConsecutiveTransfer} event is only permissible during contract creation.
     *
     * Requirements:
     *
     * - `to` cannot be the zero address.
     * - `quantity` must be greater than 0.
     *
     * Emits a {ConsecutiveTransfer} event.
     */
    function _mintERC2309(address to, uint256 quantity) internal {
        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);

        // Overflows are unrealistic due to the above check for `quantity` to be below the limit.
        unchecked {
            // Updates:
            // - `balance += quantity`.
            // - `numberMinted += quantity`.
            //
            // We can directly add to the `balance` and `numberMinted`.
            _packedAddressData[to] += quantity * ((1 << BITPOS_NUMBER_MINTED) | 1);

            // Updates:
            // - `address` to the owner.
            // - `startTimestamp` to the timestamp of minting.
            // - `burned` to `false`.
            // - `nextInitialized` to `quantity == 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);
    }

    /**
     * @dev Returns the storage slot and value for the approved address of `tokenId`.
     */
    function _getApprovedAddress(uint256 tokenId)
        private
        view
        returns (uint256 approvedAddressSlot, address approvedAddress)
    {
        mapping(uint256 => address) storage tokenApprovalsPtr = _tokenApprovals;
        // The following is equivalent to `approvedAddress = _tokenApprovals[tokenId]`.
        assembly {
            // Compute the slot.
            mstore(0x00, tokenId)
            mstore(0x20, tokenApprovalsPtr.slot)
            approvedAddressSlot := keccak256(0x00, 0x40)
            // Load the slot's value from storage.
            approvedAddress := sload(approvedAddressSlot)
        }
    }

    /**
     * @dev Returns whether the `approvedAddress` is equals to `from` or `msgSender`.
     */
    function _isOwnerOrApproved(
        address approvedAddress,
        address from,
        address msgSender
    ) private pure returns (bool result) {
        assembly {
            // Mask `from` to the lower 160 bits, in case the upper bits somehow aren't clean.
            from := and(from, BITMASK_ADDRESS)
            // Mask `msgSender` to the lower 160 bits, in case the upper bits somehow aren't clean.
            msgSender := and(msgSender, BITMASK_ADDRESS)
            // `msgSender == from || msgSender == approvedAddress`.
            result := or(eq(msgSender, from), eq(msgSender, approvedAddress))
        }
    }

    /**
     * @dev Transfers `tokenId` from `from` to `to`.
     *
     * Requirements:
     *
     * - `to` cannot be the zero address.
     * - `tokenId` token must be owned by `from`.
     *
     * Emits a {Transfer} event.
     */
    function transferFrom(
        address from,
        address to,
        uint256 tokenId
    ) public virtual override {
        uint256 prevOwnershipPacked = _packedOwnershipOf(tokenId);

        if (address(uint160(prevOwnershipPacked)) != from) revert TransferFromIncorrectOwner();

        (uint256 approvedAddressSlot, address approvedAddress) = _getApprovedAddress(tokenId);

        // The nested ifs save around 20+ gas over a compound boolean condition.
        if (!_isOwnerOrApproved(approvedAddress, from, _msgSenderERC721A()))
            if (!isApprovedForAll(from, _msgSenderERC721A())) revert TransferCallerNotOwnerNorApproved();

        if (to == address(0)) revert TransferToZeroAddress();

        _beforeTokenTransfers(from, to, tokenId, 1);

        // Clear approvals from the previous owner.
        assembly {
            if approvedAddress {
                // This is equivalent to `delete _tokenApprovals[tokenId]`.
                sstore(approvedAddressSlot, 0)
            }
        }

        // Underflow of the sender's balance is impossible because we check for
        // ownership above and the recipient's balance can't realistically overflow.
        // Counter overflow is incredibly unrealistic as tokenId would have to be 2**256.
        unchecked {
            // We can directly increment and decrement the balances.
            --_packedAddressData[from]; // Updates: `balance -= 1`.
            ++_packedAddressData[to]; // Updates: `balance += 1`.

            // Updates:
            // - `address` to the next owner.
            // - `startTimestamp` to the timestamp of transfering.
            // - `burned` to `false`.
            // - `nextInitialized` to `true`.
            _packedOwnerships[tokenId] = _packOwnershipData(
                to,
                BITMASK_NEXT_INITIALIZED | _nextExtraData(from, to, prevOwnershipPacked)
            );

            // If the next slot may not have been initialized (i.e. `nextInitialized == false`) .
            if (prevOwnershipPacked & BITMASK_NEXT_INITIALIZED == 0) {
                uint256 nextTokenId = tokenId + 1;
                // If the next slot's address is zero and not burned (i.e. packed value is zero).
                if (_packedOwnerships[nextTokenId] == 0) {
                    // If the next slot is within bounds.
                    if (nextTokenId != _currentIndex) {
                        // Initialize the next slot to maintain correctness for `ownerOf(tokenId + 1)`.
                        _packedOwnerships[nextTokenId] = prevOwnershipPacked;
                    }
                }
            }
        }

        emit Transfer(from, to, tokenId);
        _afterTokenTransfers(from, to, tokenId, 1);
    }

    /**
     * @dev Equivalent to `_burn(tokenId, false)`.
     */
    function _burn(uint256 tokenId) internal virtual {
        _burn(tokenId, false);
    }

    /**
     * @dev Destroys `tokenId`.
     * The approval is cleared when the token is burned.
     *
     * Requirements:
     *
     * - `tokenId` must exist.
     *
     * Emits a {Transfer} event.
     */
    function _burn(uint256 tokenId, bool approvalCheck) internal virtual {
        uint256 prevOwnershipPacked = _packedOwnershipOf(tokenId);

        address from = address(uint160(prevOwnershipPacked));

        (uint256 approvedAddressSlot, address approvedAddress) = _getApprovedAddress(tokenId);

        if (approvalCheck) {
            // The nested ifs save around 20+ gas over a compound boolean condition.
            if (!_isOwnerOrApproved(approvedAddress, from, _msgSenderERC721A()))
                if (!isApprovedForAll(from, _msgSenderERC721A())) revert TransferCallerNotOwnerNorApproved();
        }

        _beforeTokenTransfers(from, address(0), tokenId, 1);

        // Clear approvals from the previous owner.
        assembly {
            if approvedAddress {
                // This is equivalent to `delete _tokenApprovals[tokenId]`.
                sstore(approvedAddressSlot, 0)
            }
        }

        // Underflow of the sender's balance is impossible because we check for
        // ownership above and the recipient's balance can't realistically overflow.
        // Counter overflow is incredibly unrealistic as `tokenId` would have to be 2**256.
        unchecked {
            // Updates:
            // - `balance -= 1`.
            // - `numberBurned += 1`.
            //
            // We can directly decrement the balance, and increment the number burned.
            // This is equivalent to `packed -= 1; packed += 1 << BITPOS_NUMBER_BURNED;`.
            _packedAddressData[from] += (1 << BITPOS_NUMBER_BURNED) - 1;

            // Updates:
            // - `address` to the last owner.
            // - `startTimestamp` to the timestamp of burning.
            // - `burned` to `true`.
            // - `nextInitialized` to `true`.
            _packedOwnerships[tokenId] = _packOwnershipData(
                from,
                (BITMASK_BURNED | BITMASK_NEXT_INITIALIZED) | _nextExtraData(from, address(0), prevOwnershipPacked)
            );

            // If the next slot may not have been initialized (i.e. `nextInitialized == false`) .
            if (prevOwnershipPacked & BITMASK_NEXT_INITIALIZED == 0) {
                uint256 nextTokenId = tokenId + 1;
                // If the next slot's address is zero and not burned (i.e. packed value is zero).
                if (_packedOwnerships[nextTokenId] == 0) {
                    // If the next slot is within bounds.
                    if (nextTokenId != _currentIndex) {
                        // Initialize the next slot to maintain correctness for `ownerOf(tokenId + 1)`.
                        _packedOwnerships[nextTokenId] = prevOwnershipPacked;
                    }
                }
            }
        }

        emit Transfer(from, address(0), tokenId);
        _afterTokenTransfers(from, address(0), tokenId, 1);

        // Overflow not possible, as _burnCounter cannot be exceed _currentIndex times.
        unchecked {
            _burnCounter++;
        }
    }

    /**
     * @dev Internal function to invoke {IERC721Receiver-onERC721Received} on a target contract.
     *
     * @param from address representing the previous owner of the given token ID
     * @param to target address that will receive the tokens
     * @param tokenId uint256 ID of the token to be transferred
     * @param _data bytes optional data to send along with the call
     * @return bool whether the call correctly returned the expected magic value
     */
    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))
                }
            }
        }
    }

    /**
     * @dev Directly sets the extra data for the ownership data `index`.
     */
    function _setExtraDataAt(uint256 index, uint24 extraData) internal {
        uint256 packed = _packedOwnerships[index];
        if (packed == 0) revert OwnershipNotInitializedForExtraData();
        uint256 extraDataCasted;
        // Cast `extraData` with assembly to avoid redundant masking.
        assembly {
            extraDataCasted := extraData
        }
        packed = (packed & BITMASK_EXTRA_DATA_COMPLEMENT) | (extraDataCasted << BITPOS_EXTRA_DATA);
        _packedOwnerships[index] = packed;
    }

    /**
     * @dev Returns the next extra data for the packed ownership data.
     * The returned result is shifted into position.
     */
    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;
    }

    /**
     * @dev Called during each token transfer to set the 24bit `extraData` field.
     * Intended to be overridden by the cosumer contract.
     *
     * `previousExtraData` - the value of `extraData` before transfer.
     *
     * Calling conditions:
     *
     * - When `from` and `to` are both non-zero, `from`'s `tokenId` will be
     * transferred to `to`.
     * - When `from` is zero, `tokenId` will be minted for `to`.
     * - When `to` is zero, `tokenId` will be burned by `from`.
     * - `from` and `to` are never both zero.
     */
    function _extraData(
        address from,
        address to,
        uint24 previousExtraData
    ) internal view virtual returns (uint24) {}

    /**
     * @dev Hook that is called before a set of serially-ordered token ids are about to be transferred.
     * This includes minting.
     * And also called before burning one token.
     *
     * startTokenId - the first token id to be transferred
     * quantity - the amount to be transferred
     *
     * Calling conditions:
     *
     * - When `from` and `to` are both non-zero, `from`'s `tokenId` will be
     * transferred to `to`.
     * - When `from` is zero, `tokenId` will be minted for `to`.
     * - When `to` is zero, `tokenId` will be burned by `from`.
     * - `from` and `to` are never both zero.
     */
    function _beforeTokenTransfers(
        address from,
        address to,
        uint256 startTokenId,
        uint256 quantity
    ) internal virtual {}

    /**
     * @dev Hook that is called after a set of serially-ordered token ids have been transferred.
     * This includes minting.
     * And also called after one token has been burned.
     *
     * startTokenId - the first token id to be transferred
     * quantity - the amount to be transferred
     *
     * Calling conditions:
     *
     * - When `from` and `to` are both non-zero, `from`'s `tokenId` has been
     * transferred to `to`.
     * - When `from` is zero, `tokenId` has been minted for `to`.
     * - When `to` is zero, `tokenId` has been burned by `from`.
     * - `from` and `to` are never both zero.
     */
    function _afterTokenTransfers(
        address from,
        address to,
        uint256 startTokenId,
        uint256 quantity
    ) internal virtual {}

    /**
     * @dev Returns the message sender (defaults to `msg.sender`).
     *
     * If you are writing GSN compatible contracts, you need to override this function.
     */
    function _msgSenderERC721A() internal view virtual returns (address) {
        return msg.sender;
    }

    /**
     * @dev Converts a `uint256` to its ASCII `string` decimal representation.
     */
    function _toString(uint256 value) internal pure returns (string memory ptr) {
        assembly {
            // The maximum value of a uint256 contains 78 digits (1 byte per digit),
            // but we allocate 128 bytes to keep the free memory pointer 32-byte word aliged.
            // We will need 1 32-byte word to store the length,
            // and 3 32-byte words to store a maximum of 78 digits. Total: 32 + 3 * 32 = 128.
            ptr := add(mload(0x40), 128)
            // Update the free memory pointer to allocate.
            mstore(0x40, ptr)

            // Cache the end of the memory to calculate the length later.
            let end := ptr

            // We write the string from the rightmost digit to the leftmost digit.
            // The following is essentially a do-while loop that also handles the zero case.
            // Costs a bit more than early returning for the zero case,
            // but cheaper in terms of deployment and overall runtime costs.
            for {
                // Initialize and perform the first pass without check.
                let temp := value
                // Move the pointer 1 byte leftwards to point to an empty character slot.
                ptr := sub(ptr, 1)
                // Write the character to the pointer. 48 is the ASCII index of '0'.
                mstore8(ptr, add(48, mod(temp, 10)))
                temp := div(temp, 10)
            } temp {
                // Keep dividing `temp` until zero.
                temp := div(temp, 10)
            } {
                // Body of the for loop.
                ptr := sub(ptr, 1)
                mstore8(ptr, add(48, mod(temp, 10)))
            }

            let length := sub(end, ptr)
            // Move the pointer 32 bytes leftwards to make room for the length.
            ptr := sub(ptr, 32)
            // Store the length.
            mstore(ptr, length)
        }
    }
}

// File: contracts/PERFUN.sol


pragma solidity ^0.8.4.0;





contract PERFUN is ERC721A, Ownable {

    using ECDSA for bytes32;

    uint256 private constant FIRST_MINTER_RESERVED = 1;

    bool private isPublicMint = false;
    string private tokenUri;
    uint256 private maxMint = 300;
    address private firstMinter;
    mapping(bytes32 => address) private _usedSigs;

    modifier isOpenMintable() {
        require(isPublicMint || (msg.sender == firstMinter && balanceOf(msg.sender) < FIRST_MINTER_RESERVED), "NOT FOR PUBLIC MINT YET");
        _;
    }

    // Prevention from minting off of another contract
    modifier callerIsUser() {
        require(tx.origin == msg.sender, "The caller is another contract");
        _;
    }

    constructor(address _firstMinter) ERC721A("PERFUN", "PERFUN") {
        setFirstMinter(_firstMinter);
        updateTokenUri("https://bafybeiewzadopfcal7mglhkupk323wej6hie76qhemyv7fxuyzimn7qf3e.ipfs.nftstorage.link/");
    }

    fallback() external {}

    function setMaxMintLimit(uint256 newMaxQty) public onlyOwner {
        maxMint = newMaxQty;
    }

    function setPbublicMint(bool status) public onlyOwner {
        isPublicMint = status;
    }
    
    function setFirstMinter(address _firstMinter) public onlyOwner {
        firstMinter = _firstMinter;
    }

    function updateTokenUri(string memory _tokenUri) public onlyOwner {
        tokenUri = _tokenUri;
    }

    function mint() external isOpenMintable callerIsUser {
        require(totalSupply() + 1 <= maxMint, "MAX CAP OF PERFUN EXCEEDED");
        _mint(msg.sender, 1);
    }
    
    function mintTo(address to, uint256 quantity) external onlyOwner callerIsUser {
        require(quantity >= 1, "QUANTITY MUST EQUAL OR GREATER THEN 1");
        require(totalSupply() + quantity <= maxMint, "MAX CAP OF PERFUN EXCEEDED");
        _mint(to, quantity);
    }
    
    function mintTo(address to) external onlyOwner callerIsUser {
        require(totalSupply() + 1 <= maxMint, "MAX CAP OF PERFUN EXCEEDED");
        _mint(to, 1);
    }

    function signatureChecker(bytes32 message, bytes memory sig) internal view {
        require(message.recover(sig) == owner(), "SIGNATURE NOT VALID");
        require(_usedSigs[message] == address(0), "SIGNATURE ALREADY USED");
    }

    function mintWithSig(string memory ts, bytes memory sig) external isOpenMintable callerIsUser {
        require(totalSupply() + 1 <= maxMint, "MAX CAP OF PERFUN EXCEEDED");
        bytes32 message = keccak256(abi.encodePacked(address(this), _msgSender(), ts));
        signatureChecker(message, sig);
        _usedSigs[message] = _msgSender();
        _mint(msg.sender, 1);
    }

    function mintWithSig(string memory ts, bytes memory sig, uint256 quantity) external isOpenMintable callerIsUser {
        require(quantity >= 1, "QUANTITY MUST EQUAL OR GREATER THEN 1");
        require(totalSupply() + quantity <= maxMint, "MAX CAP OF PERFUN EXCEEDED");
        bytes32 message = keccak256(abi.encodePacked(address(this), _msgSender(), ts, quantity));
        signatureChecker(message, sig);
        _usedSigs[message] = _msgSender();
        _mint(msg.sender, quantity);
    }

    function _baseURI() override internal view virtual returns (string memory) {
        return tokenUri;
    }

    function tokenURI(uint256 tokenId) public view virtual override returns (string memory) {
        if (_exists(tokenId) == false) {
            revert("The token does not exist.");
        }

        string memory baseURI = _baseURI();
        return bytes(baseURI).length != 0 ? string(abi.encodePacked(baseURI, _toString(tokenId), ".json")) : "";
    }
}

Contract Security Audit

Contract ABI

[{"inputs":[{"internalType":"address","name":"_firstMinter","type":"address"}],"stateMutability":"nonpayable","type":"constructor"},{"inputs":[],"name":"ApprovalCallerNotOwnerNorApproved","type":"error"},{"inputs":[],"name":"ApprovalQueryForNonexistentToken","type":"error"},{"inputs":[],"name":"ApproveToCaller","type":"error"},{"inputs":[],"name":"BalanceQueryForZeroAddress","type":"error"},{"inputs":[],"name":"MintERC2309QuantityExceedsLimit","type":"error"},{"inputs":[],"name":"MintToZeroAddress","type":"error"},{"inputs":[],"name":"MintZeroQuantity","type":"error"},{"inputs":[],"name":"OwnerQueryForNonexistentToken","type":"error"},{"inputs":[],"name":"OwnershipNotInitializedForExtraData","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"},{"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":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"},{"stateMutability":"nonpayable","type":"fallback"},{"inputs":[{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"approve","outputs":[],"stateMutability":"nonpayable","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":"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":[],"name":"mint","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"quantity","type":"uint256"}],"name":"mintTo","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"to","type":"address"}],"name":"mintTo","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"string","name":"ts","type":"string"},{"internalType":"bytes","name":"sig","type":"bytes"},{"internalType":"uint256","name":"quantity","type":"uint256"}],"name":"mintWithSig","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"string","name":"ts","type":"string"},{"internalType":"bytes","name":"sig","type":"bytes"}],"name":"mintWithSig","outputs":[],"stateMutability":"nonpayable","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":"from","type":"address"},{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"safeTransferFrom","outputs":[],"stateMutability":"nonpayable","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":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"operator","type":"address"},{"internalType":"bool","name":"approved","type":"bool"}],"name":"setApprovalForAll","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_firstMinter","type":"address"}],"name":"setFirstMinter","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"newMaxQty","type":"uint256"}],"name":"setMaxMintLimit","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bool","name":"status","type":"bool"}],"name":"setPbublicMint","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":[],"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":[],"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":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"string","name":"_tokenUri","type":"string"}],"name":"updateTokenUri","outputs":[],"stateMutability":"nonpayable","type":"function"}]

60806040526008805460ff60a01b1916905561012c600a553480156200002457600080fd5b5060405162002180380380620021808339810160408190526200004791620002b3565b6040805180820182526006808252652822a9232aa760d11b60208084018281528551808701909652928552840152815191929162000088916002916200020d565b5080516200009e9060039060208401906200020d565b50506000805550620000b033620000e6565b620000bb8162000138565b620000df6040518060800160405280605981526020016200210760599139620001a9565b5062000322565b600880546001600160a01b038381166001600160a01b0319831681179093556040519116919082907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e090600090a35050565b6008546001600160a01b03163314620001875760405162461bcd60e51b815260206004820181905260248201526000805160206200216083398151915260448201526064015b60405180910390fd5b600b80546001600160a01b0319166001600160a01b0392909216919091179055565b6008546001600160a01b03163314620001f45760405162461bcd60e51b815260206004820181905260248201526000805160206200216083398151915260448201526064016200017e565b8051620002099060099060208401906200020d565b5050565b8280546200021b90620002e5565b90600052602060002090601f0160209004810192826200023f57600085556200028a565b82601f106200025a57805160ff19168380011785556200028a565b828001600101855582156200028a579182015b828111156200028a5782518255916020019190600101906200026d565b50620002989291506200029c565b5090565b5b808211156200029857600081556001016200029d565b600060208284031215620002c657600080fd5b81516001600160a01b0381168114620002de57600080fd5b9392505050565b600181811c90821680620002fa57607f821691505b602082108114156200031c57634e487b7160e01b600052602260045260246000fd5b50919050565b611dd580620003326000396000f3fe608060405234801561001057600080fd5b506004361061017f5760003560e01c806370a08231116100d5578063b88d4fde11610091578063c87b56dd1161006e578063c87b56dd14610332578063e985e9c514610345578063f2fde38b14610381578063f926b50c1461039457005b8063b88d4fde146102f9578063b99c6a6a1461030c578063c79b04c91461031f57005b806370a082311461029f578063715018a6146102b2578063755edd17146102ba5780638da5cb5b146102cd57806395d89b41146102de578063a22cb465146102e657005b80631a7ef85d1161013c57806342842e0e1161011957806342842e0e14610253578063449a52f8146102665780636081c0c8146102795780636352211e1461028c57005b80631a7ef85d1461021a57806323b872dd1461022d5780633a4818dc1461024057005b806301ffc9a71461018157806306fdde03146101a9578063081812fc146101be578063095ea7b3146101e95780631249c58b146101fc57806318160ddd14610204575b005b61019461018f36600461190a565b6103a7565b60405190151581526020015b60405180910390f35b6101b16103f9565b6040516101a09190611b9e565b6101d16101cc366004611a4a565b61048b565b6040516001600160a01b0390911681526020016101a0565b61017f6101f73660046118c5565b6104cf565b61017f61056f565b600154600054035b6040519081526020016101a0565b61017f6102283660046117a9565b610628565b61017f61023b3660046117f7565b610674565b61017f61024e3660046118ef565b610805565b61017f6102613660046117f7565b61084d565b61017f6102743660046118c5565b61086d565b61017f6102873660046119dd565b61091e565b6101d161029a366004611a4a565b610a54565b61020c6102ad3660046117a9565b610a5f565b61017f610aae565b61017f6102c83660046117a9565b610ae2565b6008546001600160a01b03166101d1565b6101b1610b6c565b61017f6102f436600461189b565b610b7b565b61017f610307366004611833565b610c11565b61017f61031a366004611979565b610c55565b61017f61032d366004611944565b610d5d565b6101b1610340366004611a4a565b610d9a565b6101946103533660046117c4565b6001600160a01b03918216600090815260076020908152604080832093909416825291909152205460ff1690565b61017f61038f3660046117a9565b610e4e565b61017f6103a2366004611a4a565b610ee6565b60006301ffc9a760e01b6001600160e01b0319831614806103d857506380ac58cd60e01b6001600160e01b03198316145b806103f35750635b5e139f60e01b6001600160e01b03198316145b92915050565b60606002805461040890611d22565b80601f016020809104026020016040519081016040528092919081815260200182805461043490611d22565b80156104815780601f1061045657610100808354040283529160200191610481565b820191906000526020600020905b81548152906001019060200180831161046457829003601f168201915b5050505050905090565b600061049682610f15565b6104b3576040516333d1c03960e21b815260040160405180910390fd5b506000908152600660205260409020546001600160a01b031690565b60006104da82610a54565b9050336001600160a01b03821614610513576104f68133610353565b610513576040516367d9dca160e11b815260040160405180910390fd5b60008281526006602052604080822080546001600160a01b0319166001600160a01b0387811691821790925591518593918516917f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b92591a4505050565b600854600160a01b900460ff16806105a45750600b546001600160a01b0316331480156105a4575060016105a233610a5f565b105b6105c95760405162461bcd60e51b81526004016105c090611c54565b60405180910390fd5b3233146105e85760405162461bcd60e51b81526004016105c090611be8565b600a54600154600054036105fd906001611cd0565b111561061b5760405162461bcd60e51b81526004016105c090611bb1565b610626336001610f3c565b565b6008546001600160a01b031633146106525760405162461bcd60e51b81526004016105c090611c1f565b600b80546001600160a01b0319166001600160a01b0392909216919091179055565b600061067f82611019565b9050836001600160a01b0316816001600160a01b0316146106b25760405162a1148160e81b815260040160405180910390fd5b60008281526006602052604090208054338082146001600160a01b038816909114176106ff576106e28633610353565b6106ff57604051632ce44b5f60e11b815260040160405180910390fd5b6001600160a01b03851661072657604051633a954ecd60e21b815260040160405180910390fd5b801561073157600082555b6001600160a01b038681166000908152600560205260408082208054600019019055918716808252919020805460010190554260a01b17600160e11b17600085815260046020526040902055600160e11b83166107bc57600184016000818152600460205260409020546107ba5760005481146107ba5760008181526004602052604090208490555b505b83856001600160a01b0316876001600160a01b03167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef60405160405180910390a4505050505050565b6008546001600160a01b0316331461082f5760405162461bcd60e51b81526004016105c090611c1f565b60088054911515600160a01b0260ff60a01b19909216919091179055565b61086883838360405180602001604052806000815250610c11565b505050565b6008546001600160a01b031633146108975760405162461bcd60e51b81526004016105c090611c1f565b3233146108b65760405162461bcd60e51b81526004016105c090611be8565b60018110156108d75760405162461bcd60e51b81526004016105c090611c8b565b600a54816108e86001546000540390565b6108f29190611cd0565b11156109105760405162461bcd60e51b81526004016105c090611bb1565b61091a8282610f3c565b5050565b600854600160a01b900460ff16806109535750600b546001600160a01b0316331480156109535750600161095133610a5f565b105b61096f5760405162461bcd60e51b81526004016105c090611c54565b32331461098e5760405162461bcd60e51b81526004016105c090611be8565b60018110156109af5760405162461bcd60e51b81526004016105c090611c8b565b600a54816109c06001546000540390565b6109ca9190611cd0565b11156109e85760405162461bcd60e51b81526004016105c090611bb1565b600030338584604051602001610a019493929190611ad5565b604051602081830303815290604052805190602001209050610a23818461107a565b6000818152600c6020526040902080546001600160a01b03191633908117909155610a4e9083610f3c565b50505050565b60006103f382611019565b60006001600160a01b038216610a88576040516323d3ad8160e21b815260040160405180910390fd5b506001600160a01b031660009081526005602052604090205467ffffffffffffffff1690565b6008546001600160a01b03163314610ad85760405162461bcd60e51b81526004016105c090611c1f565b610626600061113a565b6008546001600160a01b03163314610b0c5760405162461bcd60e51b81526004016105c090611c1f565b323314610b2b5760405162461bcd60e51b81526004016105c090611be8565b600a5460015460005403610b40906001611cd0565b1115610b5e5760405162461bcd60e51b81526004016105c090611bb1565b610b69816001610f3c565b50565b60606003805461040890611d22565b6001600160a01b038216331415610ba55760405163b06307db60e01b815260040160405180910390fd5b3360008181526007602090815260408083206001600160a01b03871680855290835292819020805460ff191686151590811790915590519081529192917f17307eab39ab6107e8899845ad3d59bd9653f200f220920489ca2b5937696c31910160405180910390a35050565b610c1c848484610674565b6001600160a01b0383163b15610a4e57610c388484848461118c565b610a4e576040516368d2bf6b60e11b815260040160405180910390fd5b600854600160a01b900460ff1680610c8a5750600b546001600160a01b031633148015610c8a57506001610c8833610a5f565b105b610ca65760405162461bcd60e51b81526004016105c090611c54565b323314610cc55760405162461bcd60e51b81526004016105c090611be8565b600a5460015460005403610cda906001611cd0565b1115610cf85760405162461bcd60e51b81526004016105c090611bb1565b6000303384604051602001610d0f93929190611a8f565b604051602081830303815290604052805190602001209050610d31818361107a565b6000818152600c6020526040902080546001600160a01b03191633908117909155610868906001610f3c565b6008546001600160a01b03163314610d875760405162461bcd60e51b81526004016105c090611c1f565b805161091a906009906020840190611657565b6060610da582610f15565b610df15760405162461bcd60e51b815260206004820152601960248201527f54686520746f6b656e20646f6573206e6f742065786973742e0000000000000060448201526064016105c0565b6000610dfb611284565b9050805160001415610e1c5760405180602001604052806000815250610e47565b80610e2684611293565b604051602001610e37929190611b22565b6040516020818303038152906040525b9392505050565b6008546001600160a01b03163314610e785760405162461bcd60e51b81526004016105c090611c1f565b6001600160a01b038116610edd5760405162461bcd60e51b815260206004820152602660248201527f4f776e61626c653a206e6577206f776e657220697320746865207a65726f206160448201526564647265737360d01b60648201526084016105c0565b610b698161113a565b6008546001600160a01b03163314610f105760405162461bcd60e51b81526004016105c090611c1f565b600a55565b60008054821080156103f3575050600090815260046020526040902054600160e01b161590565b6000546001600160a01b038316610f6557604051622e076360e81b815260040160405180910390fd5b81610f835760405163b562e8dd60e01b815260040160405180910390fd5b6001600160a01b038316600081815260056020526040902080546801000000000000000185020190554260a01b6001841460e11b1717600082815260046020526040902055808281015b6040516001830192906001600160a01b038716906000907fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef908290a4808210610fcd5760005550505050565b60008160005481101561106157600081815260046020526040902054600160e01b811661105f575b80610e47575060001901600081815260046020526040902054611041565b505b604051636f96cda160e11b815260040160405180910390fd5b6008546001600160a01b031661109083836112e2565b6001600160a01b0316146110dc5760405162461bcd60e51b815260206004820152601360248201527214d251d39055155491481393d5081590531251606a1b60448201526064016105c0565b6000828152600c60205260409020546001600160a01b03161561091a5760405162461bcd60e51b815260206004820152601660248201527514d251d390551554914810531491505116481554d15160521b60448201526064016105c0565b600880546001600160a01b038381166001600160a01b0319831681179093556040519116919082907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e090600090a35050565b604051630a85bd0160e11b81526000906001600160a01b0385169063150b7a02906111c1903390899088908890600401611b61565b602060405180830381600087803b1580156111db57600080fd5b505af192505050801561120b575060408051601f3d908101601f1916820190925261120891810190611927565b60015b611266573d808015611239576040519150601f19603f3d011682016040523d82523d6000602084013e61123e565b606091505b50805161125e576040516368d2bf6b60e11b815260040160405180910390fd5b805181602001fd5b6001600160e01b031916630a85bd0160e11b1490505b949350505050565b60606009805461040890611d22565b604080516080810191829052607f0190826030600a8206018353600a90045b80156112d057600183039250600a81066030018353600a90046112b2565b50819003601f19909101908152919050565b60008060006112f18585611306565b915091506112fe81611376565b509392505050565b60008082516041141561133d5760208301516040840151606085015160001a61133187828585611531565b9450945050505061136f565b825160401415611367576020830151604084015161135c86838361161e565b93509350505061136f565b506000905060025b9250929050565b600081600481111561138a5761138a611d5d565b14156113935750565b60018160048111156113a7576113a7611d5d565b14156113f55760405162461bcd60e51b815260206004820152601860248201527f45434453413a20696e76616c6964207369676e6174757265000000000000000060448201526064016105c0565b600281600481111561140957611409611d5d565b14156114575760405162461bcd60e51b815260206004820152601f60248201527f45434453413a20696e76616c6964207369676e6174757265206c656e6774680060448201526064016105c0565b600381600481111561146b5761146b611d5d565b14156114c45760405162461bcd60e51b815260206004820152602260248201527f45434453413a20696e76616c6964207369676e6174757265202773272076616c604482015261756560f01b60648201526084016105c0565b60048160048111156114d8576114d8611d5d565b1415610b695760405162461bcd60e51b815260206004820152602260248201527f45434453413a20696e76616c6964207369676e6174757265202776272076616c604482015261756560f01b60648201526084016105c0565b6000807f7fffffffffffffffffffffffffffffff5d576e7357a4501ddfe92f46681b20a08311156115685750600090506003611615565b8460ff16601b1415801561158057508460ff16601c14155b156115915750600090506004611615565b6040805160008082526020820180845289905260ff881692820192909252606081018690526080810185905260019060a0016020604051602081039080840390855afa1580156115e5573d6000803e3d6000fd5b5050604051601f1901519150506001600160a01b03811661160e57600060019250925050611615565b9150600090505b94509492505050565b6000806001600160ff1b0383168161163b60ff86901c601b611cd0565b905061164987828885611531565b935093505050935093915050565b82805461166390611d22565b90600052602060002090601f01602090048101928261168557600085556116cb565b82601f1061169e57805160ff19168380011785556116cb565b828001600101855582156116cb579182015b828111156116cb5782518255916020019190600101906116b0565b506116d79291506116db565b5090565b5b808211156116d757600081556001016116dc565b80356001600160a01b038116811461170757600080fd5b919050565b8035801515811461170757600080fd5b600082601f83011261172d57600080fd5b813567ffffffffffffffff8082111561174857611748611d73565b604051601f8301601f19908116603f0116810190828211818310171561177057611770611d73565b8160405283815286602085880101111561178957600080fd5b836020870160208301376000602085830101528094505050505092915050565b6000602082840312156117bb57600080fd5b610e47826116f0565b600080604083850312156117d757600080fd5b6117e0836116f0565b91506117ee602084016116f0565b90509250929050565b60008060006060848603121561180c57600080fd5b611815846116f0565b9250611823602085016116f0565b9150604084013590509250925092565b6000806000806080858703121561184957600080fd5b611852856116f0565b9350611860602086016116f0565b925060408501359150606085013567ffffffffffffffff81111561188357600080fd5b61188f8782880161171c565b91505092959194509250565b600080604083850312156118ae57600080fd5b6118b7836116f0565b91506117ee6020840161170c565b600080604083850312156118d857600080fd5b6118e1836116f0565b946020939093013593505050565b60006020828403121561190157600080fd5b610e478261170c565b60006020828403121561191c57600080fd5b8135610e4781611d89565b60006020828403121561193957600080fd5b8151610e4781611d89565b60006020828403121561195657600080fd5b813567ffffffffffffffff81111561196d57600080fd5b61127c8482850161171c565b6000806040838503121561198c57600080fd5b823567ffffffffffffffff808211156119a457600080fd5b6119b08683870161171c565b935060208501359150808211156119c657600080fd5b506119d38582860161171c565b9150509250929050565b6000806000606084860312156119f257600080fd5b833567ffffffffffffffff80821115611a0a57600080fd5b611a168783880161171c565b94506020860135915080821115611a2c57600080fd5b50611a398682870161171c565b925050604084013590509250925092565b600060208284031215611a5c57600080fd5b5035919050565b60008151808452611a7b816020860160208601611cf6565b601f01601f19169290920160200192915050565b60006bffffffffffffffffffffffff19808660601b168352808560601b166014840152508251611ac6816028850160208701611cf6565b91909101602801949350505050565b60006bffffffffffffffffffffffff19808760601b168352808660601b166014840152508351611b0c816028850160208801611cf6565b6028920191820192909252604801949350505050565b60008351611b34818460208801611cf6565b835190830190611b48818360208801611cf6565b64173539b7b760d91b9101908152600501949350505050565b6001600160a01b0385811682528416602082015260408101839052608060608201819052600090611b9490830184611a63565b9695505050505050565b602081526000610e476020830184611a63565b6020808252601a908201527f4d415820434150204f462050455246554e204558434545444544000000000000604082015260600190565b6020808252601e908201527f5468652063616c6c657220697320616e6f7468657220636f6e74726163740000604082015260600190565b6020808252818101527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604082015260600190565b60208082526017908201527f4e4f5420464f52205055424c4943204d494e5420594554000000000000000000604082015260600190565b60208082526025908201527f5155414e54495459204d55535420455155414c204f522047524541544552205460408201526448454e203160d81b606082015260800190565b60008219821115611cf157634e487b7160e01b600052601160045260246000fd5b500190565b60005b83811015611d11578181015183820152602001611cf9565b83811115610a4e5750506000910152565b600181811c90821680611d3657607f821691505b60208210811415611d5757634e487b7160e01b600052602260045260246000fd5b50919050565b634e487b7160e01b600052602160045260246000fd5b634e487b7160e01b600052604160045260246000fd5b6001600160e01b031981168114610b6957600080fdfea2646970667358221220d80baf902837fc3226913632746ab84858f4789f565fc16ff81e2454f4f23aae64736f6c6343000807003368747470733a2f2f6261667962656965777a61646f706663616c376d676c686b75706b33323377656a3668696537367168656d797637667875797a696d6e37716633652e697066732e6e667473746f726167652e6c696e6b2f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572000000000000000000000000999e2d0a2d8d22177a1b43123ca4064d56503a7e

Deployed Bytecode

0x608060405234801561001057600080fd5b506004361061017f5760003560e01c806370a08231116100d5578063b88d4fde11610091578063c87b56dd1161006e578063c87b56dd14610332578063e985e9c514610345578063f2fde38b14610381578063f926b50c1461039457005b8063b88d4fde146102f9578063b99c6a6a1461030c578063c79b04c91461031f57005b806370a082311461029f578063715018a6146102b2578063755edd17146102ba5780638da5cb5b146102cd57806395d89b41146102de578063a22cb465146102e657005b80631a7ef85d1161013c57806342842e0e1161011957806342842e0e14610253578063449a52f8146102665780636081c0c8146102795780636352211e1461028c57005b80631a7ef85d1461021a57806323b872dd1461022d5780633a4818dc1461024057005b806301ffc9a71461018157806306fdde03146101a9578063081812fc146101be578063095ea7b3146101e95780631249c58b146101fc57806318160ddd14610204575b005b61019461018f36600461190a565b6103a7565b60405190151581526020015b60405180910390f35b6101b16103f9565b6040516101a09190611b9e565b6101d16101cc366004611a4a565b61048b565b6040516001600160a01b0390911681526020016101a0565b61017f6101f73660046118c5565b6104cf565b61017f61056f565b600154600054035b6040519081526020016101a0565b61017f6102283660046117a9565b610628565b61017f61023b3660046117f7565b610674565b61017f61024e3660046118ef565b610805565b61017f6102613660046117f7565b61084d565b61017f6102743660046118c5565b61086d565b61017f6102873660046119dd565b61091e565b6101d161029a366004611a4a565b610a54565b61020c6102ad3660046117a9565b610a5f565b61017f610aae565b61017f6102c83660046117a9565b610ae2565b6008546001600160a01b03166101d1565b6101b1610b6c565b61017f6102f436600461189b565b610b7b565b61017f610307366004611833565b610c11565b61017f61031a366004611979565b610c55565b61017f61032d366004611944565b610d5d565b6101b1610340366004611a4a565b610d9a565b6101946103533660046117c4565b6001600160a01b03918216600090815260076020908152604080832093909416825291909152205460ff1690565b61017f61038f3660046117a9565b610e4e565b61017f6103a2366004611a4a565b610ee6565b60006301ffc9a760e01b6001600160e01b0319831614806103d857506380ac58cd60e01b6001600160e01b03198316145b806103f35750635b5e139f60e01b6001600160e01b03198316145b92915050565b60606002805461040890611d22565b80601f016020809104026020016040519081016040528092919081815260200182805461043490611d22565b80156104815780601f1061045657610100808354040283529160200191610481565b820191906000526020600020905b81548152906001019060200180831161046457829003601f168201915b5050505050905090565b600061049682610f15565b6104b3576040516333d1c03960e21b815260040160405180910390fd5b506000908152600660205260409020546001600160a01b031690565b60006104da82610a54565b9050336001600160a01b03821614610513576104f68133610353565b610513576040516367d9dca160e11b815260040160405180910390fd5b60008281526006602052604080822080546001600160a01b0319166001600160a01b0387811691821790925591518593918516917f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b92591a4505050565b600854600160a01b900460ff16806105a45750600b546001600160a01b0316331480156105a4575060016105a233610a5f565b105b6105c95760405162461bcd60e51b81526004016105c090611c54565b60405180910390fd5b3233146105e85760405162461bcd60e51b81526004016105c090611be8565b600a54600154600054036105fd906001611cd0565b111561061b5760405162461bcd60e51b81526004016105c090611bb1565b610626336001610f3c565b565b6008546001600160a01b031633146106525760405162461bcd60e51b81526004016105c090611c1f565b600b80546001600160a01b0319166001600160a01b0392909216919091179055565b600061067f82611019565b9050836001600160a01b0316816001600160a01b0316146106b25760405162a1148160e81b815260040160405180910390fd5b60008281526006602052604090208054338082146001600160a01b038816909114176106ff576106e28633610353565b6106ff57604051632ce44b5f60e11b815260040160405180910390fd5b6001600160a01b03851661072657604051633a954ecd60e21b815260040160405180910390fd5b801561073157600082555b6001600160a01b038681166000908152600560205260408082208054600019019055918716808252919020805460010190554260a01b17600160e11b17600085815260046020526040902055600160e11b83166107bc57600184016000818152600460205260409020546107ba5760005481146107ba5760008181526004602052604090208490555b505b83856001600160a01b0316876001600160a01b03167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef60405160405180910390a4505050505050565b6008546001600160a01b0316331461082f5760405162461bcd60e51b81526004016105c090611c1f565b60088054911515600160a01b0260ff60a01b19909216919091179055565b61086883838360405180602001604052806000815250610c11565b505050565b6008546001600160a01b031633146108975760405162461bcd60e51b81526004016105c090611c1f565b3233146108b65760405162461bcd60e51b81526004016105c090611be8565b60018110156108d75760405162461bcd60e51b81526004016105c090611c8b565b600a54816108e86001546000540390565b6108f29190611cd0565b11156109105760405162461bcd60e51b81526004016105c090611bb1565b61091a8282610f3c565b5050565b600854600160a01b900460ff16806109535750600b546001600160a01b0316331480156109535750600161095133610a5f565b105b61096f5760405162461bcd60e51b81526004016105c090611c54565b32331461098e5760405162461bcd60e51b81526004016105c090611be8565b60018110156109af5760405162461bcd60e51b81526004016105c090611c8b565b600a54816109c06001546000540390565b6109ca9190611cd0565b11156109e85760405162461bcd60e51b81526004016105c090611bb1565b600030338584604051602001610a019493929190611ad5565b604051602081830303815290604052805190602001209050610a23818461107a565b6000818152600c6020526040902080546001600160a01b03191633908117909155610a4e9083610f3c565b50505050565b60006103f382611019565b60006001600160a01b038216610a88576040516323d3ad8160e21b815260040160405180910390fd5b506001600160a01b031660009081526005602052604090205467ffffffffffffffff1690565b6008546001600160a01b03163314610ad85760405162461bcd60e51b81526004016105c090611c1f565b610626600061113a565b6008546001600160a01b03163314610b0c5760405162461bcd60e51b81526004016105c090611c1f565b323314610b2b5760405162461bcd60e51b81526004016105c090611be8565b600a5460015460005403610b40906001611cd0565b1115610b5e5760405162461bcd60e51b81526004016105c090611bb1565b610b69816001610f3c565b50565b60606003805461040890611d22565b6001600160a01b038216331415610ba55760405163b06307db60e01b815260040160405180910390fd5b3360008181526007602090815260408083206001600160a01b03871680855290835292819020805460ff191686151590811790915590519081529192917f17307eab39ab6107e8899845ad3d59bd9653f200f220920489ca2b5937696c31910160405180910390a35050565b610c1c848484610674565b6001600160a01b0383163b15610a4e57610c388484848461118c565b610a4e576040516368d2bf6b60e11b815260040160405180910390fd5b600854600160a01b900460ff1680610c8a5750600b546001600160a01b031633148015610c8a57506001610c8833610a5f565b105b610ca65760405162461bcd60e51b81526004016105c090611c54565b323314610cc55760405162461bcd60e51b81526004016105c090611be8565b600a5460015460005403610cda906001611cd0565b1115610cf85760405162461bcd60e51b81526004016105c090611bb1565b6000303384604051602001610d0f93929190611a8f565b604051602081830303815290604052805190602001209050610d31818361107a565b6000818152600c6020526040902080546001600160a01b03191633908117909155610868906001610f3c565b6008546001600160a01b03163314610d875760405162461bcd60e51b81526004016105c090611c1f565b805161091a906009906020840190611657565b6060610da582610f15565b610df15760405162461bcd60e51b815260206004820152601960248201527f54686520746f6b656e20646f6573206e6f742065786973742e0000000000000060448201526064016105c0565b6000610dfb611284565b9050805160001415610e1c5760405180602001604052806000815250610e47565b80610e2684611293565b604051602001610e37929190611b22565b6040516020818303038152906040525b9392505050565b6008546001600160a01b03163314610e785760405162461bcd60e51b81526004016105c090611c1f565b6001600160a01b038116610edd5760405162461bcd60e51b815260206004820152602660248201527f4f776e61626c653a206e6577206f776e657220697320746865207a65726f206160448201526564647265737360d01b60648201526084016105c0565b610b698161113a565b6008546001600160a01b03163314610f105760405162461bcd60e51b81526004016105c090611c1f565b600a55565b60008054821080156103f3575050600090815260046020526040902054600160e01b161590565b6000546001600160a01b038316610f6557604051622e076360e81b815260040160405180910390fd5b81610f835760405163b562e8dd60e01b815260040160405180910390fd5b6001600160a01b038316600081815260056020526040902080546801000000000000000185020190554260a01b6001841460e11b1717600082815260046020526040902055808281015b6040516001830192906001600160a01b038716906000907fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef908290a4808210610fcd5760005550505050565b60008160005481101561106157600081815260046020526040902054600160e01b811661105f575b80610e47575060001901600081815260046020526040902054611041565b505b604051636f96cda160e11b815260040160405180910390fd5b6008546001600160a01b031661109083836112e2565b6001600160a01b0316146110dc5760405162461bcd60e51b815260206004820152601360248201527214d251d39055155491481393d5081590531251606a1b60448201526064016105c0565b6000828152600c60205260409020546001600160a01b03161561091a5760405162461bcd60e51b815260206004820152601660248201527514d251d390551554914810531491505116481554d15160521b60448201526064016105c0565b600880546001600160a01b038381166001600160a01b0319831681179093556040519116919082907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e090600090a35050565b604051630a85bd0160e11b81526000906001600160a01b0385169063150b7a02906111c1903390899088908890600401611b61565b602060405180830381600087803b1580156111db57600080fd5b505af192505050801561120b575060408051601f3d908101601f1916820190925261120891810190611927565b60015b611266573d808015611239576040519150601f19603f3d011682016040523d82523d6000602084013e61123e565b606091505b50805161125e576040516368d2bf6b60e11b815260040160405180910390fd5b805181602001fd5b6001600160e01b031916630a85bd0160e11b1490505b949350505050565b60606009805461040890611d22565b604080516080810191829052607f0190826030600a8206018353600a90045b80156112d057600183039250600a81066030018353600a90046112b2565b50819003601f19909101908152919050565b60008060006112f18585611306565b915091506112fe81611376565b509392505050565b60008082516041141561133d5760208301516040840151606085015160001a61133187828585611531565b9450945050505061136f565b825160401415611367576020830151604084015161135c86838361161e565b93509350505061136f565b506000905060025b9250929050565b600081600481111561138a5761138a611d5d565b14156113935750565b60018160048111156113a7576113a7611d5d565b14156113f55760405162461bcd60e51b815260206004820152601860248201527f45434453413a20696e76616c6964207369676e6174757265000000000000000060448201526064016105c0565b600281600481111561140957611409611d5d565b14156114575760405162461bcd60e51b815260206004820152601f60248201527f45434453413a20696e76616c6964207369676e6174757265206c656e6774680060448201526064016105c0565b600381600481111561146b5761146b611d5d565b14156114c45760405162461bcd60e51b815260206004820152602260248201527f45434453413a20696e76616c6964207369676e6174757265202773272076616c604482015261756560f01b60648201526084016105c0565b60048160048111156114d8576114d8611d5d565b1415610b695760405162461bcd60e51b815260206004820152602260248201527f45434453413a20696e76616c6964207369676e6174757265202776272076616c604482015261756560f01b60648201526084016105c0565b6000807f7fffffffffffffffffffffffffffffff5d576e7357a4501ddfe92f46681b20a08311156115685750600090506003611615565b8460ff16601b1415801561158057508460ff16601c14155b156115915750600090506004611615565b6040805160008082526020820180845289905260ff881692820192909252606081018690526080810185905260019060a0016020604051602081039080840390855afa1580156115e5573d6000803e3d6000fd5b5050604051601f1901519150506001600160a01b03811661160e57600060019250925050611615565b9150600090505b94509492505050565b6000806001600160ff1b0383168161163b60ff86901c601b611cd0565b905061164987828885611531565b935093505050935093915050565b82805461166390611d22565b90600052602060002090601f01602090048101928261168557600085556116cb565b82601f1061169e57805160ff19168380011785556116cb565b828001600101855582156116cb579182015b828111156116cb5782518255916020019190600101906116b0565b506116d79291506116db565b5090565b5b808211156116d757600081556001016116dc565b80356001600160a01b038116811461170757600080fd5b919050565b8035801515811461170757600080fd5b600082601f83011261172d57600080fd5b813567ffffffffffffffff8082111561174857611748611d73565b604051601f8301601f19908116603f0116810190828211818310171561177057611770611d73565b8160405283815286602085880101111561178957600080fd5b836020870160208301376000602085830101528094505050505092915050565b6000602082840312156117bb57600080fd5b610e47826116f0565b600080604083850312156117d757600080fd5b6117e0836116f0565b91506117ee602084016116f0565b90509250929050565b60008060006060848603121561180c57600080fd5b611815846116f0565b9250611823602085016116f0565b9150604084013590509250925092565b6000806000806080858703121561184957600080fd5b611852856116f0565b9350611860602086016116f0565b925060408501359150606085013567ffffffffffffffff81111561188357600080fd5b61188f8782880161171c565b91505092959194509250565b600080604083850312156118ae57600080fd5b6118b7836116f0565b91506117ee6020840161170c565b600080604083850312156118d857600080fd5b6118e1836116f0565b946020939093013593505050565b60006020828403121561190157600080fd5b610e478261170c565b60006020828403121561191c57600080fd5b8135610e4781611d89565b60006020828403121561193957600080fd5b8151610e4781611d89565b60006020828403121561195657600080fd5b813567ffffffffffffffff81111561196d57600080fd5b61127c8482850161171c565b6000806040838503121561198c57600080fd5b823567ffffffffffffffff808211156119a457600080fd5b6119b08683870161171c565b935060208501359150808211156119c657600080fd5b506119d38582860161171c565b9150509250929050565b6000806000606084860312156119f257600080fd5b833567ffffffffffffffff80821115611a0a57600080fd5b611a168783880161171c565b94506020860135915080821115611a2c57600080fd5b50611a398682870161171c565b925050604084013590509250925092565b600060208284031215611a5c57600080fd5b5035919050565b60008151808452611a7b816020860160208601611cf6565b601f01601f19169290920160200192915050565b60006bffffffffffffffffffffffff19808660601b168352808560601b166014840152508251611ac6816028850160208701611cf6565b91909101602801949350505050565b60006bffffffffffffffffffffffff19808760601b168352808660601b166014840152508351611b0c816028850160208801611cf6565b6028920191820192909252604801949350505050565b60008351611b34818460208801611cf6565b835190830190611b48818360208801611cf6565b64173539b7b760d91b9101908152600501949350505050565b6001600160a01b0385811682528416602082015260408101839052608060608201819052600090611b9490830184611a63565b9695505050505050565b602081526000610e476020830184611a63565b6020808252601a908201527f4d415820434150204f462050455246554e204558434545444544000000000000604082015260600190565b6020808252601e908201527f5468652063616c6c657220697320616e6f7468657220636f6e74726163740000604082015260600190565b6020808252818101527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604082015260600190565b60208082526017908201527f4e4f5420464f52205055424c4943204d494e5420594554000000000000000000604082015260600190565b60208082526025908201527f5155414e54495459204d55535420455155414c204f522047524541544552205460408201526448454e203160d81b606082015260800190565b60008219821115611cf157634e487b7160e01b600052601160045260246000fd5b500190565b60005b83811015611d11578181015183820152602001611cf9565b83811115610a4e5750506000910152565b600181811c90821680611d3657607f821691505b60208210811415611d5757634e487b7160e01b600052602260045260246000fd5b50919050565b634e487b7160e01b600052602160045260246000fd5b634e487b7160e01b600052604160045260246000fd5b6001600160e01b031981168114610b6957600080fdfea2646970667358221220d80baf902837fc3226913632746ab84858f4789f565fc16ff81e2454f4f23aae64736f6c63430008070033

Constructor Arguments (ABI-Encoded and is the last bytes of the Contract Creation Code above)

000000000000000000000000999e2d0a2d8d22177a1b43123ca4064d56503a7e

-----Decoded View---------------
Arg [0] : _firstMinter (address): 0x999e2D0A2D8D22177A1B43123cA4064d56503A7E

-----Encoded View---------------
1 Constructor Arguments found :
Arg [0] : 000000000000000000000000999e2d0a2d8d22177a1b43123ca4064d56503a7e


Deployed Bytecode Sourcemap

59917:3686:0:-:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;29713:615;;;;;;:::i;:::-;;:::i;:::-;;;8154:14:1;;8147:22;8129:41;;8117:2;8102:18;29713:615:0;;;;;;;;35360:100;;;:::i;:::-;;;;;;;:::i;37306:204::-;;;;;;:::i;:::-;;:::i;:::-;;;-1:-1:-1;;;;;7452:32:1;;;7434:51;;7422:2;7407:18;37306:204:0;7288:203:1;36854:386:0;;;;;;:::i;:::-;;:::i;61332:170::-;;;:::i;28767:315::-;29033:12;;28820:7;29017:13;:28;28767:315;;;13766:25:1;;;13754:2;13739:18;28767:315:0;13620:177:1;61103:108:0;;;;;;:::i;:::-;;:::i;46571:2800::-;;;;;;:::i;:::-;;:::i;60997:94::-;;;;;;:::i;:::-;;:::i;38196:185::-;;;;;;:::i;:::-;;:::i;61514:275::-;;;;;;:::i;:::-;;:::i;62614:501::-;;;;;;:::i;:::-;;:::i;35149:144::-;;;;;;:::i;:::-;;:::i;30392:224::-;;;;;;:::i;:::-;;:::i;14304:103::-;;;:::i;61801:169::-;;;;;;:::i;:::-;;:::i;13653:87::-;13726:6;;-1:-1:-1;;;;;13726:6:0;13653:87;;35529:104;;;:::i;37582:308::-;;;;;;:::i;:::-;;:::i;38452:399::-;;;;;;:::i;:::-;;:::i;62221:385::-;;;;;;:::i;:::-;;:::i;61219:105::-;;;;;;:::i;:::-;;:::i;63240:360::-;;;;;;:::i;:::-;;:::i;37961:164::-;;;;;;:::i;:::-;-1:-1:-1;;;;;38082:25:0;;;38058:4;38082:25;;;:18;:25;;;;;;;;:35;;;;;;;;;;;;;;;37961:164;14562:201;;;;;;:::i;:::-;;:::i;60890:99::-;;;;;;:::i;:::-;;:::i;29713:615::-;29798:4;-1:-1:-1;;;;;;;;;30098:25:0;;;;:102;;-1:-1:-1;;;;;;;;;;30175:25:0;;;30098:102;:179;;;-1:-1:-1;;;;;;;;;;30252:25:0;;;30098:179;30078:199;29713:615;-1:-1:-1;;29713:615:0:o;35360:100::-;35414:13;35447:5;35440:12;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;35360:100;:::o;37306:204::-;37374:7;37399:16;37407:7;37399;:16::i;:::-;37394:64;;37424:34;;-1:-1:-1;;;37424:34:0;;;;;;;;;;;37394:64;-1:-1:-1;37478:24:0;;;;:15;:24;;;;;;-1:-1:-1;;;;;37478:24:0;;37306:204::o;36854:386::-;36927:13;36943:16;36951:7;36943;:16::i;:::-;36927:32;-1:-1:-1;57754:10:0;-1:-1:-1;;;;;36976:28:0;;;36972:175;;37024:44;37041:5;57754:10;37961:164;:::i;37024:44::-;37019:128;;37096:35;;-1:-1:-1;;;37096:35:0;;;;;;;;;;;37019:128;37159:24;;;;:15;:24;;;;;;:29;;-1:-1:-1;;;;;;37159:29:0;-1:-1:-1;;;;;37159:29:0;;;;;;;;;37204:28;;37159:24;;37204:28;;;;;;;36916:324;36854:386;;:::o;61332:170::-;60292:12;;-1:-1:-1;;;60292:12:0;;;;;:92;;-1:-1:-1;60323:11:0;;-1:-1:-1;;;;;60323:11:0;60309:10;:25;:74;;;;;60043:1;60338:21;60348:10;60338:9;:21::i;:::-;:45;60309:74;60284:128;;;;-1:-1:-1;;;60284:128:0;;;;;;;:::i;:::-;;;;;;;;;60539:9:::1;60552:10;60539:23;60531:66;;;;-1:-1:-1::0;;;60531:66:0::1;;;;;;;:::i;:::-;61425:7:::2;::::0;29033:12;;28820:7;29017:13;:28;61404:17:::2;::::0;61420:1:::2;61404:17;:::i;:::-;:28;;61396:67;;;;-1:-1:-1::0;;;61396:67:0::2;;;;;;;:::i;:::-;61474:20;61480:10;61492:1;61474:5;:20::i;:::-;61332:170::o:0;61103:108::-;13726:6;;-1:-1:-1;;;;;13726:6:0;57754:10;13873:23;13865:68;;;;-1:-1:-1;;;13865:68:0;;;;;;;:::i;:::-;61177:11:::1;:26:::0;;-1:-1:-1;;;;;;61177:26:0::1;-1:-1:-1::0;;;;;61177:26:0;;;::::1;::::0;;;::::1;::::0;;61103:108::o;46571:2800::-;46705:27;46735;46754:7;46735:18;:27::i;:::-;46705:57;;46820:4;-1:-1:-1;;;;;46779:45:0;46795:19;-1:-1:-1;;;;;46779:45:0;;46775:86;;46833:28;;-1:-1:-1;;;46833:28:0;;;;;;;;;;;46775:86;46875:27;45301:21;;;45128:15;45343:4;45336:36;45425:4;45409:21;;45515:26;;57754:10;46268:30;;;-1:-1:-1;;;;;45966:26:0;;46247:19;;;46244:55;47054:174;;47141:43;47158:4;57754:10;37961:164;:::i;47141:43::-;47136:92;;47193:35;;-1:-1:-1;;;47193:35:0;;;;;;;;;;;47136:92;-1:-1:-1;;;;;47245:16:0;;47241:52;;47270:23;;-1:-1:-1;;;47270:23:0;;;;;;;;;;;47241:52;47442:15;47439:160;;;47582:1;47561:19;47554:30;47439:160;-1:-1:-1;;;;;47977:24:0;;;;;;;:18;:24;;;;;;47975:26;;-1:-1:-1;;47975:26:0;;;48046:22;;;;;;;;;48044:24;;-1:-1:-1;48044:24:0;;;35048:11;35024:22;35020:40;35007:62;-1:-1:-1;;;35007:62:0;48339:26;;;;:17;:26;;;;;:174;-1:-1:-1;;;48633:46:0;;48629:626;;48737:1;48727:11;;48705:19;48860:30;;;:17;:30;;;;;;48856:384;;48998:13;;48983:11;:28;48979:242;;49145:30;;;;:17;:30;;;;;:52;;;48979:242;48686:569;48629:626;49302:7;49298:2;-1:-1:-1;;;;;49283:27:0;49292:4;-1:-1:-1;;;;;49283:27:0;;;;;;;;;;;46694:2677;;;46571:2800;;;:::o;60997:94::-;13726:6;;-1:-1:-1;;;;;13726:6:0;57754:10;13873:23;13865:68;;;;-1:-1:-1;;;13865:68:0;;;;;;;:::i;:::-;61062:12:::1;:21:::0;;;::::1;;-1:-1:-1::0;;;61062:21:0::1;-1:-1:-1::0;;;;61062:21:0;;::::1;::::0;;;::::1;::::0;;60997:94::o;38196:185::-;38334:39;38351:4;38357:2;38361:7;38334:39;;;;;;;;;;;;:16;:39::i;:::-;38196:185;;;:::o;61514:275::-;13726:6;;-1:-1:-1;;;;;13726:6:0;57754:10;13873:23;13865:68;;;;-1:-1:-1;;;13865:68:0;;;;;;;:::i;:::-;60539:9:::1;60552:10;60539:23;60531:66;;;;-1:-1:-1::0;;;60531:66:0::1;;;;;;;:::i;:::-;61623:1:::2;61611:8;:13;;61603:63;;;;-1:-1:-1::0;;;61603:63:0::2;;;;;;;:::i;:::-;61713:7;;61701:8;61685:13;29033:12:::0;;28820:7;29017:13;:28;;28767:315;61685:13:::2;:24;;;;:::i;:::-;:35;;61677:74;;;;-1:-1:-1::0;;;61677:74:0::2;;;;;;;:::i;:::-;61762:19;61768:2;61772:8;61762:5;:19::i;:::-;61514:275:::0;;:::o;62614:501::-;60292:12;;-1:-1:-1;;;60292:12:0;;;;;:92;;-1:-1:-1;60323:11:0;;-1:-1:-1;;;;;60323:11:0;60309:10;:25;:74;;;;;60043:1;60338:21;60348:10;60338:9;:21::i;:::-;:45;60309:74;60284:128;;;;-1:-1:-1;;;60284:128:0;;;;;;;:::i;:::-;60539:9:::1;60552:10;60539:23;60531:66;;;;-1:-1:-1::0;;;60531:66:0::1;;;;;;;:::i;:::-;62757:1:::2;62745:8;:13;;62737:63;;;;-1:-1:-1::0;;;62737:63:0::2;;;;;;;:::i;:::-;62847:7;;62835:8;62819:13;29033:12:::0;;28820:7;29017:13;:28;;28767:315;62819:13:::2;:24;;;;:::i;:::-;:35;;62811:74;;;;-1:-1:-1::0;;;62811:74:0::2;;;;;;;:::i;:::-;62896:15;62949:4;57754:10:::0;62970:2:::2;62974:8;62924:59;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;62914:70;;;;;;62896:88;;62995:30;63012:7;63021:3;62995:16;:30::i;:::-;63036:18;::::0;;;:9:::2;:18;::::0;;;;:33;;-1:-1:-1;;;;;;63036:33:0::2;57754:10:::0;63036:33;;::::2;::::0;;;63080:27:::2;::::0;63098:8;63080:5:::2;:27::i;:::-;62726:389;62614:501:::0;;;:::o;35149:144::-;35213:7;35256:27;35275:7;35256:18;:27::i;30392:224::-;30456:7;-1:-1:-1;;;;;30480:19:0;;30476:60;;30508:28;;-1:-1:-1;;;30508:28:0;;;;;;;;;;;30476:60;-1:-1:-1;;;;;;30554:25:0;;;;;:18;:25;;;;;;24947:13;30554:54;;30392:224::o;14304:103::-;13726:6;;-1:-1:-1;;;;;13726:6:0;57754:10;13873:23;13865:68;;;;-1:-1:-1;;;13865:68:0;;;;;;;:::i;:::-;14369:30:::1;14396:1;14369:18;:30::i;61801:169::-:0;13726:6;;-1:-1:-1;;;;;13726:6:0;57754:10;13873:23;13865:68;;;;-1:-1:-1;;;13865:68:0;;;;;;;:::i;:::-;60539:9:::1;60552:10;60539:23;60531:66;;;;-1:-1:-1::0;;;60531:66:0::1;;;;;;;:::i;:::-;61901:7:::2;::::0;29033:12;;28820:7;29017:13;:28;61880:17:::2;::::0;61896:1:::2;61880:17;:::i;:::-;:28;;61872:67;;;;-1:-1:-1::0;;;61872:67:0::2;;;;;;;:::i;:::-;61950:12;61956:2;61960:1;61950:5;:12::i;:::-;61801:169:::0;:::o;35529:104::-;35585:13;35618:7;35611:14;;;;;:::i;37582:308::-;-1:-1:-1;;;;;37681:31:0;;57754:10;37681:31;37677:61;;;37721:17;;-1:-1:-1;;;37721:17:0;;;;;;;;;;;37677:61;57754:10;37751:39;;;;:18;:39;;;;;;;;-1:-1:-1;;;;;37751:49:0;;;;;;;;;;;;:60;;-1:-1:-1;;37751:60:0;;;;;;;;;;37827:55;;8129:41:1;;;37751:49:0;;57754:10;37827:55;;8102:18:1;37827:55:0;;;;;;;37582:308;;:::o;38452:399::-;38619:31;38632:4;38638:2;38642:7;38619:12;:31::i;:::-;-1:-1:-1;;;;;38665:14:0;;;:19;38661:183;;38704:56;38735:4;38741:2;38745:7;38754:5;38704:30;:56::i;:::-;38699:145;;38788:40;;-1:-1:-1;;;38788:40:0;;;;;;;;;;;62221:385;60292:12;;-1:-1:-1;;;60292:12:0;;;;;:92;;-1:-1:-1;60323:11:0;;-1:-1:-1;;;;;60323:11:0;60309:10;:25;:74;;;;;60043:1;60338:21;60348:10;60338:9;:21::i;:::-;:45;60309:74;60284:128;;;;-1:-1:-1;;;60284:128:0;;;;;;;:::i;:::-;60539:9:::1;60552:10;60539:23;60531:66;;;;-1:-1:-1::0;;;60531:66:0::1;;;;;;;:::i;:::-;62355:7:::2;::::0;29033:12;;28820:7;29017:13;:28;62334:17:::2;::::0;62350:1:::2;62334:17;:::i;:::-;:28;;62326:67;;;;-1:-1:-1::0;;;62326:67:0::2;;;;;;;:::i;:::-;62404:15;62457:4;57754:10:::0;62478:2:::2;62432:49;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;62422:60;;;;;;62404:78;;62493:30;62510:7;62519:3;62493:16;:30::i;:::-;62534:18;::::0;;;:9:::2;:18;::::0;;;;:33;;-1:-1:-1;;;;;;62534:33:0::2;57754:10:::0;62534:33;;::::2;::::0;;;62578:20:::2;::::0;-1:-1:-1;62578:5:0::2;:20::i;61219:105::-:0;13726:6;;-1:-1:-1;;;;;13726:6:0;57754:10;13873:23;13865:68;;;;-1:-1:-1;;;13865:68:0;;;;;;;:::i;:::-;61296:20;;::::1;::::0;:8:::1;::::0;:20:::1;::::0;::::1;::::0;::::1;:::i;63240:360::-:0;63313:13;63343:16;63351:7;63343;:16::i;:::-;63339:93;;63385:35;;-1:-1:-1;;;63385:35:0;;11247:2:1;63385:35:0;;;11229:21:1;11286:2;11266:18;;;11259:30;11325:27;11305:18;;;11298:55;11370:18;;63385:35:0;11045:349:1;63339:93:0;63444:21;63468:10;:8;:10::i;:::-;63444:34;;63502:7;63496:21;63521:1;63496:26;;:96;;;;;;;;;;;;;;;;;63549:7;63558:18;63568:7;63558:9;:18::i;:::-;63532:54;;;;;;;;;:::i;:::-;;;;;;;;;;;;;63496:96;63489:103;63240:360;-1:-1:-1;;;63240:360:0:o;14562:201::-;13726:6;;-1:-1:-1;;;;;13726:6:0;57754:10;13873:23;13865:68;;;;-1:-1:-1;;;13865:68:0;;;;;;;:::i;:::-;-1:-1:-1;;;;;14651:22:0;::::1;14643:73;;;::::0;-1:-1:-1;;;14643:73:0;;10078:2:1;14643:73:0::1;::::0;::::1;10060:21:1::0;10117:2;10097:18;;;10090:30;10156:34;10136:18;;;10129:62;-1:-1:-1;;;10207:18:1;;;10200:36;10253:19;;14643:73:0::1;9876:402:1::0;14643:73:0::1;14727:28;14746:8;14727:18;:28::i;60890:99::-:0;13726:6;;-1:-1:-1;;;;;13726:6:0;57754:10;13873:23;13865:68;;;;-1:-1:-1;;;13865:68:0;;;;;;;:::i;:::-;60962:7:::1;:19:::0;60890:99::o;39106:273::-;39163:4;39253:13;;39243:7;:23;39200:152;;;;-1:-1:-1;;39304:26:0;;;;:17;:26;;;;;;-1:-1:-1;;;39304:43:0;:48;;39106:273::o;40937:1529::-;41002:20;41025:13;-1:-1:-1;;;;;41053:16:0;;41049:48;;41078:19;;-1:-1:-1;;;41078:19:0;;;;;;;;;;;41049:48;41112:13;41108:44;;41134:18;;-1:-1:-1;;;41134:18:0;;;;;;;;;;;41108:44;-1:-1:-1;;;;;41640:22:0;;;;;;:18;:22;;25084:2;41640:22;;:70;;41678:31;41666:44;;41640:70;;;35048:11;35024:22;35020:40;-1:-1:-1;36758:15:0;;36733:23;36729:45;35017:51;35007:62;41953:31;;;;:17;:31;;;;;:173;41971:12;42202:23;;;42240:101;42267:35;;42292:9;;;;;-1:-1:-1;;;;;42267:35:0;;;42284:1;;42267:35;;42284:1;;42267:35;42336:3;42326:7;:13;42240:101;;42357:13;:19;-1:-1:-1;38196:185:0;;;:::o;32066:1129::-;32133:7;32168;32270:13;;32263:4;:20;32259:869;;;32308:14;32325:23;;;:17;:23;;;;;;-1:-1:-1;;;32414:23:0;;32410:699;;32933:113;32940:11;32933:113;;-1:-1:-1;;;33011:6:0;32993:25;;;;:17;:25;;;;;;32933:113;;32410:699;32285:843;32259:869;33156:31;;-1:-1:-1;;;33156:31:0;;;;;;;;;;;61978:235;13726:6;;-1:-1:-1;;;;;13726:6:0;62072:20;:7;62088:3;62072:15;:20::i;:::-;-1:-1:-1;;;;;62072:31:0;;62064:63;;;;-1:-1:-1;;;62064:63:0;;12004:2:1;62064:63:0;;;11986:21:1;12043:2;12023:18;;;12016:30;-1:-1:-1;;;12062:18:1;;;12055:49;12121:18;;62064:63:0;11802:343:1;62064:63:0;62176:1;62146:18;;;:9;:18;;;;;;-1:-1:-1;;;;;62146:18:0;:32;62138:67;;;;-1:-1:-1;;;62138:67:0;;12713:2:1;62138:67:0;;;12695:21:1;12752:2;12732:18;;;12725:30;-1:-1:-1;;;12771:18:1;;;12764:52;12833:18;;62138:67:0;12511:346:1;14923:191:0;15016:6;;;-1:-1:-1;;;;;15033:17:0;;;-1:-1:-1;;;;;;15033:17:0;;;;;;;15066:40;;15016:6;;;15033:17;15016:6;;15066:40;;14997:16;;15066:40;14986:128;14923:191;:::o;53322:716::-;53506:88;;-1:-1:-1;;;53506:88:0;;53485:4;;-1:-1:-1;;;;;53506:45:0;;;;;:88;;57754:10;;53573:4;;53579:7;;53588:5;;53506:88;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;53506:88:0;;;;;;;;-1:-1:-1;;53506:88:0;;;;;;;;;;;;:::i;:::-;;;53502:529;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;53789:13:0;;53785:235;;53835:40;;-1:-1:-1;;;53835:40:0;;;;;;;;;;;53785:235;53978:6;53972:13;53963:6;53959:2;53955:15;53948:38;53502:529;-1:-1:-1;;;;;;53665:64:0;-1:-1:-1;;;53665:64:0;;-1:-1:-1;53502:529:0;53322:716;;;;;;:::o;63123:109::-;63183:13;63216:8;63209:15;;;;;:::i;57878:1960::-;58347:4;58341:11;;58354:3;58337:21;;58432:17;;;;59128:11;;;59007:5;59260:2;59274;59264:13;;59256:22;59128:11;59243:36;59315:2;59305:13;;58899:697;59334:4;58899:697;;;59525:1;59520:3;59516:11;59509:18;;59576:2;59570:4;59566:13;59562:2;59558:22;59553:3;59545:36;59429:2;59419:13;;58899:697;;;-1:-1:-1;59626:13:0;;;-1:-1:-1;;59741:12:0;;;59801:19;;;59741:12;57878:1960;-1:-1:-1;57878:1960:0:o;6581:231::-;6659:7;6680:17;6699:18;6721:27;6732:4;6738:9;6721:10;:27::i;:::-;6679:69;;;;6759:18;6771:5;6759:11;:18::i;:::-;-1:-1:-1;6795:9:0;6581:231;-1:-1:-1;;;6581:231:0:o;4471:1308::-;4552:7;4561:12;4786:9;:16;4806:2;4786:22;4782:990;;;5082:4;5067:20;;5061:27;5132:4;5117:20;;5111:27;5190:4;5175:20;;5169:27;4825:9;5161:36;5233:25;5244:4;5161:36;5061:27;5111;5233:10;:25::i;:::-;5226:32;;;;;;;;;4782:990;5280:9;:16;5300:2;5280:22;5276:496;;;5555:4;5540:20;;5534:27;5606:4;5591:20;;5585:27;5648:23;5659:4;5534:27;5585;5648:10;:23::i;:::-;5641:30;;;;;;;;5276:496;-1:-1:-1;5720:1:0;;-1:-1:-1;5724:35:0;5276:496;4471:1308;;;;;:::o;2742:643::-;2820:20;2811:5;:29;;;;;;;;:::i;:::-;;2807:571;;;2742:643;:::o;2807:571::-;2918:29;2909:5;:38;;;;;;;;:::i;:::-;;2905:473;;;2964:34;;-1:-1:-1;;;2964:34:0;;9010:2:1;2964:34:0;;;8992:21:1;9049:2;9029:18;;;9022:30;9088:26;9068:18;;;9061:54;9132:18;;2964:34:0;8808:348:1;2905:473:0;3029:35;3020:5;:44;;;;;;;;:::i;:::-;;3016:362;;;3081:41;;-1:-1:-1;;;3081:41:0;;9718:2:1;3081:41:0;;;9700:21:1;9757:2;9737:18;;;9730:30;9796:33;9776:18;;;9769:61;9847:18;;3081:41:0;9516:355:1;3016:362:0;3153:30;3144:5;:39;;;;;;;;:::i;:::-;;3140:238;;;3200:44;;-1:-1:-1;;;3200:44:0;;10485:2:1;3200:44:0;;;10467:21:1;10524:2;10504:18;;;10497:30;10563:34;10543:18;;;10536:62;-1:-1:-1;;;10614:18:1;;;10607:32;10656:19;;3200:44:0;10283:398:1;3140:238:0;3275:30;3266:5;:39;;;;;;;;:::i;:::-;;3262:116;;;3322:44;;-1:-1:-1;;;3322:44:0;;11601:2:1;3322:44:0;;;11583:21:1;11640:2;11620:18;;;11613:30;11679:34;11659:18;;;11652:62;-1:-1:-1;;;11730:18:1;;;11723:32;11772:19;;3322:44:0;11399:398:1;8033:1632:0;8164:7;;9098:66;9085:79;;9081:163;;;-1:-1:-1;9197:1:0;;-1:-1:-1;9201:30:0;9181:51;;9081:163;9258:1;:7;;9263:2;9258:7;;:18;;;;;9269:1;:7;;9274:2;9269:7;;9258:18;9254:102;;;-1:-1:-1;9309:1:0;;-1:-1:-1;9313:30:0;9293:51;;9254:102;9470:24;;;9453:14;9470:24;;;;;;;;;8408:25:1;;;8481:4;8469:17;;8449:18;;;8442:45;;;;8503:18;;;8496:34;;;8546:18;;;8539:34;;;9470:24:0;;8380:19:1;;9470:24:0;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;;9470:24:0;;-1:-1:-1;;9470:24:0;;;-1:-1:-1;;;;;;;9509:20:0;;9505:103;;9562:1;9566:29;9546:50;;;;;;;9505:103;9628:6;-1:-1:-1;9636:20:0;;-1:-1:-1;8033:1632:0;;;;;;;;:::o;7075:344::-;7189:7;;-1:-1:-1;;;;;7235:80:0;;7189:7;7342:25;7358:3;7343:18;;;7365:2;7342:25;:::i;:::-;7326:42;;7386:25;7397:4;7403:1;7406;7409;7386:10;:25::i;:::-;7379:32;;;;;;7075:344;;;;;;:::o;-1:-1:-1:-;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;:::o;:::-;;;;;;;;;;;;;;;14:173:1;82:20;;-1:-1:-1;;;;;131:31:1;;121:42;;111:70;;177:1;174;167:12;111:70;14:173;;;:::o;192:160::-;257:20;;313:13;;306:21;296:32;;286:60;;342:1;339;332:12;357:718;399:5;452:3;445:4;437:6;433:17;429:27;419:55;;470:1;467;460:12;419:55;506:6;493:20;532:18;569:2;565;562:10;559:36;;;575:18;;:::i;:::-;650:2;644:9;618:2;704:13;;-1:-1:-1;;700:22:1;;;724:2;696:31;692:40;680:53;;;748:18;;;768:22;;;745:46;742:72;;;794:18;;:::i;:::-;834:10;830:2;823:22;869:2;861:6;854:18;915:3;908:4;903:2;895:6;891:15;887:26;884:35;881:55;;;932:1;929;922:12;881:55;996:2;989:4;981:6;977:17;970:4;962:6;958:17;945:54;1043:1;1036:4;1031:2;1023:6;1019:15;1015:26;1008:37;1063:6;1054:15;;;;;;357:718;;;;:::o;1080:186::-;1139:6;1192:2;1180:9;1171:7;1167:23;1163:32;1160:52;;;1208:1;1205;1198:12;1160:52;1231:29;1250:9;1231:29;:::i;1271:260::-;1339:6;1347;1400:2;1388:9;1379:7;1375:23;1371:32;1368:52;;;1416:1;1413;1406:12;1368:52;1439:29;1458:9;1439:29;:::i;:::-;1429:39;;1487:38;1521:2;1510:9;1506:18;1487:38;:::i;:::-;1477:48;;1271:260;;;;;:::o;1536:328::-;1613:6;1621;1629;1682:2;1670:9;1661:7;1657:23;1653:32;1650:52;;;1698:1;1695;1688:12;1650:52;1721:29;1740:9;1721:29;:::i;:::-;1711:39;;1769:38;1803:2;1792:9;1788:18;1769:38;:::i;:::-;1759:48;;1854:2;1843:9;1839:18;1826:32;1816:42;;1536:328;;;;;:::o;1869:537::-;1964:6;1972;1980;1988;2041:3;2029:9;2020:7;2016:23;2012:33;2009:53;;;2058:1;2055;2048:12;2009:53;2081:29;2100:9;2081:29;:::i;:::-;2071:39;;2129:38;2163:2;2152:9;2148:18;2129:38;:::i;:::-;2119:48;;2214:2;2203:9;2199:18;2186:32;2176:42;;2269:2;2258:9;2254:18;2241:32;2296:18;2288:6;2285:30;2282:50;;;2328:1;2325;2318:12;2282:50;2351:49;2392:7;2383:6;2372:9;2368:22;2351:49;:::i;:::-;2341:59;;;1869:537;;;;;;;:::o;2411:254::-;2476:6;2484;2537:2;2525:9;2516:7;2512:23;2508:32;2505:52;;;2553:1;2550;2543:12;2505:52;2576:29;2595:9;2576:29;:::i;:::-;2566:39;;2624:35;2655:2;2644:9;2640:18;2624:35;:::i;2670:254::-;2738:6;2746;2799:2;2787:9;2778:7;2774:23;2770:32;2767:52;;;2815:1;2812;2805:12;2767:52;2838:29;2857:9;2838:29;:::i;:::-;2828:39;2914:2;2899:18;;;;2886:32;;-1:-1:-1;;;2670:254:1:o;2929:180::-;2985:6;3038:2;3026:9;3017:7;3013:23;3009:32;3006:52;;;3054:1;3051;3044:12;3006:52;3077:26;3093:9;3077:26;:::i;3114:245::-;3172:6;3225:2;3213:9;3204:7;3200:23;3196:32;3193:52;;;3241:1;3238;3231:12;3193:52;3280:9;3267:23;3299:30;3323:5;3299:30;:::i;3364:249::-;3433:6;3486:2;3474:9;3465:7;3461:23;3457:32;3454:52;;;3502:1;3499;3492:12;3454:52;3534:9;3528:16;3553:30;3577:5;3553:30;:::i;3618:321::-;3687:6;3740:2;3728:9;3719:7;3715:23;3711:32;3708:52;;;3756:1;3753;3746:12;3708:52;3796:9;3783:23;3829:18;3821:6;3818:30;3815:50;;;3861:1;3858;3851:12;3815:50;3884:49;3925:7;3916:6;3905:9;3901:22;3884:49;:::i;3944:540::-;4031:6;4039;4092:2;4080:9;4071:7;4067:23;4063:32;4060:52;;;4108:1;4105;4098:12;4060:52;4148:9;4135:23;4177:18;4218:2;4210:6;4207:14;4204:34;;;4234:1;4231;4224:12;4204:34;4257:49;4298:7;4289:6;4278:9;4274:22;4257:49;:::i;:::-;4247:59;;4359:2;4348:9;4344:18;4331:32;4315:48;;4388:2;4378:8;4375:16;4372:36;;;4404:1;4401;4394:12;4372:36;;4427:51;4470:7;4459:8;4448:9;4444:24;4427:51;:::i;:::-;4417:61;;;3944:540;;;;;:::o;4489:608::-;4585:6;4593;4601;4654:2;4642:9;4633:7;4629:23;4625:32;4622:52;;;4670:1;4667;4660:12;4622:52;4710:9;4697:23;4739:18;4780:2;4772:6;4769:14;4766:34;;;4796:1;4793;4786:12;4766:34;4819:49;4860:7;4851:6;4840:9;4836:22;4819:49;:::i;:::-;4809:59;;4921:2;4910:9;4906:18;4893:32;4877:48;;4950:2;4940:8;4937:16;4934:36;;;4966:1;4963;4956:12;4934:36;;4989:51;5032:7;5021:8;5010:9;5006:24;4989:51;:::i;:::-;4979:61;;;5087:2;5076:9;5072:18;5059:32;5049:42;;4489:608;;;;;:::o;5102:180::-;5161:6;5214:2;5202:9;5193:7;5189:23;5185:32;5182:52;;;5230:1;5227;5220:12;5182:52;-1:-1:-1;5253:23:1;;5102:180;-1:-1:-1;5102:180:1:o;5287:257::-;5328:3;5366:5;5360:12;5393:6;5388:3;5381:19;5409:63;5465:6;5458:4;5453:3;5449:14;5442:4;5435:5;5431:16;5409:63;:::i;:::-;5526:2;5505:15;-1:-1:-1;;5501:29:1;5492:39;;;;5533:4;5488:50;;5287:257;-1:-1:-1;;5287:257:1:o;5549:501::-;5736:3;5768:26;5764:31;5837:2;5828:6;5824:2;5820:15;5816:24;5811:3;5804:37;5892:2;5883:6;5879:2;5875:15;5871:24;5866:2;5861:3;5857:12;5850:46;;5925:6;5919:13;5941:62;5996:6;5991:2;5986:3;5982:12;5975:4;5967:6;5963:17;5941:62;:::i;:::-;6023:16;;;;6041:2;6019:25;;5549:501;-1:-1:-1;;;;5549:501:1:o;6055:586::-;6270:3;6302:26;6298:31;6371:2;6362:6;6358:2;6354:15;6350:24;6345:3;6338:37;6426:2;6417:6;6413:2;6409:15;6405:24;6400:2;6395:3;6391:12;6384:46;;6459:6;6453:13;6475:62;6530:6;6525:2;6520:3;6516:12;6509:4;6501:6;6497:17;6475:62;:::i;:::-;6596:2;6556:16;;6588:11;;;6581:27;;;;6632:2;6624:11;;6055:586;-1:-1:-1;;;;6055:586:1:o;6646:637::-;6926:3;6964:6;6958:13;6980:53;7026:6;7021:3;7014:4;7006:6;7002:17;6980:53;:::i;:::-;7096:13;;7055:16;;;;7118:57;7096:13;7055:16;7152:4;7140:17;;7118:57;:::i;:::-;-1:-1:-1;;;7197:20:1;;7226:22;;;7275:1;7264:13;;6646:637;-1:-1:-1;;;;6646:637:1:o;7496:488::-;-1:-1:-1;;;;;7765:15:1;;;7747:34;;7817:15;;7812:2;7797:18;;7790:43;7864:2;7849:18;;7842:34;;;7912:3;7907:2;7892:18;;7885:31;;;7690:4;;7933:45;;7958:19;;7950:6;7933:45;:::i;:::-;7925:53;7496:488;-1:-1:-1;;;;;;7496:488:1:o;8584:219::-;8733:2;8722:9;8715:21;8696:4;8753:44;8793:2;8782:9;8778:18;8770:6;8753:44;:::i;9161:350::-;9363:2;9345:21;;;9402:2;9382:18;;;9375:30;9441:28;9436:2;9421:18;;9414:56;9502:2;9487:18;;9161:350::o;10686:354::-;10888:2;10870:21;;;10927:2;10907:18;;;10900:30;10966:32;10961:2;10946:18;;10939:60;11031:2;11016:18;;10686:354::o;12150:356::-;12352:2;12334:21;;;12371:18;;;12364:30;12430:34;12425:2;12410:18;;12403:62;12497:2;12482:18;;12150:356::o;12862:347::-;13064:2;13046:21;;;13103:2;13083:18;;;13076:30;13142:25;13137:2;13122:18;;13115:53;13200:2;13185:18;;12862:347::o;13214:401::-;13416:2;13398:21;;;13455:2;13435:18;;;13428:30;13494:34;13489:2;13474:18;;13467:62;-1:-1:-1;;;13560:2:1;13545:18;;13538:35;13605:3;13590:19;;13214:401::o;13802:225::-;13842:3;13873:1;13869:6;13866:1;13863:13;13860:136;;;13918:10;13913:3;13909:20;13906:1;13899:31;13953:4;13950:1;13943:15;13981:4;13978:1;13971:15;13860:136;-1:-1:-1;14012:9:1;;13802:225::o;14032:258::-;14104:1;14114:113;14128:6;14125:1;14122:13;14114:113;;;14204:11;;;14198:18;14185:11;;;14178:39;14150:2;14143:10;14114:113;;;14245:6;14242:1;14239:13;14236:48;;;-1:-1:-1;;14280:1:1;14262:16;;14255:27;14032:258::o;14295:380::-;14374:1;14370:12;;;;14417;;;14438:61;;14492:4;14484:6;14480:17;14470:27;;14438:61;14545:2;14537:6;14534:14;14514:18;14511:38;14508:161;;;14591:10;14586:3;14582:20;14579:1;14572:31;14626:4;14623:1;14616:15;14654:4;14651:1;14644:15;14508:161;;14295:380;;;:::o;14680:127::-;14741:10;14736:3;14732:20;14729:1;14722:31;14772:4;14769:1;14762:15;14796:4;14793:1;14786:15;14812:127;14873:10;14868:3;14864:20;14861:1;14854:31;14904:4;14901:1;14894:15;14928:4;14925:1;14918:15;14944:131;-1:-1:-1;;;;;;15018:32:1;;15008:43;;14998:71;;15065:1;15062;15055:12

Swarm Source

ipfs://d80baf902837fc3226913632746ab84858f4789f565fc16ff81e2454f4f23aae
Loading...
Loading
Loading...
Loading
[ Download: CSV Export  ]
[ Download: CSV Export  ]

A token is a representation of an on-chain or off-chain asset. The token page shows information such as price, total supply, holders, transfers and social links. Learn more about this page in our Knowledge Base.