ETH Price: $2,683.28 (-2.51%)

Token

My GM Friends (MGMF)
 

Overview

Max Total Supply

44 MGMF

Holders

24

Market

Volume (24H)

N/A

Min Price (24H)

N/A

Max Price (24H)

N/A
Balance
1 MGMF
0x284db6c37ff807a3015bc7177c284dab8f30fd0e
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:
MyGMFriends

Compiler Version
v0.8.7+commit.e28d00a7

Optimization Enabled:
Yes with 200 runs

Other Settings:
default evmVersion, MIT license

Contract Source Code (Solidity Multiple files format)

File 12 of 14: MyGMFriends.sol
// SPDX-License-Identifier: MIT

pragma solidity ^0.8.0;

import "./Ownable.sol";
import "./MerkleProof.sol";
import "./ERC721A.sol";

contract MyGMFriends is ERC721A, Ownable {
    uint256 public whitelistMintPrice = 0.033 ether;
    uint256 public mintPrice = 0.0433 ether;
    uint256 public maxPerWallet = 10;
    uint256 public whitelistMaxPerWallet = 3;
    uint256 public maxSupply = 3333;
    uint256 public totalAirdropNum;
    uint256 public totalAirdrop = 33;
    uint256 public whitelistSaleStartTime = 1651154400;
    uint256 public publicSaleStartTime = 1651240800;
    bytes32 public root;

    constructor() ERC721A("My GM Friends", "MGMF") {}

    modifier withinMintableSupply(uint256 _quantity) {
        require(
            totalSupply() + _quantity + totalAirdropNum <= maxSupply,
            "Surpasses supply"
        );
        _;
    }

    modifier withinMaxPerWallet(uint256 _quantity, uint256 _limits) {
        require(
            _quantity > 0 &&  _quantity <= _limits,
            "Minting above allocation"
        );
        _;
    }

    modifier publicSaleActive() {
        require(
            publicSaleStartTime <= block.timestamp,
            "Public sale not started."
        );
        _;
    }

    modifier whitelistSaleActive() {
        require(
            whitelistSaleStartTime <= block.timestamp && block.timestamp < publicSaleStartTime,
            "Whitelist sale not started."
        );
        _;
    }

    function setPublicSaleTime(uint256 _time) external onlyOwner {
        publicSaleStartTime = _time;
    }

    function setWhitelistSaleTime(uint256 _time) external onlyOwner {
        whitelistSaleStartTime = _time;
    }

    /**
     * @dev Public minting functionality
     */
    function mintPublic(uint256 _quantity)
        external
        payable
        publicSaleActive
        withinMintableSupply(_quantity)
        withinMaxPerWallet(_quantity, maxPerWallet)
    {
        require(msg.value >= mintPrice * _quantity, "Insufficent funds.");
        
        _safeMint(msg.sender, _quantity);
    }

    modifier hasValidMerkleProof(bytes32[] calldata merkleProof) {
        require(
            MerkleProof.verify(
                merkleProof,
                root,
                keccak256(abi.encodePacked(msg.sender))
            ),
            "Address not whitelisted."
        );
        _;
    }

    function checkWhitelist(bytes32[] calldata merkleProof)
        external
        view
        returns (bool)
    {
        return
            MerkleProof.verify(
                merkleProof,
                root,
                keccak256(abi.encodePacked(msg.sender))
            );
    }

    function mintWhitelist(uint256 _quantity, bytes32[] calldata merkleProof)
        external
        payable
        whitelistSaleActive
        hasValidMerkleProof(merkleProof)
        withinMintableSupply(_quantity)
        withinMaxPerWallet(_quantity, whitelistMaxPerWallet)
    {
        require(
            msg.value >= whitelistMintPrice * _quantity,
            "Insufficent funds."
        );
        
        _safeMint(msg.sender, _quantity);
    }

    function airdrop(address[] memory _recipients, uint8[] memory _quantity)
        external
        onlyOwner
    {
        uint256 _airdropNum;

        for (uint256 i = 0; i < _recipients.length; i++) {
            _airdropNum += _quantity[i];
        }
        require(
            totalAirdropNum + _airdropNum <= totalAirdrop,
            "We've reached the maximum of airdrop limits."
        );

        require(totalSupply() + _airdropNum <= maxSupply, "Surpasses supply.");

        for (uint256 i = 0; i < _recipients.length; i++) {
            _safeMint(_recipients[i], _quantity[i]);
        }
        totalAirdropNum += _airdropNum;
    }

    /**
     * @dev Allows owner to adjust the mint price (in wei)
     */
    function setMerkleRoot(bytes32 _root) external onlyOwner {
        root = _root;
    }

    /**
     * @dev Allows owner to adjust the mint price (in wei)
     */
    function setMintPrice(uint256 _price) external onlyOwner {
        mintPrice = _price;
    }

    
    /**
     * @dev Allows owner to adjust the mint price (in wei)
     */
    function setWhitelistMintPrice(uint256 _price) external onlyOwner {
        whitelistMintPrice = _price;
    }


    /**
     * @dev Base URI for the NFT
     */
    string private baseURI;

    function setBaseURI(string memory baseURI_) external onlyOwner {
        baseURI = baseURI_;
    }

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

    function withdrawMoney() external onlyOwner {
        (bool success, ) = msg.sender.call{value: address(this).balance}("");
        require(success, "Transfer failed.");
    }
}

File 1 of 14: Address.sol
// SPDX-License-Identifier: MIT

pragma solidity ^0.8.0;

/**
 * @dev Collection of functions related to the address type
 */
library Address {
    /**
     * @dev Returns true if `account` is a contract.
     *
     * [IMPORTANT]
     * ====
     * It is unsafe to assume that an address for which this function returns
     * false is an externally-owned account (EOA) and not a contract.
     *
     * Among others, `isContract` will return false for the following
     * types of addresses:
     *
     *  - an externally-owned account
     *  - a contract in construction
     *  - an address where a contract will be created
     *  - an address where a contract lived, but was destroyed
     * ====
     */
    function isContract(address account) internal view returns (bool) {
        // This method relies on extcodesize, which returns 0 for contracts in
        // construction, since the code is only stored at the end of the
        // constructor execution.

        uint256 size;
        assembly {
            size := extcodesize(account)
        }
        return size > 0;
    }

    /**
     * @dev Replacement for Solidity's `transfer`: sends `amount` wei to
     * `recipient`, forwarding all available gas and reverting on errors.
     *
     * https://eips.ethereum.org/EIPS/eip-1884[EIP1884] increases the gas cost
     * of certain opcodes, possibly making contracts go over the 2300 gas limit
     * imposed by `transfer`, making them unable to receive funds via
     * `transfer`. {sendValue} removes this limitation.
     *
     * https://diligence.consensys.net/posts/2019/09/stop-using-soliditys-transfer-now/[Learn more].
     *
     * IMPORTANT: because control is transferred to `recipient`, care must be
     * taken to not create reentrancy vulnerabilities. Consider using
     * {ReentrancyGuard} or the
     * https://solidity.readthedocs.io/en/v0.5.11/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern].
     */
    function sendValue(address payable recipient, uint256 amount) internal {
        require(address(this).balance >= amount, "Address: insufficient balance");

        (bool success, ) = recipient.call{value: amount}("");
        require(success, "Address: unable to send value, recipient may have reverted");
    }

    /**
     * @dev Performs a Solidity function call using a low level `call`. A
     * plain `call` is an unsafe replacement for a function call: use this
     * function instead.
     *
     * If `target` reverts with a revert reason, it is bubbled up by this
     * function (like regular Solidity function calls).
     *
     * Returns the raw returned data. To convert to the expected return value,
     * use https://solidity.readthedocs.io/en/latest/units-and-global-variables.html?highlight=abi.decode#abi-encoding-and-decoding-functions[`abi.decode`].
     *
     * Requirements:
     *
     * - `target` must be a contract.
     * - calling `target` with `data` must not revert.
     *
     * _Available since v3.1._
     */
    function functionCall(address target, bytes memory data) internal returns (bytes memory) {
        return functionCall(target, data, "Address: low-level call failed");
    }

    /**
     * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], but with
     * `errorMessage` as a fallback revert reason when `target` reverts.
     *
     * _Available since v3.1._
     */
    function functionCall(
        address target,
        bytes memory data,
        string memory errorMessage
    ) internal returns (bytes memory) {
        return functionCallWithValue(target, data, 0, errorMessage);
    }

    /**
     * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
     * but also transferring `value` wei to `target`.
     *
     * Requirements:
     *
     * - the calling contract must have an ETH balance of at least `value`.
     * - the called Solidity function must be `payable`.
     *
     * _Available since v3.1._
     */
    function functionCallWithValue(
        address target,
        bytes memory data,
        uint256 value
    ) internal returns (bytes memory) {
        return functionCallWithValue(target, data, value, "Address: low-level call with value failed");
    }

    /**
     * @dev Same as {xref-Address-functionCallWithValue-address-bytes-uint256-}[`functionCallWithValue`], but
     * with `errorMessage` as a fallback revert reason when `target` reverts.
     *
     * _Available since v3.1._
     */
    function functionCallWithValue(
        address target,
        bytes memory data,
        uint256 value,
        string memory errorMessage
    ) internal returns (bytes memory) {
        require(address(this).balance >= value, "Address: insufficient balance for call");
        require(isContract(target), "Address: call to non-contract");

        (bool success, bytes memory returndata) = target.call{value: value}(data);
        return _verifyCallResult(success, returndata, errorMessage);
    }

    /**
     * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
     * but performing a static call.
     *
     * _Available since v3.3._
     */
    function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) {
        return functionStaticCall(target, data, "Address: low-level static call failed");
    }

    /**
     * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],
     * but performing a static call.
     *
     * _Available since v3.3._
     */
    function functionStaticCall(
        address target,
        bytes memory data,
        string memory errorMessage
    ) internal view returns (bytes memory) {
        require(isContract(target), "Address: static call to non-contract");

        (bool success, bytes memory returndata) = target.staticcall(data);
        return _verifyCallResult(success, returndata, errorMessage);
    }

    /**
     * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
     * but performing a delegate call.
     *
     * _Available since v3.4._
     */
    function functionDelegateCall(address target, bytes memory data) internal returns (bytes memory) {
        return functionDelegateCall(target, data, "Address: low-level delegate call failed");
    }

    /**
     * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],
     * but performing a delegate call.
     *
     * _Available since v3.4._
     */
    function functionDelegateCall(
        address target,
        bytes memory data,
        string memory errorMessage
    ) internal returns (bytes memory) {
        require(isContract(target), "Address: delegate call to non-contract");

        (bool success, bytes memory returndata) = target.delegatecall(data);
        return _verifyCallResult(success, returndata, errorMessage);
    }

    function _verifyCallResult(
        bool success,
        bytes memory returndata,
        string memory errorMessage
    ) private pure returns (bytes memory) {
        if (success) {
            return returndata;
        } else {
            // Look for revert reason and bubble it up if present
            if (returndata.length > 0) {
                // The easiest way to bubble the revert reason is using memory via assembly

                assembly {
                    let returndata_size := mload(returndata)
                    revert(add(32, returndata), returndata_size)
                }
            } else {
                revert(errorMessage);
            }
        }
    }
}

File 2 of 14: Context.sol
// SPDX-License-Identifier: MIT

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 3 of 14: ERC165.sol
// SPDX-License-Identifier: MIT

pragma solidity ^0.8.0;

import "./IERC165.sol";

/**
 * @dev Implementation of the {IERC165} interface.
 *
 * Contracts that want to implement ERC165 should inherit from this contract and override {supportsInterface} to check
 * for the additional interface id that will be supported. For example:
 *
 * ```solidity
 * function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) {
 *     return interfaceId == type(MyInterface).interfaceId || super.supportsInterface(interfaceId);
 * }
 * ```
 *
 * Alternatively, {ERC165Storage} provides an easier to use but more expensive implementation.
 */
abstract contract ERC165 is IERC165 {
    /**
     * @dev See {IERC165-supportsInterface}.
     */
    function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) {
        return interfaceId == type(IERC165).interfaceId;
    }
}

File 4 of 14: ERC721.sol
// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.5.0) (token/ERC721/ERC721.sol)

pragma solidity ^0.8.0;

import "./IERC721.sol";
import "./IERC721Receiver.sol";
import "./IERC721Metadata.sol";
import "./Address.sol";
import "./Context.sol";
import "./Strings.sol";
import "./ERC165.sol";

/**
 * @dev Implementation of https://eips.ethereum.org/EIPS/eip-721[ERC721] Non-Fungible Token Standard, including
 * the Metadata extension, but not including the Enumerable extension, which is available separately as
 * {ERC721Enumerable}.
 */
contract ERC721 is Context, ERC165, IERC721, IERC721Metadata {
    using Address for address;
    using Strings for uint256;

    // Token name
    string private _name;

    // Token symbol
    string private _symbol;

    // Mapping from token ID to owner address
    mapping(uint256 => address) private _owners;

    // Mapping owner address to token count
    mapping(address => uint256) private _balances;

    // 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;

    /**
     * @dev Initializes the contract by setting a `name` and a `symbol` to the token collection.
     */
    constructor(string memory name_, string memory symbol_) {
        _name = name_;
        _symbol = symbol_;
    }

    /**
     * @dev See {IERC165-supportsInterface}.
     */
    function supportsInterface(bytes4 interfaceId)
        public
        view
        virtual
        override(ERC165, IERC165)
        returns (bool)
    {
        return
            interfaceId == type(IERC721).interfaceId ||
            interfaceId == type(IERC721Metadata).interfaceId ||
            super.supportsInterface(interfaceId);
    }

    /**
     * @dev See {IERC721-balanceOf}.
     */
    function balanceOf(address owner)
        public
        view
        virtual
        override
        returns (uint256)
    {
        require(
            owner != address(0),
            "ERC721: address zero is not a valid owner"
        );
        return _balances[owner];
    }

    /**
     * @dev See {IERC721-ownerOf}.
     */
    function ownerOf(uint256 tokenId)
        public
        view
        virtual
        override
        returns (address)
    {
        address owner = _owners[tokenId];
        require(
            owner != address(0),
            "ERC721: owner query for nonexistent token"
        );
        return owner;
    }

    /**
     * @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)
    {
        require(
            _exists(tokenId),
            "ERC721Metadata: URI query for nonexistent token"
        );

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

    /**
     * @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, can be overridden in child contracts.
     */
    function _baseURI() internal view virtual returns (string memory) {
        return "";
    }

    /**
     * @dev See {IERC721-approve}.
     */
    function approve(address to, uint256 tokenId) public virtual override {
        address owner = ERC721.ownerOf(tokenId);
        require(to != owner, "ERC721: approval to current owner");

        require(
            _msgSender() == owner || isApprovedForAll(owner, _msgSender()),
            "ERC721: approve caller is not owner nor approved for all"
        );

        _approve(to, tokenId);
    }

    /**
     * @dev See {IERC721-getApproved}.
     */
    function getApproved(uint256 tokenId)
        public
        view
        virtual
        override
        returns (address)
    {
        require(
            _exists(tokenId),
            "ERC721: approved query for nonexistent token"
        );

        return _tokenApprovals[tokenId];
    }

    /**
     * @dev See {IERC721-setApprovalForAll}.
     */
    function setApprovalForAll(address operator, bool approved)
        public
        virtual
        override
    {
        _setApprovalForAll(_msgSender(), 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-transferFrom}.
     */
    function transferFrom(
        address from,
        address to,
        uint256 tokenId
    ) public virtual override {
        //solhint-disable-next-line max-line-length
        require(
            _isApprovedOrOwner(_msgSender(), tokenId),
            "ERC721: transfer caller is not owner nor approved"
        );

        _transfer(from, to, tokenId);
    }

    /**
     * @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 {
        require(
            _isApprovedOrOwner(_msgSender(), tokenId),
            "ERC721: transfer caller is not owner nor approved"
        );
        _safeTransfer(from, to, tokenId, _data);
    }

    /**
     * @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.
     *
     * `_data` is additional data, it has no specified format and it is sent in call to `to`.
     *
     * This internal function is equivalent to {safeTransferFrom}, and can be used to e.g.
     * implement alternative mechanisms to perform token transfer, such as signature-based.
     *
     * Requirements:
     *
     * - `from` cannot be the zero address.
     * - `to` cannot be the zero address.
     * - `tokenId` token must exist and be owned by `from`.
     * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer.
     *
     * Emits a {Transfer} event.
     */
    function _safeTransfer(
        address from,
        address to,
        uint256 tokenId,
        bytes memory _data
    ) internal virtual {
        _transfer(from, to, tokenId);
        require(
            _checkOnERC721Received(from, to, tokenId, _data),
            "ERC721: transfer to non ERC721Receiver implementer"
        );
    }

    /**
     * @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`),
     * and stop existing when they are burned (`_burn`).
     */
    function _exists(uint256 tokenId) internal view virtual returns (bool) {
        return _owners[tokenId] != address(0);
    }

    /**
     * @dev Returns whether `spender` is allowed to manage `tokenId`.
     *
     * Requirements:
     *
     * - `tokenId` must exist.
     */
    function _isApprovedOrOwner(address spender, uint256 tokenId)
        internal
        view
        virtual
        returns (bool)
    {
        require(
            _exists(tokenId),
            "ERC721: operator query for nonexistent token"
        );
        address owner = ERC721.ownerOf(tokenId);
        return (spender == owner ||
            isApprovedForAll(owner, spender) ||
            getApproved(tokenId) == spender);
    }

    /**
     * @dev Safely mints `tokenId` and transfers it to `to`.
     *
     * Requirements:
     *
     * - `tokenId` must not exist.
     * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer.
     *
     * Emits a {Transfer} event.
     */
    function _safeMint(address to, uint256 tokenId) internal virtual {
        _safeMint(to, tokenId, "");
    }

    /**
     * @dev Same as {xref-ERC721-_safeMint-address-uint256-}[`_safeMint`], with an additional `data` parameter which is
     * forwarded in {IERC721Receiver-onERC721Received} to contract recipients.
     */
    function _safeMint(
        address to,
        uint256 tokenId,
        bytes memory _data
    ) internal virtual {
        _mint(to, tokenId);
        require(
            _checkOnERC721Received(address(0), to, tokenId, _data),
            "ERC721: transfer to non ERC721Receiver implementer"
        );
    }

    /**
     * @dev Mints `tokenId` and transfers it to `to`.
     *
     * WARNING: Usage of this method is discouraged, use {_safeMint} whenever possible
     *
     * Requirements:
     *
     * - `tokenId` must not exist.
     * - `to` cannot be the zero address.
     *
     * Emits a {Transfer} event.
     */
    function _mint(address to, uint256 tokenId) internal virtual {
        require(to != address(0), "ERC721: mint to the zero address");
        require(!_exists(tokenId), "ERC721: token already minted");

        _beforeTokenTransfer(address(0), to, tokenId);

        _balances[to] += 1;
        _owners[tokenId] = to;

        emit Transfer(address(0), to, tokenId);

        _afterTokenTransfer(address(0), to, tokenId);
    }

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

        _beforeTokenTransfer(owner, address(0), tokenId);

        // Clear approvals
        _approve(address(0), tokenId);

        _balances[owner] -= 1;
        delete _owners[tokenId];

        emit Transfer(owner, address(0), tokenId);

        _afterTokenTransfer(owner, address(0), tokenId);
    }

    /**
     * @dev Transfers `tokenId` from `from` to `to`.
     *  As opposed to {transferFrom}, this imposes no restrictions on msg.sender.
     *
     * Requirements:
     *
     * - `to` cannot be the zero address.
     * - `tokenId` token must be owned by `from`.
     *
     * Emits a {Transfer} event.
     */
    function _transfer(
        address from,
        address to,
        uint256 tokenId
    ) internal virtual {
        require(
            ERC721.ownerOf(tokenId) == from,
            "ERC721: transfer from incorrect owner"
        );
        require(to != address(0), "ERC721: transfer to the zero address");

        _beforeTokenTransfer(from, to, tokenId);

        // Clear approvals from the previous owner
        _approve(address(0), tokenId);

        _balances[from] -= 1;
        _balances[to] += 1;
        _owners[tokenId] = to;

        emit Transfer(from, to, tokenId);

        _afterTokenTransfer(from, to, tokenId);
    }

    /**
     * @dev Approve `to` to operate on `tokenId`
     *
     * Emits a {Approval} event.
     */
    function _approve(address to, uint256 tokenId) internal virtual {
        _tokenApprovals[tokenId] = to;
        emit Approval(ERC721.ownerOf(tokenId), to, tokenId);
    }

    /**
     * @dev Approve `operator` to operate on all of `owner` tokens
     *
     * Emits a {ApprovalForAll} event.
     */
    function _setApprovalForAll(
        address owner,
        address operator,
        bool approved
    ) internal virtual {
        require(owner != operator, "ERC721: approve to caller");
        _operatorApprovals[owner][operator] = approved;
        emit ApprovalForAll(owner, operator, approved);
    }

    /**
     * @dev Internal function to invoke {IERC721Receiver-onERC721Received} on a target address.
     * The call is not executed if the target address is not a 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 _checkOnERC721Received(
        address from,
        address to,
        uint256 tokenId,
        bytes memory _data
    ) private returns (bool) {
        if (to.isContract()) {
            try
                IERC721Receiver(to).onERC721Received(
                    _msgSender(),
                    from,
                    tokenId,
                    _data
                )
            returns (bytes4 retval) {
                return retval == IERC721Receiver.onERC721Received.selector;
            } catch (bytes memory reason) {
                if (reason.length == 0) {
                    revert(
                        "ERC721: transfer to non ERC721Receiver implementer"
                    );
                } else {
                    assembly {
                        revert(add(32, reason), mload(reason))
                    }
                }
            }
        } else {
            return true;
        }
    }

    /**
     * @dev Hook that is called before any token transfer. This includes minting
     * and burning.
     *
     * 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, ``from``'s `tokenId` will be burned.
     * - `from` and `to` are never both zero.
     *
     * To learn more about hooks, head to xref:ROOT:extending-contracts.adoc#using-hooks[Using Hooks].
     */
    function _beforeTokenTransfer(
        address from,
        address to,
        uint256 tokenId
    ) internal virtual {}

    /**
     * @dev Hook that is called after any transfer of tokens. This includes
     * minting and burning.
     *
     * Calling conditions:
     *
     * - when `from` and `to` are both non-zero.
     * - `from` and `to` are never both zero.
     *
     * To learn more about hooks, head to xref:ROOT:extending-contracts.adoc#using-hooks[Using Hooks].
     */
    function _afterTokenTransfer(
        address from,
        address to,
        uint256 tokenId
    ) internal virtual {}
}

File 5 of 14: ERC721A.sol
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;

import "./IERC721.sol";
import "./IERC721Enumerable.sol";
import "./IERC721Metadata.sol";
import "./Strings.sol";
import "./Context.sol";
import "./ERC165.sol";
import "./Address.sol";
import "./IERC721Receiver.sol";

error ApprovalCallerNotOwnerNorApproved();
error ApprovalQueryForNonexistentToken();
error ApproveToCaller();
error ApprovalToCurrentOwner();
error BalanceQueryForZeroAddress();
error MintedQueryForZeroAddress();
error BurnedQueryForZeroAddress();
error MintToZeroAddress();
error MintZeroQuantity();
error OwnerIndexOutOfBounds();
error OwnerQueryForNonexistentToken();
error TokenIndexOutOfBounds();
error TransferCallerNotOwnerNorApproved();
error TransferFromIncorrectOwner();
error TransferToNonERC721ReceiverImplementer();
error TransferToZeroAddress();
error URIQueryForNonexistentToken();

/**
 * @dev Implementation of https://eips.ethereum.org/EIPS/eip-721[ERC721] Non-Fungible Token Standard, including
 * the Metadata and Enumerable extension. Built to optimize for lower gas during batch mints.
 *
 * Assumes serials are sequentially minted starting at 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**128 - 1 (max value of uint128).
 */
contract ERC721A is Context, ERC165, IERC721, IERC721Metadata, IERC721Enumerable {
    using Address for address;
    using Strings for uint256;

    // Compiler will pack this into a single 256bit word.
    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;
    }

    // Compiler will pack this into a single 256bit word.
    struct AddressData {
        // Realistically, 2**64-1 is more than enough.
        uint64 balance;
        // Keeps track of mint count with minimal overhead for tokenomics.
        uint64 numberMinted;
        // Keeps track of burn count with minimal overhead for tokenomics.
        uint64 numberBurned;
    }

    // Compiler will pack the following 
    // _currentIndex and _burnCounter into a single 256bit word.
    
    // The tokenId of the next token to be minted.
    uint128 internal _currentIndex;

    // The number of tokens burned.
    uint128 internal _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 ownershipOf implementation for details.
    mapping(uint256 => TokenOwnership) internal _ownerships;

    // Mapping owner address to address data
    mapping(address => AddressData) private _addressData;

    // 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_;
    }

    /**
     * @dev See {IERC721Enumerable-totalSupply}.
     */
    function totalSupply() public view override returns (uint256) {
        // Counter underflow is impossible as _burnCounter cannot be incremented
        // more than _currentIndex times
        unchecked {
            return _currentIndex - _burnCounter;    
        }
    }

    /**
     * @dev See {IERC721Enumerable-tokenByIndex}.
     * This read function is O(totalSupply). If calling from a separate contract, be sure to test gas first.
     * It may also degrade with extremely large collection sizes (e.g >> 10000), test for your use case.
     */
    function tokenByIndex(uint256 index) public view override returns (uint256) {
        uint256 numMintedSoFar = _currentIndex;
        uint256 tokenIdsIdx;

        // Counter overflow is impossible as the loop breaks when
        // uint256 i is equal to another uint256 numMintedSoFar.
        unchecked {
            for (uint256 i; i < numMintedSoFar; i++) {
                TokenOwnership memory ownership = _ownerships[i];
                if (!ownership.burned) {
                    if (tokenIdsIdx == index) {
                        return i;
                    }
                    tokenIdsIdx++;
                }
            }
        }
        revert TokenIndexOutOfBounds();
    }

    /**
     * @dev See {IERC721Enumerable-tokenOfOwnerByIndex}.
     * This read function is O(totalSupply). If calling from a separate contract, be sure to test gas first.
     * It may also degrade with extremely large collection sizes (e.g >> 10000), test for your use case.
     */
    function tokenOfOwnerByIndex(address owner, uint256 index) public view override returns (uint256) {
        if (index >= balanceOf(owner)) revert OwnerIndexOutOfBounds();
        uint256 numMintedSoFar = _currentIndex;
        uint256 tokenIdsIdx;
        address currOwnershipAddr;

        // Counter overflow is impossible as the loop breaks when
        // uint256 i is equal to another uint256 numMintedSoFar.
        unchecked {
            for (uint256 i; i < numMintedSoFar; i++) {
                TokenOwnership memory ownership = _ownerships[i];
                if (ownership.burned) {
                    continue;
                }
                if (ownership.addr != address(0)) {
                    currOwnershipAddr = ownership.addr;
                }
                if (currOwnershipAddr == owner) {
                    if (tokenIdsIdx == index) {
                        return i;
                    }
                    tokenIdsIdx++;
                }
            }
        }

        // Execution should never reach this point.
        revert();
    }

    /**
     * @dev See {IERC165-supportsInterface}.
     */
    function supportsInterface(bytes4 interfaceId) public view virtual override(ERC165, IERC165) returns (bool) {
        return
            interfaceId == type(IERC721).interfaceId ||
            interfaceId == type(IERC721Metadata).interfaceId ||
            interfaceId == type(IERC721Enumerable).interfaceId ||
            super.supportsInterface(interfaceId);
    }

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

    function _numberMinted(address owner) internal view returns (uint256) {
        if (owner == address(0)) revert MintedQueryForZeroAddress();
        return uint256(_addressData[owner].numberMinted);
    }

    function _numberBurned(address owner) internal view returns (uint256) {
        if (owner == address(0)) revert BurnedQueryForZeroAddress();
        return uint256(_addressData[owner].numberBurned);
    }

    /**
     * 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) {
        uint256 curr = tokenId;

        unchecked {
            if (curr < _currentIndex) {
                TokenOwnership memory ownership = _ownerships[curr];
                if (!ownership.burned) {
                    if (ownership.addr != address(0)) {
                        return ownership;
                    }
                    // 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.
                    while (true) {
                        curr--;
                        ownership = _ownerships[curr];
                        if (ownership.addr != address(0)) {
                            return ownership;
                        }
                    }
                }
            }
        }
        revert OwnerQueryForNonexistentToken();
    }

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

    /**
     * @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, tokenId.toString())) : '';
    }

    /**
     * @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, can be overriden in child contracts.
     */
    function _baseURI() internal view virtual returns (string memory) {
        return '';
    }

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

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

        _approve(to, tokenId, owner);
    }

    /**
     * @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 override {
        if (operator == _msgSender()) revert ApproveToCaller();

        _operatorApprovals[_msgSender()][operator] = approved;
        emit ApprovalForAll(_msgSender(), 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-transferFrom}.
     */
    function transferFrom(
        address from,
        address to,
        uint256 tokenId
    ) public virtual override {
        _transfer(from, to, tokenId);
    }

    /**
     * @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 {
        _transfer(from, to, tokenId);
        if (!_checkOnERC721Received(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 tokenId < _currentIndex && !_ownerships[tokenId].burned;
    }

    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.
     *
     * Emits a {Transfer} event.
     */
    function _safeMint(
        address to,
        uint256 quantity,
        bytes memory _data
    ) internal {
        _mint(to, quantity, _data, true);
    }

    /**
     * @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.
     */
    function _mint(
        address to,
        uint256 quantity,
        bytes memory _data,
        bool safe
    ) 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 or numberMinted overflow if current value of either + quantity > 3.4e38 (2**128) - 1
        // updatedIndex overflows if _currentIndex + quantity > 3.4e38 (2**128) - 1
        unchecked {
            _addressData[to].balance += uint64(quantity);
            _addressData[to].numberMinted += uint64(quantity);

            _ownerships[startTokenId].addr = to;
            _ownerships[startTokenId].startTimestamp = uint64(block.timestamp);

            uint256 updatedIndex = startTokenId;

            for (uint256 i; i < quantity; i++) {
                emit Transfer(address(0), to, updatedIndex);
                if (safe && !_checkOnERC721Received(address(0), to, updatedIndex, _data)) {
                    revert TransferToNonERC721ReceiverImplementer();
                }
                updatedIndex++;
            }

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

    /**
     * @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 _transfer(
        address from,
        address to,
        uint256 tokenId
    ) private {
        TokenOwnership memory prevOwnership = ownershipOf(tokenId);

        bool isApprovedOrOwner = (_msgSender() == prevOwnership.addr ||
            isApprovedForAll(prevOwnership.addr, _msgSender()) ||
            getApproved(tokenId) == _msgSender());

        if (!isApprovedOrOwner) revert TransferCallerNotOwnerNorApproved();
        if (prevOwnership.addr != from) revert TransferFromIncorrectOwner();
        if (to == address(0)) revert TransferToZeroAddress();

        _beforeTokenTransfers(from, to, tokenId, 1);

        // Clear approvals from the previous owner
        _approve(address(0), tokenId, prevOwnership.addr);

        // 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**128.
        unchecked {
            _addressData[from].balance -= 1;
            _addressData[to].balance += 1;

            _ownerships[tokenId].addr = to;
            _ownerships[tokenId].startTimestamp = uint64(block.timestamp);

            // If the ownership slot of tokenId+1 is not explicitly set, that means the transfer initiator owns it.
            // Set the slot of tokenId+1 explicitly in storage to maintain correctness for ownerOf(tokenId+1) calls.
            uint256 nextTokenId = tokenId + 1;
            if (_ownerships[nextTokenId].addr == address(0)) {
                // This will suffice for checking _exists(nextTokenId),
                // as a burned slot cannot contain the zero address.
                if (nextTokenId < _currentIndex) {
                    _ownerships[nextTokenId].addr = prevOwnership.addr;
                    _ownerships[nextTokenId].startTimestamp = prevOwnership.startTimestamp;
                }
            }
        }

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

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

        _beforeTokenTransfers(prevOwnership.addr, address(0), tokenId, 1);

        // Clear approvals from the previous owner
        _approve(address(0), tokenId, prevOwnership.addr);

        // 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**128.
        unchecked {
            _addressData[prevOwnership.addr].balance -= 1;
            _addressData[prevOwnership.addr].numberBurned += 1;

            // Keep track of who burned the token, and the timestamp of burning.
            _ownerships[tokenId].addr = prevOwnership.addr;
            _ownerships[tokenId].startTimestamp = uint64(block.timestamp);
            _ownerships[tokenId].burned = true;

            // If the ownership slot of tokenId+1 is not explicitly set, that means the burn initiator owns it.
            // Set the slot of tokenId+1 explicitly in storage to maintain correctness for ownerOf(tokenId+1) calls.
            uint256 nextTokenId = tokenId + 1;
            if (_ownerships[nextTokenId].addr == address(0)) {
                // This will suffice for checking _exists(nextTokenId),
                // as a burned slot cannot contain the zero address.
                if (nextTokenId < _currentIndex) {
                    _ownerships[nextTokenId].addr = prevOwnership.addr;
                    _ownerships[nextTokenId].startTimestamp = prevOwnership.startTimestamp;
                }
            }
        }

        emit Transfer(prevOwnership.addr, address(0), tokenId);
        _afterTokenTransfers(prevOwnership.addr, address(0), tokenId, 1);

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

    /**
     * @dev Approve `to` to operate on `tokenId`
     *
     * Emits a {Approval} event.
     */
    function _approve(
        address to,
        uint256 tokenId,
        address owner
    ) private {
        _tokenApprovals[tokenId] = to;
        emit Approval(owner, to, tokenId);
    }

    /**
     * @dev Internal function to invoke {IERC721Receiver-onERC721Received} on a target address.
     * The call is not executed if the target address is not a 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 _checkOnERC721Received(
        address from,
        address to,
        uint256 tokenId,
        bytes memory _data
    ) private returns (bool) {
        if (to.isContract()) {
            try IERC721Receiver(to).onERC721Received(_msgSender(), from, tokenId, _data) returns (bytes4 retval) {
                return retval == IERC721Receiver(to).onERC721Received.selector;
            } catch (bytes memory reason) {
                if (reason.length == 0) {
                    revert TransferToNonERC721ReceiverImplementer();
                } else {
                    assembly {
                        revert(add(32, reason), mload(reason))
                    }
                }
            }
        } else {
            return true;
        }
    }

    /**
     * @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 {}
}

File 6 of 14: IERC165.sol
// SPDX-License-Identifier: MIT

pragma solidity ^0.8.0;

/**
 * @dev Interface of the ERC165 standard, as defined in the
 * https://eips.ethereum.org/EIPS/eip-165[EIP].
 *
 * Implementers can declare support of contract interfaces, which can then be
 * queried by others ({ERC165Checker}).
 *
 * For an implementation, see {ERC165}.
 */
interface 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);
}

File 7 of 14: IERC721.sol
// SPDX-License-Identifier: MIT

pragma solidity ^0.8.0;

import "./IERC165.sol";

/**
 * @dev Required interface of an ERC721 compliant contract.
 */
interface IERC721 is IERC165 {
    /**
     * @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`, 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 Returns the account approved for `tokenId` token.
     *
     * Requirements:
     *
     * - `tokenId` must exist.
     */
    function getApproved(uint256 tokenId) external view returns (address operator);

    /**
     * @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 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);

    /**
     * @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;
}

File 8 of 14: IERC721Enumerable.sol
// SPDX-License-Identifier: MIT

pragma solidity ^0.8.0;

import "./IERC721.sol";

/**
 * @title ERC-721 Non-Fungible Token Standard, optional enumeration extension
 * @dev See https://eips.ethereum.org/EIPS/eip-721
 */
interface IERC721Enumerable is IERC721 {
    /**
     * @dev Returns the total amount of tokens stored by the contract.
     */
    function totalSupply() external view returns (uint256);

    /**
     * @dev Returns a token ID owned by `owner` at a given `index` of its token list.
     * Use along with {balanceOf} to enumerate all of ``owner``'s tokens.
     */
    function tokenOfOwnerByIndex(address owner, uint256 index) external view returns (uint256 tokenId);

    /**
     * @dev Returns a token ID at a given `index` of all the tokens stored by the contract.
     * Use along with {totalSupply} to enumerate all tokens.
     */
    function tokenByIndex(uint256 index) external view returns (uint256);
}

File 9 of 14: IERC721Metadata.sol
// SPDX-License-Identifier: MIT

pragma solidity ^0.8.0;

import "./IERC721.sol";

/**
 * @title ERC-721 Non-Fungible Token Standard, optional metadata extension
 * @dev See https://eips.ethereum.org/EIPS/eip-721
 */
interface IERC721Metadata is IERC721 {
    /**
     * @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);
}

File 10 of 14: IERC721Receiver.sol
// SPDX-License-Identifier: MIT

pragma solidity ^0.8.0;

/**
 * @title ERC721 token receiver interface
 * @dev Interface for any contract that wants to support safeTransfers
 * from ERC721 asset contracts.
 */
interface IERC721Receiver {
    /**
     * @dev Whenever an {IERC721} `tokenId` token is transferred to this contract via {IERC721-safeTransferFrom}
     * by `operator` from `from`, this function is called.
     *
     * It must return its Solidity selector to confirm the token transfer.
     * If any other value is returned or the interface is not implemented by the recipient, the transfer will be reverted.
     *
     * The selector can be obtained in Solidity with `IERC721.onERC721Received.selector`.
     */
    function onERC721Received(
        address operator,
        address from,
        uint256 tokenId,
        bytes calldata data
    ) external returns (bytes4);
}

File 11 of 14: MerkleProof.sol
// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts v4.4.1 (utils/cryptography/MerkleProof.sol)

pragma solidity ^0.8.0;

/**
 * @dev These functions deal with verification of Merkle Trees proofs.
 *
 * The proofs can be generated using the JavaScript library
 * https://github.com/miguelmota/merkletreejs[merkletreejs].
 * Note: the hashing algorithm should be keccak256 and pair sorting should be enabled.
 *
 * See `test/utils/cryptography/MerkleProof.test.js` for some examples.
 */
library MerkleProof {
    /**
     * @dev Returns true if a `leaf` can be proved to be a part of a Merkle tree
     * defined by `root`. For this, a `proof` must be provided, containing
     * sibling hashes on the branch from the leaf to the root of the tree. Each
     * pair of leaves and each pair of pre-images are assumed to be sorted.
     */
    function verify(
        bytes32[] memory proof,
        bytes32 root,
        bytes32 leaf
    ) internal pure returns (bool) {
        return processProof(proof, leaf) == root;
    }

    /**
     * @dev Returns the rebuilt hash obtained by traversing a Merklee tree up
     * from `leaf` using `proof`. A `proof` is valid if and only if the rebuilt
     * hash matches the root of the tree. When processing the proof, the pairs
     * of leafs & pre-images are assumed to be sorted.
     *
     * _Available since v4.4._
     */
    function processProof(bytes32[] memory proof, bytes32 leaf) internal pure returns (bytes32) {
        bytes32 computedHash = leaf;
        for (uint256 i = 0; i < proof.length; i++) {
            bytes32 proofElement = proof[i];
            if (computedHash <= proofElement) {
                // Hash(current computed hash + current element of the proof)
                computedHash = keccak256(abi.encodePacked(computedHash, proofElement));
            } else {
                // Hash(current element of the proof + current computed hash)
                computedHash = keccak256(abi.encodePacked(proofElement, computedHash));
            }
        }
        return computedHash;
    }
}

File 13 of 14: Ownable.sol
// SPDX-License-Identifier: MIT

pragma solidity ^0.8.0;

import "./Context.sol";

/**
 * @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() {
        _setOwner(_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 {
        _setOwner(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");
        _setOwner(newOwner);
    }

    function _setOwner(address newOwner) private {
        address oldOwner = _owner;
        _owner = newOwner;
        emit OwnershipTransferred(oldOwner, newOwner);
    }
}

File 14 of 14: Strings.sol
// SPDX-License-Identifier: MIT

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

Contract Security Audit

Contract ABI

[{"inputs":[],"stateMutability":"nonpayable","type":"constructor"},{"inputs":[],"name":"ApprovalCallerNotOwnerNorApproved","type":"error"},{"inputs":[],"name":"ApprovalQueryForNonexistentToken","type":"error"},{"inputs":[],"name":"ApprovalToCurrentOwner","type":"error"},{"inputs":[],"name":"ApproveToCaller","type":"error"},{"inputs":[],"name":"BalanceQueryForZeroAddress","type":"error"},{"inputs":[],"name":"MintToZeroAddress","type":"error"},{"inputs":[],"name":"MintZeroQuantity","type":"error"},{"inputs":[],"name":"OwnerIndexOutOfBounds","type":"error"},{"inputs":[],"name":"OwnerQueryForNonexistentToken","type":"error"},{"inputs":[],"name":"TokenIndexOutOfBounds","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":"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"},{"inputs":[{"internalType":"address[]","name":"_recipients","type":"address[]"},{"internalType":"uint8[]","name":"_quantity","type":"uint8[]"}],"name":"airdrop","outputs":[],"stateMutability":"nonpayable","type":"function"},{"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":"bytes32[]","name":"merkleProof","type":"bytes32[]"}],"name":"checkWhitelist","outputs":[{"internalType":"bool","name":"","type":"bool"}],"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":"maxPerWallet","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"maxSupply","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"mintPrice","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_quantity","type":"uint256"}],"name":"mintPublic","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_quantity","type":"uint256"},{"internalType":"bytes32[]","name":"merkleProof","type":"bytes32[]"}],"name":"mintWhitelist","outputs":[],"stateMutability":"payable","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":"publicSaleStartTime","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"root","outputs":[{"internalType":"bytes32","name":"","type":"bytes32"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"from","type":"address"},{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"safeTransferFrom","outputs":[],"stateMutability":"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":"string","name":"baseURI_","type":"string"}],"name":"setBaseURI","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bytes32","name":"_root","type":"bytes32"}],"name":"setMerkleRoot","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_price","type":"uint256"}],"name":"setMintPrice","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_time","type":"uint256"}],"name":"setPublicSaleTime","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_price","type":"uint256"}],"name":"setWhitelistMintPrice","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_time","type":"uint256"}],"name":"setWhitelistSaleTime","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":"index","type":"uint256"}],"name":"tokenByIndex","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"},{"internalType":"uint256","name":"index","type":"uint256"}],"name":"tokenOfOwnerByIndex","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"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":"totalAirdrop","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"totalAirdropNum","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"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":[],"name":"whitelistMaxPerWallet","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"whitelistMintPrice","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"whitelistSaleStartTime","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"withdrawMoney","outputs":[],"stateMutability":"nonpayable","type":"function"}]

608060405266753d533d9680006008556699d51edec64000600955600a80556003600b55610d05600c556021600e5563626a9de0600f5563626bef606010553480156200004b57600080fd5b50604080518082018252600d81526c4d7920474d20467269656e647360981b60208083019182528351808501909452600484526326a3a6a360e11b9084015281519192916200009d916001916200012c565b508051620000b39060029060208401906200012c565b505050620000d0620000ca620000d660201b60201c565b620000da565b6200020f565b3390565b600780546001600160a01b038381166001600160a01b0319831681179093556040519116919082907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e090600090a35050565b8280546200013a90620001d2565b90600052602060002090601f0160209004810192826200015e5760008555620001a9565b82601f106200017957805160ff1916838001178555620001a9565b82800160010185558215620001a9579182015b82811115620001a95782518255916020019190600101906200018c565b50620001b7929150620001bb565b5090565b5b80821115620001b75760008155600101620001bc565b600181811c90821680620001e757607f821691505b602082108114156200020957634e487b7160e01b600052602260045260246000fd5b50919050565b6125b3806200021f6000396000f3fe60806040526004361061023b5760003560e01c806370a082311161012e578063b88d4fde116100ab578063ebf0c7171161006f578063ebf0c7171461067d578063efd0cbf914610693578063f2fde38b146106a6578063f4a0a528146106c6578063f6000c7a146106e657600080fd5b8063b88d4fde146105be578063b8dde112146105de578063c87b56dd146105fe578063d5abeb011461061e578063e985e9c51461063457600080fd5b8063a061a03a116100f2578063a061a03a14610529578063a22cb46514610549578063a611708e14610569578063ac44600214610589578063afdea6c21461059e57600080fd5b806370a08231146104a1578063715018a6146104c15780637cb64759146104d65780638da5cb5b146104f657806395d89b411461051457600080fd5b806335c6aaf8116101bc57806355f804b31161018057806355f804b31461041f5780635ce97dbb1461043f5780636352211e146104555780636817c76c146104755780636bb7b1d91461048b57600080fd5b806335c6aaf81461039d57806342842e0e146103b3578063453c2310146103d35780634f6ccce7146103e957806350d549151461040957600080fd5b806311b7e5e71161020357806311b7e5e71461030457806318160ddd14610324578063230b43f41461034757806323b872dd1461035d5780632f745c591461037d57600080fd5b806301ffc9a714610240578063061431a81461027557806306fdde031461028a578063081812fc146102ac578063095ea7b3146102e4575b600080fd5b34801561024c57600080fd5b5061026061025b366004612217565b6106fc565b60405190151581526020015b60405180910390f35b610288610283366004612299565b610769565b005b34801561029657600080fd5b5061029f6109b7565b60405161026c919061237c565b3480156102b857600080fd5b506102cc6102c73660046121fe565b610a49565b6040516001600160a01b03909116815260200161026c565b3480156102f057600080fd5b506102886102ff3660046120cd565b610a8d565b34801561031057600080fd5b5061028861031f3660046121fe565b610b1b565b34801561033057600080fd5b50610339610b4a565b60405190815260200161026c565b34801561035357600080fd5b50610339600f5481565b34801561036957600080fd5b50610288610378366004611fda565b610b69565b34801561038957600080fd5b506103396103983660046120cd565b610b74565b3480156103a957600080fd5b5061033960085481565b3480156103bf57600080fd5b506102886103ce366004611fda565b610c70565b3480156103df57600080fd5b50610339600a5481565b3480156103f557600080fd5b506103396104043660046121fe565b610c8b565b34801561041557600080fd5b50610339600d5481565b34801561042b57600080fd5b5061028861043a366004612251565b610d35565b34801561044b57600080fd5b50610339600e5481565b34801561046157600080fd5b506102cc6104703660046121fe565b610d76565b34801561048157600080fd5b5061033960095481565b34801561049757600080fd5b5061033960105481565b3480156104ad57600080fd5b506103396104bc366004611f8c565b610d88565b3480156104cd57600080fd5b50610288610dd6565b3480156104e257600080fd5b506102886104f13660046121fe565b610e0c565b34801561050257600080fd5b506007546001600160a01b03166102cc565b34801561052057600080fd5b5061029f610e3b565b34801561053557600080fd5b506102886105443660046120f7565b610e4a565b34801561055557600080fd5b50610288610564366004612091565b611004565b34801561057557600080fd5b506102886105843660046121fe565b61109a565b34801561059557600080fd5b506102886110c9565b3480156105aa57600080fd5b506102886105b93660046121fe565b611181565b3480156105ca57600080fd5b506102886105d9366004612016565b6111b0565b3480156105ea57600080fd5b506102606105f93660046121bd565b6111ea565b34801561060a57600080fd5b5061029f6106193660046121fe565b611252565b34801561062a57600080fd5b50610339600c5481565b34801561064057600080fd5b5061026061064f366004611fa7565b6001600160a01b03918216600090815260066020908152604080832093909416825291909152205460ff1690565b34801561068957600080fd5b5061033960115481565b6102886106a13660046121fe565b6112d6565b3480156106b257600080fd5b506102886106c1366004611f8c565b611444565b3480156106d257600080fd5b506102886106e13660046121fe565b6114dc565b3480156106f257600080fd5b50610339600b5481565b60006001600160e01b031982166380ac58cd60e01b148061072d57506001600160e01b03198216635b5e139f60e01b145b8061074857506001600160e01b0319821663780e9d6360e01b145b8061076357506301ffc9a760e01b6001600160e01b03198316145b92915050565b42600f541115801561077c575060105442105b6107cd5760405162461bcd60e51b815260206004820152601b60248201527f57686974656c6973742073616c65206e6f7420737461727465642e000000000060448201526064015b60405180910390fd5b8181610845828280806020026020016040519081016040528093929190818152602001838360200280828437600092019190915250506011546040516bffffffffffffffffffffffff193360601b16602082015290925060340190505b6040516020818303038152906040528051906020012061150b565b6108915760405162461bcd60e51b815260206004820152601860248201527f41646472657373206e6f742077686974656c69737465642e000000000000000060448201526064016107c4565b84600c54600d54826108a1610b4a565b6108ab9190612417565b6108b59190612417565b11156108f65760405162461bcd60e51b815260206004820152601060248201526f53757270617373657320737570706c7960801b60448201526064016107c4565b85600b5460008211801561090a5750808211155b6109515760405162461bcd60e51b815260206004820152601860248201527726b4b73a34b7339030b137bb329030b63637b1b0ba34b7b760411b60448201526064016107c4565b8760085461095f9190612443565b3410156109a35760405162461bcd60e51b815260206004820152601260248201527124b739bab33334b1b2b73a10333ab732399760711b60448201526064016107c4565b6109ad3389611521565b5050505050505050565b6060600180546109c6906124a5565b80601f01602080910402602001604051908101604052809291908181526020018280546109f2906124a5565b8015610a3f5780601f10610a1457610100808354040283529160200191610a3f565b820191906000526020600020905b815481529060010190602001808311610a2257829003601f168201915b5050505050905090565b6000610a548261153b565b610a71576040516333d1c03960e21b815260040160405180910390fd5b506000908152600560205260409020546001600160a01b031690565b6000610a9882610d76565b9050806001600160a01b0316836001600160a01b03161415610acd5760405163250fdee360e21b815260040160405180910390fd5b336001600160a01b03821614801590610aed5750610aeb813361064f565b155b15610b0b576040516367d9dca160e11b815260040160405180910390fd5b610b1683838361156f565b505050565b6007546001600160a01b03163314610b455760405162461bcd60e51b81526004016107c49061238f565b601055565b6000546001600160801b03600160801b82048116918116919091031690565b610b168383836115cb565b6000610b7f83610d88565b8210610b9e576040516306ed618760e11b815260040160405180910390fd5b600080546001600160801b03169080805b83811015610c6a57600081815260036020908152604091829020825160608101845290546001600160a01b0381168252600160a01b81046001600160401b031692820192909252600160e01b90910460ff161580159282019290925290610c165750610c62565b80516001600160a01b031615610c2b57805192505b876001600160a01b0316836001600160a01b03161415610c605786841415610c595750935061076392505050565b6001909301925b505b600101610baf565b50600080fd5b610b16838383604051806020016040528060008152506111b0565b600080546001600160801b031681805b82811015610d1b57600081815260036020908152604091829020825160608101845290546001600160a01b0381168252600160a01b81046001600160401b031692820192909252600160e01b90910460ff16151591810182905290610d125785831415610d0b5750949350505050565b6001909201915b50600101610c9b565b506040516329c8c00760e21b815260040160405180910390fd5b6007546001600160a01b03163314610d5f5760405162461bcd60e51b81526004016107c49061238f565b8051610d72906012906020840190611db3565b5050565b6000610d81826117e8565b5192915050565b60006001600160a01b038216610db1576040516323d3ad8160e21b815260040160405180910390fd5b506001600160a01b03166000908152600460205260409020546001600160401b031690565b6007546001600160a01b03163314610e005760405162461bcd60e51b81526004016107c49061238f565b610e0a600061190a565b565b6007546001600160a01b03163314610e365760405162461bcd60e51b81526004016107c49061238f565b601155565b6060600280546109c6906124a5565b6007546001600160a01b03163314610e745760405162461bcd60e51b81526004016107c49061238f565b6000805b8351811015610ebd57828181518110610e9357610e9361253b565b602002602001015160ff1682610ea99190612417565b915080610eb5816124e0565b915050610e78565b50600e5481600d54610ecf9190612417565b1115610f325760405162461bcd60e51b815260206004820152602c60248201527f5765277665207265616368656420746865206d6178696d756d206f662061697260448201526b323937b8103634b6b4ba399760a11b60648201526084016107c4565b600c5481610f3e610b4a565b610f489190612417565b1115610f8a5760405162461bcd60e51b815260206004820152601160248201527029bab93830b9b9b2b99039bab838363c9760791b60448201526064016107c4565b60005b8351811015610fe757610fd5848281518110610fab57610fab61253b565b6020026020010151848381518110610fc557610fc561253b565b602002602001015160ff16611521565b80610fdf816124e0565b915050610f8d565b5080600d6000828254610ffa9190612417565b9091555050505050565b6001600160a01b03821633141561102e5760405163b06307db60e01b815260040160405180910390fd5b3360008181526006602090815260408083206001600160a01b03871680855290835292819020805460ff191686151590811790915590519081529192917f17307eab39ab6107e8899845ad3d59bd9653f200f220920489ca2b5937696c31910160405180910390a35050565b6007546001600160a01b031633146110c45760405162461bcd60e51b81526004016107c49061238f565b600855565b6007546001600160a01b031633146110f35760405162461bcd60e51b81526004016107c49061238f565b604051600090339047908381818185875af1925050503d8060008114611135576040519150601f19603f3d011682016040523d82523d6000602084013e61113a565b606091505b505090508061117e5760405162461bcd60e51b815260206004820152601060248201526f2a3930b739b332b9103330b4b632b21760811b60448201526064016107c4565b50565b6007546001600160a01b031633146111ab5760405162461bcd60e51b81526004016107c49061238f565b600f55565b6111bb8484846115cb565b6111c78484848461195c565b6111e4576040516368d2bf6b60e11b815260040160405180910390fd5b50505050565b600061124b838380806020026020016040519081016040528093929190818152602001838360200280828437600092019190915250506011546040516bffffffffffffffffffffffff193360601b166020820152909250603401905061082a565b9392505050565b606061125d8261153b565b61127a57604051630a14c4b560e41b815260040160405180910390fd5b6000611284611a6b565b90508051600014156112a5576040518060200160405280600081525061124b565b806112af84611a7a565b6040516020016112c0929190612310565b6040516020818303038152906040529392505050565b4260105411156113285760405162461bcd60e51b815260206004820152601860248201527f5075626c69632073616c65206e6f7420737461727465642e000000000000000060448201526064016107c4565b80600c54600d5482611338610b4a565b6113429190612417565b61134c9190612417565b111561138d5760405162461bcd60e51b815260206004820152601060248201526f53757270617373657320737570706c7960801b60448201526064016107c4565b81600a546000821180156113a15750808211155b6113e85760405162461bcd60e51b815260206004820152601860248201527726b4b73a34b7339030b137bb329030b63637b1b0ba34b7b760411b60448201526064016107c4565b836009546113f69190612443565b34101561143a5760405162461bcd60e51b815260206004820152601260248201527124b739bab33334b1b2b73a10333ab732399760711b60448201526064016107c4565b6111e43385611521565b6007546001600160a01b0316331461146e5760405162461bcd60e51b81526004016107c49061238f565b6001600160a01b0381166114d35760405162461bcd60e51b815260206004820152602660248201527f4f776e61626c653a206e6577206f776e657220697320746865207a65726f206160448201526564647265737360d01b60648201526084016107c4565b61117e8161190a565b6007546001600160a01b031633146115065760405162461bcd60e51b81526004016107c49061238f565b600955565b6000826115188584611b77565b14949350505050565b610d72828260405180602001604052806000815250611c23565b600080546001600160801b031682108015610763575050600090815260036020526040902054600160e01b900460ff161590565b60008281526005602052604080822080546001600160a01b0319166001600160a01b0387811691821790925591518593918516917f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b92591a4505050565b60006115d6826117e8565b80519091506000906001600160a01b0316336001600160a01b0316148061160457508151611604903361064f565b8061161f57503361161484610a49565b6001600160a01b0316145b90508061163f57604051632ce44b5f60e11b815260040160405180910390fd5b846001600160a01b031682600001516001600160a01b0316146116745760405162a1148160e81b815260040160405180910390fd5b6001600160a01b03841661169b57604051633a954ecd60e21b815260040160405180910390fd5b6116ab600084846000015161156f565b6001600160a01b038581166000908152600460209081526040808320805467ffffffffffffffff198082166001600160401b0392831660001901831617909255898616808652838620805493841693831660019081018416949094179055898652600390945282852080546001600160e01b031916909417600160a01b42909216919091021790925590860180835291205490911661179e576000546001600160801b031681101561179e57825160008281526003602090815260409091208054918601516001600160401b0316600160a01b026001600160e01b03199092166001600160a01b03909316929092171790555b5082846001600160a01b0316866001600160a01b03167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef60405160405180910390a45b5050505050565b60408051606081018252600080825260208201819052918101829052905482906001600160801b03168110156118f157600081815260036020908152604091829020825160608101845290546001600160a01b0381168252600160a01b81046001600160401b031692820192909252600160e01b90910460ff161515918101829052906118ef5780516001600160a01b031615611886579392505050565b5060001901600081815260036020908152604091829020825160608101845290546001600160a01b038116808352600160a01b82046001600160401b031693830193909352600160e01b900460ff16151592810192909252156118ea579392505050565b611886565b505b604051636f96cda160e11b815260040160405180910390fd5b600780546001600160a01b038381166001600160a01b0319831681179093556040519116919082907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e090600090a35050565b60006001600160a01b0384163b15611a5f57604051630a85bd0160e11b81526001600160a01b0385169063150b7a02906119a090339089908890889060040161233f565b602060405180830381600087803b1580156119ba57600080fd5b505af19250505080156119ea575060408051601f3d908101601f191682019092526119e791810190612234565b60015b611a45573d808015611a18576040519150601f19603f3d011682016040523d82523d6000602084013e611a1d565b606091505b508051611a3d576040516368d2bf6b60e11b815260040160405180910390fd5b805181602001fd5b6001600160e01b031916630a85bd0160e11b149050611a63565b5060015b949350505050565b6060601280546109c6906124a5565b606081611a9e5750506040805180820190915260018152600360fc1b602082015290565b8160005b8115611ac85780611ab2816124e0565b9150611ac19050600a8361242f565b9150611aa2565b6000816001600160401b03811115611ae257611ae2612551565b6040519080825280601f01601f191660200182016040528015611b0c576020820181803683370190505b5090505b8415611a6357611b21600183612462565b9150611b2e600a866124fb565b611b39906030612417565b60f81b818381518110611b4e57611b4e61253b565b60200101906001600160f81b031916908160001a905350611b70600a8661242f565b9450611b10565b600081815b8451811015611c1b576000858281518110611b9957611b9961253b565b60200260200101519050808311611bdb576040805160208101859052908101829052606001604051602081830303815290604052805190602001209250611c08565b60408051602081018390529081018490526060016040516020818303038152906040528051906020012092505b5080611c13816124e0565b915050611b7c565b509392505050565b610b1683838360016000546001600160801b03166001600160a01b038516611c5d57604051622e076360e81b815260040160405180910390fd5b83611c7b5760405163b562e8dd60e01b815260040160405180910390fd5b6001600160a01b038516600081815260046020908152604080832080546001600160801b031981166001600160401b038083168c0181169182176801000000000000000067ffffffffffffffff1990941690921783900481168c018116909202179091558584526003909252822080546001600160e01b031916909317600160a01b42909216919091021790915581905b85811015611d8d5760405182906001600160a01b038916906000907fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef908290a4838015611d635750611d61600088848861195c565b155b15611d81576040516368d2bf6b60e11b815260040160405180910390fd5b60019182019101611d0c565b50600080546001600160801b0319166001600160801b03929092169190911790556117e1565b828054611dbf906124a5565b90600052602060002090601f016020900481019282611de15760008555611e27565b82601f10611dfa57805160ff1916838001178555611e27565b82800160010185558215611e27579182015b82811115611e27578251825591602001919060010190611e0c565b50611e33929150611e37565b5090565b5b80821115611e335760008155600101611e38565b60006001600160401b03831115611e6557611e65612551565b611e78601f8401601f19166020016123c4565b9050828152838383011115611e8c57600080fd5b828260208301376000602084830101529392505050565b80356001600160a01b0381168114611eba57600080fd5b919050565b60008083601f840112611ed157600080fd5b5081356001600160401b03811115611ee857600080fd5b6020830191508360208260051b8501011115611f0357600080fd5b9250929050565b600082601f830112611f1b57600080fd5b81356020611f30611f2b836123f4565b6123c4565b80838252828201915082860187848660051b8901011115611f5057600080fd5b6000805b86811015611f7e57823560ff81168114611f6c578283fd5b85529385019391850191600101611f54565b509198975050505050505050565b600060208284031215611f9e57600080fd5b61124b82611ea3565b60008060408385031215611fba57600080fd5b611fc383611ea3565b9150611fd160208401611ea3565b90509250929050565b600080600060608486031215611fef57600080fd5b611ff884611ea3565b925061200660208501611ea3565b9150604084013590509250925092565b6000806000806080858703121561202c57600080fd5b61203585611ea3565b935061204360208601611ea3565b92506040850135915060608501356001600160401b0381111561206557600080fd5b8501601f8101871361207657600080fd5b61208587823560208401611e4c565b91505092959194509250565b600080604083850312156120a457600080fd5b6120ad83611ea3565b9150602083013580151581146120c257600080fd5b809150509250929050565b600080604083850312156120e057600080fd5b6120e983611ea3565b946020939093013593505050565b6000806040838503121561210a57600080fd5b82356001600160401b038082111561212157600080fd5b818501915085601f83011261213557600080fd5b81356020612145611f2b836123f4565b8083825282820191508286018a848660051b890101111561216557600080fd5b600096505b8487101561218f5761217b81611ea3565b83526001969096019591830191830161216a565b50965050860135925050808211156121a657600080fd5b506121b385828601611f0a565b9150509250929050565b600080602083850312156121d057600080fd5b82356001600160401b038111156121e657600080fd5b6121f285828601611ebf565b90969095509350505050565b60006020828403121561221057600080fd5b5035919050565b60006020828403121561222957600080fd5b813561124b81612567565b60006020828403121561224657600080fd5b815161124b81612567565b60006020828403121561226357600080fd5b81356001600160401b0381111561227957600080fd5b8201601f8101841361228a57600080fd5b611a6384823560208401611e4c565b6000806000604084860312156122ae57600080fd5b8335925060208401356001600160401b038111156122cb57600080fd5b6122d786828701611ebf565b9497909650939450505050565b600081518084526122fc816020860160208601612479565b601f01601f19169290920160200192915050565b60008351612322818460208801612479565b835190830190612336818360208801612479565b01949350505050565b6001600160a01b0385811682528416602082015260408101839052608060608201819052600090612372908301846122e4565b9695505050505050565b60208152600061124b60208301846122e4565b6020808252818101527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604082015260600190565b604051601f8201601f191681016001600160401b03811182821017156123ec576123ec612551565b604052919050565b60006001600160401b0382111561240d5761240d612551565b5060051b60200190565b6000821982111561242a5761242a61250f565b500190565b60008261243e5761243e612525565b500490565b600081600019048311821515161561245d5761245d61250f565b500290565b6000828210156124745761247461250f565b500390565b60005b8381101561249457818101518382015260200161247c565b838111156111e45750506000910152565b600181811c908216806124b957607f821691505b602082108114156124da57634e487b7160e01b600052602260045260246000fd5b50919050565b60006000198214156124f4576124f461250f565b5060010190565b60008261250a5761250a612525565b500690565b634e487b7160e01b600052601160045260246000fd5b634e487b7160e01b600052601260045260246000fd5b634e487b7160e01b600052603260045260246000fd5b634e487b7160e01b600052604160045260246000fd5b6001600160e01b03198116811461117e57600080fdfea26469706673582212206dee3fc8f2422bee2e2d4351e7638eeaf49b00b891ef5e74bddedc07c533713364736f6c63430008070033

Deployed Bytecode

0x60806040526004361061023b5760003560e01c806370a082311161012e578063b88d4fde116100ab578063ebf0c7171161006f578063ebf0c7171461067d578063efd0cbf914610693578063f2fde38b146106a6578063f4a0a528146106c6578063f6000c7a146106e657600080fd5b8063b88d4fde146105be578063b8dde112146105de578063c87b56dd146105fe578063d5abeb011461061e578063e985e9c51461063457600080fd5b8063a061a03a116100f2578063a061a03a14610529578063a22cb46514610549578063a611708e14610569578063ac44600214610589578063afdea6c21461059e57600080fd5b806370a08231146104a1578063715018a6146104c15780637cb64759146104d65780638da5cb5b146104f657806395d89b411461051457600080fd5b806335c6aaf8116101bc57806355f804b31161018057806355f804b31461041f5780635ce97dbb1461043f5780636352211e146104555780636817c76c146104755780636bb7b1d91461048b57600080fd5b806335c6aaf81461039d57806342842e0e146103b3578063453c2310146103d35780634f6ccce7146103e957806350d549151461040957600080fd5b806311b7e5e71161020357806311b7e5e71461030457806318160ddd14610324578063230b43f41461034757806323b872dd1461035d5780632f745c591461037d57600080fd5b806301ffc9a714610240578063061431a81461027557806306fdde031461028a578063081812fc146102ac578063095ea7b3146102e4575b600080fd5b34801561024c57600080fd5b5061026061025b366004612217565b6106fc565b60405190151581526020015b60405180910390f35b610288610283366004612299565b610769565b005b34801561029657600080fd5b5061029f6109b7565b60405161026c919061237c565b3480156102b857600080fd5b506102cc6102c73660046121fe565b610a49565b6040516001600160a01b03909116815260200161026c565b3480156102f057600080fd5b506102886102ff3660046120cd565b610a8d565b34801561031057600080fd5b5061028861031f3660046121fe565b610b1b565b34801561033057600080fd5b50610339610b4a565b60405190815260200161026c565b34801561035357600080fd5b50610339600f5481565b34801561036957600080fd5b50610288610378366004611fda565b610b69565b34801561038957600080fd5b506103396103983660046120cd565b610b74565b3480156103a957600080fd5b5061033960085481565b3480156103bf57600080fd5b506102886103ce366004611fda565b610c70565b3480156103df57600080fd5b50610339600a5481565b3480156103f557600080fd5b506103396104043660046121fe565b610c8b565b34801561041557600080fd5b50610339600d5481565b34801561042b57600080fd5b5061028861043a366004612251565b610d35565b34801561044b57600080fd5b50610339600e5481565b34801561046157600080fd5b506102cc6104703660046121fe565b610d76565b34801561048157600080fd5b5061033960095481565b34801561049757600080fd5b5061033960105481565b3480156104ad57600080fd5b506103396104bc366004611f8c565b610d88565b3480156104cd57600080fd5b50610288610dd6565b3480156104e257600080fd5b506102886104f13660046121fe565b610e0c565b34801561050257600080fd5b506007546001600160a01b03166102cc565b34801561052057600080fd5b5061029f610e3b565b34801561053557600080fd5b506102886105443660046120f7565b610e4a565b34801561055557600080fd5b50610288610564366004612091565b611004565b34801561057557600080fd5b506102886105843660046121fe565b61109a565b34801561059557600080fd5b506102886110c9565b3480156105aa57600080fd5b506102886105b93660046121fe565b611181565b3480156105ca57600080fd5b506102886105d9366004612016565b6111b0565b3480156105ea57600080fd5b506102606105f93660046121bd565b6111ea565b34801561060a57600080fd5b5061029f6106193660046121fe565b611252565b34801561062a57600080fd5b50610339600c5481565b34801561064057600080fd5b5061026061064f366004611fa7565b6001600160a01b03918216600090815260066020908152604080832093909416825291909152205460ff1690565b34801561068957600080fd5b5061033960115481565b6102886106a13660046121fe565b6112d6565b3480156106b257600080fd5b506102886106c1366004611f8c565b611444565b3480156106d257600080fd5b506102886106e13660046121fe565b6114dc565b3480156106f257600080fd5b50610339600b5481565b60006001600160e01b031982166380ac58cd60e01b148061072d57506001600160e01b03198216635b5e139f60e01b145b8061074857506001600160e01b0319821663780e9d6360e01b145b8061076357506301ffc9a760e01b6001600160e01b03198316145b92915050565b42600f541115801561077c575060105442105b6107cd5760405162461bcd60e51b815260206004820152601b60248201527f57686974656c6973742073616c65206e6f7420737461727465642e000000000060448201526064015b60405180910390fd5b8181610845828280806020026020016040519081016040528093929190818152602001838360200280828437600092019190915250506011546040516bffffffffffffffffffffffff193360601b16602082015290925060340190505b6040516020818303038152906040528051906020012061150b565b6108915760405162461bcd60e51b815260206004820152601860248201527f41646472657373206e6f742077686974656c69737465642e000000000000000060448201526064016107c4565b84600c54600d54826108a1610b4a565b6108ab9190612417565b6108b59190612417565b11156108f65760405162461bcd60e51b815260206004820152601060248201526f53757270617373657320737570706c7960801b60448201526064016107c4565b85600b5460008211801561090a5750808211155b6109515760405162461bcd60e51b815260206004820152601860248201527726b4b73a34b7339030b137bb329030b63637b1b0ba34b7b760411b60448201526064016107c4565b8760085461095f9190612443565b3410156109a35760405162461bcd60e51b815260206004820152601260248201527124b739bab33334b1b2b73a10333ab732399760711b60448201526064016107c4565b6109ad3389611521565b5050505050505050565b6060600180546109c6906124a5565b80601f01602080910402602001604051908101604052809291908181526020018280546109f2906124a5565b8015610a3f5780601f10610a1457610100808354040283529160200191610a3f565b820191906000526020600020905b815481529060010190602001808311610a2257829003601f168201915b5050505050905090565b6000610a548261153b565b610a71576040516333d1c03960e21b815260040160405180910390fd5b506000908152600560205260409020546001600160a01b031690565b6000610a9882610d76565b9050806001600160a01b0316836001600160a01b03161415610acd5760405163250fdee360e21b815260040160405180910390fd5b336001600160a01b03821614801590610aed5750610aeb813361064f565b155b15610b0b576040516367d9dca160e11b815260040160405180910390fd5b610b1683838361156f565b505050565b6007546001600160a01b03163314610b455760405162461bcd60e51b81526004016107c49061238f565b601055565b6000546001600160801b03600160801b82048116918116919091031690565b610b168383836115cb565b6000610b7f83610d88565b8210610b9e576040516306ed618760e11b815260040160405180910390fd5b600080546001600160801b03169080805b83811015610c6a57600081815260036020908152604091829020825160608101845290546001600160a01b0381168252600160a01b81046001600160401b031692820192909252600160e01b90910460ff161580159282019290925290610c165750610c62565b80516001600160a01b031615610c2b57805192505b876001600160a01b0316836001600160a01b03161415610c605786841415610c595750935061076392505050565b6001909301925b505b600101610baf565b50600080fd5b610b16838383604051806020016040528060008152506111b0565b600080546001600160801b031681805b82811015610d1b57600081815260036020908152604091829020825160608101845290546001600160a01b0381168252600160a01b81046001600160401b031692820192909252600160e01b90910460ff16151591810182905290610d125785831415610d0b5750949350505050565b6001909201915b50600101610c9b565b506040516329c8c00760e21b815260040160405180910390fd5b6007546001600160a01b03163314610d5f5760405162461bcd60e51b81526004016107c49061238f565b8051610d72906012906020840190611db3565b5050565b6000610d81826117e8565b5192915050565b60006001600160a01b038216610db1576040516323d3ad8160e21b815260040160405180910390fd5b506001600160a01b03166000908152600460205260409020546001600160401b031690565b6007546001600160a01b03163314610e005760405162461bcd60e51b81526004016107c49061238f565b610e0a600061190a565b565b6007546001600160a01b03163314610e365760405162461bcd60e51b81526004016107c49061238f565b601155565b6060600280546109c6906124a5565b6007546001600160a01b03163314610e745760405162461bcd60e51b81526004016107c49061238f565b6000805b8351811015610ebd57828181518110610e9357610e9361253b565b602002602001015160ff1682610ea99190612417565b915080610eb5816124e0565b915050610e78565b50600e5481600d54610ecf9190612417565b1115610f325760405162461bcd60e51b815260206004820152602c60248201527f5765277665207265616368656420746865206d6178696d756d206f662061697260448201526b323937b8103634b6b4ba399760a11b60648201526084016107c4565b600c5481610f3e610b4a565b610f489190612417565b1115610f8a5760405162461bcd60e51b815260206004820152601160248201527029bab93830b9b9b2b99039bab838363c9760791b60448201526064016107c4565b60005b8351811015610fe757610fd5848281518110610fab57610fab61253b565b6020026020010151848381518110610fc557610fc561253b565b602002602001015160ff16611521565b80610fdf816124e0565b915050610f8d565b5080600d6000828254610ffa9190612417565b9091555050505050565b6001600160a01b03821633141561102e5760405163b06307db60e01b815260040160405180910390fd5b3360008181526006602090815260408083206001600160a01b03871680855290835292819020805460ff191686151590811790915590519081529192917f17307eab39ab6107e8899845ad3d59bd9653f200f220920489ca2b5937696c31910160405180910390a35050565b6007546001600160a01b031633146110c45760405162461bcd60e51b81526004016107c49061238f565b600855565b6007546001600160a01b031633146110f35760405162461bcd60e51b81526004016107c49061238f565b604051600090339047908381818185875af1925050503d8060008114611135576040519150601f19603f3d011682016040523d82523d6000602084013e61113a565b606091505b505090508061117e5760405162461bcd60e51b815260206004820152601060248201526f2a3930b739b332b9103330b4b632b21760811b60448201526064016107c4565b50565b6007546001600160a01b031633146111ab5760405162461bcd60e51b81526004016107c49061238f565b600f55565b6111bb8484846115cb565b6111c78484848461195c565b6111e4576040516368d2bf6b60e11b815260040160405180910390fd5b50505050565b600061124b838380806020026020016040519081016040528093929190818152602001838360200280828437600092019190915250506011546040516bffffffffffffffffffffffff193360601b166020820152909250603401905061082a565b9392505050565b606061125d8261153b565b61127a57604051630a14c4b560e41b815260040160405180910390fd5b6000611284611a6b565b90508051600014156112a5576040518060200160405280600081525061124b565b806112af84611a7a565b6040516020016112c0929190612310565b6040516020818303038152906040529392505050565b4260105411156113285760405162461bcd60e51b815260206004820152601860248201527f5075626c69632073616c65206e6f7420737461727465642e000000000000000060448201526064016107c4565b80600c54600d5482611338610b4a565b6113429190612417565b61134c9190612417565b111561138d5760405162461bcd60e51b815260206004820152601060248201526f53757270617373657320737570706c7960801b60448201526064016107c4565b81600a546000821180156113a15750808211155b6113e85760405162461bcd60e51b815260206004820152601860248201527726b4b73a34b7339030b137bb329030b63637b1b0ba34b7b760411b60448201526064016107c4565b836009546113f69190612443565b34101561143a5760405162461bcd60e51b815260206004820152601260248201527124b739bab33334b1b2b73a10333ab732399760711b60448201526064016107c4565b6111e43385611521565b6007546001600160a01b0316331461146e5760405162461bcd60e51b81526004016107c49061238f565b6001600160a01b0381166114d35760405162461bcd60e51b815260206004820152602660248201527f4f776e61626c653a206e6577206f776e657220697320746865207a65726f206160448201526564647265737360d01b60648201526084016107c4565b61117e8161190a565b6007546001600160a01b031633146115065760405162461bcd60e51b81526004016107c49061238f565b600955565b6000826115188584611b77565b14949350505050565b610d72828260405180602001604052806000815250611c23565b600080546001600160801b031682108015610763575050600090815260036020526040902054600160e01b900460ff161590565b60008281526005602052604080822080546001600160a01b0319166001600160a01b0387811691821790925591518593918516917f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b92591a4505050565b60006115d6826117e8565b80519091506000906001600160a01b0316336001600160a01b0316148061160457508151611604903361064f565b8061161f57503361161484610a49565b6001600160a01b0316145b90508061163f57604051632ce44b5f60e11b815260040160405180910390fd5b846001600160a01b031682600001516001600160a01b0316146116745760405162a1148160e81b815260040160405180910390fd5b6001600160a01b03841661169b57604051633a954ecd60e21b815260040160405180910390fd5b6116ab600084846000015161156f565b6001600160a01b038581166000908152600460209081526040808320805467ffffffffffffffff198082166001600160401b0392831660001901831617909255898616808652838620805493841693831660019081018416949094179055898652600390945282852080546001600160e01b031916909417600160a01b42909216919091021790925590860180835291205490911661179e576000546001600160801b031681101561179e57825160008281526003602090815260409091208054918601516001600160401b0316600160a01b026001600160e01b03199092166001600160a01b03909316929092171790555b5082846001600160a01b0316866001600160a01b03167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef60405160405180910390a45b5050505050565b60408051606081018252600080825260208201819052918101829052905482906001600160801b03168110156118f157600081815260036020908152604091829020825160608101845290546001600160a01b0381168252600160a01b81046001600160401b031692820192909252600160e01b90910460ff161515918101829052906118ef5780516001600160a01b031615611886579392505050565b5060001901600081815260036020908152604091829020825160608101845290546001600160a01b038116808352600160a01b82046001600160401b031693830193909352600160e01b900460ff16151592810192909252156118ea579392505050565b611886565b505b604051636f96cda160e11b815260040160405180910390fd5b600780546001600160a01b038381166001600160a01b0319831681179093556040519116919082907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e090600090a35050565b60006001600160a01b0384163b15611a5f57604051630a85bd0160e11b81526001600160a01b0385169063150b7a02906119a090339089908890889060040161233f565b602060405180830381600087803b1580156119ba57600080fd5b505af19250505080156119ea575060408051601f3d908101601f191682019092526119e791810190612234565b60015b611a45573d808015611a18576040519150601f19603f3d011682016040523d82523d6000602084013e611a1d565b606091505b508051611a3d576040516368d2bf6b60e11b815260040160405180910390fd5b805181602001fd5b6001600160e01b031916630a85bd0160e11b149050611a63565b5060015b949350505050565b6060601280546109c6906124a5565b606081611a9e5750506040805180820190915260018152600360fc1b602082015290565b8160005b8115611ac85780611ab2816124e0565b9150611ac19050600a8361242f565b9150611aa2565b6000816001600160401b03811115611ae257611ae2612551565b6040519080825280601f01601f191660200182016040528015611b0c576020820181803683370190505b5090505b8415611a6357611b21600183612462565b9150611b2e600a866124fb565b611b39906030612417565b60f81b818381518110611b4e57611b4e61253b565b60200101906001600160f81b031916908160001a905350611b70600a8661242f565b9450611b10565b600081815b8451811015611c1b576000858281518110611b9957611b9961253b565b60200260200101519050808311611bdb576040805160208101859052908101829052606001604051602081830303815290604052805190602001209250611c08565b60408051602081018390529081018490526060016040516020818303038152906040528051906020012092505b5080611c13816124e0565b915050611b7c565b509392505050565b610b1683838360016000546001600160801b03166001600160a01b038516611c5d57604051622e076360e81b815260040160405180910390fd5b83611c7b5760405163b562e8dd60e01b815260040160405180910390fd5b6001600160a01b038516600081815260046020908152604080832080546001600160801b031981166001600160401b038083168c0181169182176801000000000000000067ffffffffffffffff1990941690921783900481168c018116909202179091558584526003909252822080546001600160e01b031916909317600160a01b42909216919091021790915581905b85811015611d8d5760405182906001600160a01b038916906000907fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef908290a4838015611d635750611d61600088848861195c565b155b15611d81576040516368d2bf6b60e11b815260040160405180910390fd5b60019182019101611d0c565b50600080546001600160801b0319166001600160801b03929092169190911790556117e1565b828054611dbf906124a5565b90600052602060002090601f016020900481019282611de15760008555611e27565b82601f10611dfa57805160ff1916838001178555611e27565b82800160010185558215611e27579182015b82811115611e27578251825591602001919060010190611e0c565b50611e33929150611e37565b5090565b5b80821115611e335760008155600101611e38565b60006001600160401b03831115611e6557611e65612551565b611e78601f8401601f19166020016123c4565b9050828152838383011115611e8c57600080fd5b828260208301376000602084830101529392505050565b80356001600160a01b0381168114611eba57600080fd5b919050565b60008083601f840112611ed157600080fd5b5081356001600160401b03811115611ee857600080fd5b6020830191508360208260051b8501011115611f0357600080fd5b9250929050565b600082601f830112611f1b57600080fd5b81356020611f30611f2b836123f4565b6123c4565b80838252828201915082860187848660051b8901011115611f5057600080fd5b6000805b86811015611f7e57823560ff81168114611f6c578283fd5b85529385019391850191600101611f54565b509198975050505050505050565b600060208284031215611f9e57600080fd5b61124b82611ea3565b60008060408385031215611fba57600080fd5b611fc383611ea3565b9150611fd160208401611ea3565b90509250929050565b600080600060608486031215611fef57600080fd5b611ff884611ea3565b925061200660208501611ea3565b9150604084013590509250925092565b6000806000806080858703121561202c57600080fd5b61203585611ea3565b935061204360208601611ea3565b92506040850135915060608501356001600160401b0381111561206557600080fd5b8501601f8101871361207657600080fd5b61208587823560208401611e4c565b91505092959194509250565b600080604083850312156120a457600080fd5b6120ad83611ea3565b9150602083013580151581146120c257600080fd5b809150509250929050565b600080604083850312156120e057600080fd5b6120e983611ea3565b946020939093013593505050565b6000806040838503121561210a57600080fd5b82356001600160401b038082111561212157600080fd5b818501915085601f83011261213557600080fd5b81356020612145611f2b836123f4565b8083825282820191508286018a848660051b890101111561216557600080fd5b600096505b8487101561218f5761217b81611ea3565b83526001969096019591830191830161216a565b50965050860135925050808211156121a657600080fd5b506121b385828601611f0a565b9150509250929050565b600080602083850312156121d057600080fd5b82356001600160401b038111156121e657600080fd5b6121f285828601611ebf565b90969095509350505050565b60006020828403121561221057600080fd5b5035919050565b60006020828403121561222957600080fd5b813561124b81612567565b60006020828403121561224657600080fd5b815161124b81612567565b60006020828403121561226357600080fd5b81356001600160401b0381111561227957600080fd5b8201601f8101841361228a57600080fd5b611a6384823560208401611e4c565b6000806000604084860312156122ae57600080fd5b8335925060208401356001600160401b038111156122cb57600080fd5b6122d786828701611ebf565b9497909650939450505050565b600081518084526122fc816020860160208601612479565b601f01601f19169290920160200192915050565b60008351612322818460208801612479565b835190830190612336818360208801612479565b01949350505050565b6001600160a01b0385811682528416602082015260408101839052608060608201819052600090612372908301846122e4565b9695505050505050565b60208152600061124b60208301846122e4565b6020808252818101527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604082015260600190565b604051601f8201601f191681016001600160401b03811182821017156123ec576123ec612551565b604052919050565b60006001600160401b0382111561240d5761240d612551565b5060051b60200190565b6000821982111561242a5761242a61250f565b500190565b60008261243e5761243e612525565b500490565b600081600019048311821515161561245d5761245d61250f565b500290565b6000828210156124745761247461250f565b500390565b60005b8381101561249457818101518382015260200161247c565b838111156111e45750506000910152565b600181811c908216806124b957607f821691505b602082108114156124da57634e487b7160e01b600052602260045260246000fd5b50919050565b60006000198214156124f4576124f461250f565b5060010190565b60008261250a5761250a612525565b500690565b634e487b7160e01b600052601160045260246000fd5b634e487b7160e01b600052601260045260246000fd5b634e487b7160e01b600052603260045260246000fd5b634e487b7160e01b600052604160045260246000fd5b6001600160e01b03198116811461117e57600080fdfea26469706673582212206dee3fc8f2422bee2e2d4351e7638eeaf49b00b891ef5e74bddedc07c533713364736f6c63430008070033

Deployed Bytecode Sourcemap

143:4847:11:-:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;6201:372:4;;;;;;;;;;-1:-1:-1;6201:372:4;;;;;:::i;:::-;;:::i;:::-;;;9586:14:14;;9579:22;9561:41;;9549:2;9534:18;6201:372:4;;;;;;;;2786:471:11;;;;;;:::i;:::-;;:::i;:::-;;8811:100:4;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;10314:204::-;;;;;;;;;;-1:-1:-1;10314:204:4;;;;;:::i;:::-;;:::i;:::-;;;-1:-1:-1;;;;;8884:32:14;;;8866:51;;8854:2;8839:18;10314:204:4;8720:203:14;9877:371:4;;;;;;;;;;-1:-1:-1;9877:371:4;;;;;:::i;:::-;;:::i;1520:107:11:-;;;;;;;;;;-1:-1:-1;1520:107:11;;;;;:::i;:::-;;:::i;3438:280:4:-;;;;;;;;;;;;;:::i;:::-;;;9759:25:14;;;9747:2;9732:18;3438:280:4;9613:177:14;491:50:11;;;;;;;;;;;;;;;;11171:170:4;;;;;;;;;;-1:-1:-1;11171:170:4;;;;;:::i;:::-;;:::i;5024:1105::-;;;;;;;;;;-1:-1:-1;5024:1105:4;;;;;:::i;:::-;;:::i;191:47:11:-;;;;;;;;;;;;;;;;11412:185:4;;;;;;;;;;-1:-1:-1;11412:185:4;;;;;:::i;:::-;;:::i;291:32:11:-;;;;;;;;;;;;;;;;4011:713:4;;;;;;;;;;-1:-1:-1;4011:713:4;;;;;:::i;:::-;;:::i;415:30:11:-;;;;;;;;;;;;;;;;4585:100;;;;;;;;;;-1:-1:-1;4585:100:11;;;;;:::i;:::-;;:::i;452:32::-;;;;;;;;;;;;;;;;8620:124:4;;;;;;;;;;-1:-1:-1;8620:124:4;;;;;:::i;:::-;;:::i;245:39:11:-;;;;;;;;;;;;;;;;548:47;;;;;;;;;;;;;;;;6637:206:4;;;;;;;;;;-1:-1:-1;6637:206:4;;;;;:::i;:::-;;:::i;1650:94:12:-;;;;;;;;;;;;;:::i;4020:88:11:-;;;;;;;;;;-1:-1:-1;4020:88:11;;;;;:::i;:::-;;:::i;999:87:12:-;;;;;;;;;;-1:-1:-1;1072:6:12;;-1:-1:-1;;;;;1072:6:12;999:87;;8980:104:4;;;;;;;;;;;;;:::i;3265:669:11:-;;;;;;;;;;-1:-1:-1;3265:669:11;;;;;:::i;:::-;;:::i;10590:279:4:-;;;;;;;;;;-1:-1:-1;10590:279:4;;;;;:::i;:::-;;:::i;4380:112:11:-;;;;;;;;;;-1:-1:-1;4380:112:11;;;;;:::i;:::-;;:::i;4809:178::-;;;;;;;;;;;;;:::i;1635:113::-;;;;;;;;;;-1:-1:-1;1635:113:11;;;;;:::i;:::-;;:::i;11668:342:4:-;;;;;;;;;;-1:-1:-1;11668:342:4;;;;;:::i;:::-;;:::i;2478:300:11:-;;;;;;;;;;-1:-1:-1;2478:300:11;;;;;:::i;:::-;;:::i;9155:318:4:-;;;;;;;;;;-1:-1:-1;9155:318:4;;;;;:::i;:::-;;:::i;377:31:11:-;;;;;;;;;;;;;;;;10940:164:4;;;;;;;;;;-1:-1:-1;10940:164:4;;;;;:::i;:::-;-1:-1:-1;;;;;11061:25:4;;;11037:4;11061:25;;;:18;:25;;;;;;;;:35;;;;;;;;;;;;;;;10940:164;602:19:11;;;;;;;;;;;;;;;;1816:336;;;;;;:::i;:::-;;:::i;1899:192:12:-;;;;;;;;;;-1:-1:-1;1899:192:12;;;;;:::i;:::-;;:::i;4194:94:11:-;;;;;;;;;;-1:-1:-1;4194:94:11;;;;;:::i;:::-;;:::i;330:40::-;;;;;;;;;;;;;;;;6201:372:4;6303:4;-1:-1:-1;;;;;;6340:40:4;;-1:-1:-1;;;6340:40:4;;:105;;-1:-1:-1;;;;;;;6397:48:4;;-1:-1:-1;;;6397:48:4;6340:105;:172;;;-1:-1:-1;;;;;;;6462:50:4;;-1:-1:-1;;;6462:50:4;6340:172;:225;;;-1:-1:-1;;;;;;;;;;896:40:2;;;6529:36:4;6320:245;6201:372;-1:-1:-1;;6201:372:4:o;2786:471:11:-;1381:15;1355:22;;:41;;:82;;;;;1418:19;;1400:15;:37;1355:82;1333:159;;;;-1:-1:-1;;;1333:159:11;;10628:2:14;1333:159:11;;;10610:21:14;10667:2;10647:18;;;10640:30;10706:29;10686:18;;;10679:57;10753:18;;1333:159:11;;;;;;;;;2953:11:::1;;2254:144;2291:11;;2254:144;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;::::0;::::1;::::0;;;;-1:-1:-1;;2321:4:11::1;::::0;2354:28:::1;::::0;-1:-1:-1;;2371:10:11::1;7698:2:14::0;7694:15;7690:53;2354:28:11::1;::::0;::::1;7678:66:14::0;2321:4:11;;-1:-1:-1;7760:12:14;;;-1:-1:-1;2354:28:11::1;;;;;;;;;;;;;2344:39;;;;;;2254:18;:144::i;:::-;2232:218;;;::::0;-1:-1:-1;;;2232:218:11;;12105:2:14;2232:218:11::1;::::0;::::1;12087:21:14::0;12144:2;12124:18;;;12117:30;12183:26;12163:18;;;12156:54;12227:18;;2232:218:11::1;11903:348:14::0;2232:218:11::1;2996:9:::2;816;;797:15;;785:9;769:13;:11;:13::i;:::-;:25;;;;:::i;:::-;:43;;;;:::i;:::-;:56;;747:122;;;::::0;-1:-1:-1;;;747:122:11;;13855:2:14;747:122:11::2;::::0;::::2;13837:21:14::0;13894:2;13874:18;;;13867:30;-1:-1:-1;;;13913:18:14;;;13906:46;13969:18;;747:122:11::2;13653:340:14::0;747:122:11::2;3035:9:::3;3046:21;;1006:1;994:9;:13;:38;;;;;1025:7;1012:9;:20;;994:38;972:112;;;::::0;-1:-1:-1;;;972:112:11;;13149:2:14;972:112:11::3;::::0;::::3;13131:21:14::0;13188:2;13168:18;;;13161:30;-1:-1:-1;;;13207:18:14;;;13200:54;13271:18;;972:112:11::3;12947:348:14::0;972:112:11::3;3141:9:::4;3120:18;;:30;;;;:::i;:::-;3107:9;:43;;3085:111;;;::::0;-1:-1:-1;;;3085:111:11;;10984:2:14;3085:111:11::4;::::0;::::4;10966:21:14::0;11023:2;11003:18;;;10996:30;-1:-1:-1;;;11042:18:14;;;11035:48;11100:18;;3085:111:11::4;10782:342:14::0;3085:111:11::4;3217:32;3227:10;3239:9;3217;:32::i;:::-;880:1:::3;;2461::::2;1503::::1;;2786:471:::0;;;:::o;8811:100:4:-;8865:13;8898:5;8891:12;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;8811:100;:::o;10314:204::-;10382:7;10407:16;10415:7;10407;:16::i;:::-;10402:64;;10432:34;;-1:-1:-1;;;10432:34:4;;;;;;;;;;;10402:64;-1:-1:-1;10486:24:4;;;;:15;:24;;;;;;-1:-1:-1;;;;;10486:24:4;;10314:204::o;9877:371::-;9950:13;9966:24;9982:7;9966:15;:24::i;:::-;9950:40;;10011:5;-1:-1:-1;;;;;10005:11:4;:2;-1:-1:-1;;;;;10005:11:4;;10001:48;;;10025:24;;-1:-1:-1;;;10025:24:4;;;;;;;;;;;10001:48;681:10:1;-1:-1:-1;;;;;10066:21:4;;;;;;:63;;-1:-1:-1;10092:37:4;10109:5;681:10:1;10940:164:4;:::i;10092:37::-;10091:38;10066:63;10062:138;;;10153:35;;-1:-1:-1;;;10153:35:4;;;;;;;;;;;10062:138;10212:28;10221:2;10225:7;10234:5;10212:8;:28::i;:::-;9939:309;9877:371;;:::o;1520:107:11:-;1072:6:12;;-1:-1:-1;;;;;1072:6:12;681:10:1;1219:23:12;1211:68;;;;-1:-1:-1;;;1211:68:12;;;;;;;:::i;:::-;1592:19:11::1;:27:::0;1520:107::o;3438:280:4:-;3491:7;3683:12;-1:-1:-1;;;;;;;;3683:12:4;;;;3667:13;;;:28;;;;3660:35;;3438:280::o;11171:170::-;11305:28;11315:4;11321:2;11325:7;11305:9;:28::i;5024:1105::-;5113:7;5146:16;5156:5;5146:9;:16::i;:::-;5137:5;:25;5133:61;;5171:23;;-1:-1:-1;;;5171:23:4;;;;;;;;;;;5133:61;5205:22;5230:13;;-1:-1:-1;;;;;5230:13:4;;5205:22;;5480:557;5500:14;5496:1;:18;5480:557;;;5540:31;5574:14;;;:11;:14;;;;;;;;;5540:48;;;;;;;;;-1:-1:-1;;;;;5540:48:4;;;;-1:-1:-1;;;5540:48:4;;-1:-1:-1;;;;;5540:48:4;;;;;;;;-1:-1:-1;;;5540:48:4;;;;;;;;;;;;;;;;5607:73;;5652:8;;;5607:73;5702:14;;-1:-1:-1;;;;;5702:28:4;;5698:111;;5775:14;;;-1:-1:-1;5698:111:4;5852:5;-1:-1:-1;;;;;5831:26:4;:17;-1:-1:-1;;;;;5831:26:4;;5827:195;;;5901:5;5886:11;:20;5882:85;;;-1:-1:-1;5942:1:4;-1:-1:-1;5935:8:4;;-1:-1:-1;;;5935:8:4;5882:85;5989:13;;;;;5827:195;5521:516;5480:557;5516:3;;5480:557;;;;6113:8;;;11412:185;11550:39;11567:4;11573:2;11577:7;11550:39;;;;;;;;;;;;:16;:39::i;4011:713::-;4078:7;4123:13;;-1:-1:-1;;;;;4123:13:4;4078:7;;4337:328;4357:14;4353:1;:18;4337:328;;;4397:31;4431:14;;;:11;:14;;;;;;;;;4397:48;;;;;;;;;-1:-1:-1;;;;;4397:48:4;;;;-1:-1:-1;;;4397:48:4;;-1:-1:-1;;;;;4397:48:4;;;;;;;;-1:-1:-1;;;4397:48:4;;;;;;;;;;;;;;4464:186;;4529:5;4514:11;:20;4510:85;;;-1:-1:-1;4570:1:4;4011:713;-1:-1:-1;;;;4011:713:4:o;4510:85::-;4617:13;;;;;4464:186;-1:-1:-1;4373:3:4;;4337:328;;;;4693:23;;-1:-1:-1;;;4693:23:4;;;;;;;;;;;4585:100:11;1072:6:12;;-1:-1:-1;;;;;1072:6:12;681:10:1;1219:23:12;1211:68;;;;-1:-1:-1;;;1211:68:12;;;;;;;:::i;:::-;4659:18:11;;::::1;::::0;:7:::1;::::0;:18:::1;::::0;::::1;::::0;::::1;:::i;:::-;;4585:100:::0;:::o;8620:124:4:-;8684:7;8711:20;8723:7;8711:11;:20::i;:::-;:25;;8620:124;-1:-1:-1;;8620:124:4:o;6637:206::-;6701:7;-1:-1:-1;;;;;6725:19:4;;6721:60;;6753:28;;-1:-1:-1;;;6753:28:4;;;;;;;;;;;6721:60;-1:-1:-1;;;;;;6807:19:4;;;;;:12;:19;;;;;:27;-1:-1:-1;;;;;6807:27:4;;6637:206::o;1650:94:12:-;1072:6;;-1:-1:-1;;;;;1072:6:12;681:10:1;1219:23:12;1211:68;;;;-1:-1:-1;;;1211:68:12;;;;;;;:::i;:::-;1715:21:::1;1733:1;1715:9;:21::i;:::-;1650:94::o:0;4020:88:11:-;1072:6:12;;-1:-1:-1;;;;;1072:6:12;681:10:1;1219:23:12;1211:68;;;;-1:-1:-1;;;1211:68:12;;;;;;;:::i;:::-;4088:4:11::1;:12:::0;4020:88::o;8980:104:4:-;9036:13;9069:7;9062:14;;;;;:::i;3265:669:11:-;1072:6:12;;-1:-1:-1;;;;;1072:6:12;681:10:1;1219:23:12;1211:68;;;;-1:-1:-1;;;1211:68:12;;;;;;;:::i;:::-;3391:19:11::1;::::0;3423:103:::1;3447:11;:18;3443:1;:22;3423:103;;;3502:9;3512:1;3502:12;;;;;;;;:::i;:::-;;;;;;;3487:27;;;;;;;:::i;:::-;::::0;-1:-1:-1;3467:3:11;::::1;::::0;::::1;:::i;:::-;;;;3423:103;;;;3591:12;;3576:11;3558:15;;:29;;;;:::i;:::-;:45;;3536:139;;;::::0;-1:-1:-1;;;3536:139:11;;11331:2:14;3536:139:11::1;::::0;::::1;11313:21:14::0;11370:2;11350:18;;;11343:30;11409:34;11389:18;;;11382:62;-1:-1:-1;;;11460:18:14;;;11453:42;11512:19;;3536:139:11::1;11129:408:14::0;3536:139:11::1;3727:9;;3712:11;3696:13;:11;:13::i;:::-;:27;;;;:::i;:::-;:40;;3688:70;;;::::0;-1:-1:-1;;;3688:70:11;;12458:2:14;3688:70:11::1;::::0;::::1;12440:21:14::0;12497:2;12477:18;;;12470:30;-1:-1:-1;;;12516:18:14;;;12509:47;12573:18;;3688:70:11::1;12256:341:14::0;3688:70:11::1;3776:9;3771:115;3795:11;:18;3791:1;:22;3771:115;;;3835:39;3845:11;3857:1;3845:14;;;;;;;;:::i;:::-;;;;;;;3861:9;3871:1;3861:12;;;;;;;;:::i;:::-;;;;;;;3835:39;;:9;:39::i;:::-;3815:3:::0;::::1;::::0;::::1;:::i;:::-;;;;3771:115;;;;3915:11;3896:15;;:30;;;;;;;:::i;:::-;::::0;;;-1:-1:-1;;;;;3265:669:11:o;10590:279:4:-;-1:-1:-1;;;;;10681:24:4;;681:10:1;10681:24:4;10677:54;;;10714:17;;-1:-1:-1;;;10714:17:4;;;;;;;;;;;10677:54;681:10:1;10744:32:4;;;;:18;:32;;;;;;;;-1:-1:-1;;;;;10744:42:4;;;;;;;;;;;;:53;;-1:-1:-1;;10744:53:4;;;;;;;;;;10813:48;;9561:41:14;;;10744:42:4;;681:10:1;10813:48:4;;9534:18:14;10813:48:4;;;;;;;10590:279;;:::o;4380:112:11:-;1072:6:12;;-1:-1:-1;;;;;1072:6:12;681:10:1;1219:23:12;1211:68;;;;-1:-1:-1;;;1211:68:12;;;;;;;:::i;:::-;4457:18:11::1;:27:::0;4380:112::o;4809:178::-;1072:6:12;;-1:-1:-1;;;;;1072:6:12;681:10:1;1219:23:12;1211:68;;;;-1:-1:-1;;;1211:68:12;;;;;;;:::i;:::-;4883:49:11::1;::::0;4865:12:::1;::::0;4883:10:::1;::::0;4906:21:::1;::::0;4865:12;4883:49;4865:12;4883:49;4906:21;4883:10;:49:::1;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;4864:68;;;4951:7;4943:36;;;::::0;-1:-1:-1;;;4943:36:11;;12804:2:14;4943:36:11::1;::::0;::::1;12786:21:14::0;12843:2;12823:18;;;12816:30;-1:-1:-1;;;12862:18:14;;;12855:46;12918:18;;4943:36:11::1;12602:340:14::0;4943:36:11::1;4853:134;4809:178::o:0;1635:113::-;1072:6:12;;-1:-1:-1;;;;;1072:6:12;681:10:1;1219:23:12;1211:68;;;;-1:-1:-1;;;1211:68:12;;;;;;;:::i;:::-;1710:22:11::1;:30:::0;1635:113::o;11668:342:4:-;11835:28;11845:4;11851:2;11855:7;11835:9;:28::i;:::-;11879:48;11902:4;11908:2;11912:7;11921:5;11879:22;:48::i;:::-;11874:129;;11951:40;;-1:-1:-1;;;11951:40:4;;;;;;;;;;;11874:129;11668:342;;;;:::o;2478:300:11:-;2584:4;2626:144;2663:11;;2626:144;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;;2693:4:11;;2726:28;;-1:-1:-1;;2743:10:11;7698:2:14;7694:15;7690:53;2726:28:11;;;7678:66:14;2693:4:11;;-1:-1:-1;7760:12:14;;;-1:-1:-1;2726:28:11;7549:229:14;2626:144:11;2606:164;2478:300;-1:-1:-1;;;2478:300:11:o;9155:318:4:-;9228:13;9259:16;9267:7;9259;:16::i;:::-;9254:59;;9284:29;;-1:-1:-1;;;9284:29:4;;;;;;;;;;;9254:59;9326:21;9350:10;:8;:10::i;:::-;9326:34;;9384:7;9378:21;9403:1;9378:26;;:87;;;;;;;;;;;;;;;;;9431:7;9440:18;:7;:16;:18::i;:::-;9414:45;;;;;;;;;:::i;:::-;;;;;;;;;;;;;9371:94;9155:318;-1:-1:-1;;;9155:318:4:o;1816:336:11:-;1196:15;1173:19;;:38;;1151:112;;;;-1:-1:-1;;;1151:112:11;;13502:2:14;1151:112:11;;;13484:21:14;13541:2;13521:18;;;13514:30;13580:26;13560:18;;;13553:54;13624:18;;1151:112:11;13300:348:14;1151:112:11;1946:9:::1;816;;797:15;;785:9;769:13;:11;:13::i;:::-;:25;;;;:::i;:::-;:43;;;;:::i;:::-;:56;;747:122;;;::::0;-1:-1:-1;;;747:122:11;;13855:2:14;747:122:11::1;::::0;::::1;13837:21:14::0;13894:2;13874:18;;;13867:30;-1:-1:-1;;;13913:18:14;;;13906:46;13969:18;;747:122:11::1;13653:340:14::0;747:122:11::1;1985:9:::2;1996:12;;1006:1;994:9;:13;:38;;;;;1025:7;1012:9;:20;;994:38;972:112;;;::::0;-1:-1:-1;;;972:112:11;;13149:2:14;972:112:11::2;::::0;::::2;13131:21:14::0;13188:2;13168:18;;;13161:30;-1:-1:-1;;;13207:18:14;;;13200:54;13271:18;;972:112:11::2;12947:348:14::0;972:112:11::2;2059:9:::3;2047;;:21;;;;:::i;:::-;2034:9;:34;;2026:65;;;::::0;-1:-1:-1;;;2026:65:11;;10984:2:14;2026:65:11::3;::::0;::::3;10966:21:14::0;11023:2;11003:18;;;10996:30;-1:-1:-1;;;11042:18:14;;;11035:48;11100:18;;2026:65:11::3;10782:342:14::0;2026:65:11::3;2112:32;2122:10;2134:9;2112;:32::i;1899:192:12:-:0;1072:6;;-1:-1:-1;;;;;1072:6:12;681:10:1;1219:23:12;1211:68;;;;-1:-1:-1;;;1211:68:12;;;;;;;:::i;:::-;-1:-1:-1;;;;;1988:22:12;::::1;1980:73;;;::::0;-1:-1:-1;;;1980:73:12;;10221:2:14;1980:73:12::1;::::0;::::1;10203:21:14::0;10260:2;10240:18;;;10233:30;10299:34;10279:18;;;10272:62;-1:-1:-1;;;10350:18:14;;;10343:36;10396:19;;1980:73:12::1;10019:402:14::0;1980:73:12::1;2064:19;2074:8;2064:9;:19::i;4194:94:11:-:0;1072:6:12;;-1:-1:-1;;;;;1072:6:12;681:10:1;1219:23:12;1211:68;;;;-1:-1:-1;;;1211:68:12;;;;;;;:::i;:::-;4262:9:11::1;:18:::0;4194:94::o;868:190:10:-;993:4;1046;1017:25;1030:5;1037:4;1017:12;:25::i;:::-;:33;;868:190;-1:-1:-1;;;;868:190:10:o;12417:104:4:-;12486:27;12496:2;12500:8;12486:27;;;;;;;;;;;;:9;:27::i;12265:144::-;12322:4;12356:13;;-1:-1:-1;;;;;12356:13:4;12346:23;;:55;;;;-1:-1:-1;;12374:20:4;;;;:11;:20;;;;;:27;-1:-1:-1;;;12374:27:4;;;;12373:28;;12265:144::o;19481:196::-;19596:24;;;;:15;:24;;;;;;:29;;-1:-1:-1;;;;;;19596:29:4;-1:-1:-1;;;;;19596:29:4;;;;;;;;;19641:28;;19596:24;;19641:28;;;;;;;19481:196;;;:::o;14982:2112::-;15097:35;15135:20;15147:7;15135:11;:20::i;:::-;15210:18;;15097:58;;-1:-1:-1;15168:22:4;;-1:-1:-1;;;;;15194:34:4;681:10:1;-1:-1:-1;;;;;15194:34:4;;:101;;;-1:-1:-1;15262:18:4;;15245:50;;681:10:1;10940:164:4;:::i;15245:50::-;15194:154;;;-1:-1:-1;681:10:1;15312:20:4;15324:7;15312:11;:20::i;:::-;-1:-1:-1;;;;;15312:36:4;;15194:154;15168:181;;15367:17;15362:66;;15393:35;;-1:-1:-1;;;15393:35:4;;;;;;;;;;;15362:66;15465:4;-1:-1:-1;;;;;15443:26:4;:13;:18;;;-1:-1:-1;;;;;15443:26:4;;15439:67;;15478:28;;-1:-1:-1;;;15478:28:4;;;;;;;;;;;15439:67;-1:-1:-1;;;;;15521:16:4;;15517:52;;15546:23;;-1:-1:-1;;;15546:23:4;;;;;;;;;;;15517:52;15690:49;15707:1;15711:7;15720:13;:18;;;15690:8;:49::i;:::-;-1:-1:-1;;;;;16035:18:4;;;;;;;:12;:18;;;;;;;;:31;;-1:-1:-1;;16035:31:4;;;-1:-1:-1;;;;;16035:31:4;;;-1:-1:-1;;16035:31:4;;;;;;;16081:16;;;;;;;;;:29;;;;;;;;-1:-1:-1;16081:29:4;;;;;;;;;;;16127:20;;;:11;:20;;;;;;:30;;-1:-1:-1;;;;;;16172:61:4;;;;-1:-1:-1;;;16217:15:4;16172:61;;;;;;;;;;;16507:11;;;16537:24;;;;;:29;16507:11;;16537:29;16533:445;;16762:13;;-1:-1:-1;;;;;16762:13:4;16748:27;;16744:219;;;16832:18;;;16800:24;;;:11;:24;;;;;;;;:50;;16915:28;;;;-1:-1:-1;;;;;16873:70:4;-1:-1:-1;;;16873:70:4;-1:-1:-1;;;;;;16873:70:4;;;-1:-1:-1;;;;;16800:50:4;;;16873:70;;;;;;;16744:219;16010:979;17025:7;17021:2;-1:-1:-1;;;;;17006:27:4;17015:4;-1:-1:-1;;;;;17006:27:4;;;;;;;;;;;17044:42;15086:2008;;14982:2112;;;:::o;7475:1083::-;-1:-1:-1;;;;;;;;;;;;;;;;;;;;;;;;7641:13:4;;7585:7;;-1:-1:-1;;;;;7641:13:4;7634:20;;7630:861;;;7675:31;7709:17;;;:11;:17;;;;;;;;;7675:51;;;;;;;;;-1:-1:-1;;;;;7675:51:4;;;;-1:-1:-1;;;7675:51:4;;-1:-1:-1;;;;;7675:51:4;;;;;;;;-1:-1:-1;;;7675:51:4;;;;;;;;;;;;;;7745:731;;7795:14;;-1:-1:-1;;;;;7795:28:4;;7791:101;;7859:9;7475:1083;-1:-1:-1;;;7475:1083:4:o;7791:101::-;-1:-1:-1;;;8236:6:4;8281:17;;;;:11;:17;;;;;;;;;8269:29;;;;;;;;;-1:-1:-1;;;;;8269:29:4;;;;;-1:-1:-1;;;8269:29:4;;-1:-1:-1;;;;;8269:29:4;;;;;;;;-1:-1:-1;;;8269:29:4;;;;;;;;;;;;;8329:28;8325:109;;8397:9;7475:1083;-1:-1:-1;;;7475:1083:4:o;8325:109::-;8196:261;;;7656:835;7630:861;8519:31;;-1:-1:-1;;;8519:31:4;;;;;;;;;;;2099:173:12;2174:6;;;-1:-1:-1;;;;;2191:17:12;;;-1:-1:-1;;;;;;2191:17:12;;;;;;;2224:40;;2174:6;;;2191:17;2174:6;;2224:40;;2155:16;;2224:40;2144:128;2099:173;:::o;20242:790:4:-;20397:4;-1:-1:-1;;;;;20418:13:4;;1066:20:0;1114:8;20414:611:4;;20454:72;;-1:-1:-1;;;20454:72:4;;-1:-1:-1;;;;;20454:36:4;;;;;:72;;681:10:1;;20505:4:4;;20511:7;;20520:5;;20454:72;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;20454:72:4;;;;;;;;-1:-1:-1;;20454:72:4;;;;;;;;;;;;:::i;:::-;;;20450:520;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;20700:13:4;;20696:259;;20750:40;;-1:-1:-1;;;20750:40:4;;;;;;;;;;;20696:259;20905:6;20899:13;20890:6;20886:2;20882:15;20875:38;20450:520;-1:-1:-1;;;;;;20577:55:4;-1:-1:-1;;;20577:55:4;;-1:-1:-1;20570:62:4;;20414:611;-1:-1:-1;21009:4:4;20414:611;20242:790;;;;;;:::o;4693:108:11:-;4753:13;4786:7;4779:14;;;;;:::i;288:723:13:-;344:13;565:10;561:53;;-1:-1:-1;;592:10:13;;;;;;;;;;;;-1:-1:-1;;;592:10:13;;;;;288:723::o;561:53::-;639:5;624:12;680:78;687:9;;680:78;;713:8;;;;:::i;:::-;;-1:-1:-1;736:10:13;;-1:-1:-1;744:2:13;736:10;;:::i;:::-;;;680:78;;;768:19;800:6;-1:-1:-1;;;;;790:17:13;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;790:17:13;;768:39;;818:154;825:10;;818:154;;852:11;862:1;852:11;;:::i;:::-;;-1:-1:-1;921:10:13;929:2;921:5;:10;:::i;:::-;908:24;;:2;:24;:::i;:::-;895:39;;878:6;885;878:14;;;;;;;;:::i;:::-;;;;:56;-1:-1:-1;;;;;878:56:13;;;;;;;;-1:-1:-1;949:11:13;958:2;949:11;;:::i;:::-;;;818:154;;1420:701:10;1503:7;1546:4;1503:7;1561:523;1585:5;:12;1581:1;:16;1561:523;;;1619:20;1642:5;1648:1;1642:8;;;;;;;;:::i;:::-;;;;;;;1619:31;;1685:12;1669;:28;1665:408;;1822:44;;;;;;7940:19:14;;;7975:12;;;7968:28;;;8012:12;;1822:44:10;;;;;;;;;;;;1812:55;;;;;;1797:70;;1665:408;;;2012:44;;;;;;7940:19:14;;;7975:12;;;7968:28;;;8012:12;;2012:44:10;;;;;;;;;;;;2002:55;;;;;;1987:70;;1665:408;-1:-1:-1;1599:3:10;;;;:::i;:::-;;;;1561:523;;;-1:-1:-1;2101:12:10;1420:701;-1:-1:-1;;;1420:701:10:o;12884:163:4:-;13007:32;13013:2;13017:8;13027:5;13034:4;13445:20;13468:13;-1:-1:-1;;;;;13468:13:4;-1:-1:-1;;;;;13496:16:4;;13492:48;;13521:19;;-1:-1:-1;;;13521:19:4;;;;;;;;;;;13492:48;13555:13;13551:44;;13577:18;;-1:-1:-1;;;13577:18:4;;;;;;;;;;;13551:44;-1:-1:-1;;;;;13947:16:4;;;;;;:12;:16;;;;;;;;:44;;-1:-1:-1;;;;;;14006:49:4;;-1:-1:-1;;;;;13947:44:4;;;;;;;14006:49;;;;-1:-1:-1;;13947:44:4;;;;;;14006:49;;;;;;;;;;;;;;;;14072:25;;;:11;:25;;;;;:35;;-1:-1:-1;;;;;;14122:66:4;;;;-1:-1:-1;;;14172:15:4;14122:66;;;;;;;;;;;14072:25;;14257:328;14277:8;14273:1;:12;14257:328;;;14316:38;;14341:12;;-1:-1:-1;;;;;14316:38:4;;;14333:1;;14316:38;;14333:1;;14316:38;14377:4;:68;;;;;14386:59;14417:1;14421:2;14425:12;14439:5;14386:22;:59::i;:::-;14385:60;14377:68;14373:164;;;14477:40;;-1:-1:-1;;;14477:40:4;;;;;;;;;;;14373:164;14555:14;;;;;14287:3;14257:328;;;-1:-1:-1;14601:13:4;:37;;-1:-1:-1;;;;;;14601:37:4;-1:-1:-1;;;;;14601:37:4;;;;;;;;;;14660:60;11668:342;-1:-1:-1;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;:::o;:::-;;;;;;;;;;;;;;;14:406:14;78:5;-1:-1:-1;;;;;104:6:14;101:30;98:56;;;134:18;;:::i;:::-;172:57;217:2;196:15;;-1:-1:-1;;192:29:14;223:4;188:40;172:57;:::i;:::-;163:66;;252:6;245:5;238:21;292:3;283:6;278:3;274:16;271:25;268:45;;;309:1;306;299:12;268:45;358:6;353:3;346:4;339:5;335:16;322:43;412:1;405:4;396:6;389:5;385:18;381:29;374:40;14:406;;;;;:::o;425:173::-;493:20;;-1:-1:-1;;;;;542:31:14;;532:42;;522:70;;588:1;585;578:12;522:70;425:173;;;:::o;603:367::-;666:8;676:6;730:3;723:4;715:6;711:17;707:27;697:55;;748:1;745;738:12;697:55;-1:-1:-1;771:20:14;;-1:-1:-1;;;;;803:30:14;;800:50;;;846:1;843;836:12;800:50;883:4;875:6;871:17;859:29;;943:3;936:4;926:6;923:1;919:14;911:6;907:27;903:38;900:47;897:67;;;960:1;957;950:12;897:67;603:367;;;;;:::o;975:797::-;1027:5;1080:3;1073:4;1065:6;1061:17;1057:27;1047:55;;1098:1;1095;1088:12;1047:55;1134:6;1121:20;1160:4;1184:60;1200:43;1240:2;1200:43;:::i;:::-;1184:60;:::i;:::-;1266:3;1290:2;1285:3;1278:15;1318:2;1313:3;1309:12;1302:19;;1353:2;1345:6;1341:15;1405:3;1400:2;1394;1391:1;1387:10;1379:6;1375:23;1371:32;1368:41;1365:61;;;1422:1;1419;1412:12;1365:61;1444:1;1465;1475:268;1491:2;1486:3;1483:11;1475:268;;;1566:3;1553:17;1614:4;1607:5;1603:16;1596:5;1593:27;1583:55;;1634:1;1631;1624:12;1583:55;1651:18;;1689:12;;;;1721;;;;1513:1;1504:11;1475:268;;;-1:-1:-1;1761:5:14;;975:797;-1:-1:-1;;;;;;;;975:797:14:o;1777:186::-;1836:6;1889:2;1877:9;1868:7;1864:23;1860:32;1857:52;;;1905:1;1902;1895:12;1857:52;1928:29;1947:9;1928:29;:::i;1968:260::-;2036:6;2044;2097:2;2085:9;2076:7;2072:23;2068:32;2065:52;;;2113:1;2110;2103:12;2065:52;2136:29;2155:9;2136:29;:::i;:::-;2126:39;;2184:38;2218:2;2207:9;2203:18;2184:38;:::i;:::-;2174:48;;1968:260;;;;;:::o;2233:328::-;2310:6;2318;2326;2379:2;2367:9;2358:7;2354:23;2350:32;2347:52;;;2395:1;2392;2385:12;2347:52;2418:29;2437:9;2418:29;:::i;:::-;2408:39;;2466:38;2500:2;2489:9;2485:18;2466:38;:::i;:::-;2456:48;;2551:2;2540:9;2536:18;2523:32;2513:42;;2233:328;;;;;:::o;2566:666::-;2661:6;2669;2677;2685;2738:3;2726:9;2717:7;2713:23;2709:33;2706:53;;;2755:1;2752;2745:12;2706:53;2778:29;2797:9;2778:29;:::i;:::-;2768:39;;2826:38;2860:2;2849:9;2845:18;2826:38;:::i;:::-;2816:48;;2911:2;2900:9;2896:18;2883:32;2873:42;;2966:2;2955:9;2951:18;2938:32;-1:-1:-1;;;;;2985:6:14;2982:30;2979:50;;;3025:1;3022;3015:12;2979:50;3048:22;;3101:4;3093:13;;3089:27;-1:-1:-1;3079:55:14;;3130:1;3127;3120:12;3079:55;3153:73;3218:7;3213:2;3200:16;3195:2;3191;3187:11;3153:73;:::i;:::-;3143:83;;;2566:666;;;;;;;:::o;3237:347::-;3302:6;3310;3363:2;3351:9;3342:7;3338:23;3334:32;3331:52;;;3379:1;3376;3369:12;3331:52;3402:29;3421:9;3402:29;:::i;:::-;3392:39;;3481:2;3470:9;3466:18;3453:32;3528:5;3521:13;3514:21;3507:5;3504:32;3494:60;;3550:1;3547;3540:12;3494:60;3573:5;3563:15;;;3237:347;;;;;:::o;3589:254::-;3657:6;3665;3718:2;3706:9;3697:7;3693:23;3689:32;3686:52;;;3734:1;3731;3724:12;3686:52;3757:29;3776:9;3757:29;:::i;:::-;3747:39;3833:2;3818:18;;;;3805:32;;-1:-1:-1;;;3589:254:14:o;3848:1153::-;3964:6;3972;4025:2;4013:9;4004:7;4000:23;3996:32;3993:52;;;4041:1;4038;4031:12;3993:52;4081:9;4068:23;-1:-1:-1;;;;;4151:2:14;4143:6;4140:14;4137:34;;;4167:1;4164;4157:12;4137:34;4205:6;4194:9;4190:22;4180:32;;4250:7;4243:4;4239:2;4235:13;4231:27;4221:55;;4272:1;4269;4262:12;4221:55;4308:2;4295:16;4330:4;4354:60;4370:43;4410:2;4370:43;:::i;4354:60::-;4436:3;4460:2;4455:3;4448:15;4488:2;4483:3;4479:12;4472:19;;4519:2;4515;4511:11;4567:7;4562:2;4556;4553:1;4549:10;4545:2;4541:19;4537:28;4534:41;4531:61;;;4588:1;4585;4578:12;4531:61;4610:1;4601:10;;4620:169;4634:2;4631:1;4628:9;4620:169;;;4691:23;4710:3;4691:23;:::i;:::-;4679:36;;4652:1;4645:9;;;;;4735:12;;;;4767;;4620:169;;;-1:-1:-1;4808:5:14;-1:-1:-1;;4851:18:14;;4838:32;;-1:-1:-1;;4882:16:14;;;4879:36;;;4911:1;4908;4901:12;4879:36;;4934:61;4987:7;4976:8;4965:9;4961:24;4934:61;:::i;:::-;4924:71;;;3848:1153;;;;;:::o;5006:437::-;5092:6;5100;5153:2;5141:9;5132:7;5128:23;5124:32;5121:52;;;5169:1;5166;5159:12;5121:52;5209:9;5196:23;-1:-1:-1;;;;;5234:6:14;5231:30;5228:50;;;5274:1;5271;5264:12;5228:50;5313:70;5375:7;5366:6;5355:9;5351:22;5313:70;:::i;:::-;5402:8;;5287:96;;-1:-1:-1;5006:437:14;-1:-1:-1;;;;5006:437:14:o;5448:180::-;5507:6;5560:2;5548:9;5539:7;5535:23;5531:32;5528:52;;;5576:1;5573;5566:12;5528:52;-1:-1:-1;5599:23:14;;5448:180;-1:-1:-1;5448:180:14:o;5633:245::-;5691:6;5744:2;5732:9;5723:7;5719:23;5715:32;5712:52;;;5760:1;5757;5750:12;5712:52;5799:9;5786:23;5818:30;5842:5;5818:30;:::i;5883:249::-;5952:6;6005:2;5993:9;5984:7;5980:23;5976:32;5973:52;;;6021:1;6018;6011:12;5973:52;6053:9;6047:16;6072:30;6096:5;6072:30;:::i;6137:450::-;6206:6;6259:2;6247:9;6238:7;6234:23;6230:32;6227:52;;;6275:1;6272;6265:12;6227:52;6315:9;6302:23;-1:-1:-1;;;;;6340:6:14;6337:30;6334:50;;;6380:1;6377;6370:12;6334:50;6403:22;;6456:4;6448:13;;6444:27;-1:-1:-1;6434:55:14;;6485:1;6482;6475:12;6434:55;6508:73;6573:7;6568:2;6555:16;6550:2;6546;6542:11;6508:73;:::i;6777:505::-;6872:6;6880;6888;6941:2;6929:9;6920:7;6916:23;6912:32;6909:52;;;6957:1;6954;6947:12;6909:52;6993:9;6980:23;6970:33;;7054:2;7043:9;7039:18;7026:32;-1:-1:-1;;;;;7073:6:14;7070:30;7067:50;;;7113:1;7110;7103:12;7067:50;7152:70;7214:7;7205:6;7194:9;7190:22;7152:70;:::i;:::-;6777:505;;7241:8;;-1:-1:-1;7126:96:14;;-1:-1:-1;;;;6777:505:14:o;7287:257::-;7328:3;7366:5;7360:12;7393:6;7388:3;7381:19;7409:63;7465:6;7458:4;7453:3;7449:14;7442:4;7435:5;7431:16;7409:63;:::i;:::-;7526:2;7505:15;-1:-1:-1;;7501:29:14;7492:39;;;;7533:4;7488:50;;7287:257;-1:-1:-1;;7287:257:14:o;8035:470::-;8214:3;8252:6;8246:13;8268:53;8314:6;8309:3;8302:4;8294:6;8290:17;8268:53;:::i;:::-;8384:13;;8343:16;;;;8406:57;8384:13;8343:16;8440:4;8428:17;;8406:57;:::i;:::-;8479:20;;8035:470;-1:-1:-1;;;;8035:470:14:o;8928:488::-;-1:-1:-1;;;;;9197:15:14;;;9179:34;;9249:15;;9244:2;9229:18;;9222:43;9296:2;9281:18;;9274:34;;;9344:3;9339:2;9324:18;;9317:31;;;9122:4;;9365:45;;9390:19;;9382:6;9365:45;:::i;:::-;9357:53;8928:488;-1:-1:-1;;;;;;8928:488:14:o;9795:219::-;9944:2;9933:9;9926:21;9907:4;9964:44;10004:2;9993:9;9989:18;9981:6;9964:44;:::i;11542:356::-;11744:2;11726:21;;;11763:18;;;11756:30;11822:34;11817:2;11802:18;;11795:62;11889:2;11874:18;;11542:356::o;14180:275::-;14251:2;14245:9;14316:2;14297:13;;-1:-1:-1;;14293:27:14;14281:40;;-1:-1:-1;;;;;14336:34:14;;14372:22;;;14333:62;14330:88;;;14398:18;;:::i;:::-;14434:2;14427:22;14180:275;;-1:-1:-1;14180:275:14:o;14460:183::-;14520:4;-1:-1:-1;;;;;14545:6:14;14542:30;14539:56;;;14575:18;;:::i;:::-;-1:-1:-1;14620:1:14;14616:14;14632:4;14612:25;;14460:183::o;14648:128::-;14688:3;14719:1;14715:6;14712:1;14709:13;14706:39;;;14725:18;;:::i;:::-;-1:-1:-1;14761:9:14;;14648:128::o;14781:120::-;14821:1;14847;14837:35;;14852:18;;:::i;:::-;-1:-1:-1;14886:9:14;;14781:120::o;14906:168::-;14946:7;15012:1;15008;15004:6;15000:14;14997:1;14994:21;14989:1;14982:9;14975:17;14971:45;14968:71;;;15019:18;;:::i;:::-;-1:-1:-1;15059:9:14;;14906:168::o;15079:125::-;15119:4;15147:1;15144;15141:8;15138:34;;;15152:18;;:::i;:::-;-1:-1:-1;15189:9:14;;15079:125::o;15209:258::-;15281:1;15291:113;15305:6;15302:1;15299:13;15291:113;;;15381:11;;;15375:18;15362:11;;;15355:39;15327:2;15320:10;15291:113;;;15422:6;15419:1;15416:13;15413:48;;;-1:-1:-1;;15457:1:14;15439:16;;15432:27;15209:258::o;15472:380::-;15551:1;15547:12;;;;15594;;;15615:61;;15669:4;15661:6;15657:17;15647:27;;15615:61;15722:2;15714:6;15711:14;15691:18;15688:38;15685:161;;;15768:10;15763:3;15759:20;15756:1;15749:31;15803:4;15800:1;15793:15;15831:4;15828:1;15821:15;15685:161;;15472:380;;;:::o;15857:135::-;15896:3;-1:-1:-1;;15917:17:14;;15914:43;;;15937:18;;:::i;:::-;-1:-1:-1;15984:1:14;15973:13;;15857:135::o;15997:112::-;16029:1;16055;16045:35;;16060:18;;:::i;:::-;-1:-1:-1;16094:9:14;;15997:112::o;16114:127::-;16175:10;16170:3;16166:20;16163:1;16156:31;16206:4;16203:1;16196:15;16230:4;16227:1;16220:15;16246:127;16307:10;16302:3;16298:20;16295:1;16288:31;16338:4;16335:1;16328:15;16362:4;16359:1;16352:15;16378:127;16439:10;16434:3;16430:20;16427:1;16420:31;16470:4;16467:1;16460:15;16494:4;16491:1;16484:15;16510:127;16571:10;16566:3;16562:20;16559:1;16552:31;16602:4;16599:1;16592:15;16626:4;16623:1;16616:15;16642:131;-1:-1:-1;;;;;;16716:32:14;;16706:43;;16696:71;;16763:1;16760;16753:12

Swarm Source

ipfs://6dee3fc8f2422bee2e2d4351e7638eeaf49b00b891ef5e74bddedc07c5337133
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.