ETH Price: $2,536.61 (+1.24%)

Token

Lovers Under The Moon (LOVER)
 

Overview

Max Total Supply

117 LOVER

Holders

68

Market

Volume (24H)

N/A

Min Price (24H)

N/A

Max Price (24H)

N/A
Balance
1 LOVER
0xd9c25cfab63b80ae9ae52a42341d7c0a022b0cc7
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:
LoversNft

Compiler Version
v0.8.4+commit.c7e474f2

Optimization Enabled:
Yes with 200 runs

Other Settings:
default evmVersion
File 1 of 9 : LoversNft.sol
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.4;

import "erc721a/contracts/ERC721A.sol";
import "@openzeppelin/contracts/access/Ownable.sol";
import "@openzeppelin/contracts/security/ReentrancyGuard.sol";
import "@openzeppelin/contracts/security/Pausable.sol";
import "./ILoversNft.sol";
import "./Adminable.sol";

//  ╭╮╱╱╱╱╱╱╱╱╱╱╱╱╱╱╱╱╱╭╮╱╭╮╱╱╱╱╭╮╱╱╱╱╱╭━━━━┳╮╱╱╱╱╱╭━╮╭━╮
//  ┃┃╱╱╱╱╱╱╱╱╱╱╱╱╱╱╱╱╱┃┃╱┃┃╱╱╱╱┃┃╱╱╱╱╱┃╭╮╭╮┃┃╱╱╱╱╱┃┃╰╯┃┃
//  ┃┃╱╱╭━━┳╮╭┳━━┳━┳━━╮┃┃╱┃┣━╮╭━╯┣━━┳━╮╰╯┃┃╰┫╰━┳━━╮┃╭╮╭╮┣━━┳━━┳━╮
//  ┃┃╱╭┫╭╮┃╰╯┃┃━┫╭┫━━┫┃┃╱┃┃╭╮┫╭╮┃┃━┫╭╯╱╱┃┃╱┃╭╮┃┃━┫┃┃┃┃┃┃╭╮┃╭╮┃╭╮╮
//  ┃╰━╯┃╰╯┣╮╭┫┃━┫┃┣━━┃┃╰━╯┃┃┃┃╰╯┃┃━┫┃╱╱╱┃┃╱┃┃┃┃┃━┫┃┃┃┃┃┃╰╯┃╰╯┃┃┃┃
//  ╰━━━┻━━╯╰╯╰━━┻╯╰━━╯╰━━━┻╯╰┻━━┻━━┻╯╱╱╱╰╯╱╰╯╰┻━━╯╰╯╰╯╰┻━━┻━━┻╯╰╯

contract LoversNft is ILoversNft, ERC721A, Adminable, ReentrancyGuard, Pausable {
    /**
     * Maximum number of supply
     */
    uint256 public maximumSupply;

    /**
     *  To make token uri immutable permanently
     */
    bool public permanent;

    modifier notPermanent() {
        if (permanent) {
            revert ImmutableState();
        }
        _;
    }

    /**
     *  Minting rounds
     */
    mapping(uint16 => Round) rounds;

    uint256 public revealBlockOffset = 3000;

    /**
     *  Current round number
     */
    uint16 public currentRoundNumber;

    address private _receiver;

    /**
     * Base uri for token uri
     */
    string public baseURI;

    /**
     * Default unrevealed uri
     */
    string private defaultUnrevealedURI;

    constructor(
        address receiver,
        uint256 maxSupply,
        string memory _baseUri,
        string memory _defaultUnrevealedURI,
        address admin_
    ) ERC721A("Lovers Under The Moon", "LOVER")
        Adminable(admin_)
        Pausable()
    {
        _receiver = receiver;
        maximumSupply = maxSupply;
        baseURI = _baseUri;
        defaultUnrevealedURI = _defaultUnrevealedURI;
    }

    /**
     *  Create a new minting round
     *  For owner only
     */
    function newRound(
        MintingType mintingType,
        uint256 supply,   // total supply for the round
        uint256 startTime, // round starting time
        uint256 endTime, // round ending time
        uint256 mintingFee, // minting fee
        uint16 maxMintingQuantity, // maximum number of minting quantity per address
        bool revealed // reveal image or not
    ) external onlyOwner notPermanent {
        // wrap-up the existing round
        if (currentRoundNumber > 0) {
            endRound();
        }

        uint256 maxMintingId = _nextTokenId() + supply - 1;

        // the maxMintingId of new round can NOT exceed the maximumSupply
        if (maxMintingId > maximumSupply) {
            revert BadRequest("maxMintingId exceed the maximumSupply");
        }

        if (startTime >= endTime) {
            revert BadRequest("endTime should be bigger");
        }

        uint16 newRoundNumber = ++currentRoundNumber;

        rounds[newRoundNumber] = Round({
            roundNumber: newRoundNumber,
            mintingType: mintingType,
            startTime: startTime,
            endTime: endTime,
            maxMintingQuantity: maxMintingQuantity,
            mintingFee: mintingFee,

            maxMintingId: maxMintingId,
            startId: _nextTokenId(),
            lastMintedId: 0,
            tokenURIPrefix: "",
            revealed: revealed,
            revealBlockNumber: 0,
            randomSelection: 0,
            closedTime: 0
        });

        emit NewRoundCreated(newRoundNumber);
    }

    /**
     * End the current round
     * For owner and admin
     */
    function endRound() public onlyOwnerOrAdmin {
        Round storage currentRound = rounds[currentRoundNumber];
        currentRound.lastMintedId = _nextTokenId() - 1;
        currentRound.closedTime = block.timestamp;

        emit RoundEnded(currentRoundNumber);
    }

    /**
     *  Get the detail of the current round
     */
    function getCurrentRound() public view returns (Round memory) {
        return getRound(currentRoundNumber);
    }

    /**
     *  Get round detail
     */
    function getRound(uint16 roundNumber) public view returns (Round memory) {
        return rounds[roundNumber];
    }

    /**
     *  Get the STATE of the contract
     */
    function getState() public view returns (State) {
        if (currentRoundNumber == 0) {
            return State.DEPLOYED;
        }

        Round memory currentRound = rounds[currentRoundNumber];

        uint256 closedTime = currentRound.closedTime;
        uint256 startTime = currentRound.startTime;
        uint256 endTime = currentRound.endTime;
        uint256 currentTime = block.timestamp;

        State currentState;

        if (closedTime != 0) {
            currentState = State.END_MINTING;
        } else {
            if (currentTime < startTime) {
                currentState = State.PREPARE_MINTING;
            } else if (currentTime < endTime) {
                if (currentRound.maxMintingId > totalSupply()) {
                    currentState = State.ON_MINTING;
                } else {
                    currentState = State.END_MINTING;
                }
            } else {
                currentState = State.END_MINTING;
            }
        }

        return currentState;
    }

    /**
     * Minting
     */
    function mint(uint256 quantity)
        external
        payable
        nonReentrant
        whenNotPaused
        notPermanent
    {
        Round memory currentRound = getRound(currentRoundNumber);

        _sanityCheckForMinting(currentRound, quantity);

        require(safeMint(msg.sender, quantity));
    }

    /**
     * Admin minting
     * Only for owner or admin
     */
    function mint(address to, uint256 quantity)
        external
        payable
        onlyOwnerOrAdmin
        notPermanent
    {
        Round memory currentRound = rounds[currentRoundNumber];

        _sanityCheckForMinting(currentRound, quantity);

        require(safeMint(to, quantity));
    }

    /**
     * Change Minting type
     * For only owner or admin
     */
    function changeMintingType(uint16 roundNumber, MintingType mintingType) 
        external onlyOwnerOrAdmin notPermanent {
        Round storage round = rounds[roundNumber];
        round.mintingType = mintingType;

        emit MintingTypeChanged(roundNumber, mintingType);
    }

    /**
     *  Change the maximum minting id for the specified round
     *  For owner and admin
     */
    function changeMaxMintingId(uint16 roundNumber, uint256 maxId)
        public onlyOwnerOrAdmin notPermanent
    {
        if (maxId < _nextTokenId()) {
            revert MaxMintingIdLowerThanCurrentId();
        }

        if (maxId > maximumSupply) {
            revert ExceedMaximumSupply();
        }

        Round storage round = rounds[roundNumber];
        round.maxMintingId = maxId;

        emit MaxMintingIdChanged(roundNumber, maxId);
    }

    /**
     *  Change minting start time for the specified round
     *  Only for owner and admin
     */
    function changeStartTime(uint16 roundNumber, uint256 time)
        public onlyOwnerOrAdmin notPermanent
    {
        Round storage round = rounds[roundNumber];
        round.startTime = time;

        emit StartTimeChanged(roundNumber, time);
    }

    /**
     *  Change minting end time for the specified round
     *  Only for owner and admin
     */
    function changeEndTime(uint16 roundNumber, uint256 time)
        public onlyOwnerOrAdmin notPermanent
    {
        Round storage round = rounds[roundNumber];
        round.endTime = time;

        emit EndTimeChanged(roundNumber, time);
    }

    /**
     *  Change maximum minting quantity for an account
     *  For owner and admin
     */
    function changeMaxMintingQuantity(uint16 roundNumber, uint16 quantity)
        public onlyOwnerOrAdmin notPermanent
    {
        Round storage round = rounds[roundNumber];
        round.maxMintingQuantity = quantity;

        emit MaxMintingQuantityChanged(roundNumber, quantity);
    }

    /**
     *  Set the minting for the specified round
     *  OnlyOwner functions
     */
    function changeMintingFee(uint16 roundNumber, uint256 fee)
        external onlyOwnerOrAdmin notPermanent
    {
        Round storage round = rounds[roundNumber];
        round.mintingFee = fee;

        emit MintingFeeChanged(roundNumber, fee);
    }

    /**
     *  Set the token uri prefix for the specified round
     *  For owner and admin
     */
    function setTokenURIPrefix(uint16 roundNumber, string memory prefix)
        public onlyOwnerOrAdmin notPermanent
    {
        Round storage round = rounds[roundNumber];
        round.tokenURIPrefix = prefix;

        emit TokenURIPrefixUpdated(roundNumber, prefix);
    }

    function safeMint(address receiver, uint256 quantity)
        private
        returns (bool)
    {
        _safeMint(receiver, quantity);
        return true;
    }



    /**
     * Minting and transfer tokens
     * Only for owner
     */
    function adminMintTo(address[] calldata tos, uint256[] calldata quantities)
        external
        payable
        onlyOwnerOrAdmin 
        notPermanent
    {
        uint256 length = tos.length;
        if (length != quantities.length) {
            revert BadRequest("Input size not match");
        }

        uint256 totalQuantity = 0;

        for (uint256 i = 0; i < tos.length; i++) {
            totalQuantity += quantities[i];
        }

        Round memory currentRound = rounds[currentRoundNumber];
        // check if minting does not exceed the maximum tokens for the round
        uint256 maxMintingId = currentRound.maxMintingId;

        if (!_isTokenAvailable(totalQuantity, maxMintingId)) {
            revert ExceedMaximumForTheRound();
        }

        for (uint256 i = 0; i < length; i++) {
            require(safeMint(tos[i], quantities[i]));
        }
    }

    /**
     * Transfer multiple tokens to an account
     */
    function transferBatch(uint256[] calldata tokenIds, address to)
        external
        nonReentrant
        notPermanent
    {
        for (uint256 i = 0; i < tokenIds.length; i++) {
            safeTransferFrom(_msgSender(), to, tokenIds[i]);
        }
    }

    /**
     * Get token uri
     */
    function tokenURI(uint256 tokenId)
        public
        view
        override
        returns (string memory)
    {
        Round memory round = _getRoundByTokenId(tokenId);

        if (
            round.revealed &&
            keccak256(abi.encodePacked(round.tokenURIPrefix)) !=
            keccak256(abi.encodePacked(""))
        ) {
            return
                string(
                    abi.encodePacked(
                        baseURI,
                        round.tokenURIPrefix,
                        "/",
                        _toString(tokenId),
                        ".json"
                    )
                );
        } else {
            return
                string(
                    abi.encodePacked(
                        defaultUnrevealedURI,
                        _toString(tokenId),
                        ".json"
                    )
                );
        }
    }

    /**
     *  Trigger reveal process for the specified round
     *  For owner and admin
     */
    function setRevealBlock(uint16 roundNumber)
        public onlyOwnerOrAdmin notPermanent
    {
        Round storage round = rounds[roundNumber];

        if (round.lastMintedId == 0) {
            revert BadRequest("Round should be closed");
        }

        round.revealBlockNumber = block.number + revealBlockOffset;

        emit SetRevealBlock(round.revealBlockNumber);
    }

    /**
     *  Set random selection number based on entropy
     *  It set the reveal on
     */
    function setRandomSelection(uint16 roundNumber) public notPermanent {
        Round storage round = rounds[roundNumber];

        if (round.revealed) {
            revert AlreadyRevealed();
        }

        uint256 revealBlockNumber = round.revealBlockNumber;

        if (revealBlockNumber > block.number) {
            revert BadRequest("Random selection is not ready");
        }

        bytes32 entropy;

        if (blockhash(revealBlockNumber - 1) != 0) {
            entropy = keccak256(
                abi.encodePacked(
                    blockhash(revealBlockNumber),
                    blockhash(revealBlockNumber - 1),
                    block.timestamp
                )
            );
        } else {
            entropy = keccak256(
                abi.encodePacked(
                    blockhash(block.number - 1),
                    blockhash(block.number - 2),
                    block.timestamp
                )
            );
        }

        round.revealed = true;

        uint256 selected = _getRandomInRange(
            entropy,
            round.startId,
            round.lastMintedId
        );

        round.randomSelection = selected;

        emit Revealed(roundNumber);
    }

    /**
     *  Internal function to override the ERC721A _baseURI()
     */
    function _baseURI() internal view override returns (string memory) {
        return baseURI;
    }

    /**
     *  Get the base uri
     */
    function getBaseURI() external view returns (string memory) {
        return _baseURI();
    }

    /**
     * Set the base uri
     * Only for the owner
     */
    function setBaseURI(string memory _newBaseURI) external onlyOwner notPermanent {
        baseURI = _newBaseURI;
        emit BaseURIUpdated(_newBaseURI);
    }

    /**
     * Set the default unrevealed uri
     * Only for the owner
     */
    function setDefaultUnrevealedURI(string memory _defaultUnrevealedURI)
        external onlyOwner notPermanent
    {
        defaultUnrevealedURI = _defaultUnrevealedURI;
        emit DefaultUnrevealedURIUpdated(_defaultUnrevealedURI);
    }

    /**
     *  Paused the contract
     *  Only for the owner
     */
    function pause() external onlyOwner {
        _pause();
    }

    /**
     *  Unpause the contract
     *  Only for the owner
     */
    function unpause() external onlyOwner {
        _unpause();
    }

    /**
     *  Set the receiver
     *  Only for the owner
     */
    function setReceiver(address receiver) external onlyOwner {
        _receiver = receiver;
    }

    /**
     * Withdraw the balance in the contract
     */
    function withdraw() external payable onlyOwner {
        uint256 amount = address(this).balance;
        (bool success, ) = _receiver.call{value: amount}("");
        if (!success) {
            revert FailedToSendBalance();
        }
        emit Withdraw(_receiver, amount);
    }

    /**
     *  Fallback function
     */
    fallback() external payable {
        emit Received(_msgSender(), msg.value);
    }

    /**
     *  Fallback function
     */
    receive() external payable {
        emit Received(_msgSender(), msg.value);
    }

    /**
     * Set the 'perment' to true to prevent token uri change
     */
    function setPermanent() external onlyOwner {
        permanent = true;
    }

    function updateRevealBlockOffset(uint256 newOffset) external onlyOwner notPermanent {
        revealBlockOffset = newOffset;
    }

    function changeMaximumSupply(uint256 newMaximum) external onlyOwner notPermanent {
        maximumSupply = newMaximum;
    }

    /**
     *  Sanity check before transfer tokens
     */
    function _sanityCheckForMinting(Round memory currentRound, uint256 quantity)
        private
    {
        State currentState = getState();

        if (currentState != State.ON_MINTING) {
            revert MintingNotAllowed();
        }

        if (currentRound.mintingType == MintingType.ADMIN_ONLY 
            && owner() != _msgSender() && admin() != _msgSender()) {
            revert NotOwnerNorAdmin();
        }

        // check if not exceeding the maxAllowedMintingQuantity
        if (currentRound.maxMintingQuantity < quantity) {
            revert ExceedAllowedQuantity(
                currentRound.maxMintingQuantity
            );
        }

        // check if minting does not exceed the maximum tokens for the round
        if (!_isTokenAvailable(quantity, currentRound.maxMintingId)) {
            revert ExceedMaximumForTheRound();
        }

        // check if proper fee is received (except owner or admin)
        if (currentRound.mintingType == MintingType.REGULAR 
            && owner() != _msgSender() && admin() != _msgSender()) {
            uint256 neededFee = currentRound.mintingFee * quantity;
            if (neededFee != msg.value) revert NoMatchingFee();
        }
    }

    /**
     *  Check the availability of tokens mintable for the round
     */
    function _isTokenAvailable(uint256 quantity, uint256 maxId)
        private
        view
        returns (bool)
    {
        // check if minting does not exceed the maximum tokens for the round
        if ((_nextTokenId() + quantity) > (maxId + 1)) {
            return false;
        }

        return true;
    }

    /**
     *  Private function to get the round number with token id
     */
    function _getRoundByTokenId(uint256 tokenId)
        private
        view
        returns (Round memory r)
    {
        if (!_exists(tokenId)) revert NonExistingToken(tokenId);

        uint16 roundNumber = 1;

        while (roundNumber <= currentRoundNumber) {
            r = rounds[roundNumber];
            uint256 roundMax = r.lastMintedId != 0
                ? r.lastMintedId
                : r.maxMintingId;
            if (tokenId > roundMax) {
                roundNumber++;
                continue;
            }
            return r;
        }
    }

    function _getRandomInRange(
        bytes32 hash,
        uint256 begin,
        uint256 end
    ) private pure returns (uint256) {
        uint256 diff = end - begin + 1;
        return (uint256(hash) % diff) + begin;
    }

    function _startTokenId() internal pure override returns (uint256) {
        return 1;
    }
}

File 2 of 9 : ERC721A.sol
// SPDX-License-Identifier: MIT
// ERC721A Contracts v4.0.0
// Creator: Chiru Labs

pragma solidity ^0.8.4;

import './IERC721A.sol';

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

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

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

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

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

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

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

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

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

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

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

    // Token name
    string private _name;

    // Token symbol
    string private _symbol;

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

    /**
     * Returns the unpacked `TokenOwnership` struct from `packed`.
     */
    function _unpackedOwnership(uint256 packed) private pure returns (TokenOwnership memory ownership) {
        ownership.addr = address(uint160(packed));
        ownership.startTimestamp = uint64(packed >> BITPOS_START_TIMESTAMP);
        ownership.burned = packed & BITMASK_BURNED != 0;
    }

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

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

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

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

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

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

    /**
     * @dev See {IERC721Metadata-tokenURI}.
     */
    function tokenURI(uint256 tokenId) public view virtual override returns (string memory) {
        if (!_exists(tokenId)) revert URIQueryForNonexistentToken();

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

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

    /**
     * @dev Casts the address to uint256 without masking.
     */
    function _addressToUint256(address value) private pure returns (uint256 result) {
        assembly {
            result := value
        }
    }

    /**
     * @dev Casts the boolean to uint256 without branching.
     */
    function _boolToUint256(bool value) private pure returns (uint256 result) {
        assembly {
            result := value
        }
    }

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

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

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

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

        return _tokenApprovals[tokenId];
    }

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

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

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

    /**
     * @dev See {IERC721-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 (to.code.length != 0)
            if (!_checkContractOnERC721Received(from, to, tokenId, _data)) {
                revert TransferToNonERC721ReceiverImplementer();
            }
    }

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

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

    /**
     * @dev Safely mints `quantity` tokens and transfers them to `to`.
     *
     * Requirements:
     *
     * - If `to` refers to a smart contract, it must implement
     *   {IERC721Receiver-onERC721Received}, which is called for each safe transfer.
     * - `quantity` must be greater than 0.
     *
     * Emits a {Transfer} event.
     */
    function _safeMint(
        address to,
        uint256 quantity,
        bytes memory _data
    ) 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 > 1.8e19 (2**64) - 1
        // updatedIndex overflows if _currentIndex + quantity > 1.2e77 (2**256) - 1
        unchecked {
            // Updates:
            // - `balance += quantity`.
            // - `numberMinted += quantity`.
            //
            // We can directly add to the balance and number minted.
            _packedAddressData[to] += quantity * ((1 << BITPOS_NUMBER_MINTED) | 1);

            // Updates:
            // - `address` to the owner.
            // - `startTimestamp` to the timestamp of minting.
            // - `burned` to `false`.
            // - `nextInitialized` to `quantity == 1`.
            _packedOwnerships[startTokenId] =
                _addressToUint256(to) |
                (block.timestamp << BITPOS_START_TIMESTAMP) |
                (_boolToUint256(quantity == 1) << BITPOS_NEXT_INITIALIZED);

            uint256 updatedIndex = startTokenId;
            uint256 end = updatedIndex + quantity;

            if (to.code.length != 0) {
                do {
                    emit Transfer(address(0), to, updatedIndex);
                    if (!_checkContractOnERC721Received(address(0), to, updatedIndex++, _data)) {
                        revert TransferToNonERC721ReceiverImplementer();
                    }
                } while (updatedIndex < end);
                // Reentrancy protection
                if (_currentIndex != startTokenId) revert();
            } else {
                do {
                    emit Transfer(address(0), to, updatedIndex++);
                } while (updatedIndex < end);
            }
            _currentIndex = updatedIndex;
        }
        _afterTokenTransfers(address(0), to, startTokenId, quantity);
    }

    /**
     * @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) 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 > 1.8e19 (2**64) - 1
        // updatedIndex overflows if _currentIndex + quantity > 1.2e77 (2**256) - 1
        unchecked {
            // Updates:
            // - `balance += quantity`.
            // - `numberMinted += quantity`.
            //
            // We can directly add to the balance and number minted.
            _packedAddressData[to] += quantity * ((1 << BITPOS_NUMBER_MINTED) | 1);

            // Updates:
            // - `address` to the owner.
            // - `startTimestamp` to the timestamp of minting.
            // - `burned` to `false`.
            // - `nextInitialized` to `quantity == 1`.
            _packedOwnerships[startTokenId] =
                _addressToUint256(to) |
                (block.timestamp << BITPOS_START_TIMESTAMP) |
                (_boolToUint256(quantity == 1) << BITPOS_NEXT_INITIALIZED);

            uint256 updatedIndex = startTokenId;
            uint256 end = updatedIndex + quantity;

            do {
                emit Transfer(address(0), to, updatedIndex++);
            } while (updatedIndex < end);

            _currentIndex = 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 {
        uint256 prevOwnershipPacked = _packedOwnershipOf(tokenId);

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

        bool isApprovedOrOwner = (_msgSenderERC721A() == from ||
            isApprovedForAll(from, _msgSenderERC721A()) ||
            getApproved(tokenId) == _msgSenderERC721A());

        if (!isApprovedOrOwner) revert TransferCallerNotOwnerNorApproved();
        if (to == address(0)) revert TransferToZeroAddress();

        _beforeTokenTransfers(from, to, tokenId, 1);

        // Clear approvals from the previous owner.
        delete _tokenApprovals[tokenId];

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

            // Updates:
            // - `address` to the next owner.
            // - `startTimestamp` to the timestamp of transfering.
            // - `burned` to `false`.
            // - `nextInitialized` to `true`.
            _packedOwnerships[tokenId] =
                _addressToUint256(to) |
                (block.timestamp << BITPOS_START_TIMESTAMP) |
                BITMASK_NEXT_INITIALIZED;

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

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

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

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

        address from = address(uint160(prevOwnershipPacked));

        if (approvalCheck) {
            bool isApprovedOrOwner = (_msgSenderERC721A() == from ||
                isApprovedForAll(from, _msgSenderERC721A()) ||
                getApproved(tokenId) == _msgSenderERC721A());

            if (!isApprovedOrOwner) revert TransferCallerNotOwnerNorApproved();
        }

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

        // Clear approvals from the previous owner.
        delete _tokenApprovals[tokenId];

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

            // Updates:
            // - `address` to the last owner.
            // - `startTimestamp` to the timestamp of burning.
            // - `burned` to `true`.
            // - `nextInitialized` to `true`.
            _packedOwnerships[tokenId] =
                _addressToUint256(from) |
                (block.timestamp << BITPOS_START_TIMESTAMP) |
                BITMASK_BURNED | 
                BITMASK_NEXT_INITIALIZED;

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

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

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

    /**
     * @dev Internal function to invoke {IERC721Receiver-onERC721Received} on a target contract.
     *
     * @param from address representing the previous owner of the given token ID
     * @param to target address that will receive the tokens
     * @param tokenId uint256 ID of the token to be transferred
     * @param _data bytes optional data to send along with the call
     * @return bool whether the call correctly returned the expected magic value
     */
    function _checkContractOnERC721Received(
        address from,
        address to,
        uint256 tokenId,
        bytes memory _data
    ) private returns (bool) {
        try ERC721A__IERC721Receiver(to).onERC721Received(_msgSenderERC721A(), from, tokenId, _data) returns (
            bytes4 retval
        ) {
            return retval == ERC721A__IERC721Receiver(to).onERC721Received.selector;
        } catch (bytes memory reason) {
            if (reason.length == 0) {
                revert TransferToNonERC721ReceiverImplementer();
            } else {
                assembly {
                    revert(add(32, reason), mload(reason))
                }
            }
        }
    }

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

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

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

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

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

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

File 3 of 9 : Ownable.sol
// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts v4.4.1 (access/Ownable.sol)

pragma solidity ^0.8.0;

import "../utils/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() {
        _transferOwnership(_msgSender());
    }

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

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

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

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

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

File 4 of 9 : ReentrancyGuard.sol
// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts v4.4.1 (security/ReentrancyGuard.sol)

pragma solidity ^0.8.0;

/**
 * @dev Contract module that helps prevent reentrant calls to a function.
 *
 * Inheriting from `ReentrancyGuard` will make the {nonReentrant} modifier
 * available, which can be applied to functions to make sure there are no nested
 * (reentrant) calls to them.
 *
 * Note that because there is a single `nonReentrant` guard, functions marked as
 * `nonReentrant` may not call one another. This can be worked around by making
 * those functions `private`, and then adding `external` `nonReentrant` entry
 * points to them.
 *
 * TIP: If you would like to learn more about reentrancy and alternative ways
 * to protect against it, check out our blog post
 * https://blog.openzeppelin.com/reentrancy-after-istanbul/[Reentrancy After Istanbul].
 */
abstract contract ReentrancyGuard {
    // Booleans are more expensive than uint256 or any type that takes up a full
    // word because each write operation emits an extra SLOAD to first read the
    // slot's contents, replace the bits taken up by the boolean, and then write
    // back. This is the compiler's defense against contract upgrades and
    // pointer aliasing, and it cannot be disabled.

    // The values being non-zero value makes deployment a bit more expensive,
    // but in exchange the refund on every call to nonReentrant will be lower in
    // amount. Since refunds are capped to a percentage of the total
    // transaction's gas, it is best to keep them low in cases like this one, to
    // increase the likelihood of the full refund coming into effect.
    uint256 private constant _NOT_ENTERED = 1;
    uint256 private constant _ENTERED = 2;

    uint256 private _status;

    constructor() {
        _status = _NOT_ENTERED;
    }

    /**
     * @dev Prevents a contract from calling itself, directly or indirectly.
     * Calling a `nonReentrant` function from another `nonReentrant`
     * function is not supported. It is possible to prevent this from happening
     * by making the `nonReentrant` function external, and making it call a
     * `private` function that does the actual work.
     */
    modifier nonReentrant() {
        // On the first call to nonReentrant, _notEntered will be true
        require(_status != _ENTERED, "ReentrancyGuard: reentrant call");

        // Any calls to nonReentrant after this point will fail
        _status = _ENTERED;

        _;

        // By storing the original value once again, a refund is triggered (see
        // https://eips.ethereum.org/EIPS/eip-2200)
        _status = _NOT_ENTERED;
    }
}

File 5 of 9 : Pausable.sol
// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts v4.4.1 (security/Pausable.sol)

pragma solidity ^0.8.0;

import "../utils/Context.sol";

/**
 * @dev Contract module which allows children to implement an emergency stop
 * mechanism that can be triggered by an authorized account.
 *
 * This module is used through inheritance. It will make available the
 * modifiers `whenNotPaused` and `whenPaused`, which can be applied to
 * the functions of your contract. Note that they will not be pausable by
 * simply including this module, only once the modifiers are put in place.
 */
abstract contract Pausable is Context {
    /**
     * @dev Emitted when the pause is triggered by `account`.
     */
    event Paused(address account);

    /**
     * @dev Emitted when the pause is lifted by `account`.
     */
    event Unpaused(address account);

    bool private _paused;

    /**
     * @dev Initializes the contract in unpaused state.
     */
    constructor() {
        _paused = false;
    }

    /**
     * @dev Returns true if the contract is paused, and false otherwise.
     */
    function paused() public view virtual returns (bool) {
        return _paused;
    }

    /**
     * @dev Modifier to make a function callable only when the contract is not paused.
     *
     * Requirements:
     *
     * - The contract must not be paused.
     */
    modifier whenNotPaused() {
        require(!paused(), "Pausable: paused");
        _;
    }

    /**
     * @dev Modifier to make a function callable only when the contract is paused.
     *
     * Requirements:
     *
     * - The contract must be paused.
     */
    modifier whenPaused() {
        require(paused(), "Pausable: not paused");
        _;
    }

    /**
     * @dev Triggers stopped state.
     *
     * Requirements:
     *
     * - The contract must not be paused.
     */
    function _pause() internal virtual whenNotPaused {
        _paused = true;
        emit Paused(_msgSender());
    }

    /**
     * @dev Returns to normal state.
     *
     * Requirements:
     *
     * - The contract must be paused.
     */
    function _unpause() internal virtual whenPaused {
        _paused = false;
        emit Unpaused(_msgSender());
    }
}

File 6 of 9 : ILoversNft.sol
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.4;

interface ILoversNft {
    error MintingNotAllowed();
    error ExceedMaximumForTheRound();
    error ExceedMaximumSupply();
    error ExceedAllowedQuantity(uint16 maximumAllowedQuantity);
    error MaxMintingIdLowerThanCurrentId();
    error NoMatchingFee();
    error NonExistingToken(uint256 tokenId);
    error FailedToSendBalance();
    error BadRequest(string reason);
    error AlreadyRevealed();
    error ImmutableState();

    enum MintingType {
        REGULAR,
        ADMIN_ONLY
    }

    enum State {
        DEPLOYED,
        PREPARE_MINTING,
        ON_MINTING,
        END_MINTING,
        ALL_MINTING_DONE
    }

    struct Round {
        uint16 roundNumber; // round number
        MintingType mintingType;
        uint256 startTime; // round start time
        uint256 endTime; // round end time (if zero, no end time)
        uint16 maxMintingQuantity; // max number of tokens for an account (if zero, no limit)
        uint256 mintingFee; // minting for the round

        uint256 maxMintingId; // maximum token id for this round
        uint256 startId; // beginning of the tokenId for the round
        uint256 lastMintedId; // last token id actually minted before the next round starts
        string tokenURIPrefix; // directory hash value for token uri
        bool revealed; // released token is revealed or not
        uint256 revealBlockNumber; // blocknubmer which entropy will calculated
        uint256 randomSelection;
        uint256 closedTime;  // round closed timestamp
    }

    event NewRoundCreated(uint16 roundNumber);
    event RoundEnded(uint16 roundNumber);
    event MintingTypeChanged(uint16 roundNumber, MintingType mintingType);
    event MaxMintingIdChanged(uint16 roundNumber, uint256 maxId);
    event StartTimeChanged(uint16 roundNumber, uint256 time);
    event EndTimeChanged(uint16 roundNumber, uint256 time);
    event MaxMintingQuantityChanged(uint16 roundNumber, uint16 count);
    event MintingFeeChanged(uint16 roundNumber, uint256 newFee);
    event AdminUpdated(uint16 roundNumber, address admin);
    event TokenURIPrefixUpdated(uint16 roundNumber, string prefix);
    event SetRevealBlock(uint256 revealBlockNumber);
    event Revealed(uint16 roundNumber);
    event BaseURIUpdated(string baseURI);
    event DefaultUnrevealedURIUpdated(string defaultUnrevealedURI);
    event Received(address called, uint256 amount);
    event Withdraw(address receiver, uint256 amount);
}

File 7 of 9 : Adminable.sol
// SPDX-License-Identifier: MIT

pragma solidity ^0.8.0;

import "@openzeppelin/contracts/access/Ownable.sol";

abstract contract Adminable is Ownable {
    address private _admin;

    event AdminTransferred(address indexed previousAdmin, address indexed newAdmin);

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

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

    error NotAdmin();
    error NotOwnerNorAdmin();

    /**
     * @dev Throws if called by any account other than the admin.
     */
    modifier onlyAdmin() {
        if(admin() != _msgSender()) {
            revert NotAdmin();
        }
        _;
        // require(admin() == _msgSender(), "Adminable: caller is not the admin");
        // _;
    }

    /**
     * @dev Throws if called by any account other than the owner nor the admin.
     */
    modifier onlyOwnerOrAdmin() {
        if(admin() != _msgSender() && owner() != _msgSender()) {
            revert NotOwnerNorAdmin();
        }
        _;

        // require(admin() == _msgSender() || owner() == _msgSender(), "Adminable: caller is not the owner nor the admin");
        // _;
    }

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

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

    /**
     * @dev Transfers admin of the contract to a new account (`newAdmin`).
     * Internal function without access restriction.
     */
    function _transferAdmin(address newAdmin) internal virtual {
        address oldAdmin = _admin;
        _admin = newAdmin;
        emit AdminTransferred(oldAdmin, newAdmin);
    }
}

File 8 of 9 : IERC721A.sol
// SPDX-License-Identifier: MIT
// ERC721A Contracts v4.0.0
// Creator: Chiru Labs

pragma solidity ^0.8.4;

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

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

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

    /**
     * The caller cannot approve to the current owner.
     */
    error ApprovalToCurrentOwner();

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

File 9 of 9 : Context.sol
// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts v4.4.1 (utils/Context.sol)

pragma solidity ^0.8.0;

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

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

Settings
{
  "optimizer": {
    "enabled": true,
    "runs": 200
  },
  "outputSelection": {
    "*": {
      "*": [
        "evm.bytecode",
        "evm.deployedBytecode",
        "devdoc",
        "userdoc",
        "metadata",
        "abi"
      ]
    }
  },
  "libraries": {}
}

Contract Security Audit

Contract ABI

[{"inputs":[{"internalType":"address","name":"receiver","type":"address"},{"internalType":"uint256","name":"maxSupply","type":"uint256"},{"internalType":"string","name":"_baseUri","type":"string"},{"internalType":"string","name":"_defaultUnrevealedURI","type":"string"},{"internalType":"address","name":"admin_","type":"address"}],"stateMutability":"nonpayable","type":"constructor"},{"inputs":[],"name":"AlreadyRevealed","type":"error"},{"inputs":[],"name":"ApprovalCallerNotOwnerNorApproved","type":"error"},{"inputs":[],"name":"ApprovalQueryForNonexistentToken","type":"error"},{"inputs":[],"name":"ApprovalToCurrentOwner","type":"error"},{"inputs":[],"name":"ApproveToCaller","type":"error"},{"inputs":[{"internalType":"string","name":"reason","type":"string"}],"name":"BadRequest","type":"error"},{"inputs":[],"name":"BalanceQueryForZeroAddress","type":"error"},{"inputs":[{"internalType":"uint16","name":"maximumAllowedQuantity","type":"uint16"}],"name":"ExceedAllowedQuantity","type":"error"},{"inputs":[],"name":"ExceedMaximumForTheRound","type":"error"},{"inputs":[],"name":"ExceedMaximumSupply","type":"error"},{"inputs":[],"name":"FailedToSendBalance","type":"error"},{"inputs":[],"name":"ImmutableState","type":"error"},{"inputs":[],"name":"MaxMintingIdLowerThanCurrentId","type":"error"},{"inputs":[],"name":"MintToZeroAddress","type":"error"},{"inputs":[],"name":"MintZeroQuantity","type":"error"},{"inputs":[],"name":"MintingNotAllowed","type":"error"},{"inputs":[],"name":"NoMatchingFee","type":"error"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"NonExistingToken","type":"error"},{"inputs":[],"name":"NotAdmin","type":"error"},{"inputs":[],"name":"NotOwnerNorAdmin","type":"error"},{"inputs":[],"name":"OwnerQueryForNonexistentToken","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":"previousAdmin","type":"address"},{"indexed":true,"internalType":"address","name":"newAdmin","type":"address"}],"name":"AdminTransferred","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint16","name":"roundNumber","type":"uint16"},{"indexed":false,"internalType":"address","name":"admin","type":"address"}],"name":"AdminUpdated","type":"event"},{"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":false,"internalType":"string","name":"baseURI","type":"string"}],"name":"BaseURIUpdated","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"string","name":"defaultUnrevealedURI","type":"string"}],"name":"DefaultUnrevealedURIUpdated","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint16","name":"roundNumber","type":"uint16"},{"indexed":false,"internalType":"uint256","name":"time","type":"uint256"}],"name":"EndTimeChanged","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint16","name":"roundNumber","type":"uint16"},{"indexed":false,"internalType":"uint256","name":"maxId","type":"uint256"}],"name":"MaxMintingIdChanged","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint16","name":"roundNumber","type":"uint16"},{"indexed":false,"internalType":"uint16","name":"count","type":"uint16"}],"name":"MaxMintingQuantityChanged","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint16","name":"roundNumber","type":"uint16"},{"indexed":false,"internalType":"uint256","name":"newFee","type":"uint256"}],"name":"MintingFeeChanged","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint16","name":"roundNumber","type":"uint16"},{"indexed":false,"internalType":"enum ILoversNft.MintingType","name":"mintingType","type":"uint8"}],"name":"MintingTypeChanged","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint16","name":"roundNumber","type":"uint16"}],"name":"NewRoundCreated","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"previousOwner","type":"address"},{"indexed":true,"internalType":"address","name":"newOwner","type":"address"}],"name":"OwnershipTransferred","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"account","type":"address"}],"name":"Paused","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"called","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"Received","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint16","name":"roundNumber","type":"uint16"}],"name":"Revealed","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint16","name":"roundNumber","type":"uint16"}],"name":"RoundEnded","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"revealBlockNumber","type":"uint256"}],"name":"SetRevealBlock","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint16","name":"roundNumber","type":"uint16"},{"indexed":false,"internalType":"uint256","name":"time","type":"uint256"}],"name":"StartTimeChanged","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint16","name":"roundNumber","type":"uint16"},{"indexed":false,"internalType":"string","name":"prefix","type":"string"}],"name":"TokenURIPrefixUpdated","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"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"account","type":"address"}],"name":"Unpaused","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"receiver","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"Withdraw","type":"event"},{"stateMutability":"payable","type":"fallback"},{"inputs":[],"name":"admin","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address[]","name":"tos","type":"address[]"},{"internalType":"uint256[]","name":"quantities","type":"uint256[]"}],"name":"adminMintTo","outputs":[],"stateMutability":"payable","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":[],"name":"baseURI","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint16","name":"roundNumber","type":"uint16"},{"internalType":"uint256","name":"time","type":"uint256"}],"name":"changeEndTime","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint16","name":"roundNumber","type":"uint16"},{"internalType":"uint256","name":"maxId","type":"uint256"}],"name":"changeMaxMintingId","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint16","name":"roundNumber","type":"uint16"},{"internalType":"uint16","name":"quantity","type":"uint16"}],"name":"changeMaxMintingQuantity","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"newMaximum","type":"uint256"}],"name":"changeMaximumSupply","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint16","name":"roundNumber","type":"uint16"},{"internalType":"uint256","name":"fee","type":"uint256"}],"name":"changeMintingFee","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint16","name":"roundNumber","type":"uint16"},{"internalType":"enum ILoversNft.MintingType","name":"mintingType","type":"uint8"}],"name":"changeMintingType","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint16","name":"roundNumber","type":"uint16"},{"internalType":"uint256","name":"time","type":"uint256"}],"name":"changeStartTime","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"currentRoundNumber","outputs":[{"internalType":"uint16","name":"","type":"uint16"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"endRound","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"getApproved","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getBaseURI","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getCurrentRound","outputs":[{"components":[{"internalType":"uint16","name":"roundNumber","type":"uint16"},{"internalType":"enum ILoversNft.MintingType","name":"mintingType","type":"uint8"},{"internalType":"uint256","name":"startTime","type":"uint256"},{"internalType":"uint256","name":"endTime","type":"uint256"},{"internalType":"uint16","name":"maxMintingQuantity","type":"uint16"},{"internalType":"uint256","name":"mintingFee","type":"uint256"},{"internalType":"uint256","name":"maxMintingId","type":"uint256"},{"internalType":"uint256","name":"startId","type":"uint256"},{"internalType":"uint256","name":"lastMintedId","type":"uint256"},{"internalType":"string","name":"tokenURIPrefix","type":"string"},{"internalType":"bool","name":"revealed","type":"bool"},{"internalType":"uint256","name":"revealBlockNumber","type":"uint256"},{"internalType":"uint256","name":"randomSelection","type":"uint256"},{"internalType":"uint256","name":"closedTime","type":"uint256"}],"internalType":"struct ILoversNft.Round","name":"","type":"tuple"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint16","name":"roundNumber","type":"uint16"}],"name":"getRound","outputs":[{"components":[{"internalType":"uint16","name":"roundNumber","type":"uint16"},{"internalType":"enum ILoversNft.MintingType","name":"mintingType","type":"uint8"},{"internalType":"uint256","name":"startTime","type":"uint256"},{"internalType":"uint256","name":"endTime","type":"uint256"},{"internalType":"uint16","name":"maxMintingQuantity","type":"uint16"},{"internalType":"uint256","name":"mintingFee","type":"uint256"},{"internalType":"uint256","name":"maxMintingId","type":"uint256"},{"internalType":"uint256","name":"startId","type":"uint256"},{"internalType":"uint256","name":"lastMintedId","type":"uint256"},{"internalType":"string","name":"tokenURIPrefix","type":"string"},{"internalType":"bool","name":"revealed","type":"bool"},{"internalType":"uint256","name":"revealBlockNumber","type":"uint256"},{"internalType":"uint256","name":"randomSelection","type":"uint256"},{"internalType":"uint256","name":"closedTime","type":"uint256"}],"internalType":"struct ILoversNft.Round","name":"","type":"tuple"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getState","outputs":[{"internalType":"enum ILoversNft.State","name":"","type":"uint8"}],"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":"maximumSupply","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"quantity","type":"uint256"}],"name":"mint","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"uint256","name":"quantity","type":"uint256"}],"name":"mint","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[],"name":"name","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"enum ILoversNft.MintingType","name":"mintingType","type":"uint8"},{"internalType":"uint256","name":"supply","type":"uint256"},{"internalType":"uint256","name":"startTime","type":"uint256"},{"internalType":"uint256","name":"endTime","type":"uint256"},{"internalType":"uint256","name":"mintingFee","type":"uint256"},{"internalType":"uint16","name":"maxMintingQuantity","type":"uint16"},{"internalType":"bool","name":"revealed","type":"bool"}],"name":"newRound","outputs":[],"stateMutability":"nonpayable","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":"pause","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"paused","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"permanent","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"renounceAdmin","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"revealBlockOffset","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":"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":"_newBaseURI","type":"string"}],"name":"setBaseURI","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"string","name":"_defaultUnrevealedURI","type":"string"}],"name":"setDefaultUnrevealedURI","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"setPermanent","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint16","name":"roundNumber","type":"uint16"}],"name":"setRandomSelection","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"receiver","type":"address"}],"name":"setReceiver","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint16","name":"roundNumber","type":"uint16"}],"name":"setRevealBlock","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint16","name":"roundNumber","type":"uint16"},{"internalType":"string","name":"prefix","type":"string"}],"name":"setTokenURIPrefix","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bytes4","name":"interfaceId","type":"bytes4"}],"name":"supportsInterface","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"symbol","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"tokenURI","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"totalSupply","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"newAdmin","type":"address"}],"name":"transferAdmin","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256[]","name":"tokenIds","type":"uint256[]"},{"internalType":"address","name":"to","type":"address"}],"name":"transferBatch","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"from","type":"address"},{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"transferFrom","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"unpause","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"newOffset","type":"uint256"}],"name":"updateRevealBlockOffset","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"withdraw","outputs":[],"stateMutability":"payable","type":"function"},{"stateMutability":"payable","type":"receive"}]

6080604052610bb8600f553480156200001757600080fd5b506040516200460e3803806200460e8339810160408190526200003a9162000364565b604080518082018252601581527f4c6f7665727320556e64657220546865204d6f6f6e00000000000000000000006020808301918252835180850190945260058452642627ab22a960d91b9084015281518493916200009d9160029190620001ee565b508051620000b3906003906020840190620001ee565b5050600160005550620000c6336200014a565b620000d1336200014a565b620000dc816200019c565b506001600a55600b805460ff191690556010805462010000600160b01b031916620100006001600160a01b03881602179055600c849055825162000128906011906020860190620001ee565b5081516200013e906012906020850190620001ee565b5050505050506200044e565b600880546001600160a01b038381166001600160a01b0319831681179093556040519116919082907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e090600090a35050565b600980546001600160a01b038381166001600160a01b0319831681179093556040519116919082907ff8ccb027dfcd135e000e9d45e6cc2d662578a8825d4c45b5e32e0adf67e79ec690600090a35050565b828054620001fc90620003fb565b90600052602060002090601f0160209004810192826200022057600085556200026b565b82601f106200023b57805160ff19168380011785556200026b565b828001600101855582156200026b579182015b828111156200026b5782518255916020019190600101906200024e565b50620002799291506200027d565b5090565b5b808211156200027957600081556001016200027e565b80516001600160a01b0381168114620002ac57600080fd5b919050565b600082601f830112620002c2578081fd5b81516001600160401b0380821115620002df57620002df62000438565b604051601f8301601f19908116603f011681019082821181831017156200030a576200030a62000438565b8160405283815260209250868385880101111562000326578485fd5b8491505b838210156200034957858201830151818301840152908201906200032a565b838211156200035a57848385830101525b9695505050505050565b600080600080600060a086880312156200037c578081fd5b620003878662000294565b6020870151604088015191965094506001600160401b0380821115620003ab578283fd5b620003b989838a01620002b1565b94506060880151915080821115620003cf578283fd5b50620003de88828901620002b1565b925050620003ef6080870162000294565b90509295509295909350565b600181811c908216806200041057607f821691505b602082108114156200043257634e487b7160e01b600052602260045260246000fd5b50919050565b634e487b7160e01b600052604160045260246000fd5b6141b0806200045e6000396000f3fe6080604052600436106103395760003560e01c806370a08231116101ab578063a32bf597116100f7578063c87b56dd11610095578063e985e9c51161006f578063e985e9c51461097c578063ec342b7f1461099c578063f2fde38b146109b2578063f851a440146109d257610383565b8063c87b56dd1461091c578063caf47a731461093c578063d7cac6961461095c57610383565b8063b824df9e116100d1578063b824df9e146108a2578063b88d4fde146108c2578063c33ac6bb146108e2578063c6ad32001461090257610383565b8063a32bf59714610840578063a9a8baa514610862578063b2cebd8d1461088257610383565b80638456cb591161016457806395d89b411161013e57806395d89b41146107ca5780639c4780d8146107df578063a0712d681461080d578063a22cb4651461082057610383565b80638456cb59146107825780638bad0c0a146107975780638da5cb5b146107ac57610383565b806370a08231146106e3578063714c539814610703578063715018a614610718578063718da7ee1461072d578063749aa2d91461074d57806375829def1461076257610383565b80633ccfd60b116102855780634a61ccfd1161022357806356dba9f7116101fd57806356dba9f7146106765780635c975abb146106965780636352211e146106ae5780636c0360eb146106ce57610383565b80634a61ccfd146106235780634fa81f181461063657806355f804b31461065657610383565b806340d342851161025f57806340d34285146105ae57806342842e0e146105c357806346a6fb7b146105e357806348d0120d1461060357610383565b80633ccfd60b1461057e5780633f4ba83a1461058657806340c10f191461059b57610383565b806311cbe346116102f257806323b872dd116102cc57806323b872dd146104fe5780632e0b30061461051e578063315c17081461053e57806334d15ae51461055e57610383565b806311cbe3461461049f57806318160ddd146104bf5780631865c57d146104dc57610383565b806301ffc9a7146103aa5780630202def1146103df5780630480e58b1461040157806306fdde0314610425578063081812fc14610447578063095ea7b31461047f57610383565b36610383577f88a5966d370b9919b20f3e2c13ff65706f196a4e32cc2c12bf57088f88525874335b604080516001600160a01b0390921682523460208301520160405180910390a1005b7f88a5966d370b9919b20f3e2c13ff65706f196a4e32cc2c12bf57088f8852587433610361565b3480156103b657600080fd5b506103ca6103c5366004613b27565b6109f0565b60405190151581526020015b60405180910390f35b3480156103eb57600080fd5b506103ff6103fa366004613ad5565b610a42565b005b34801561040d57600080fd5b50610417600c5481565b6040519081526020016103d6565b34801561043157600080fd5b5061043a610b1c565b6040516103d69190613eb4565b34801561045357600080fd5b50610467610462366004613cce565b610bae565b6040516001600160a01b0390911681526020016103d6565b34801561048b57600080fd5b506103ff61049a366004613a43565b610bf2565b3480156104ab57600080fd5b506103ff6104ba366004613cb3565b610cc5565b3480156104cb57600080fd5b506001546000540360001901610417565b3480156104e857600080fd5b506104f1610d88565b6040516103d69190613e9a565b34801561050a57600080fd5b506103ff610519366004613966565b610fa2565b34801561052a57600080fd5b506103ff610539366004613c3e565b610fb2565b34801561054a57600080fd5b506103ff610559366004613c8a565b611077565b34801561056a57600080fd5b506103ff610579366004613cce565b61113e565b6103ff611191565b34801561059257600080fd5b506103ff61128d565b6103ff6105a9366004613a43565b6112c1565b3480156105ba57600080fd5b506103ff6114d4565b3480156105cf57600080fd5b506103ff6105de366004613966565b61150d565b3480156105ef57600080fd5b506103ff6105fe366004613bc8565b611528565b34801561060f57600080fd5b506103ff61061e366004613cb3565b6115c4565b6103ff610631366004613a6c565b61167e565b34801561064257600080fd5b506103ff610651366004613cce565b6119d1565b34801561066257600080fd5b506103ff610671366004613bc8565b611a24565b34801561068257600080fd5b506103ff610691366004613c15565b611ab5565b3480156106a257600080fd5b50600b5460ff166103ca565b3480156106ba57600080fd5b506104676106c9366004613cce565b611b96565b3480156106da57600080fd5b5061043a611ba1565b3480156106ef57600080fd5b506104176106fe36600461391a565b611c2f565b34801561070f57600080fd5b5061043a611c7e565b34801561072457600080fd5b506103ff611c8d565b34801561073957600080fd5b506103ff61074836600461391a565b611cc1565b34801561075957600080fd5b506103ff611d15565b34801561076e57600080fd5b506103ff61077d36600461391a565b611dc6565b34801561078e57600080fd5b506103ff611e63565b3480156107a357600080fd5b506103ff611e95565b3480156107b857600080fd5b506008546001600160a01b0316610467565b3480156107d657600080fd5b5061043a611ec9565b3480156107eb57600080fd5b506010546107fa9061ffff1681565b60405161ffff90911681526020016103d6565b6103ff61081b366004613cce565b611ed8565b34801561082c57600080fd5b506103ff61083b366004613a1a565b611fd5565b34801561084c57600080fd5b5061085561206b565b6040516103d69190613efc565b34801561086e57600080fd5b506103ff61087d366004613bfb565b612083565b34801561088e57600080fd5b506103ff61089d366004613bfb565b61218d565b3480156108ae57600080fd5b506103ff6108bd366004613b5f565b612353565b3480156108ce57600080fd5b506103ff6108dd3660046139a1565b6126aa565b3480156108ee57600080fd5b506108556108fd366004613bfb565b6126f4565b34801561090e57600080fd5b50600d546103ca9060ff1681565b34801561092857600080fd5b5061043a610937366004613cce565b61288a565b34801561094857600080fd5b506103ff610957366004613cb3565b612945565b34801561096857600080fd5b506103ff610977366004613cb3565b6129ff565b34801561098857600080fd5b506103ca610997366004613934565b612aff565b3480156109a857600080fd5b50610417600f5481565b3480156109be57600080fd5b506103ff6109cd36600461391a565b612b2d565b3480156109de57600080fd5b506009546001600160a01b0316610467565b60006301ffc9a760e01b6001600160e01b031983161480610a2157506380ac58cd60e01b6001600160e01b03198316145b80610a3c5750635b5e139f60e01b6001600160e01b03198316145b92915050565b6002600a541415610a9a5760405162461bcd60e51b815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c0060448201526064015b60405180910390fd5b6002600a55600d5460ff1615610ac357604051636bf43ac160e11b815260040160405180910390fd5b60005b82811015610b1157610aff3383868685818110610af357634e487b7160e01b600052603260045260246000fd5b9050602002013561150d565b80610b09816140e7565b915050610ac6565b50506001600a555050565b606060028054610b2b90614090565b80601f0160208091040260200160405190810160405280929190818152602001828054610b5790614090565b8015610ba45780601f10610b7957610100808354040283529160200191610ba4565b820191906000526020600020905b815481529060010190602001808311610b8757829003601f168201915b5050505050905090565b6000610bb982612bc5565b610bd6576040516333d1c03960e21b815260040160405180910390fd5b506000908152600660205260409020546001600160a01b031690565b6000610bfd82612bfa565b9050806001600160a01b0316836001600160a01b03161415610c325760405163250fdee360e21b815260040160405180910390fd5b336001600160a01b03821614610c6957610c4c8133612aff565b610c69576040516367d9dca160e11b815260040160405180910390fd5b60008281526006602052604080822080546001600160a01b0319166001600160a01b0387811691821790925591518593918516917f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b92591a4505050565b6009546001600160a01b03163314801590610ceb57506008546001600160a01b03163314155b15610d0957604051630ba9cf7d60e01b815260040160405180910390fd5b600d5460ff1615610d2d57604051636bf43ac160e11b815260040160405180910390fd5b61ffff82166000818152600e6020908152604091829020600281018590558251938452908301849052917f2cd014b38887dc9fbc71a1b75d178be9fed12ae7e297861fa1dda0584dca10aa91015b60405180910390a1505050565b60105460009061ffff16610d9c5750600090565b60105461ffff9081166000908152600e6020908152604080832081516101c0810190925280549485168252929390929183019062010000900460ff166001811115610df757634e487b7160e01b600052602160045260246000fd5b6001811115610e1657634e487b7160e01b600052602160045260246000fd5b81526001820154602082015260028201546040820152600382015461ffff16606082015260048201546080820152600582015460a0820152600682015460c0820152600782015460e082015260088201805461010090920191610e7890614090565b80601f0160208091040260200160405190810160405280929190818152602001828054610ea490614090565b8015610ef15780601f10610ec657610100808354040283529160200191610ef1565b820191906000526020600020905b815481529060010190602001808311610ed457829003601f168201915b5050509183525050600982015460ff1615156020820152600a820154604080830191909152600b830154606080840191909152600c909301546080909201919091526101a08301519083015191830151929350914260008415610f5657506003610f98565b83821015610f6657506001610f98565b82821015610f945760015460005403600019018660c001511115610f8c57506002610f98565b506003610f98565b5060035b9695505050505050565b610fad838383612c6a565b505050565b6009546001600160a01b03163314801590610fd857506008546001600160a01b03163314155b15610ff657604051630ba9cf7d60e01b815260040160405180910390fd5b600d5460ff161561101a57604051636bf43ac160e11b815260040160405180910390fd5b61ffff82166000908152600e60209081526040909120825190916110459160088401918501906136df565b507fd1c3bef23efc23b7b878bc09751c78345ceda36e3ba298e9caf49f7b728a15788383604051610d7b929190613ff9565b6009546001600160a01b0316331480159061109d57506008546001600160a01b03163314155b156110bb57604051630ba9cf7d60e01b815260040160405180910390fd5b600d5460ff16156110df57604051636bf43ac160e11b815260040160405180910390fd5b61ffff8281166000818152600e602090815260409182902060038101805461ffff191695871695861790558251938452908301939093527f1416ea16c319d3dabc21a0c35202bc32b48f2d36feef66ca6a7d42fa4ed0e2029101610d7b565b6008546001600160a01b031633146111685760405162461bcd60e51b8152600401610a9190613ec7565b600d5460ff161561118c57604051636bf43ac160e11b815260040160405180910390fd5b600f55565b6008546001600160a01b031633146111bb5760405162461bcd60e51b8152600401610a9190613ec7565b6010546040514791600091620100009091046001600160a01b031690839060006040518083038185875af1925050503d8060008114611216576040519150601f19603f3d011682016040523d82523d6000602084013e61121b565b606091505b505090508061123d5760405163ce43c9c560e01b815260040160405180910390fd5b60105460408051620100009092046001600160a01b03168252602082018490527f884edad9ce6fa2440d8a54cc123490eb96d2768479d49ff9c7366125a942436491015b60405180910390a15050565b6008546001600160a01b031633146112b75760405162461bcd60e51b8152600401610a9190613ec7565b6112bf612e0d565b565b6009546001600160a01b031633148015906112e757506008546001600160a01b03163314155b1561130557604051630ba9cf7d60e01b815260040160405180910390fd5b600d5460ff161561132957604051636bf43ac160e11b815260040160405180910390fd5b60105461ffff9081166000908152600e6020908152604080832081516101c0810190925280549485168252929390929183019062010000900460ff16600181111561138457634e487b7160e01b600052602160045260246000fd5b60018111156113a357634e487b7160e01b600052602160045260246000fd5b81526001820154602082015260028201546040820152600382015461ffff16606082015260048201546080820152600582015460a0820152600682015460c0820152600782015460e08201526008820180546101009092019161140590614090565b80601f016020809104026020016040519081016040528092919081815260200182805461143190614090565b801561147e5780601f106114535761010080835404028352916020019161147e565b820191906000526020600020905b81548152906001019060200180831161146157829003601f168201915b5050509183525050600982015460ff1615156020820152600a8201546040820152600b8201546060820152600c9091015460809091015290506114c18183612ea0565b6114cb838361304d565b610fad57600080fd5b6008546001600160a01b031633146114fe5760405162461bcd60e51b8152600401610a9190613ec7565b600d805460ff19166001179055565b610fad838383604051806020016040528060008152506126aa565b6008546001600160a01b031633146115525760405162461bcd60e51b8152600401610a9190613ec7565b600d5460ff161561157657604051636bf43ac160e11b815260040160405180910390fd5b80516115899060129060208401906136df565b507fa5d40bbdde39fa9ed704b3ede3470a370dedabb286ed798d27b843adedae454e816040516115b99190613eb4565b60405180910390a150565b6009546001600160a01b031633148015906115ea57506008546001600160a01b03163314155b1561160857604051630ba9cf7d60e01b815260040160405180910390fd5b600d5460ff161561162c57604051636bf43ac160e11b815260040160405180910390fd5b61ffff82166000818152600e6020908152604091829020600481018590558251938452908301849052917f97b667b092a348c9eec2a1a69ce3d0372eb5caf02143dafe2604fe2e8acb23819101610d7b565b6009546001600160a01b031633148015906116a457506008546001600160a01b03163314155b156116c257604051630ba9cf7d60e01b815260040160405180910390fd5b600d5460ff16156116e657604051636bf43ac160e11b815260040160405180910390fd5b8281811461172e57604051632c73a0bf60e11b8152602060048201526014602482015273092dce0eae840e6d2f4ca40dcdee840dac2e8c6d60631b6044820152606401610a91565b6000805b858110156117805784848281811061175a57634e487b7160e01b600052603260045260246000fd5b905060200201358261176c9190614016565b915080611778816140e7565b915050611732565b5060105461ffff9081166000908152600e6020908152604080832081516101c0810190925280549485168252929390929183019062010000900460ff1660018111156117dc57634e487b7160e01b600052602160045260246000fd5b60018111156117fb57634e487b7160e01b600052602160045260246000fd5b81526001820154602082015260028201546040820152600382015461ffff16606082015260048201546080820152600582015460a0820152600682015460c0820152600782015460e08201526008820180546101009092019161185d90614090565b80601f016020809104026020016040519081016040528092919081815260200182805461188990614090565b80156118d65780601f106118ab576101008083540402835291602001916118d6565b820191906000526020600020905b8154815290600101906020018083116118b957829003601f168201915b5050509183525050600982015460ff1615156020820152600a8201546040820152600b8201546060820152600c9091015460809091015260c081015190915061191f8382613062565b61193c57604051636f5606f560e01b815260040160405180910390fd5b60005b848110156119c6576119ab89898381811061196a57634e487b7160e01b600052603260045260246000fd5b905060200201602081019061197f919061391a565b88888481811061199f57634e487b7160e01b600052603260045260246000fd5b9050602002013561304d565b6119b457600080fd5b806119be816140e7565b91505061193f565b505050505050505050565b6008546001600160a01b031633146119fb5760405162461bcd60e51b8152600401610a9190613ec7565b600d5460ff1615611a1f57604051636bf43ac160e11b815260040160405180910390fd5b600c55565b6008546001600160a01b03163314611a4e5760405162461bcd60e51b8152600401610a9190613ec7565b600d5460ff1615611a7257604051636bf43ac160e11b815260040160405180910390fd5b8051611a859060119060208401906136df565b507f6741b2fc379fad678116fe3d4d4b9a1a184ab53ba36b86ad0fa66340b1ab41ad816040516115b99190613eb4565b6009546001600160a01b03163314801590611adb57506008546001600160a01b03163314155b15611af957604051630ba9cf7d60e01b815260040160405180910390fd5b600d5460ff1615611b1d57604051636bf43ac160e11b815260040160405180910390fd5b61ffff82166000908152600e6020526040902080548290829062ff0000191662010000836001811115611b6057634e487b7160e01b600052602160045260246000fd5b02179055507fcc44668b6fa33b0ad467f10b798b408603a7da8b11619465490e341aab4124118383604051610d7b929190613fe1565b6000610a3c82612bfa565b60118054611bae90614090565b80601f0160208091040260200160405190810160405280929190818152602001828054611bda90614090565b8015611c275780601f10611bfc57610100808354040283529160200191611c27565b820191906000526020600020905b815481529060010190602001808311611c0a57829003601f168201915b505050505081565b60006001600160a01b038216611c58576040516323d3ad8160e21b815260040160405180910390fd5b506001600160a01b031660009081526005602052604090205467ffffffffffffffff1690565b6060611c88613091565b905090565b6008546001600160a01b03163314611cb75760405162461bcd60e51b8152600401610a9190613ec7565b6112bf60006130a0565b6008546001600160a01b03163314611ceb5760405162461bcd60e51b8152600401610a9190613ec7565b601080546001600160a01b03909216620100000262010000600160b01b0319909216919091179055565b6009546001600160a01b03163314801590611d3b57506008546001600160a01b03163314155b15611d5957604051630ba9cf7d60e01b815260040160405180910390fd5b60105461ffff166000908152600e602052604090206001611d7960005490565b611d83919061404d565b600782015542600c82015560105460405161ffff90911681527fef9474fd3990648f429c1f27b976775088ba1974353959272281ac075ea11d76906020016115b9565b6008546001600160a01b03163314611df05760405162461bcd60e51b8152600401610a9190613ec7565b6001600160a01b038116611e575760405162461bcd60e51b815260206004820152602860248201527f41646d696e61626c653a206e65772061646d696e20697320746865207a65726f604482015267206164647265737360c01b6064820152608401610a91565b611e60816130f2565b50565b6008546001600160a01b03163314611e8d5760405162461bcd60e51b8152600401610a9190613ec7565b6112bf613144565b6008546001600160a01b03163314611ebf5760405162461bcd60e51b8152600401610a9190613ec7565b6112bf60006130f2565b606060038054610b2b90614090565b6002600a541415611f2b5760405162461bcd60e51b815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c006044820152606401610a91565b6002600a55600b5460ff1615611f765760405162461bcd60e51b815260206004820152601060248201526f14185d5cd8589b194e881c185d5cd95960821b6044820152606401610a91565b600d5460ff1615611f9a57604051636bf43ac160e11b815260040160405180910390fd5b601054600090611fad9061ffff166126f4565b9050611fb98183612ea0565b611fc3338361304d565b611fcc57600080fd5b50506001600a55565b6001600160a01b038216331415611fff5760405163b06307db60e01b815260040160405180910390fd5b3360008181526007602090815260408083206001600160a01b03871680855290835292819020805460ff191686151590811790915590519081529192917f17307eab39ab6107e8899845ad3d59bd9653f200f220920489ca2b5937696c31910160405180910390a35050565b612073613763565b601054611c889061ffff166126f4565b6009546001600160a01b031633148015906120a957506008546001600160a01b03163314155b156120c757604051630ba9cf7d60e01b815260040160405180910390fd5b600d5460ff16156120eb57604051636bf43ac160e11b815260040160405180910390fd5b61ffff81166000908152600e60205260409020600781015461214957604051632c73a0bf60e11b8152602060048201526016602482015275149bdd5b99081cda1bdd5b190818994818db1bdcd95960521b6044820152606401610a91565b600f546121569043614016565b600a82018190556040519081527ffd1cd879b90803328042915a0dab567886d80637d84c7875df6a3e4495c379ac90602001611281565b600d5460ff16156121b157604051636bf43ac160e11b815260040160405180910390fd5b61ffff81166000908152600e60205260409020600981015460ff16156121ea5760405163a89ac15160e01b815260040160405180910390fd5b600a8101544381111561224057604051632c73a0bf60e11b815260206004820152601d60248201527f52616e646f6d2073656c656374696f6e206973206e6f742072656164790000006044820152606401610a91565b600061224d60018361404d565b401561229857814061226060018461404d565b6040805160208101939093529040908201524260608201526080016040516020818303038152906040528051906020012090506122e3565b6122a360014361404d565b406122af60024361404d565b6040805160208101939093529040908201524260608201526080016040516020818303038152906040528051906020012090505b60098301805460ff191660011790556006830154600784015460009161230b918491906131bf565b600b850181905560405161ffff871681529091507fd104aada4b066eb003582f2d87f8450d182d2a2306ab2aad08ab10d5564fe4c69060200160405180910390a15050505050565b6008546001600160a01b0316331461237d5760405162461bcd60e51b8152600401610a9190613ec7565b600d5460ff16156123a157604051636bf43ac160e11b815260040160405180910390fd5b60105461ffff16156123b5576123b5611d15565b60006001876123c360005490565b6123cd9190614016565b6123d7919061404d565b9050600c5481111561243a57604051632c73a0bf60e11b815260206004820152602560248201527f6d61784d696e74696e6749642065786365656420746865206d6178696d756d536044820152647570706c7960d81b6064820152608401610a91565b84861061248a57604051632c73a0bf60e11b815260206004820152601860248201527f656e6454696d652073686f756c642062652062696767657200000000000000006044820152606401610a91565b601080546000919082906124a19061ffff166140c5565b91906101000a81548161ffff021916908361ffff16021790559050604051806101c001604052808261ffff1681526020018a60018111156124f257634e487b7160e01b600052602160045260246000fd5b81526020018881526020018781526020018561ffff16815260200186815260200183815260200161252260005490565b8152600060208083018290526040805180830182528381528185015287151560608501526080840183905260a0840183905260c090930182905261ffff8086168352600e82529290912083518154931661ffff198416811782559184015190929091839162ffffff191617620100008360018111156125b157634e487b7160e01b600052602160045260246000fd5b02179055506040820151600182015560608201516002820155608082015160038201805461ffff191661ffff90921691909117905560a0820151600482015560c0820151600582015560e082015160068201556101008201516007820155610120820151805161262b9160088401916020909101906136df565b5061014082015160098201805460ff1916911515919091179055610160820151600a820155610180820151600b8201556101a090910151600c9091015560405161ffff821681527f812de79a73e727eb2b77c67cd0d01ab1559b5a9a841d0f247d28517f37ea16c89060200160405180910390a1505050505050505050565b6126b5848484612c6a565b6001600160a01b0383163b156126ee576126d1848484846131f7565b6126ee576040516368d2bf6b60e11b815260040160405180910390fd5b50505050565b6126fc613763565b61ffff8281166000908152600e602090815260409182902082516101c081019093528054938416835291929083019062010000900460ff16600181111561275357634e487b7160e01b600052602160045260246000fd5b600181111561277257634e487b7160e01b600052602160045260246000fd5b81526001820154602082015260028201546040820152600382015461ffff16606082015260048201546080820152600582015460a0820152600682015460c0820152600782015460e0820152600882018054610100909201916127d490614090565b80601f016020809104026020016040519081016040528092919081815260200182805461280090614090565b801561284d5780601f106128225761010080835404028352916020019161284d565b820191906000526020600020905b81548152906001019060200180831161283057829003601f168201915b5050509183525050600982015460ff1615156020820152600a8201546040820152600b8201546060820152600c9091015460809091015292915050565b60606000612897836132ef565b905080610140015180156128e457506040805160008152602081018083528151902061012084015190926128cb9201613dbe565b6040516020818303038152906040528051906020012014155b156129235760118161012001516128fa85613501565b60405160200161290c93929190613e0f565b604051602081830303815290604052915050919050565b601261292e84613501565b60405160200161290c929190613dda565b50919050565b6009546001600160a01b0316331480159061296b57506008546001600160a01b03163314155b1561298957604051630ba9cf7d60e01b815260040160405180910390fd5b600d5460ff16156129ad57604051636bf43ac160e11b815260040160405180910390fd5b61ffff82166000818152600e6020908152604091829020600181018590558251938452908301849052917f7196d16a1c019f6d894f4e212e61954e22117dbeaaf8573558b943c990db29479101610d7b565b6009546001600160a01b03163314801590612a2557506008546001600160a01b03163314155b15612a4357604051630ba9cf7d60e01b815260040160405180910390fd5b600d5460ff1615612a6757604051636bf43ac160e11b815260040160405180910390fd5b600054811015612a8a576040516335d5065160e11b815260040160405180910390fd5b600c54811115612aad576040516316b0321960e01b815260040160405180910390fd5b61ffff82166000818152600e6020908152604091829020600581018590558251938452908301849052917ffab54bb645c4d4a86ffcfe9885b75722d72fca682dadad89f1abe2f84bfe76929101610d7b565b6001600160a01b03918216600090815260076020908152604080832093909416825291909152205460ff1690565b6008546001600160a01b03163314612b575760405162461bcd60e51b8152600401610a9190613ec7565b6001600160a01b038116612bbc5760405162461bcd60e51b815260206004820152602660248201527f4f776e61626c653a206e6577206f776e657220697320746865207a65726f206160448201526564647265737360d01b6064820152608401610a91565b611e60816130a0565b600081600111158015612bd9575060005482105b8015610a3c575050600090815260046020526040902054600160e01b161590565b60008180600111612c5157600054811015612c5157600081815260046020526040902054600160e01b8116612c4f575b80612c48575060001901600081815260046020526040902054612c2a565b9392505050565b505b604051636f96cda160e11b815260040160405180910390fd5b6000612c7582612bfa565b9050836001600160a01b0316816001600160a01b031614612ca85760405162a1148160e81b815260040160405180910390fd5b6000336001600160a01b0386161480612cc65750612cc68533612aff565b80612ce1575033612cd684610bae565b6001600160a01b0316145b905080612d0157604051632ce44b5f60e11b815260040160405180910390fd5b6001600160a01b038416612d2857604051633a954ecd60e21b815260040160405180910390fd5b600083815260066020908152604080832080546001600160a01b03191690556001600160a01b038881168452600583528184208054600019019055871683528083208054600101905585835260049091529020600160e11b4260a01b861781179091558216612dc55760018301600081815260046020526040902054612dc3576000548114612dc35760008181526004602052604090208390555b505b82846001600160a01b0316866001600160a01b03167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef60405160405180910390a45050505050565b600b5460ff16612e565760405162461bcd60e51b815260206004820152601460248201527314185d5cd8589b194e881b9bdd081c185d5cd95960621b6044820152606401610a91565b600b805460ff191690557f5db9ee0a495bf2e6ff9c91a7834c1ba4fdd244a5e8aa4e537bd38aeae4b073aa335b6040516001600160a01b03909116815260200160405180910390a1565b6000612eaa610d88565b90506002816004811115612ece57634e487b7160e01b600052602160045260246000fd5b14612eec57604051632ac9194b60e21b815260040160405180910390fd5b600183602001516001811115612f1257634e487b7160e01b600052602160045260246000fd5b148015612f2a57506008546001600160a01b03163314155b8015612f4157506009546001600160a01b03163314155b15612f5f57604051630ba9cf7d60e01b815260040160405180910390fd5b81836080015161ffff161015612f9457608083015160405163e70937a360e01b815261ffff9091166004820152602401610a91565b612fa2828460c00151613062565b612fbf57604051636f5606f560e01b815260040160405180910390fd5b600083602001516001811115612fe557634e487b7160e01b600052602160045260246000fd5b148015612ffd57506008546001600160a01b03163314155b801561301457506009546001600160a01b03163314155b15610fad576000828460a0015161302b919061402e565b90503481146126ee5760405163bfa3573760e01b815260040160405180910390fd5b60006130598383613550565b50600192915050565b600061306f826001614016565b8361307960005490565b6130839190614016565b111561305957506000610a3c565b606060118054610b2b90614090565b600880546001600160a01b038381166001600160a01b0319831681179093556040519116919082907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e090600090a35050565b600980546001600160a01b038381166001600160a01b0319831681179093556040519116919082907ff8ccb027dfcd135e000e9d45e6cc2d662578a8825d4c45b5e32e0adf67e79ec690600090a35050565b600b5460ff161561318a5760405162461bcd60e51b815260206004820152601060248201526f14185d5cd8589b194e881c185d5cd95960821b6044820152606401610a91565b600b805460ff191660011790557f62e78cea01bee320cd4e420270b5ea74000d11b0c9f74754ebdbfc544b05a258612e833390565b6000806131cc848461404d565b6131d7906001614016565b9050836131e48287614102565b6131ee9190614016565b95945050505050565b604051630a85bd0160e11b81526000906001600160a01b0385169063150b7a029061322c903390899088908890600401613e67565b602060405180830381600087803b15801561324657600080fd5b505af1925050508015613276575060408051601f3d908101601f1916820190925261327391810190613b43565b60015b6132d1573d8080156132a4576040519150601f19603f3d011682016040523d82523d6000602084013e6132a9565b606091505b5080516132c9576040516368d2bf6b60e11b815260040160405180910390fd5b805181602001fd5b6001600160e01b031916630a85bd0160e11b1490505b949350505050565b6132f7613763565b61330082612bc5565b6133205760405163b5706f4f60e01b815260048101839052602401610a91565b60015b60105461ffff9081169082161161293f5761ffff8181166000908152600e602090815260409182902082516101c081019093528054938416835291929083019062010000900460ff16600181111561338b57634e487b7160e01b600052602160045260246000fd5b60018111156133aa57634e487b7160e01b600052602160045260246000fd5b81526001820154602082015260028201546040820152600382015461ffff16606082015260048201546080820152600582015460a0820152600682015460c0820152600782015460e08201526008820180546101009092019161340c90614090565b80601f016020809104026020016040519081016040528092919081815260200182805461343890614090565b80156134855780601f1061345a57610100808354040283529160200191613485565b820191906000526020600020905b81548152906001019060200180831161346857829003601f168201915b5050509183525050600982015460ff1615156020820152600a8201546040820152600b8201546060820152600c909101546080909101526101008101519092506000906134d6578260c001516134dd565b8261010001515b9050808411156134fa57816134f1816140c5565b92505050613323565b5050919050565b604080516080810191829052607f0190826030600a8206018353600a90045b801561353e57600183039250600a81066030018353600a9004613520565b50819003601f19909101908152919050565b61356a82826040518060200160405280600081525061356e565b5050565b6000546001600160a01b03841661359757604051622e076360e81b815260040160405180910390fd5b826135b55760405163b562e8dd60e01b815260040160405180910390fd5b6001600160a01b03841660008181526005602090815260408083208054680100000000000000018902019055848352600490915290204260a01b86176001861460e11b1790558190818501903b1561368a575b60405182906001600160a01b038816906000907fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef908290a461365360008784806001019550876131f7565b613670576040516368d2bf6b60e11b815260040160405180910390fd5b80821061360857826000541461368557600080fd5b6136cf565b5b6040516001830192906001600160a01b038816906000907fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef908290a480821061368b575b5060009081556126ee9085838684565b8280546136eb90614090565b90600052602060002090601f01602090048101928261370d5760008555613753565b82601f1061372657805160ff1916838001178555613753565b82800160010185558215613753579182015b82811115613753578251825591602001919060010190613738565b5061375f9291506137d9565b5090565b604080516101c081019091526000808252602082019081526020016000815260200160008152602001600061ffff16815260200160008152602001600081526020016000815260200160008152602001606081526020016000151581526020016000815260200160008152602001600081525090565b5b8082111561375f57600081556001016137da565b600067ffffffffffffffff808411156138095761380961414e565b604051601f8501601f19908116603f011681019082821181831017156138315761383161414e565b8160405280935085815286868601111561384a57600080fd5b858560208301376000602087830101525050509392505050565b80356001600160a01b038116811461387b57600080fd5b919050565b60008083601f840112613891578081fd5b50813567ffffffffffffffff8111156138a8578182fd5b6020830191508360208260051b85010111156138c357600080fd5b9250929050565b8035801515811461387b57600080fd5b80356002811061387b57600080fd5b600082601f8301126138f9578081fd5b612c48838335602085016137ee565b803561ffff8116811461387b57600080fd5b60006020828403121561392b578081fd5b612c4882613864565b60008060408385031215613946578081fd5b61394f83613864565b915061395d60208401613864565b90509250929050565b60008060006060848603121561397a578081fd5b61398384613864565b925061399160208501613864565b9150604084013590509250925092565b600080600080608085870312156139b6578081fd5b6139bf85613864565b93506139cd60208601613864565b925060408501359150606085013567ffffffffffffffff8111156139ef578182fd5b8501601f810187136139ff578182fd5b613a0e878235602084016137ee565b91505092959194509250565b60008060408385031215613a2c578182fd5b613a3583613864565b915061395d602084016138ca565b60008060408385031215613a55578182fd5b613a5e83613864565b946020939093013593505050565b60008060008060408587031215613a81578384fd5b843567ffffffffffffffff80821115613a98578586fd5b613aa488838901613880565b90965094506020870135915080821115613abc578384fd5b50613ac987828801613880565b95989497509550505050565b600080600060408486031215613ae9578283fd5b833567ffffffffffffffff811115613aff578384fd5b613b0b86828701613880565b9094509250613b1e905060208501613864565b90509250925092565b600060208284031215613b38578081fd5b8135612c4881614164565b600060208284031215613b54578081fd5b8151612c4881614164565b600080600080600080600060e0888a031215613b79578485fd5b613b82886138da565b965060208801359550604088013594506060880135935060808801359250613bac60a08901613908565b9150613bba60c089016138ca565b905092959891949750929550565b600060208284031215613bd9578081fd5b813567ffffffffffffffff811115613bef578182fd5b6132e7848285016138e9565b600060208284031215613c0c578081fd5b612c4882613908565b60008060408385031215613c27578182fd5b613c3083613908565b915061395d602084016138da565b60008060408385031215613c50578182fd5b613c5983613908565b9150602083013567ffffffffffffffff811115613c74578182fd5b613c80858286016138e9565b9150509250929050565b60008060408385031215613c9c578182fd5b613ca583613908565b915061395d60208401613908565b60008060408385031215613cc5578182fd5b613a5e83613908565b600060208284031215613cdf578081fd5b5035919050565b60008151808452613cfe816020860160208601614064565b601f01601f19169290920160200192915050565b60028110613d2257613d22614138565b9052565b8054600090600181811c9080831680613d4057607f831692505b6020808410821415613d6057634e487b7160e01b86526022600452602486fd5b818015613d745760018114613d8557613db2565b60ff19861689528489019650613db2565b60008881526020902060005b86811015613daa5781548b820152908501908301613d91565b505084890196505b50505050505092915050565b60008251613dd0818460208701614064565b9190910192915050565b6000613de68285613d26565b8351613df6818360208801614064565b64173539b7b760d91b9101908152600501949350505050565b6000613e1b8286613d26565b8451613e2b818360208901614064565b602f60f81b91019081528351613e48816001840160208801614064565b64173539b7b760d91b6001929091019182015260060195945050505050565b6001600160a01b0385811682528416602082015260408101839052608060608201819052600090610f9890830184613ce6565b6020810160058310613eae57613eae614138565b91905290565b602081526000612c486020830184613ce6565b6020808252818101527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604082015260600190565b60208152613f1160208201835161ffff169052565b60006020830151613f256040840182613d12565b5060408301516060830152606083015160808301526080830151613f4f60a084018261ffff169052565b5060a083015160c083015260c083015160e083015260e08301516101008181850152808501519150506101208181850152808501519150506101c06101408181860152613fa06101e0860184613ce6565b90860151909250610160613fb78682018315159052565b860151610180868101919091528601516101a0808701919091529095015193019290925250919050565b61ffff8316815260408101612c486020830184613d12565b61ffff831681526040602082015260006132e76040830184613ce6565b6000821982111561402957614029614122565b500190565b600081600019048311821515161561404857614048614122565b500290565b60008282101561405f5761405f614122565b500390565b60005b8381101561407f578181015183820152602001614067565b838111156126ee5750506000910152565b600181811c908216806140a457607f821691505b6020821081141561293f57634e487b7160e01b600052602260045260246000fd5b600061ffff808316818114156140dd576140dd614122565b6001019392505050565b60006000198214156140fb576140fb614122565b5060010190565b60008261411d57634e487b7160e01b81526012600452602481fd5b500690565b634e487b7160e01b600052601160045260246000fd5b634e487b7160e01b600052602160045260246000fd5b634e487b7160e01b600052604160045260246000fd5b6001600160e01b031981168114611e6057600080fdfea2646970667358221220432d56aba4fe74d60caeda2ea31b4cad4b5ea4997b032804cfa22624e148da6c64736f6c634300080400330000000000000000000000006c9e30f0340e32fde08deade9be98ad83df05e73000000000000000000000000000000000000000000000000000000000000047800000000000000000000000000000000000000000000000000000000000000a000000000000000000000000000000000000000000000000000000000000000e00000000000000000000000006c9e30f0340e32fde08deade9be98ad83df05e73000000000000000000000000000000000000000000000000000000000000001b68747470733a2f2f6b6d6d2e6d7970696e6174612e636c6f75642f0000000000000000000000000000000000000000000000000000000000000000000000002568747470733a2f2f676174657761792d697066732e61746f6d726967732e696f2f6b6d6d2f000000000000000000000000000000000000000000000000000000

Deployed Bytecode

0x6080604052600436106103395760003560e01c806370a08231116101ab578063a32bf597116100f7578063c87b56dd11610095578063e985e9c51161006f578063e985e9c51461097c578063ec342b7f1461099c578063f2fde38b146109b2578063f851a440146109d257610383565b8063c87b56dd1461091c578063caf47a731461093c578063d7cac6961461095c57610383565b8063b824df9e116100d1578063b824df9e146108a2578063b88d4fde146108c2578063c33ac6bb146108e2578063c6ad32001461090257610383565b8063a32bf59714610840578063a9a8baa514610862578063b2cebd8d1461088257610383565b80638456cb591161016457806395d89b411161013e57806395d89b41146107ca5780639c4780d8146107df578063a0712d681461080d578063a22cb4651461082057610383565b80638456cb59146107825780638bad0c0a146107975780638da5cb5b146107ac57610383565b806370a08231146106e3578063714c539814610703578063715018a614610718578063718da7ee1461072d578063749aa2d91461074d57806375829def1461076257610383565b80633ccfd60b116102855780634a61ccfd1161022357806356dba9f7116101fd57806356dba9f7146106765780635c975abb146106965780636352211e146106ae5780636c0360eb146106ce57610383565b80634a61ccfd146106235780634fa81f181461063657806355f804b31461065657610383565b806340d342851161025f57806340d34285146105ae57806342842e0e146105c357806346a6fb7b146105e357806348d0120d1461060357610383565b80633ccfd60b1461057e5780633f4ba83a1461058657806340c10f191461059b57610383565b806311cbe346116102f257806323b872dd116102cc57806323b872dd146104fe5780632e0b30061461051e578063315c17081461053e57806334d15ae51461055e57610383565b806311cbe3461461049f57806318160ddd146104bf5780631865c57d146104dc57610383565b806301ffc9a7146103aa5780630202def1146103df5780630480e58b1461040157806306fdde0314610425578063081812fc14610447578063095ea7b31461047f57610383565b36610383577f88a5966d370b9919b20f3e2c13ff65706f196a4e32cc2c12bf57088f88525874335b604080516001600160a01b0390921682523460208301520160405180910390a1005b7f88a5966d370b9919b20f3e2c13ff65706f196a4e32cc2c12bf57088f8852587433610361565b3480156103b657600080fd5b506103ca6103c5366004613b27565b6109f0565b60405190151581526020015b60405180910390f35b3480156103eb57600080fd5b506103ff6103fa366004613ad5565b610a42565b005b34801561040d57600080fd5b50610417600c5481565b6040519081526020016103d6565b34801561043157600080fd5b5061043a610b1c565b6040516103d69190613eb4565b34801561045357600080fd5b50610467610462366004613cce565b610bae565b6040516001600160a01b0390911681526020016103d6565b34801561048b57600080fd5b506103ff61049a366004613a43565b610bf2565b3480156104ab57600080fd5b506103ff6104ba366004613cb3565b610cc5565b3480156104cb57600080fd5b506001546000540360001901610417565b3480156104e857600080fd5b506104f1610d88565b6040516103d69190613e9a565b34801561050a57600080fd5b506103ff610519366004613966565b610fa2565b34801561052a57600080fd5b506103ff610539366004613c3e565b610fb2565b34801561054a57600080fd5b506103ff610559366004613c8a565b611077565b34801561056a57600080fd5b506103ff610579366004613cce565b61113e565b6103ff611191565b34801561059257600080fd5b506103ff61128d565b6103ff6105a9366004613a43565b6112c1565b3480156105ba57600080fd5b506103ff6114d4565b3480156105cf57600080fd5b506103ff6105de366004613966565b61150d565b3480156105ef57600080fd5b506103ff6105fe366004613bc8565b611528565b34801561060f57600080fd5b506103ff61061e366004613cb3565b6115c4565b6103ff610631366004613a6c565b61167e565b34801561064257600080fd5b506103ff610651366004613cce565b6119d1565b34801561066257600080fd5b506103ff610671366004613bc8565b611a24565b34801561068257600080fd5b506103ff610691366004613c15565b611ab5565b3480156106a257600080fd5b50600b5460ff166103ca565b3480156106ba57600080fd5b506104676106c9366004613cce565b611b96565b3480156106da57600080fd5b5061043a611ba1565b3480156106ef57600080fd5b506104176106fe36600461391a565b611c2f565b34801561070f57600080fd5b5061043a611c7e565b34801561072457600080fd5b506103ff611c8d565b34801561073957600080fd5b506103ff61074836600461391a565b611cc1565b34801561075957600080fd5b506103ff611d15565b34801561076e57600080fd5b506103ff61077d36600461391a565b611dc6565b34801561078e57600080fd5b506103ff611e63565b3480156107a357600080fd5b506103ff611e95565b3480156107b857600080fd5b506008546001600160a01b0316610467565b3480156107d657600080fd5b5061043a611ec9565b3480156107eb57600080fd5b506010546107fa9061ffff1681565b60405161ffff90911681526020016103d6565b6103ff61081b366004613cce565b611ed8565b34801561082c57600080fd5b506103ff61083b366004613a1a565b611fd5565b34801561084c57600080fd5b5061085561206b565b6040516103d69190613efc565b34801561086e57600080fd5b506103ff61087d366004613bfb565b612083565b34801561088e57600080fd5b506103ff61089d366004613bfb565b61218d565b3480156108ae57600080fd5b506103ff6108bd366004613b5f565b612353565b3480156108ce57600080fd5b506103ff6108dd3660046139a1565b6126aa565b3480156108ee57600080fd5b506108556108fd366004613bfb565b6126f4565b34801561090e57600080fd5b50600d546103ca9060ff1681565b34801561092857600080fd5b5061043a610937366004613cce565b61288a565b34801561094857600080fd5b506103ff610957366004613cb3565b612945565b34801561096857600080fd5b506103ff610977366004613cb3565b6129ff565b34801561098857600080fd5b506103ca610997366004613934565b612aff565b3480156109a857600080fd5b50610417600f5481565b3480156109be57600080fd5b506103ff6109cd36600461391a565b612b2d565b3480156109de57600080fd5b506009546001600160a01b0316610467565b60006301ffc9a760e01b6001600160e01b031983161480610a2157506380ac58cd60e01b6001600160e01b03198316145b80610a3c5750635b5e139f60e01b6001600160e01b03198316145b92915050565b6002600a541415610a9a5760405162461bcd60e51b815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c0060448201526064015b60405180910390fd5b6002600a55600d5460ff1615610ac357604051636bf43ac160e11b815260040160405180910390fd5b60005b82811015610b1157610aff3383868685818110610af357634e487b7160e01b600052603260045260246000fd5b9050602002013561150d565b80610b09816140e7565b915050610ac6565b50506001600a555050565b606060028054610b2b90614090565b80601f0160208091040260200160405190810160405280929190818152602001828054610b5790614090565b8015610ba45780601f10610b7957610100808354040283529160200191610ba4565b820191906000526020600020905b815481529060010190602001808311610b8757829003601f168201915b5050505050905090565b6000610bb982612bc5565b610bd6576040516333d1c03960e21b815260040160405180910390fd5b506000908152600660205260409020546001600160a01b031690565b6000610bfd82612bfa565b9050806001600160a01b0316836001600160a01b03161415610c325760405163250fdee360e21b815260040160405180910390fd5b336001600160a01b03821614610c6957610c4c8133612aff565b610c69576040516367d9dca160e11b815260040160405180910390fd5b60008281526006602052604080822080546001600160a01b0319166001600160a01b0387811691821790925591518593918516917f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b92591a4505050565b6009546001600160a01b03163314801590610ceb57506008546001600160a01b03163314155b15610d0957604051630ba9cf7d60e01b815260040160405180910390fd5b600d5460ff1615610d2d57604051636bf43ac160e11b815260040160405180910390fd5b61ffff82166000818152600e6020908152604091829020600281018590558251938452908301849052917f2cd014b38887dc9fbc71a1b75d178be9fed12ae7e297861fa1dda0584dca10aa91015b60405180910390a1505050565b60105460009061ffff16610d9c5750600090565b60105461ffff9081166000908152600e6020908152604080832081516101c0810190925280549485168252929390929183019062010000900460ff166001811115610df757634e487b7160e01b600052602160045260246000fd5b6001811115610e1657634e487b7160e01b600052602160045260246000fd5b81526001820154602082015260028201546040820152600382015461ffff16606082015260048201546080820152600582015460a0820152600682015460c0820152600782015460e082015260088201805461010090920191610e7890614090565b80601f0160208091040260200160405190810160405280929190818152602001828054610ea490614090565b8015610ef15780601f10610ec657610100808354040283529160200191610ef1565b820191906000526020600020905b815481529060010190602001808311610ed457829003601f168201915b5050509183525050600982015460ff1615156020820152600a820154604080830191909152600b830154606080840191909152600c909301546080909201919091526101a08301519083015191830151929350914260008415610f5657506003610f98565b83821015610f6657506001610f98565b82821015610f945760015460005403600019018660c001511115610f8c57506002610f98565b506003610f98565b5060035b9695505050505050565b610fad838383612c6a565b505050565b6009546001600160a01b03163314801590610fd857506008546001600160a01b03163314155b15610ff657604051630ba9cf7d60e01b815260040160405180910390fd5b600d5460ff161561101a57604051636bf43ac160e11b815260040160405180910390fd5b61ffff82166000908152600e60209081526040909120825190916110459160088401918501906136df565b507fd1c3bef23efc23b7b878bc09751c78345ceda36e3ba298e9caf49f7b728a15788383604051610d7b929190613ff9565b6009546001600160a01b0316331480159061109d57506008546001600160a01b03163314155b156110bb57604051630ba9cf7d60e01b815260040160405180910390fd5b600d5460ff16156110df57604051636bf43ac160e11b815260040160405180910390fd5b61ffff8281166000818152600e602090815260409182902060038101805461ffff191695871695861790558251938452908301939093527f1416ea16c319d3dabc21a0c35202bc32b48f2d36feef66ca6a7d42fa4ed0e2029101610d7b565b6008546001600160a01b031633146111685760405162461bcd60e51b8152600401610a9190613ec7565b600d5460ff161561118c57604051636bf43ac160e11b815260040160405180910390fd5b600f55565b6008546001600160a01b031633146111bb5760405162461bcd60e51b8152600401610a9190613ec7565b6010546040514791600091620100009091046001600160a01b031690839060006040518083038185875af1925050503d8060008114611216576040519150601f19603f3d011682016040523d82523d6000602084013e61121b565b606091505b505090508061123d5760405163ce43c9c560e01b815260040160405180910390fd5b60105460408051620100009092046001600160a01b03168252602082018490527f884edad9ce6fa2440d8a54cc123490eb96d2768479d49ff9c7366125a942436491015b60405180910390a15050565b6008546001600160a01b031633146112b75760405162461bcd60e51b8152600401610a9190613ec7565b6112bf612e0d565b565b6009546001600160a01b031633148015906112e757506008546001600160a01b03163314155b1561130557604051630ba9cf7d60e01b815260040160405180910390fd5b600d5460ff161561132957604051636bf43ac160e11b815260040160405180910390fd5b60105461ffff9081166000908152600e6020908152604080832081516101c0810190925280549485168252929390929183019062010000900460ff16600181111561138457634e487b7160e01b600052602160045260246000fd5b60018111156113a357634e487b7160e01b600052602160045260246000fd5b81526001820154602082015260028201546040820152600382015461ffff16606082015260048201546080820152600582015460a0820152600682015460c0820152600782015460e08201526008820180546101009092019161140590614090565b80601f016020809104026020016040519081016040528092919081815260200182805461143190614090565b801561147e5780601f106114535761010080835404028352916020019161147e565b820191906000526020600020905b81548152906001019060200180831161146157829003601f168201915b5050509183525050600982015460ff1615156020820152600a8201546040820152600b8201546060820152600c9091015460809091015290506114c18183612ea0565b6114cb838361304d565b610fad57600080fd5b6008546001600160a01b031633146114fe5760405162461bcd60e51b8152600401610a9190613ec7565b600d805460ff19166001179055565b610fad838383604051806020016040528060008152506126aa565b6008546001600160a01b031633146115525760405162461bcd60e51b8152600401610a9190613ec7565b600d5460ff161561157657604051636bf43ac160e11b815260040160405180910390fd5b80516115899060129060208401906136df565b507fa5d40bbdde39fa9ed704b3ede3470a370dedabb286ed798d27b843adedae454e816040516115b99190613eb4565b60405180910390a150565b6009546001600160a01b031633148015906115ea57506008546001600160a01b03163314155b1561160857604051630ba9cf7d60e01b815260040160405180910390fd5b600d5460ff161561162c57604051636bf43ac160e11b815260040160405180910390fd5b61ffff82166000818152600e6020908152604091829020600481018590558251938452908301849052917f97b667b092a348c9eec2a1a69ce3d0372eb5caf02143dafe2604fe2e8acb23819101610d7b565b6009546001600160a01b031633148015906116a457506008546001600160a01b03163314155b156116c257604051630ba9cf7d60e01b815260040160405180910390fd5b600d5460ff16156116e657604051636bf43ac160e11b815260040160405180910390fd5b8281811461172e57604051632c73a0bf60e11b8152602060048201526014602482015273092dce0eae840e6d2f4ca40dcdee840dac2e8c6d60631b6044820152606401610a91565b6000805b858110156117805784848281811061175a57634e487b7160e01b600052603260045260246000fd5b905060200201358261176c9190614016565b915080611778816140e7565b915050611732565b5060105461ffff9081166000908152600e6020908152604080832081516101c0810190925280549485168252929390929183019062010000900460ff1660018111156117dc57634e487b7160e01b600052602160045260246000fd5b60018111156117fb57634e487b7160e01b600052602160045260246000fd5b81526001820154602082015260028201546040820152600382015461ffff16606082015260048201546080820152600582015460a0820152600682015460c0820152600782015460e08201526008820180546101009092019161185d90614090565b80601f016020809104026020016040519081016040528092919081815260200182805461188990614090565b80156118d65780601f106118ab576101008083540402835291602001916118d6565b820191906000526020600020905b8154815290600101906020018083116118b957829003601f168201915b5050509183525050600982015460ff1615156020820152600a8201546040820152600b8201546060820152600c9091015460809091015260c081015190915061191f8382613062565b61193c57604051636f5606f560e01b815260040160405180910390fd5b60005b848110156119c6576119ab89898381811061196a57634e487b7160e01b600052603260045260246000fd5b905060200201602081019061197f919061391a565b88888481811061199f57634e487b7160e01b600052603260045260246000fd5b9050602002013561304d565b6119b457600080fd5b806119be816140e7565b91505061193f565b505050505050505050565b6008546001600160a01b031633146119fb5760405162461bcd60e51b8152600401610a9190613ec7565b600d5460ff1615611a1f57604051636bf43ac160e11b815260040160405180910390fd5b600c55565b6008546001600160a01b03163314611a4e5760405162461bcd60e51b8152600401610a9190613ec7565b600d5460ff1615611a7257604051636bf43ac160e11b815260040160405180910390fd5b8051611a859060119060208401906136df565b507f6741b2fc379fad678116fe3d4d4b9a1a184ab53ba36b86ad0fa66340b1ab41ad816040516115b99190613eb4565b6009546001600160a01b03163314801590611adb57506008546001600160a01b03163314155b15611af957604051630ba9cf7d60e01b815260040160405180910390fd5b600d5460ff1615611b1d57604051636bf43ac160e11b815260040160405180910390fd5b61ffff82166000908152600e6020526040902080548290829062ff0000191662010000836001811115611b6057634e487b7160e01b600052602160045260246000fd5b02179055507fcc44668b6fa33b0ad467f10b798b408603a7da8b11619465490e341aab4124118383604051610d7b929190613fe1565b6000610a3c82612bfa565b60118054611bae90614090565b80601f0160208091040260200160405190810160405280929190818152602001828054611bda90614090565b8015611c275780601f10611bfc57610100808354040283529160200191611c27565b820191906000526020600020905b815481529060010190602001808311611c0a57829003601f168201915b505050505081565b60006001600160a01b038216611c58576040516323d3ad8160e21b815260040160405180910390fd5b506001600160a01b031660009081526005602052604090205467ffffffffffffffff1690565b6060611c88613091565b905090565b6008546001600160a01b03163314611cb75760405162461bcd60e51b8152600401610a9190613ec7565b6112bf60006130a0565b6008546001600160a01b03163314611ceb5760405162461bcd60e51b8152600401610a9190613ec7565b601080546001600160a01b03909216620100000262010000600160b01b0319909216919091179055565b6009546001600160a01b03163314801590611d3b57506008546001600160a01b03163314155b15611d5957604051630ba9cf7d60e01b815260040160405180910390fd5b60105461ffff166000908152600e602052604090206001611d7960005490565b611d83919061404d565b600782015542600c82015560105460405161ffff90911681527fef9474fd3990648f429c1f27b976775088ba1974353959272281ac075ea11d76906020016115b9565b6008546001600160a01b03163314611df05760405162461bcd60e51b8152600401610a9190613ec7565b6001600160a01b038116611e575760405162461bcd60e51b815260206004820152602860248201527f41646d696e61626c653a206e65772061646d696e20697320746865207a65726f604482015267206164647265737360c01b6064820152608401610a91565b611e60816130f2565b50565b6008546001600160a01b03163314611e8d5760405162461bcd60e51b8152600401610a9190613ec7565b6112bf613144565b6008546001600160a01b03163314611ebf5760405162461bcd60e51b8152600401610a9190613ec7565b6112bf60006130f2565b606060038054610b2b90614090565b6002600a541415611f2b5760405162461bcd60e51b815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c006044820152606401610a91565b6002600a55600b5460ff1615611f765760405162461bcd60e51b815260206004820152601060248201526f14185d5cd8589b194e881c185d5cd95960821b6044820152606401610a91565b600d5460ff1615611f9a57604051636bf43ac160e11b815260040160405180910390fd5b601054600090611fad9061ffff166126f4565b9050611fb98183612ea0565b611fc3338361304d565b611fcc57600080fd5b50506001600a55565b6001600160a01b038216331415611fff5760405163b06307db60e01b815260040160405180910390fd5b3360008181526007602090815260408083206001600160a01b03871680855290835292819020805460ff191686151590811790915590519081529192917f17307eab39ab6107e8899845ad3d59bd9653f200f220920489ca2b5937696c31910160405180910390a35050565b612073613763565b601054611c889061ffff166126f4565b6009546001600160a01b031633148015906120a957506008546001600160a01b03163314155b156120c757604051630ba9cf7d60e01b815260040160405180910390fd5b600d5460ff16156120eb57604051636bf43ac160e11b815260040160405180910390fd5b61ffff81166000908152600e60205260409020600781015461214957604051632c73a0bf60e11b8152602060048201526016602482015275149bdd5b99081cda1bdd5b190818994818db1bdcd95960521b6044820152606401610a91565b600f546121569043614016565b600a82018190556040519081527ffd1cd879b90803328042915a0dab567886d80637d84c7875df6a3e4495c379ac90602001611281565b600d5460ff16156121b157604051636bf43ac160e11b815260040160405180910390fd5b61ffff81166000908152600e60205260409020600981015460ff16156121ea5760405163a89ac15160e01b815260040160405180910390fd5b600a8101544381111561224057604051632c73a0bf60e11b815260206004820152601d60248201527f52616e646f6d2073656c656374696f6e206973206e6f742072656164790000006044820152606401610a91565b600061224d60018361404d565b401561229857814061226060018461404d565b6040805160208101939093529040908201524260608201526080016040516020818303038152906040528051906020012090506122e3565b6122a360014361404d565b406122af60024361404d565b6040805160208101939093529040908201524260608201526080016040516020818303038152906040528051906020012090505b60098301805460ff191660011790556006830154600784015460009161230b918491906131bf565b600b850181905560405161ffff871681529091507fd104aada4b066eb003582f2d87f8450d182d2a2306ab2aad08ab10d5564fe4c69060200160405180910390a15050505050565b6008546001600160a01b0316331461237d5760405162461bcd60e51b8152600401610a9190613ec7565b600d5460ff16156123a157604051636bf43ac160e11b815260040160405180910390fd5b60105461ffff16156123b5576123b5611d15565b60006001876123c360005490565b6123cd9190614016565b6123d7919061404d565b9050600c5481111561243a57604051632c73a0bf60e11b815260206004820152602560248201527f6d61784d696e74696e6749642065786365656420746865206d6178696d756d536044820152647570706c7960d81b6064820152608401610a91565b84861061248a57604051632c73a0bf60e11b815260206004820152601860248201527f656e6454696d652073686f756c642062652062696767657200000000000000006044820152606401610a91565b601080546000919082906124a19061ffff166140c5565b91906101000a81548161ffff021916908361ffff16021790559050604051806101c001604052808261ffff1681526020018a60018111156124f257634e487b7160e01b600052602160045260246000fd5b81526020018881526020018781526020018561ffff16815260200186815260200183815260200161252260005490565b8152600060208083018290526040805180830182528381528185015287151560608501526080840183905260a0840183905260c090930182905261ffff8086168352600e82529290912083518154931661ffff198416811782559184015190929091839162ffffff191617620100008360018111156125b157634e487b7160e01b600052602160045260246000fd5b02179055506040820151600182015560608201516002820155608082015160038201805461ffff191661ffff90921691909117905560a0820151600482015560c0820151600582015560e082015160068201556101008201516007820155610120820151805161262b9160088401916020909101906136df565b5061014082015160098201805460ff1916911515919091179055610160820151600a820155610180820151600b8201556101a090910151600c9091015560405161ffff821681527f812de79a73e727eb2b77c67cd0d01ab1559b5a9a841d0f247d28517f37ea16c89060200160405180910390a1505050505050505050565b6126b5848484612c6a565b6001600160a01b0383163b156126ee576126d1848484846131f7565b6126ee576040516368d2bf6b60e11b815260040160405180910390fd5b50505050565b6126fc613763565b61ffff8281166000908152600e602090815260409182902082516101c081019093528054938416835291929083019062010000900460ff16600181111561275357634e487b7160e01b600052602160045260246000fd5b600181111561277257634e487b7160e01b600052602160045260246000fd5b81526001820154602082015260028201546040820152600382015461ffff16606082015260048201546080820152600582015460a0820152600682015460c0820152600782015460e0820152600882018054610100909201916127d490614090565b80601f016020809104026020016040519081016040528092919081815260200182805461280090614090565b801561284d5780601f106128225761010080835404028352916020019161284d565b820191906000526020600020905b81548152906001019060200180831161283057829003601f168201915b5050509183525050600982015460ff1615156020820152600a8201546040820152600b8201546060820152600c9091015460809091015292915050565b60606000612897836132ef565b905080610140015180156128e457506040805160008152602081018083528151902061012084015190926128cb9201613dbe565b6040516020818303038152906040528051906020012014155b156129235760118161012001516128fa85613501565b60405160200161290c93929190613e0f565b604051602081830303815290604052915050919050565b601261292e84613501565b60405160200161290c929190613dda565b50919050565b6009546001600160a01b0316331480159061296b57506008546001600160a01b03163314155b1561298957604051630ba9cf7d60e01b815260040160405180910390fd5b600d5460ff16156129ad57604051636bf43ac160e11b815260040160405180910390fd5b61ffff82166000818152600e6020908152604091829020600181018590558251938452908301849052917f7196d16a1c019f6d894f4e212e61954e22117dbeaaf8573558b943c990db29479101610d7b565b6009546001600160a01b03163314801590612a2557506008546001600160a01b03163314155b15612a4357604051630ba9cf7d60e01b815260040160405180910390fd5b600d5460ff1615612a6757604051636bf43ac160e11b815260040160405180910390fd5b600054811015612a8a576040516335d5065160e11b815260040160405180910390fd5b600c54811115612aad576040516316b0321960e01b815260040160405180910390fd5b61ffff82166000818152600e6020908152604091829020600581018590558251938452908301849052917ffab54bb645c4d4a86ffcfe9885b75722d72fca682dadad89f1abe2f84bfe76929101610d7b565b6001600160a01b03918216600090815260076020908152604080832093909416825291909152205460ff1690565b6008546001600160a01b03163314612b575760405162461bcd60e51b8152600401610a9190613ec7565b6001600160a01b038116612bbc5760405162461bcd60e51b815260206004820152602660248201527f4f776e61626c653a206e6577206f776e657220697320746865207a65726f206160448201526564647265737360d01b6064820152608401610a91565b611e60816130a0565b600081600111158015612bd9575060005482105b8015610a3c575050600090815260046020526040902054600160e01b161590565b60008180600111612c5157600054811015612c5157600081815260046020526040902054600160e01b8116612c4f575b80612c48575060001901600081815260046020526040902054612c2a565b9392505050565b505b604051636f96cda160e11b815260040160405180910390fd5b6000612c7582612bfa565b9050836001600160a01b0316816001600160a01b031614612ca85760405162a1148160e81b815260040160405180910390fd5b6000336001600160a01b0386161480612cc65750612cc68533612aff565b80612ce1575033612cd684610bae565b6001600160a01b0316145b905080612d0157604051632ce44b5f60e11b815260040160405180910390fd5b6001600160a01b038416612d2857604051633a954ecd60e21b815260040160405180910390fd5b600083815260066020908152604080832080546001600160a01b03191690556001600160a01b038881168452600583528184208054600019019055871683528083208054600101905585835260049091529020600160e11b4260a01b861781179091558216612dc55760018301600081815260046020526040902054612dc3576000548114612dc35760008181526004602052604090208390555b505b82846001600160a01b0316866001600160a01b03167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef60405160405180910390a45050505050565b600b5460ff16612e565760405162461bcd60e51b815260206004820152601460248201527314185d5cd8589b194e881b9bdd081c185d5cd95960621b6044820152606401610a91565b600b805460ff191690557f5db9ee0a495bf2e6ff9c91a7834c1ba4fdd244a5e8aa4e537bd38aeae4b073aa335b6040516001600160a01b03909116815260200160405180910390a1565b6000612eaa610d88565b90506002816004811115612ece57634e487b7160e01b600052602160045260246000fd5b14612eec57604051632ac9194b60e21b815260040160405180910390fd5b600183602001516001811115612f1257634e487b7160e01b600052602160045260246000fd5b148015612f2a57506008546001600160a01b03163314155b8015612f4157506009546001600160a01b03163314155b15612f5f57604051630ba9cf7d60e01b815260040160405180910390fd5b81836080015161ffff161015612f9457608083015160405163e70937a360e01b815261ffff9091166004820152602401610a91565b612fa2828460c00151613062565b612fbf57604051636f5606f560e01b815260040160405180910390fd5b600083602001516001811115612fe557634e487b7160e01b600052602160045260246000fd5b148015612ffd57506008546001600160a01b03163314155b801561301457506009546001600160a01b03163314155b15610fad576000828460a0015161302b919061402e565b90503481146126ee5760405163bfa3573760e01b815260040160405180910390fd5b60006130598383613550565b50600192915050565b600061306f826001614016565b8361307960005490565b6130839190614016565b111561305957506000610a3c565b606060118054610b2b90614090565b600880546001600160a01b038381166001600160a01b0319831681179093556040519116919082907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e090600090a35050565b600980546001600160a01b038381166001600160a01b0319831681179093556040519116919082907ff8ccb027dfcd135e000e9d45e6cc2d662578a8825d4c45b5e32e0adf67e79ec690600090a35050565b600b5460ff161561318a5760405162461bcd60e51b815260206004820152601060248201526f14185d5cd8589b194e881c185d5cd95960821b6044820152606401610a91565b600b805460ff191660011790557f62e78cea01bee320cd4e420270b5ea74000d11b0c9f74754ebdbfc544b05a258612e833390565b6000806131cc848461404d565b6131d7906001614016565b9050836131e48287614102565b6131ee9190614016565b95945050505050565b604051630a85bd0160e11b81526000906001600160a01b0385169063150b7a029061322c903390899088908890600401613e67565b602060405180830381600087803b15801561324657600080fd5b505af1925050508015613276575060408051601f3d908101601f1916820190925261327391810190613b43565b60015b6132d1573d8080156132a4576040519150601f19603f3d011682016040523d82523d6000602084013e6132a9565b606091505b5080516132c9576040516368d2bf6b60e11b815260040160405180910390fd5b805181602001fd5b6001600160e01b031916630a85bd0160e11b1490505b949350505050565b6132f7613763565b61330082612bc5565b6133205760405163b5706f4f60e01b815260048101839052602401610a91565b60015b60105461ffff9081169082161161293f5761ffff8181166000908152600e602090815260409182902082516101c081019093528054938416835291929083019062010000900460ff16600181111561338b57634e487b7160e01b600052602160045260246000fd5b60018111156133aa57634e487b7160e01b600052602160045260246000fd5b81526001820154602082015260028201546040820152600382015461ffff16606082015260048201546080820152600582015460a0820152600682015460c0820152600782015460e08201526008820180546101009092019161340c90614090565b80601f016020809104026020016040519081016040528092919081815260200182805461343890614090565b80156134855780601f1061345a57610100808354040283529160200191613485565b820191906000526020600020905b81548152906001019060200180831161346857829003601f168201915b5050509183525050600982015460ff1615156020820152600a8201546040820152600b8201546060820152600c909101546080909101526101008101519092506000906134d6578260c001516134dd565b8261010001515b9050808411156134fa57816134f1816140c5565b92505050613323565b5050919050565b604080516080810191829052607f0190826030600a8206018353600a90045b801561353e57600183039250600a81066030018353600a9004613520565b50819003601f19909101908152919050565b61356a82826040518060200160405280600081525061356e565b5050565b6000546001600160a01b03841661359757604051622e076360e81b815260040160405180910390fd5b826135b55760405163b562e8dd60e01b815260040160405180910390fd5b6001600160a01b03841660008181526005602090815260408083208054680100000000000000018902019055848352600490915290204260a01b86176001861460e11b1790558190818501903b1561368a575b60405182906001600160a01b038816906000907fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef908290a461365360008784806001019550876131f7565b613670576040516368d2bf6b60e11b815260040160405180910390fd5b80821061360857826000541461368557600080fd5b6136cf565b5b6040516001830192906001600160a01b038816906000907fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef908290a480821061368b575b5060009081556126ee9085838684565b8280546136eb90614090565b90600052602060002090601f01602090048101928261370d5760008555613753565b82601f1061372657805160ff1916838001178555613753565b82800160010185558215613753579182015b82811115613753578251825591602001919060010190613738565b5061375f9291506137d9565b5090565b604080516101c081019091526000808252602082019081526020016000815260200160008152602001600061ffff16815260200160008152602001600081526020016000815260200160008152602001606081526020016000151581526020016000815260200160008152602001600081525090565b5b8082111561375f57600081556001016137da565b600067ffffffffffffffff808411156138095761380961414e565b604051601f8501601f19908116603f011681019082821181831017156138315761383161414e565b8160405280935085815286868601111561384a57600080fd5b858560208301376000602087830101525050509392505050565b80356001600160a01b038116811461387b57600080fd5b919050565b60008083601f840112613891578081fd5b50813567ffffffffffffffff8111156138a8578182fd5b6020830191508360208260051b85010111156138c357600080fd5b9250929050565b8035801515811461387b57600080fd5b80356002811061387b57600080fd5b600082601f8301126138f9578081fd5b612c48838335602085016137ee565b803561ffff8116811461387b57600080fd5b60006020828403121561392b578081fd5b612c4882613864565b60008060408385031215613946578081fd5b61394f83613864565b915061395d60208401613864565b90509250929050565b60008060006060848603121561397a578081fd5b61398384613864565b925061399160208501613864565b9150604084013590509250925092565b600080600080608085870312156139b6578081fd5b6139bf85613864565b93506139cd60208601613864565b925060408501359150606085013567ffffffffffffffff8111156139ef578182fd5b8501601f810187136139ff578182fd5b613a0e878235602084016137ee565b91505092959194509250565b60008060408385031215613a2c578182fd5b613a3583613864565b915061395d602084016138ca565b60008060408385031215613a55578182fd5b613a5e83613864565b946020939093013593505050565b60008060008060408587031215613a81578384fd5b843567ffffffffffffffff80821115613a98578586fd5b613aa488838901613880565b90965094506020870135915080821115613abc578384fd5b50613ac987828801613880565b95989497509550505050565b600080600060408486031215613ae9578283fd5b833567ffffffffffffffff811115613aff578384fd5b613b0b86828701613880565b9094509250613b1e905060208501613864565b90509250925092565b600060208284031215613b38578081fd5b8135612c4881614164565b600060208284031215613b54578081fd5b8151612c4881614164565b600080600080600080600060e0888a031215613b79578485fd5b613b82886138da565b965060208801359550604088013594506060880135935060808801359250613bac60a08901613908565b9150613bba60c089016138ca565b905092959891949750929550565b600060208284031215613bd9578081fd5b813567ffffffffffffffff811115613bef578182fd5b6132e7848285016138e9565b600060208284031215613c0c578081fd5b612c4882613908565b60008060408385031215613c27578182fd5b613c3083613908565b915061395d602084016138da565b60008060408385031215613c50578182fd5b613c5983613908565b9150602083013567ffffffffffffffff811115613c74578182fd5b613c80858286016138e9565b9150509250929050565b60008060408385031215613c9c578182fd5b613ca583613908565b915061395d60208401613908565b60008060408385031215613cc5578182fd5b613a5e83613908565b600060208284031215613cdf578081fd5b5035919050565b60008151808452613cfe816020860160208601614064565b601f01601f19169290920160200192915050565b60028110613d2257613d22614138565b9052565b8054600090600181811c9080831680613d4057607f831692505b6020808410821415613d6057634e487b7160e01b86526022600452602486fd5b818015613d745760018114613d8557613db2565b60ff19861689528489019650613db2565b60008881526020902060005b86811015613daa5781548b820152908501908301613d91565b505084890196505b50505050505092915050565b60008251613dd0818460208701614064565b9190910192915050565b6000613de68285613d26565b8351613df6818360208801614064565b64173539b7b760d91b9101908152600501949350505050565b6000613e1b8286613d26565b8451613e2b818360208901614064565b602f60f81b91019081528351613e48816001840160208801614064565b64173539b7b760d91b6001929091019182015260060195945050505050565b6001600160a01b0385811682528416602082015260408101839052608060608201819052600090610f9890830184613ce6565b6020810160058310613eae57613eae614138565b91905290565b602081526000612c486020830184613ce6565b6020808252818101527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604082015260600190565b60208152613f1160208201835161ffff169052565b60006020830151613f256040840182613d12565b5060408301516060830152606083015160808301526080830151613f4f60a084018261ffff169052565b5060a083015160c083015260c083015160e083015260e08301516101008181850152808501519150506101208181850152808501519150506101c06101408181860152613fa06101e0860184613ce6565b90860151909250610160613fb78682018315159052565b860151610180868101919091528601516101a0808701919091529095015193019290925250919050565b61ffff8316815260408101612c486020830184613d12565b61ffff831681526040602082015260006132e76040830184613ce6565b6000821982111561402957614029614122565b500190565b600081600019048311821515161561404857614048614122565b500290565b60008282101561405f5761405f614122565b500390565b60005b8381101561407f578181015183820152602001614067565b838111156126ee5750506000910152565b600181811c908216806140a457607f821691505b6020821081141561293f57634e487b7160e01b600052602260045260246000fd5b600061ffff808316818114156140dd576140dd614122565b6001019392505050565b60006000198214156140fb576140fb614122565b5060010190565b60008261411d57634e487b7160e01b81526012600452602481fd5b500690565b634e487b7160e01b600052601160045260246000fd5b634e487b7160e01b600052602160045260246000fd5b634e487b7160e01b600052604160045260246000fd5b6001600160e01b031981168114611e6057600080fdfea2646970667358221220432d56aba4fe74d60caeda2ea31b4cad4b5ea4997b032804cfa22624e148da6c64736f6c63430008040033

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

0000000000000000000000006c9e30f0340e32fde08deade9be98ad83df05e73000000000000000000000000000000000000000000000000000000000000047800000000000000000000000000000000000000000000000000000000000000a000000000000000000000000000000000000000000000000000000000000000e00000000000000000000000006c9e30f0340e32fde08deade9be98ad83df05e73000000000000000000000000000000000000000000000000000000000000001b68747470733a2f2f6b6d6d2e6d7970696e6174612e636c6f75642f0000000000000000000000000000000000000000000000000000000000000000000000002568747470733a2f2f676174657761792d697066732e61746f6d726967732e696f2f6b6d6d2f000000000000000000000000000000000000000000000000000000

-----Decoded View---------------
Arg [0] : receiver (address): 0x6c9e30F0340e32fDe08deADE9bE98ad83df05e73
Arg [1] : maxSupply (uint256): 1144
Arg [2] : _baseUri (string): https://kmm.mypinata.cloud/
Arg [3] : _defaultUnrevealedURI (string): https://gateway-ipfs.atomrigs.io/kmm/
Arg [4] : admin_ (address): 0x6c9e30F0340e32fDe08deADE9bE98ad83df05e73

-----Encoded View---------------
10 Constructor Arguments found :
Arg [0] : 0000000000000000000000006c9e30f0340e32fde08deade9be98ad83df05e73
Arg [1] : 0000000000000000000000000000000000000000000000000000000000000478
Arg [2] : 00000000000000000000000000000000000000000000000000000000000000a0
Arg [3] : 00000000000000000000000000000000000000000000000000000000000000e0
Arg [4] : 0000000000000000000000006c9e30f0340e32fde08deade9be98ad83df05e73
Arg [5] : 000000000000000000000000000000000000000000000000000000000000001b
Arg [6] : 68747470733a2f2f6b6d6d2e6d7970696e6174612e636c6f75642f0000000000
Arg [7] : 0000000000000000000000000000000000000000000000000000000000000025
Arg [8] : 68747470733a2f2f676174657761792d697066732e61746f6d726967732e696f
Arg [9] : 2f6b6d6d2f000000000000000000000000000000000000000000000000000000


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.