ETH Price: $2,155.55 (+0.71%)

Transaction Decoder

Block:
14940390 at Jun-10-2022 08:28:24 PM +UTC
Transaction Fee:
0.00586729377852399 ETH $12.65
Gas Used:
123,930 Gas / 47.343611543 Gwei

Emitted Events:

Account State Difference:

  Address   Before After State Difference Code
0x3E509376...0AB1b37aD
(F2Pool Old)
3,303.027008739566255483 Eth3,303.027194634566255483 Eth0.000185895
0x93408577...F0e269b10
0xBC3D193b...B4976c486
0.140912555023327374 Eth
Nonce: 9
0.135045261244803384 Eth
Nonce: 10
0.00586729377852399

Execution Trace

0x3e509376de25c78bba1ab37ab86027c0ab1b37ad.c63ff8dd( )
  • Null: 0x000...001.fdf6f9d6( )
  • DropERC721.balanceOf( owner=0xBC3D193baE8e351e39891cCfF3258bAB4976c486 ) => ( 1 )
    • DropERC721.balanceOf( owner=0xBC3D193baE8e351e39891cCfF3258bAB4976c486 ) => ( 1 )
    • DropERC721.tokenOfOwnerByIndex( owner=0xBC3D193baE8e351e39891cCfF3258bAB4976c486, index=0 ) => ( 365 )
      • DropERC721.tokenOfOwnerByIndex( owner=0xBC3D193baE8e351e39891cCfF3258bAB4976c486, index=0 ) => ( 365 )
      • Edu.balanceOf( account=0x3E509376dE25c78BBA1aB37Ab86027C0AB1b37aD ) => ( 34180000000000000000000 )
      • Edu.transfer( to=0xBC3D193baE8e351e39891cCfF3258bAB4976c486, amount=60000000000000000000 ) => ( True )
        File 1 of 3: Edu
        // SPDX-License-Identifier: MIT
        pragma solidity ^0.8.9;
        import { IERC20 } from "@openzeppelin/contracts/token/ERC20/IERC20.sol";
        import { IERC165 } from "@openzeppelin/contracts/utils/introspection/IERC165.sol";
        interface IL2StandardERC20 is IERC20, IERC165 {
            function l1Token() external returns (address);
            function mint(address _to, uint256 _amount) external;
            function burn(address _from, uint256 _amount) external;
            event Mint(address indexed _account, uint256 _amount);
            event Burn(address indexed _account, uint256 _amount);
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v4.6.0) (token/ERC20/ERC20.sol)
        pragma solidity ^0.8.0;
        import "./IERC20.sol";
        import "./extensions/IERC20Metadata.sol";
        import "../../utils/Context.sol";
        /**
         * @dev Implementation of the {IERC20} interface.
         *
         * This implementation is agnostic to the way tokens are created. This means
         * that a supply mechanism has to be added in a derived contract using {_mint}.
         * For a generic mechanism see {ERC20PresetMinterPauser}.
         *
         * TIP: For a detailed writeup see our guide
         * https://forum.zeppelin.solutions/t/how-to-implement-erc20-supply-mechanisms/226[How
         * to implement supply mechanisms].
         *
         * We have followed general OpenZeppelin Contracts guidelines: functions revert
         * instead returning `false` on failure. This behavior is nonetheless
         * conventional and does not conflict with the expectations of ERC20
         * applications.
         *
         * Additionally, an {Approval} event is emitted on calls to {transferFrom}.
         * This allows applications to reconstruct the allowance for all accounts just
         * by listening to said events. Other implementations of the EIP may not emit
         * these events, as it isn't required by the specification.
         *
         * Finally, the non-standard {decreaseAllowance} and {increaseAllowance}
         * functions have been added to mitigate the well-known issues around setting
         * allowances. See {IERC20-approve}.
         */
        contract ERC20 is Context, IERC20, IERC20Metadata {
            mapping(address => uint256) private _balances;
            mapping(address => mapping(address => uint256)) private _allowances;
            uint256 private _totalSupply;
            string private _name;
            string private _symbol;
            /**
             * @dev Sets the values for {name} and {symbol}.
             *
             * The default value of {decimals} is 18. To select a different value for
             * {decimals} you should overload it.
             *
             * All two of these values are immutable: they can only be set once during
             * construction.
             */
            constructor(string memory name_, string memory symbol_) {
                _name = name_;
                _symbol = symbol_;
            }
            /**
             * @dev Returns the name of the token.
             */
            function name() public view virtual override returns (string memory) {
                return _name;
            }
            /**
             * @dev Returns the symbol of the token, usually a shorter version of the
             * name.
             */
            function symbol() public view virtual override returns (string memory) {
                return _symbol;
            }
            /**
             * @dev Returns the number of decimals used to get its user representation.
             * For example, if `decimals` equals `2`, a balance of `505` tokens should
             * be displayed to a user as `5.05` (`505 / 10 ** 2`).
             *
             * Tokens usually opt for a value of 18, imitating the relationship between
             * Ether and Wei. This is the value {ERC20} uses, unless this function is
             * overridden;
             *
             * NOTE: This information is only used for _display_ purposes: it in
             * no way affects any of the arithmetic of the contract, including
             * {IERC20-balanceOf} and {IERC20-transfer}.
             */
            function decimals() public view virtual override returns (uint8) {
                return 18;
            }
            /**
             * @dev See {IERC20-totalSupply}.
             */
            function totalSupply() public view virtual override returns (uint256) {
                return _totalSupply;
            }
            /**
             * @dev See {IERC20-balanceOf}.
             */
            function balanceOf(address account) public view virtual override returns (uint256) {
                return _balances[account];
            }
            /**
             * @dev See {IERC20-transfer}.
             *
             * Requirements:
             *
             * - `to` cannot be the zero address.
             * - the caller must have a balance of at least `amount`.
             */
            function transfer(address to, uint256 amount) public virtual override returns (bool) {
                address owner = _msgSender();
                _transfer(owner, to, amount);
                return true;
            }
            /**
             * @dev See {IERC20-allowance}.
             */
            function allowance(address owner, address spender) public view virtual override returns (uint256) {
                return _allowances[owner][spender];
            }
            /**
             * @dev See {IERC20-approve}.
             *
             * NOTE: If `amount` is the maximum `uint256`, the allowance is not updated on
             * `transferFrom`. This is semantically equivalent to an infinite approval.
             *
             * Requirements:
             *
             * - `spender` cannot be the zero address.
             */
            function approve(address spender, uint256 amount) public virtual override returns (bool) {
                address owner = _msgSender();
                _approve(owner, spender, amount);
                return true;
            }
            /**
             * @dev See {IERC20-transferFrom}.
             *
             * Emits an {Approval} event indicating the updated allowance. This is not
             * required by the EIP. See the note at the beginning of {ERC20}.
             *
             * NOTE: Does not update the allowance if the current allowance
             * is the maximum `uint256`.
             *
             * Requirements:
             *
             * - `from` and `to` cannot be the zero address.
             * - `from` must have a balance of at least `amount`.
             * - the caller must have allowance for ``from``'s tokens of at least
             * `amount`.
             */
            function transferFrom(
                address from,
                address to,
                uint256 amount
            ) public virtual override returns (bool) {
                address spender = _msgSender();
                _spendAllowance(from, spender, amount);
                _transfer(from, to, amount);
                return true;
            }
            /**
             * @dev Atomically increases the allowance granted to `spender` by the caller.
             *
             * This is an alternative to {approve} that can be used as a mitigation for
             * problems described in {IERC20-approve}.
             *
             * Emits an {Approval} event indicating the updated allowance.
             *
             * Requirements:
             *
             * - `spender` cannot be the zero address.
             */
            function increaseAllowance(address spender, uint256 addedValue) public virtual returns (bool) {
                address owner = _msgSender();
                _approve(owner, spender, allowance(owner, spender) + addedValue);
                return true;
            }
            /**
             * @dev Atomically decreases the allowance granted to `spender` by the caller.
             *
             * This is an alternative to {approve} that can be used as a mitigation for
             * problems described in {IERC20-approve}.
             *
             * Emits an {Approval} event indicating the updated allowance.
             *
             * Requirements:
             *
             * - `spender` cannot be the zero address.
             * - `spender` must have allowance for the caller of at least
             * `subtractedValue`.
             */
            function decreaseAllowance(address spender, uint256 subtractedValue) public virtual returns (bool) {
                address owner = _msgSender();
                uint256 currentAllowance = allowance(owner, spender);
                require(currentAllowance >= subtractedValue, "ERC20: decreased allowance below zero");
                unchecked {
                    _approve(owner, spender, currentAllowance - subtractedValue);
                }
                return true;
            }
            /**
             * @dev Moves `amount` of tokens from `sender` to `recipient`.
             *
             * This internal function is equivalent to {transfer}, and can be used to
             * e.g. implement automatic token fees, slashing mechanisms, etc.
             *
             * Emits a {Transfer} event.
             *
             * Requirements:
             *
             * - `from` cannot be the zero address.
             * - `to` cannot be the zero address.
             * - `from` must have a balance of at least `amount`.
             */
            function _transfer(
                address from,
                address to,
                uint256 amount
            ) internal virtual {
                require(from != address(0), "ERC20: transfer from the zero address");
                require(to != address(0), "ERC20: transfer to the zero address");
                _beforeTokenTransfer(from, to, amount);
                uint256 fromBalance = _balances[from];
                require(fromBalance >= amount, "ERC20: transfer amount exceeds balance");
                unchecked {
                    _balances[from] = fromBalance - amount;
                }
                _balances[to] += amount;
                emit Transfer(from, to, amount);
                _afterTokenTransfer(from, to, amount);
            }
            /** @dev Creates `amount` tokens and assigns them to `account`, increasing
             * the total supply.
             *
             * Emits a {Transfer} event with `from` set to the zero address.
             *
             * Requirements:
             *
             * - `account` cannot be the zero address.
             */
            function _mint(address account, uint256 amount) internal virtual {
                require(account != address(0), "ERC20: mint to the zero address");
                _beforeTokenTransfer(address(0), account, amount);
                _totalSupply += amount;
                _balances[account] += amount;
                emit Transfer(address(0), account, amount);
                _afterTokenTransfer(address(0), account, amount);
            }
            /**
             * @dev Destroys `amount` tokens from `account`, reducing the
             * total supply.
             *
             * Emits a {Transfer} event with `to` set to the zero address.
             *
             * Requirements:
             *
             * - `account` cannot be the zero address.
             * - `account` must have at least `amount` tokens.
             */
            function _burn(address account, uint256 amount) internal virtual {
                require(account != address(0), "ERC20: burn from the zero address");
                _beforeTokenTransfer(account, address(0), amount);
                uint256 accountBalance = _balances[account];
                require(accountBalance >= amount, "ERC20: burn amount exceeds balance");
                unchecked {
                    _balances[account] = accountBalance - amount;
                }
                _totalSupply -= amount;
                emit Transfer(account, address(0), amount);
                _afterTokenTransfer(account, address(0), amount);
            }
            /**
             * @dev Sets `amount` as the allowance of `spender` over the `owner` s tokens.
             *
             * This internal function is equivalent to `approve`, and can be used to
             * e.g. set automatic allowances for certain subsystems, etc.
             *
             * Emits an {Approval} event.
             *
             * Requirements:
             *
             * - `owner` cannot be the zero address.
             * - `spender` cannot be the zero address.
             */
            function _approve(
                address owner,
                address spender,
                uint256 amount
            ) internal virtual {
                require(owner != address(0), "ERC20: approve from the zero address");
                require(spender != address(0), "ERC20: approve to the zero address");
                _allowances[owner][spender] = amount;
                emit Approval(owner, spender, amount);
            }
            /**
             * @dev Updates `owner` s allowance for `spender` based on spent `amount`.
             *
             * Does not update the allowance amount in case of infinite allowance.
             * Revert if not enough allowance is available.
             *
             * Might emit an {Approval} event.
             */
            function _spendAllowance(
                address owner,
                address spender,
                uint256 amount
            ) internal virtual {
                uint256 currentAllowance = allowance(owner, spender);
                if (currentAllowance != type(uint256).max) {
                    require(currentAllowance >= amount, "ERC20: insufficient allowance");
                    unchecked {
                        _approve(owner, spender, currentAllowance - amount);
                    }
                }
            }
            /**
             * @dev Hook that is called before any transfer of tokens. This includes
             * minting and burning.
             *
             * Calling conditions:
             *
             * - when `from` and `to` are both non-zero, `amount` of ``from``'s tokens
             * will be transferred to `to`.
             * - when `from` is zero, `amount` tokens will be minted for `to`.
             * - when `to` is zero, `amount` of ``from``'s tokens will be burned.
             * - `from` and `to` are never both zero.
             *
             * To learn more about hooks, head to xref:ROOT:extending-contracts.adoc#using-hooks[Using Hooks].
             */
            function _beforeTokenTransfer(
                address from,
                address to,
                uint256 amount
            ) internal virtual {}
            /**
             * @dev Hook that is called after any transfer of tokens. This includes
             * minting and burning.
             *
             * Calling conditions:
             *
             * - when `from` and `to` are both non-zero, `amount` of ``from``'s tokens
             * has been transferred to `to`.
             * - when `from` is zero, `amount` tokens have been minted for `to`.
             * - when `to` is zero, `amount` of ``from``'s tokens have been burned.
             * - `from` and `to` are never both zero.
             *
             * To learn more about hooks, head to xref:ROOT:extending-contracts.adoc#using-hooks[Using Hooks].
             */
            function _afterTokenTransfer(
                address from,
                address to,
                uint256 amount
            ) internal virtual {}
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v4.6.0) (token/ERC20/IERC20.sol)
        pragma solidity ^0.8.0;
        /**
         * @dev Interface of the ERC20 standard as defined in the EIP.
         */
        interface IERC20 {
            /**
             * @dev Emitted when `value` tokens are moved from one account (`from`) to
             * another (`to`).
             *
             * Note that `value` may be zero.
             */
            event Transfer(address indexed from, address indexed to, uint256 value);
            /**
             * @dev Emitted when the allowance of a `spender` for an `owner` is set by
             * a call to {approve}. `value` is the new allowance.
             */
            event Approval(address indexed owner, address indexed spender, uint256 value);
            /**
             * @dev Returns the amount of tokens in existence.
             */
            function totalSupply() external view returns (uint256);
            /**
             * @dev Returns the amount of tokens owned by `account`.
             */
            function balanceOf(address account) external view returns (uint256);
            /**
             * @dev Moves `amount` tokens from the caller's account to `to`.
             *
             * Returns a boolean value indicating whether the operation succeeded.
             *
             * Emits a {Transfer} event.
             */
            function transfer(address to, uint256 amount) external returns (bool);
            /**
             * @dev Returns the remaining number of tokens that `spender` will be
             * allowed to spend on behalf of `owner` through {transferFrom}. This is
             * zero by default.
             *
             * This value changes when {approve} or {transferFrom} are called.
             */
            function allowance(address owner, address spender) external view returns (uint256);
            /**
             * @dev Sets `amount` as the allowance of `spender` over the caller's tokens.
             *
             * Returns a boolean value indicating whether the operation succeeded.
             *
             * IMPORTANT: Beware that changing an allowance with this method brings the risk
             * that someone may use both the old and the new allowance by unfortunate
             * transaction ordering. One possible solution to mitigate this race
             * condition is to first reduce the spender's allowance to 0 and set the
             * desired value afterwards:
             * https://github.com/ethereum/EIPs/issues/20#issuecomment-263524729
             *
             * Emits an {Approval} event.
             */
            function approve(address spender, uint256 amount) external returns (bool);
            /**
             * @dev Moves `amount` tokens from `from` to `to` using the
             * allowance mechanism. `amount` is then deducted from the caller's
             * allowance.
             *
             * Returns a boolean value indicating whether the operation succeeded.
             *
             * Emits a {Transfer} event.
             */
            function transferFrom(
                address from,
                address to,
                uint256 amount
            ) external returns (bool);
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v4.5.0) (token/ERC20/extensions/ERC20Burnable.sol)
        pragma solidity ^0.8.0;
        import "../ERC20.sol";
        import "../../../utils/Context.sol";
        /**
         * @dev Extension of {ERC20} that allows token holders to destroy both their own
         * tokens and those that they have an allowance for, in a way that can be
         * recognized off-chain (via event analysis).
         */
        abstract contract ERC20Burnable is Context, ERC20 {
            /**
             * @dev Destroys `amount` tokens from the caller.
             *
             * See {ERC20-_burn}.
             */
            function burn(uint256 amount) public virtual {
                _burn(_msgSender(), amount);
            }
            /**
             * @dev Destroys `amount` tokens from `account`, deducting from the caller's
             * allowance.
             *
             * See {ERC20-_burn} and {ERC20-allowance}.
             *
             * Requirements:
             *
             * - the caller must have allowance for ``accounts``'s tokens of at least
             * `amount`.
             */
            function burnFrom(address account, uint256 amount) public virtual {
                _spendAllowance(account, _msgSender(), amount);
                _burn(account, amount);
            }
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts v4.4.1 (token/ERC20/extensions/IERC20Metadata.sol)
        pragma solidity ^0.8.0;
        import "../IERC20.sol";
        /**
         * @dev Interface for the optional metadata functions from the ERC20 standard.
         *
         * _Available since v4.1._
         */
        interface IERC20Metadata is IERC20 {
            /**
             * @dev Returns the name of the token.
             */
            function name() external view returns (string memory);
            /**
             * @dev Returns the symbol of the token.
             */
            function symbol() external view returns (string memory);
            /**
             * @dev Returns the decimals places of the token.
             */
            function decimals() external view returns (uint8);
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v4.6.0) (token/ERC20/extensions/draft-ERC20Permit.sol)
        pragma solidity ^0.8.0;
        import "./draft-IERC20Permit.sol";
        import "../ERC20.sol";
        import "../../../utils/cryptography/draft-EIP712.sol";
        import "../../../utils/cryptography/ECDSA.sol";
        import "../../../utils/Counters.sol";
        /**
         * @dev Implementation of the ERC20 Permit extension allowing approvals to be made via signatures, as defined in
         * https://eips.ethereum.org/EIPS/eip-2612[EIP-2612].
         *
         * Adds the {permit} method, which can be used to change an account's ERC20 allowance (see {IERC20-allowance}) by
         * presenting a message signed by the account. By not relying on `{IERC20-approve}`, the token holder account doesn't
         * need to send a transaction, and thus is not required to hold Ether at all.
         *
         * _Available since v3.4._
         */
        abstract contract ERC20Permit is ERC20, IERC20Permit, EIP712 {
            using Counters for Counters.Counter;
            mapping(address => Counters.Counter) private _nonces;
            // solhint-disable-next-line var-name-mixedcase
            bytes32 private constant _PERMIT_TYPEHASH =
                keccak256("Permit(address owner,address spender,uint256 value,uint256 nonce,uint256 deadline)");
            /**
             * @dev In previous versions `_PERMIT_TYPEHASH` was declared as `immutable`.
             * However, to ensure consistency with the upgradeable transpiler, we will continue
             * to reserve a slot.
             * @custom:oz-renamed-from _PERMIT_TYPEHASH
             */
            // solhint-disable-next-line var-name-mixedcase
            bytes32 private _PERMIT_TYPEHASH_DEPRECATED_SLOT;
            /**
             * @dev Initializes the {EIP712} domain separator using the `name` parameter, and setting `version` to `"1"`.
             *
             * It's a good idea to use the same `name` that is defined as the ERC20 token name.
             */
            constructor(string memory name) EIP712(name, "1") {}
            /**
             * @dev See {IERC20Permit-permit}.
             */
            function permit(
                address owner,
                address spender,
                uint256 value,
                uint256 deadline,
                uint8 v,
                bytes32 r,
                bytes32 s
            ) public virtual override {
                require(block.timestamp <= deadline, "ERC20Permit: expired deadline");
                bytes32 structHash = keccak256(abi.encode(_PERMIT_TYPEHASH, owner, spender, value, _useNonce(owner), deadline));
                bytes32 hash = _hashTypedDataV4(structHash);
                address signer = ECDSA.recover(hash, v, r, s);
                require(signer == owner, "ERC20Permit: invalid signature");
                _approve(owner, spender, value);
            }
            /**
             * @dev See {IERC20Permit-nonces}.
             */
            function nonces(address owner) public view virtual override returns (uint256) {
                return _nonces[owner].current();
            }
            /**
             * @dev See {IERC20Permit-DOMAIN_SEPARATOR}.
             */
            // solhint-disable-next-line func-name-mixedcase
            function DOMAIN_SEPARATOR() external view override returns (bytes32) {
                return _domainSeparatorV4();
            }
            /**
             * @dev "Consume a nonce": return the current value and increment.
             *
             * _Available since v4.1._
             */
            function _useNonce(address owner) internal virtual returns (uint256 current) {
                Counters.Counter storage nonce = _nonces[owner];
                current = nonce.current();
                nonce.increment();
            }
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts v4.4.1 (token/ERC20/extensions/draft-IERC20Permit.sol)
        pragma solidity ^0.8.0;
        /**
         * @dev Interface of the ERC20 Permit extension allowing approvals to be made via signatures, as defined in
         * https://eips.ethereum.org/EIPS/eip-2612[EIP-2612].
         *
         * Adds the {permit} method, which can be used to change an account's ERC20 allowance (see {IERC20-allowance}) by
         * presenting a message signed by the account. By not relying on {IERC20-approve}, the token holder account doesn't
         * need to send a transaction, and thus is not required to hold Ether at all.
         */
        interface IERC20Permit {
            /**
             * @dev Sets `value` as the allowance of `spender` over ``owner``'s tokens,
             * given ``owner``'s signed approval.
             *
             * IMPORTANT: The same issues {IERC20-approve} has related to transaction
             * ordering also apply here.
             *
             * Emits an {Approval} event.
             *
             * Requirements:
             *
             * - `spender` cannot be the zero address.
             * - `deadline` must be a timestamp in the future.
             * - `v`, `r` and `s` must be a valid `secp256k1` signature from `owner`
             * over the EIP712-formatted function arguments.
             * - the signature must use ``owner``'s current nonce (see {nonces}).
             *
             * For more information on the signature format, see the
             * https://eips.ethereum.org/EIPS/eip-2612#specification[relevant EIP
             * section].
             */
            function permit(
                address owner,
                address spender,
                uint256 value,
                uint256 deadline,
                uint8 v,
                bytes32 r,
                bytes32 s
            ) external;
            /**
             * @dev Returns the current nonce for `owner`. This value must be
             * included whenever a signature is generated for {permit}.
             *
             * Every successful call to {permit} increases ``owner``'s nonce by one. This
             * prevents a signature from being used multiple times.
             */
            function nonces(address owner) external view returns (uint256);
            /**
             * @dev Returns the domain separator used in the encoding of the signature for {permit}, as defined by {EIP712}.
             */
            // solhint-disable-next-line func-name-mixedcase
            function DOMAIN_SEPARATOR() external view returns (bytes32);
        }
        // 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;
            }
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts v4.4.1 (utils/Counters.sol)
        pragma solidity ^0.8.0;
        /**
         * @title Counters
         * @author Matt Condon (@shrugs)
         * @dev Provides counters that can only be incremented, decremented or reset. This can be used e.g. to track the number
         * of elements in a mapping, issuing ERC721 ids, or counting request ids.
         *
         * Include with `using Counters for Counters.Counter;`
         */
        library Counters {
            struct Counter {
                // This variable should never be directly accessed by users of the library: interactions must be restricted to
                // the library's function. As of Solidity v0.5.2, this cannot be enforced, though there is a proposal to add
                // this feature: see https://github.com/ethereum/solidity/issues/4637
                uint256 _value; // default: 0
            }
            function current(Counter storage counter) internal view returns (uint256) {
                return counter._value;
            }
            function increment(Counter storage counter) internal {
                unchecked {
                    counter._value += 1;
                }
            }
            function decrement(Counter storage counter) internal {
                uint256 value = counter._value;
                require(value > 0, "Counter: decrement overflow");
                unchecked {
                    counter._value = value - 1;
                }
            }
            function reset(Counter storage counter) internal {
                counter._value = 0;
            }
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts v4.4.1 (utils/Strings.sol)
        pragma solidity ^0.8.0;
        /**
         * @dev String operations.
         */
        library Strings {
            bytes16 private constant _HEX_SYMBOLS = "0123456789abcdef";
            /**
             * @dev Converts a `uint256` to its ASCII `string` decimal representation.
             */
            function toString(uint256 value) internal pure returns (string memory) {
                // Inspired by OraclizeAPI's implementation - MIT licence
                // https://github.com/oraclize/ethereum-api/blob/b42146b063c7d6ee1358846c198246239e9360e8/oraclizeAPI_0.4.25.sol
                if (value == 0) {
                    return "0";
                }
                uint256 temp = value;
                uint256 digits;
                while (temp != 0) {
                    digits++;
                    temp /= 10;
                }
                bytes memory buffer = new bytes(digits);
                while (value != 0) {
                    digits -= 1;
                    buffer[digits] = bytes1(uint8(48 + uint256(value % 10)));
                    value /= 10;
                }
                return string(buffer);
            }
            /**
             * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation.
             */
            function toHexString(uint256 value) internal pure returns (string memory) {
                if (value == 0) {
                    return "0x00";
                }
                uint256 temp = value;
                uint256 length = 0;
                while (temp != 0) {
                    length++;
                    temp >>= 8;
                }
                return toHexString(value, length);
            }
            /**
             * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation with fixed length.
             */
            function toHexString(uint256 value, uint256 length) internal pure returns (string memory) {
                bytes memory buffer = new bytes(2 * length + 2);
                buffer[0] = "0";
                buffer[1] = "x";
                for (uint256 i = 2 * length + 1; i > 1; --i) {
                    buffer[i] = _HEX_SYMBOLS[value & 0xf];
                    value >>= 4;
                }
                require(value == 0, "Strings: hex length insufficient");
                return string(buffer);
            }
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v4.5.0) (utils/cryptography/ECDSA.sol)
        pragma solidity ^0.8.0;
        import "../Strings.sol";
        /**
         * @dev Elliptic Curve Digital Signature Algorithm (ECDSA) operations.
         *
         * These functions can be used to verify that a message was signed by the holder
         * of the private keys of a given address.
         */
        library ECDSA {
            enum RecoverError {
                NoError,
                InvalidSignature,
                InvalidSignatureLength,
                InvalidSignatureS,
                InvalidSignatureV
            }
            function _throwError(RecoverError error) private pure {
                if (error == RecoverError.NoError) {
                    return; // no error: do nothing
                } else if (error == RecoverError.InvalidSignature) {
                    revert("ECDSA: invalid signature");
                } else if (error == RecoverError.InvalidSignatureLength) {
                    revert("ECDSA: invalid signature length");
                } else if (error == RecoverError.InvalidSignatureS) {
                    revert("ECDSA: invalid signature 's' value");
                } else if (error == RecoverError.InvalidSignatureV) {
                    revert("ECDSA: invalid signature 'v' value");
                }
            }
            /**
             * @dev Returns the address that signed a hashed message (`hash`) with
             * `signature` or error string. This address can then be used for verification purposes.
             *
             * The `ecrecover` EVM opcode allows for malleable (non-unique) signatures:
             * this function rejects them by requiring the `s` value to be in the lower
             * half order, and the `v` value to be either 27 or 28.
             *
             * IMPORTANT: `hash` _must_ be the result of a hash operation for the
             * verification to be secure: it is possible to craft signatures that
             * recover to arbitrary addresses for non-hashed data. A safe way to ensure
             * this is by receiving a hash of the original message (which may otherwise
             * be too long), and then calling {toEthSignedMessageHash} on it.
             *
             * Documentation for signature generation:
             * - with https://web3js.readthedocs.io/en/v1.3.4/web3-eth-accounts.html#sign[Web3.js]
             * - with https://docs.ethers.io/v5/api/signer/#Signer-signMessage[ethers]
             *
             * _Available since v4.3._
             */
            function tryRecover(bytes32 hash, bytes memory signature) internal pure returns (address, RecoverError) {
                // Check the signature length
                // - case 65: r,s,v signature (standard)
                // - case 64: r,vs signature (cf https://eips.ethereum.org/EIPS/eip-2098) _Available since v4.1._
                if (signature.length == 65) {
                    bytes32 r;
                    bytes32 s;
                    uint8 v;
                    // ecrecover takes the signature parameters, and the only way to get them
                    // currently is to use assembly.
                    assembly {
                        r := mload(add(signature, 0x20))
                        s := mload(add(signature, 0x40))
                        v := byte(0, mload(add(signature, 0x60)))
                    }
                    return tryRecover(hash, v, r, s);
                } else if (signature.length == 64) {
                    bytes32 r;
                    bytes32 vs;
                    // ecrecover takes the signature parameters, and the only way to get them
                    // currently is to use assembly.
                    assembly {
                        r := mload(add(signature, 0x20))
                        vs := mload(add(signature, 0x40))
                    }
                    return tryRecover(hash, r, vs);
                } else {
                    return (address(0), RecoverError.InvalidSignatureLength);
                }
            }
            /**
             * @dev Returns the address that signed a hashed message (`hash`) with
             * `signature`. This address can then be used for verification purposes.
             *
             * The `ecrecover` EVM opcode allows for malleable (non-unique) signatures:
             * this function rejects them by requiring the `s` value to be in the lower
             * half order, and the `v` value to be either 27 or 28.
             *
             * IMPORTANT: `hash` _must_ be the result of a hash operation for the
             * verification to be secure: it is possible to craft signatures that
             * recover to arbitrary addresses for non-hashed data. A safe way to ensure
             * this is by receiving a hash of the original message (which may otherwise
             * be too long), and then calling {toEthSignedMessageHash} on it.
             */
            function recover(bytes32 hash, bytes memory signature) internal pure returns (address) {
                (address recovered, RecoverError error) = tryRecover(hash, signature);
                _throwError(error);
                return recovered;
            }
            /**
             * @dev Overload of {ECDSA-tryRecover} that receives the `r` and `vs` short-signature fields separately.
             *
             * See https://eips.ethereum.org/EIPS/eip-2098[EIP-2098 short signatures]
             *
             * _Available since v4.3._
             */
            function tryRecover(
                bytes32 hash,
                bytes32 r,
                bytes32 vs
            ) internal pure returns (address, RecoverError) {
                bytes32 s = vs & bytes32(0x7fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff);
                uint8 v = uint8((uint256(vs) >> 255) + 27);
                return tryRecover(hash, v, r, s);
            }
            /**
             * @dev Overload of {ECDSA-recover} that receives the `r and `vs` short-signature fields separately.
             *
             * _Available since v4.2._
             */
            function recover(
                bytes32 hash,
                bytes32 r,
                bytes32 vs
            ) internal pure returns (address) {
                (address recovered, RecoverError error) = tryRecover(hash, r, vs);
                _throwError(error);
                return recovered;
            }
            /**
             * @dev Overload of {ECDSA-tryRecover} that receives the `v`,
             * `r` and `s` signature fields separately.
             *
             * _Available since v4.3._
             */
            function tryRecover(
                bytes32 hash,
                uint8 v,
                bytes32 r,
                bytes32 s
            ) internal pure returns (address, RecoverError) {
                // EIP-2 still allows signature malleability for ecrecover(). Remove this possibility and make the signature
                // unique. Appendix F in the Ethereum Yellow paper (https://ethereum.github.io/yellowpaper/paper.pdf), defines
                // the valid range for s in (301): 0 < s < secp256k1n ÷ 2 + 1, and for v in (302): v ∈ {27, 28}. Most
                // signatures from current libraries generate a unique signature with an s-value in the lower half order.
                //
                // If your library generates malleable signatures, such as s-values in the upper range, calculate a new s-value
                // with 0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEBAAEDCE6AF48A03BBFD25E8CD0364141 - s1 and flip v from 27 to 28 or
                // vice versa. If your library also generates signatures with 0/1 for v instead 27/28, add 27 to v to accept
                // these malleable signatures as well.
                if (uint256(s) > 0x7FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF5D576E7357A4501DDFE92F46681B20A0) {
                    return (address(0), RecoverError.InvalidSignatureS);
                }
                if (v != 27 && v != 28) {
                    return (address(0), RecoverError.InvalidSignatureV);
                }
                // If the signature is valid (and not malleable), return the signer address
                address signer = ecrecover(hash, v, r, s);
                if (signer == address(0)) {
                    return (address(0), RecoverError.InvalidSignature);
                }
                return (signer, RecoverError.NoError);
            }
            /**
             * @dev Overload of {ECDSA-recover} that receives the `v`,
             * `r` and `s` signature fields separately.
             */
            function recover(
                bytes32 hash,
                uint8 v,
                bytes32 r,
                bytes32 s
            ) internal pure returns (address) {
                (address recovered, RecoverError error) = tryRecover(hash, v, r, s);
                _throwError(error);
                return recovered;
            }
            /**
             * @dev Returns an Ethereum Signed Message, created from a `hash`. This
             * produces hash corresponding to the one signed with the
             * https://eth.wiki/json-rpc/API#eth_sign[`eth_sign`]
             * JSON-RPC method as part of EIP-191.
             *
             * See {recover}.
             */
            function toEthSignedMessageHash(bytes32 hash) internal pure returns (bytes32) {
                // 32 is the length in bytes of hash,
                // enforced by the type signature above
                return keccak256(abi.encodePacked("\\x19Ethereum Signed Message:\
        32", hash));
            }
            /**
             * @dev Returns an Ethereum Signed Message, created from `s`. This
             * produces hash corresponding to the one signed with the
             * https://eth.wiki/json-rpc/API#eth_sign[`eth_sign`]
             * JSON-RPC method as part of EIP-191.
             *
             * See {recover}.
             */
            function toEthSignedMessageHash(bytes memory s) internal pure returns (bytes32) {
                return keccak256(abi.encodePacked("\\x19Ethereum Signed Message:\
        ", Strings.toString(s.length), s));
            }
            /**
             * @dev Returns an Ethereum Signed Typed Data, created from a
             * `domainSeparator` and a `structHash`. This produces hash corresponding
             * to the one signed with the
             * https://eips.ethereum.org/EIPS/eip-712[`eth_signTypedData`]
             * JSON-RPC method as part of EIP-712.
             *
             * See {recover}.
             */
            function toTypedDataHash(bytes32 domainSeparator, bytes32 structHash) internal pure returns (bytes32) {
                return keccak256(abi.encodePacked("\\x19\\x01", domainSeparator, structHash));
            }
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts v4.4.1 (utils/cryptography/draft-EIP712.sol)
        pragma solidity ^0.8.0;
        import "./ECDSA.sol";
        /**
         * @dev https://eips.ethereum.org/EIPS/eip-712[EIP 712] is a standard for hashing and signing of typed structured data.
         *
         * The encoding specified in the EIP is very generic, and such a generic implementation in Solidity is not feasible,
         * thus this contract does not implement the encoding itself. Protocols need to implement the type-specific encoding
         * they need in their contracts using a combination of `abi.encode` and `keccak256`.
         *
         * This contract implements the EIP 712 domain separator ({_domainSeparatorV4}) that is used as part of the encoding
         * scheme, and the final step of the encoding to obtain the message digest that is then signed via ECDSA
         * ({_hashTypedDataV4}).
         *
         * The implementation of the domain separator was designed to be as efficient as possible while still properly updating
         * the chain id to protect against replay attacks on an eventual fork of the chain.
         *
         * NOTE: This contract implements the version of the encoding known as "v4", as implemented by the JSON RPC method
         * https://docs.metamask.io/guide/signing-data.html[`eth_signTypedDataV4` in MetaMask].
         *
         * _Available since v3.4._
         */
        abstract contract EIP712 {
            /* solhint-disable var-name-mixedcase */
            // Cache the domain separator as an immutable value, but also store the chain id that it corresponds to, in order to
            // invalidate the cached domain separator if the chain id changes.
            bytes32 private immutable _CACHED_DOMAIN_SEPARATOR;
            uint256 private immutable _CACHED_CHAIN_ID;
            address private immutable _CACHED_THIS;
            bytes32 private immutable _HASHED_NAME;
            bytes32 private immutable _HASHED_VERSION;
            bytes32 private immutable _TYPE_HASH;
            /* solhint-enable var-name-mixedcase */
            /**
             * @dev Initializes the domain separator and parameter caches.
             *
             * The meaning of `name` and `version` is specified in
             * https://eips.ethereum.org/EIPS/eip-712#definition-of-domainseparator[EIP 712]:
             *
             * - `name`: the user readable name of the signing domain, i.e. the name of the DApp or the protocol.
             * - `version`: the current major version of the signing domain.
             *
             * NOTE: These parameters cannot be changed except through a xref:learn::upgrading-smart-contracts.adoc[smart
             * contract upgrade].
             */
            constructor(string memory name, string memory version) {
                bytes32 hashedName = keccak256(bytes(name));
                bytes32 hashedVersion = keccak256(bytes(version));
                bytes32 typeHash = keccak256(
                    "EIP712Domain(string name,string version,uint256 chainId,address verifyingContract)"
                );
                _HASHED_NAME = hashedName;
                _HASHED_VERSION = hashedVersion;
                _CACHED_CHAIN_ID = block.chainid;
                _CACHED_DOMAIN_SEPARATOR = _buildDomainSeparator(typeHash, hashedName, hashedVersion);
                _CACHED_THIS = address(this);
                _TYPE_HASH = typeHash;
            }
            /**
             * @dev Returns the domain separator for the current chain.
             */
            function _domainSeparatorV4() internal view returns (bytes32) {
                if (address(this) == _CACHED_THIS && block.chainid == _CACHED_CHAIN_ID) {
                    return _CACHED_DOMAIN_SEPARATOR;
                } else {
                    return _buildDomainSeparator(_TYPE_HASH, _HASHED_NAME, _HASHED_VERSION);
                }
            }
            function _buildDomainSeparator(
                bytes32 typeHash,
                bytes32 nameHash,
                bytes32 versionHash
            ) private view returns (bytes32) {
                return keccak256(abi.encode(typeHash, nameHash, versionHash, block.chainid, address(this)));
            }
            /**
             * @dev Given an already https://eips.ethereum.org/EIPS/eip-712#definition-of-hashstruct[hashed struct], this
             * function returns the hash of the fully encoded EIP712 message for this domain.
             *
             * This hash can be used together with {ECDSA-recover} to obtain the signer of a message. For example:
             *
             * ```solidity
             * bytes32 digest = _hashTypedDataV4(keccak256(abi.encode(
             *     keccak256("Mail(address to,string contents)"),
             *     mailTo,
             *     keccak256(bytes(mailContents))
             * )));
             * address signer = ECDSA.recover(digest, signature);
             * ```
             */
            function _hashTypedDataV4(bytes32 structHash) internal view virtual returns (bytes32) {
                return ECDSA.toTypedDataHash(_domainSeparatorV4(), structHash);
            }
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts v4.4.1 (utils/introspection/IERC165.sol)
        pragma solidity ^0.8.0;
        /**
         * @dev Interface of the ERC165 standard, as defined in the
         * https://eips.ethereum.org/EIPS/eip-165[EIP].
         *
         * Implementers can declare support of contract interfaces, which can then be
         * queried by others ({ERC165Checker}).
         *
         * For an implementation, see {ERC165}.
         */
        interface IERC165 {
            /**
             * @dev Returns true if this contract implements the interface defined by
             * `interfaceId`. See the corresponding
             * https://eips.ethereum.org/EIPS/eip-165#how-interfaces-are-identified[EIP section]
             * to learn more about how these ids are created.
             *
             * This function call must use less than 30 000 gas.
             */
            function supportsInterface(bytes4 interfaceId) external view returns (bool);
        }
        // SPDX-License-Identifier: MIT
        pragma solidity ^0.8.9;
        import "@openzeppelin/contracts/token/ERC20/ERC20.sol";
        import "@openzeppelin/contracts/token/ERC20/extensions/ERC20Burnable.sol";
        import "@openzeppelin/contracts/token/ERC20/extensions/draft-ERC20Permit.sol";
        import {IHeyEduToken, IL2StandardERC20} from "./interfaces/IHeyEduToken.sol";
        error HeyEdu_Edu_OnlyOwner();
        error HeyEdu_Edu_OnlyOwnerCandidate();
        error HeyEdu_Edu_OnlyMinter();
        error HeyEdu_Edu_MinterTimelocked();
        error HeyEdu_Edu_WrongNewMinter();
        contract Edu is IHeyEduToken, ERC20, ERC20Burnable, ERC20Permit {
            string public constant VERSION = "1";
            address public owner;
            address public ownerCandidate;
            address public minter = address(0);
            uint256 public minterUpdatedAt;
            uint256 public constant MINT_TIMELOCK_AFTER_MINTER_CHANGE = 7 days;
            address public immutable l1Token;
            address public l2Bridge;
            constructor(address _owner, address _l2Bridge) ERC20("Edu", "EDU") ERC20Permit("HeyEdu Edu") {
                owner = _owner;
                l1Token = address(this);
                if (block.chainid != 1) {
                    l2Bridge = _l2Bridge;
                }
            }
            /*//////////////////////////////////////////////////////////////
                                    MINT and BURN
            //////////////////////////////////////////////////////////////*/
            function mint(address _to, uint256 _amount) external {
                if (msg.sender != minter && msg.sender != l2Bridge) {
                    revert HeyEdu_Edu_OnlyMinter();
                }
                if (msg.sender == minter && block.timestamp < minterUpdatedAt + MINT_TIMELOCK_AFTER_MINTER_CHANGE) {
                    revert HeyEdu_Edu_MinterTimelocked();
                }
                _mint(_to, _amount);
                emit Mint(_to, _amount);
            }
            function burn(address _from, uint256 _amount) external {
                if (msg.sender != minter && msg.sender != l2Bridge) {
                    revert HeyEdu_Edu_OnlyMinter();
                }
                _burn(_from, _amount);
                emit Burn(_from, _amount);
            }
            /*//////////////////////////////////////////////////////////////
                                    OWNERSHIP
            //////////////////////////////////////////////////////////////*/
            event OwnershipTransferred(address indexed _previousOwner, address indexed _newOwner);
            event OwnerCandidateChosen(address indexed _ownerCandidate);
            event NewMinterSet(address indexed _newMinter);
            modifier onlyOwner() {
                if (msg.sender != owner) {
                    revert HeyEdu_Edu_OnlyOwner();
                }
                _;
            }
            function setMinter(address newMinter) external onlyOwner {
                if (newMinter == address(0) || newMinter == minter) {
                    revert HeyEdu_Edu_WrongNewMinter();
                }
                if (minter != address(0)) {
                    minterUpdatedAt = block.timestamp;
                }
                minter = newMinter;
                emit NewMinterSet(newMinter);
            }
            function setOwnerCandidate(address _ownerCandidate) external onlyOwner {
                ownerCandidate = _ownerCandidate;
                emit OwnerCandidateChosen(ownerCandidate);
            }
            function claimOwnership() external {
                if (msg.sender != ownerCandidate) {
                    revert HeyEdu_Edu_OnlyOwner();
                }
                emit OwnershipTransferred(owner, ownerCandidate);
                owner = ownerCandidate;
                ownerCandidate = address(0);
            }
            function rescueTokens(address _recipient) external onlyOwner {
                uint256 amount = balanceOf(address(this));
                transfer(_recipient, amount);
            }
            /*//////////////////////////////////////////////////////////////
                                    ERC165
            //////////////////////////////////////////////////////////////*/
            // slither-disable-next-line external-function
            function supportsInterface(bytes4 _interfaceId) public pure returns (bool) {
                bytes4 firstSupportedInterface = bytes4(keccak256("supportsInterface(bytes4)")); // ERC165
                bytes4 secondSupportedInterface = IL2StandardERC20.l1Token.selector ^
                    IL2StandardERC20.mint.selector ^
                    IL2StandardERC20.burn.selector;
                return _interfaceId == firstSupportedInterface || _interfaceId == secondSupportedInterface;
            }
        }
        // SPDX-License-Identifier: MIT
        pragma solidity ^0.8.9;
        import {IL2StandardERC20} from "@eth-optimism/contracts/standards/IL2StandardERC20.sol";
        interface IHeyEduToken is IL2StandardERC20 {
            function mint(address _to, uint256 _value) external;
            function claimOwnership() external;
            function setMinter(address newMinter) external;
        }
        

        File 2 of 3: DropERC721
        // SPDX-License-Identifier: Apache-2.0
        pragma solidity ^0.8.11;
        //  ==========  External imports    ==========
        import "@openzeppelin/contracts-upgradeable/token/ERC721/extensions/ERC721EnumerableUpgradeable.sol";
        import "@openzeppelin/contracts-upgradeable/interfaces/IERC2981Upgradeable.sol";
        import "@openzeppelin/contracts-upgradeable/access/AccessControlEnumerableUpgradeable.sol";
        import "@openzeppelin/contracts-upgradeable/security/ReentrancyGuardUpgradeable.sol";
        import "@openzeppelin/contracts-upgradeable/utils/structs/BitMapsUpgradeable.sol";
        import "@openzeppelin/contracts-upgradeable/utils/MulticallUpgradeable.sol";
        import "@openzeppelin/contracts-upgradeable/utils/StringsUpgradeable.sol";
        //  ==========  Internal imports    ==========
        import { IDropERC721 } from "../interfaces/drop/IDropERC721.sol";
        import { ITWFee } from "../interfaces/ITWFee.sol";
        import "../openzeppelin-presets/metatx/ERC2771ContextUpgradeable.sol";
        import "../lib/CurrencyTransferLib.sol";
        import "../lib/FeeType.sol";
        import "../lib/MerkleProof.sol";
        contract DropERC721 is
            Initializable,
            ReentrancyGuardUpgradeable,
            ERC2771ContextUpgradeable,
            MulticallUpgradeable,
            AccessControlEnumerableUpgradeable,
            ERC721EnumerableUpgradeable,
            IDropERC721
        {
            using BitMapsUpgradeable for BitMapsUpgradeable.BitMap;
            using StringsUpgradeable for uint256;
            /*///////////////////////////////////////////////////////////////
                                    State variables
            //////////////////////////////////////////////////////////////*/
            bytes32 private constant MODULE_TYPE = bytes32("DropERC721");
            uint256 private constant VERSION = 1;
            /// @dev Only transfers to or from TRANSFER_ROLE holders are valid, when transfers are restricted.
            bytes32 private constant TRANSFER_ROLE = keccak256("TRANSFER_ROLE");
            /// @dev Only MINTER_ROLE holders can lazy mint NFTs.
            bytes32 private constant MINTER_ROLE = keccak256("MINTER_ROLE");
            /// @dev Max bps in the thirdweb system.
            uint256 private constant MAX_BPS = 10_000;
            /// @dev The thirdweb contract with fee related information.
            ITWFee public immutable thirdwebFee;
            /// @dev Owner of the contract (purpose: OpenSea compatibility)
            address private _owner;
            /// @dev The next token ID of the NFT to "lazy mint".
            uint256 public nextTokenIdToMint;
            /// @dev The next token ID of the NFT that can be claimed.
            uint256 public nextTokenIdToClaim;
            /// @dev The address that receives all primary sales value.
            address public primarySaleRecipient;
            /// @dev The max number of NFTs a wallet can claim.
            uint256 public maxWalletClaimCount;
            /// @dev Global max total supply of NFTs.
            uint256 public maxTotalSupply;
            /// @dev The address that receives all platform fees from all sales.
            address private platformFeeRecipient;
            /// @dev The (default) address that receives all royalty value.
            address private royaltyRecipient;
            /// @dev The (default) % of a sale to take as royalty (in basis points).
            uint128 private royaltyBps;
            /// @dev The % of primary sales collected as platform fees.
            uint128 private platformFeeBps;
            /// @dev Contract level metadata.
            string public contractURI;
            /// @dev Largest tokenId of each batch of tokens with the same baseURI
            uint256[] public baseURIIndices;
            /// @dev The set of all claim conditions, at any given moment.
            ClaimConditionList public claimCondition;
            /*///////////////////////////////////////////////////////////////
                                        Mappings
            //////////////////////////////////////////////////////////////*/
            /**
             *  @dev Mapping from 'Largest tokenId of a batch of tokens with the same baseURI'
             *       to base URI for the respective batch of tokens.
             **/
            mapping(uint256 => string) private baseURI;
            /**
             *  @dev Mapping from 'Largest tokenId of a batch of 'delayed-reveal' tokens with
             *       the same baseURI' to encrypted base URI for the respective batch of tokens.
             **/
            mapping(uint256 => bytes) public encryptedBaseURI;
            /// @dev Mapping from address => total number of NFTs a wallet has claimed.
            mapping(address => uint256) public walletClaimCount;
            /// @dev Token ID => royalty recipient and bps for token
            mapping(uint256 => RoyaltyInfo) private royaltyInfoForToken;
            /*///////////////////////////////////////////////////////////////
                            Constructor + initializer logic
            //////////////////////////////////////////////////////////////*/
            constructor(address _thirdwebFee) initializer {
                thirdwebFee = ITWFee(_thirdwebFee);
            }
            /// @dev Initiliazes the contract, like a constructor.
            function initialize(
                address _defaultAdmin,
                string memory _name,
                string memory _symbol,
                string memory _contractURI,
                address[] memory _trustedForwarders,
                address _saleRecipient,
                address _royaltyRecipient,
                uint128 _royaltyBps,
                uint128 _platformFeeBps,
                address _platformFeeRecipient
            ) external initializer {
                // Initialize inherited contracts, most base-like -> most derived.
                __ReentrancyGuard_init();
                __ERC2771Context_init(_trustedForwarders);
                __ERC721_init(_name, _symbol);
                // Initialize this contract's state.
                royaltyRecipient = _royaltyRecipient;
                royaltyBps = _royaltyBps;
                platformFeeRecipient = _platformFeeRecipient;
                platformFeeBps = _platformFeeBps;
                primarySaleRecipient = _saleRecipient;
                contractURI = _contractURI;
                _owner = _defaultAdmin;
                _setupRole(DEFAULT_ADMIN_ROLE, _defaultAdmin);
                _setupRole(MINTER_ROLE, _defaultAdmin);
                _setupRole(TRANSFER_ROLE, _defaultAdmin);
                _setupRole(TRANSFER_ROLE, address(0));
            }
            /*///////////////////////////////////////////////////////////////
                                Generic contract logic
            //////////////////////////////////////////////////////////////*/
            /// @dev Returns the type of the contract.
            function contractType() external pure returns (bytes32) {
                return MODULE_TYPE;
            }
            /// @dev Returns the version of the contract.
            function contractVersion() external pure returns (uint8) {
                return uint8(VERSION);
            }
            /**
             * @dev Returns the address of the current owner.
             */
            function owner() public view returns (address) {
                return hasRole(DEFAULT_ADMIN_ROLE, _owner) ? _owner : address(0);
            }
            /*///////////////////////////////////////////////////////////////
                                ERC 165 / 721 / 2981 logic
            //////////////////////////////////////////////////////////////*/
            /// @dev Returns the URI for a given tokenId.
            function tokenURI(uint256 _tokenId) public view override returns (string memory) {
                for (uint256 i = 0; i < baseURIIndices.length; i += 1) {
                    if (_tokenId < baseURIIndices[i]) {
                        if (encryptedBaseURI[baseURIIndices[i]].length != 0) {
                            return string(abi.encodePacked(baseURI[baseURIIndices[i]], "0"));
                        } else {
                            return string(abi.encodePacked(baseURI[baseURIIndices[i]], _tokenId.toString()));
                        }
                    }
                }
                return "";
            }
            /// @dev See ERC 165
            function supportsInterface(bytes4 interfaceId)
                public
                view
                virtual
                override(ERC721EnumerableUpgradeable, AccessControlEnumerableUpgradeable, IERC165Upgradeable)
                returns (bool)
            {
                return super.supportsInterface(interfaceId) || type(IERC2981Upgradeable).interfaceId == interfaceId;
            }
            /// @dev Returns the royalty recipient and amount, given a tokenId and sale price.
            function royaltyInfo(uint256 tokenId, uint256 salePrice)
                external
                view
                virtual
                returns (address receiver, uint256 royaltyAmount)
            {
                (address recipient, uint256 bps) = getRoyaltyInfoForToken(tokenId);
                receiver = recipient;
                royaltyAmount = (salePrice * bps) / MAX_BPS;
            }
            /*///////////////////////////////////////////////////////////////
                            Minting + delayed-reveal logic
            //////////////////////////////////////////////////////////////*/
            /**
             *  @dev Lets an account with `MINTER_ROLE` lazy mint 'n' NFTs.
             *       The URIs for each token is the provided `_baseURIForTokens` + `{tokenId}`.
             */
            function lazyMint(
                uint256 _amount,
                string calldata _baseURIForTokens,
                bytes calldata _encryptedBaseURI
            ) external onlyRole(MINTER_ROLE) {
                uint256 startId = nextTokenIdToMint;
                uint256 baseURIIndex = startId + _amount;
                nextTokenIdToMint = baseURIIndex;
                baseURI[baseURIIndex] = _baseURIForTokens;
                baseURIIndices.push(baseURIIndex);
                if (_encryptedBaseURI.length != 0) {
                    encryptedBaseURI[baseURIIndex] = _encryptedBaseURI;
                }
                emit TokensLazyMinted(startId, startId + _amount - 1, _baseURIForTokens, _encryptedBaseURI);
            }
            /// @dev Lets an account with `MINTER_ROLE` reveal the URI for a batch of 'delayed-reveal' NFTs.
            function reveal(uint256 index, bytes calldata _key)
                external
                onlyRole(MINTER_ROLE)
                returns (string memory revealedURI)
            {
                require(index < baseURIIndices.length, "invalid index.");
                uint256 _index = baseURIIndices[index];
                bytes memory encryptedURI = encryptedBaseURI[_index];
                require(encryptedURI.length != 0, "nothing to reveal.");
                revealedURI = string(encryptDecrypt(encryptedURI, _key));
                baseURI[_index] = revealedURI;
                delete encryptedBaseURI[_index];
                emit NFTRevealed(_index, revealedURI);
                return revealedURI;
            }
            /// @dev See: https://ethereum.stackexchange.com/questions/69825/decrypt-message-on-chain
            function encryptDecrypt(bytes memory data, bytes calldata key) public pure returns (bytes memory result) {
                // Store data length on stack for later use
                uint256 length = data.length;
                // solhint-disable-next-line no-inline-assembly
                assembly {
                    // Set result to free memory pointer
                    result := mload(0x40)
                    // Increase free memory pointer by lenght + 32
                    mstore(0x40, add(add(result, length), 32))
                    // Set result length
                    mstore(result, length)
                }
                // Iterate over the data stepping by 32 bytes
                for (uint256 i = 0; i < length; i += 32) {
                    // Generate hash of the key and offset
                    bytes32 hash = keccak256(abi.encodePacked(key, i));
                    bytes32 chunk;
                    // solhint-disable-next-line no-inline-assembly
                    assembly {
                        // Read 32-bytes data chunk
                        chunk := mload(add(data, add(i, 32)))
                    }
                    // XOR the chunk with hash
                    chunk ^= hash;
                    // solhint-disable-next-line no-inline-assembly
                    assembly {
                        // Write 32-byte encrypted chunk
                        mstore(add(result, add(i, 32)), chunk)
                    }
                }
            }
            /*///////////////////////////////////////////////////////////////
                                    Claim logic
            //////////////////////////////////////////////////////////////*/
            /// @dev Lets an account claim NFTs.
            function claim(
                address _receiver,
                uint256 _quantity,
                address _currency,
                uint256 _pricePerToken,
                bytes32[] calldata _proofs,
                uint256 _proofMaxQuantityPerTransaction
            ) external payable nonReentrant {
                uint256 tokenIdToClaim = nextTokenIdToClaim;
                // Get the claim conditions.
                uint256 activeConditionId = getActiveClaimConditionId();
                // Verify claim validity. If not valid, revert.
                verifyClaim(activeConditionId, _msgSender(), _quantity, _currency, _pricePerToken);
                // Verify inclusion in allowlist
                (bool validMerkleProof, uint256 merkleProofIndex) = verifyClaimMerkleProof(
                    activeConditionId,
                    _msgSender(),
                    _quantity,
                    _proofs,
                    _proofMaxQuantityPerTransaction
                );
                if (validMerkleProof && _proofMaxQuantityPerTransaction > 0) {
                    // Mark the claimer's use of their position in the allowlist.
                    claimCondition.limitMerkleProofClaim[activeConditionId].set(merkleProofIndex);
                }
                // If there's a price, collect price.
                collectClaimPrice(_quantity, _currency, _pricePerToken);
                // Mint the relevant NFTs to claimer.
                transferClaimedTokens(_receiver, activeConditionId, _quantity);
                emit TokensClaimed(activeConditionId, _msgSender(), _receiver, tokenIdToClaim, _quantity);
            }
            /// @dev Lets a contract admin (account with `DEFAULT_ADMIN_ROLE`) set claim conditions.
            function setClaimConditions(ClaimCondition[] calldata _phases, bool _resetClaimEligibility)
                external
                onlyRole(DEFAULT_ADMIN_ROLE)
            {
                uint256 existingStartIndex = claimCondition.currentStartId;
                uint256 existingPhaseCount = claimCondition.count;
                /**
                 *  `limitLastClaimTimestamp` and `limitMerkleProofClaim` are mappings that use a
                 *  claim condition's UID as a key.
                 *
                 *  If `_resetClaimEligibility == true`, we assign completely new UIDs to the claim
                 *  conditions in `_phases`, effectively resetting the restrictions on claims expressed
                 *  by `limitLastClaimTimestamp` and `limitMerkleProofClaim`.
                 */
                uint256 newStartIndex = existingStartIndex;
                if (_resetClaimEligibility) {
                    newStartIndex = existingStartIndex + existingPhaseCount;
                }
                claimCondition.count = _phases.length;
                claimCondition.currentStartId = newStartIndex;
                uint256 lastConditionStartTimestamp;
                for (uint256 i = 0; i < _phases.length; i++) {
                    require(
                        i == 0 || lastConditionStartTimestamp < _phases[i].startTimestamp,
                        "startTimestamp must be in ascending order."
                    );
                    claimCondition.phases[newStartIndex + i] = _phases[i];
                    claimCondition.phases[newStartIndex + i].supplyClaimed = 0;
                    lastConditionStartTimestamp = _phases[i].startTimestamp;
                }
                /**
                 *  Gas refunds (as much as possible)
                 *
                 *  If `_resetClaimEligibility == true`, we assign completely new UIDs to the claim
                 *  conditions in `_phases`. So, we delete claim conditions with UID < `newStartIndex`.
                 *
                 *  If `_resetClaimEligibility == false`, and there are more existing claim conditions
                 *  than in `_phases`, we delete the existing claim conditions that don't get replaced
                 *  by the conditions in `_phases`.
                 */
                if (_resetClaimEligibility) {
                    for (uint256 i = existingStartIndex; i < newStartIndex; i++) {
                        delete claimCondition.phases[i];
                        delete claimCondition.limitMerkleProofClaim[i];
                    }
                } else {
                    if (existingPhaseCount > _phases.length) {
                        for (uint256 i = _phases.length; i < existingPhaseCount; i++) {
                            delete claimCondition.phases[newStartIndex + i];
                            delete claimCondition.limitMerkleProofClaim[newStartIndex + i];
                        }
                    }
                }
                emit ClaimConditionsUpdated(_phases);
            }
            /// @dev Collects and distributes the primary sale value of NFTs being claimed.
            function collectClaimPrice(
                uint256 _quantityToClaim,
                address _currency,
                uint256 _pricePerToken
            ) internal {
                if (_pricePerToken == 0) {
                    return;
                }
                uint256 totalPrice = _quantityToClaim * _pricePerToken;
                uint256 platformFees = (totalPrice * platformFeeBps) / MAX_BPS;
                (address twFeeRecipient, uint256 twFeeBps) = thirdwebFee.getFeeInfo(address(this), FeeType.PRIMARY_SALE);
                uint256 twFee = (totalPrice * twFeeBps) / MAX_BPS;
                if (_currency == CurrencyTransferLib.NATIVE_TOKEN) {
                    require(msg.value == totalPrice, "must send total price.");
                }
                CurrencyTransferLib.transferCurrency(_currency, _msgSender(), platformFeeRecipient, platformFees);
                CurrencyTransferLib.transferCurrency(_currency, _msgSender(), twFeeRecipient, twFee);
                CurrencyTransferLib.transferCurrency(
                    _currency,
                    _msgSender(),
                    primarySaleRecipient,
                    totalPrice - platformFees - twFee
                );
            }
            /// @dev Transfers the NFTs being claimed.
            function transferClaimedTokens(
                address _to,
                uint256 _conditionId,
                uint256 _quantityBeingClaimed
            ) internal {
                // Update the supply minted under mint condition.
                claimCondition.phases[_conditionId].supplyClaimed += _quantityBeingClaimed;
                // if transfer claimed tokens is called when `to != msg.sender`, it'd use msg.sender's limits.
                // behavior would be similar to `msg.sender` mint for itself, then transfer to `_to`.
                claimCondition.limitLastClaimTimestamp[_conditionId][_msgSender()] = block.timestamp;
                walletClaimCount[_msgSender()] += _quantityBeingClaimed;
                uint256 tokenIdToClaim = nextTokenIdToClaim;
                for (uint256 i = 0; i < _quantityBeingClaimed; i += 1) {
                    _mint(_to, tokenIdToClaim);
                    tokenIdToClaim += 1;
                }
                nextTokenIdToClaim = tokenIdToClaim;
            }
            /// @dev Checks a request to claim NFTs against the active claim condition's criteria.
            function verifyClaim(
                uint256 _conditionId,
                address _claimer,
                uint256 _quantity,
                address _currency,
                uint256 _pricePerToken
            ) public view {
                ClaimCondition memory currentClaimPhase = claimCondition.phases[_conditionId];
                require(
                    _currency == currentClaimPhase.currency && _pricePerToken == currentClaimPhase.pricePerToken,
                    "invalid currency or price specified."
                );
                require(
                    _quantity > 0 && _quantity <= currentClaimPhase.quantityLimitPerTransaction,
                    "invalid quantity claimed."
                );
                require(
                    currentClaimPhase.supplyClaimed + _quantity <= currentClaimPhase.maxClaimableSupply,
                    "exceed max mint supply."
                );
                require(nextTokenIdToClaim + _quantity <= nextTokenIdToMint, "not enough minted tokens.");
                require(maxTotalSupply == 0 || nextTokenIdToClaim + _quantity <= maxTotalSupply, "exceed max total supply.");
                require(
                    maxWalletClaimCount == 0 || walletClaimCount[_claimer] + _quantity <= maxWalletClaimCount,
                    "exceed claim limit for wallet"
                );
                (uint256 lastClaimTimestamp, uint256 nextValidClaimTimestamp) = getClaimTimestamp(_conditionId, _claimer);
                require(lastClaimTimestamp == 0 || block.timestamp >= nextValidClaimTimestamp, "cannot claim yet.");
            }
            /// @dev Checks whether a claimer meets the claim condition's allowlist criteria.
            function verifyClaimMerkleProof(
                uint256 _conditionId,
                address _claimer,
                uint256 _quantity,
                bytes32[] calldata _proofs,
                uint256 _proofMaxQuantityPerTransaction
            ) public view returns (bool validMerkleProof, uint256 merkleProofIndex) {
                ClaimCondition memory currentClaimPhase = claimCondition.phases[_conditionId];
                if (currentClaimPhase.merkleRoot != bytes32(0)) {
                    (validMerkleProof, merkleProofIndex) = MerkleProof.verify(
                        _proofs,
                        currentClaimPhase.merkleRoot,
                        keccak256(abi.encodePacked(_claimer, _proofMaxQuantityPerTransaction))
                    );
                    require(validMerkleProof, "not in whitelist.");
                    require(!claimCondition.limitMerkleProofClaim[_conditionId].get(merkleProofIndex), "proof claimed.");
                    require(
                        _proofMaxQuantityPerTransaction == 0 || _quantity <= _proofMaxQuantityPerTransaction,
                        "invalid quantity proof."
                    );
                }
            }
            /*///////////////////////////////////////////////////////////////
                                Getter functions
            //////////////////////////////////////////////////////////////*/
            /// @dev At any given moment, returns the uid for the active claim condition.
            function getActiveClaimConditionId() public view returns (uint256) {
                for (uint256 i = claimCondition.currentStartId + claimCondition.count; i > claimCondition.currentStartId; i--) {
                    if (block.timestamp >= claimCondition.phases[i - 1].startTimestamp) {
                        return i - 1;
                    }
                }
                revert("no active mint condition.");
            }
            /// @dev Returns the royalty recipient and bps for a particular token Id.
            function getRoyaltyInfoForToken(uint256 _tokenId) public view returns (address, uint16) {
                RoyaltyInfo memory royaltyForToken = royaltyInfoForToken[_tokenId];
                return
                    royaltyForToken.recipient == address(0)
                        ? (royaltyRecipient, uint16(royaltyBps))
                        : (royaltyForToken.recipient, uint16(royaltyForToken.bps));
            }
            /// @dev Returns the platform fee recipient and bps.
            function getPlatformFeeInfo() external view returns (address, uint16) {
                return (platformFeeRecipient, uint16(platformFeeBps));
            }
            /// @dev Returns the default royalty recipient and bps.
            function getDefaultRoyaltyInfo() external view returns (address, uint16) {
                return (royaltyRecipient, uint16(royaltyBps));
            }
            /// @dev Returns the timestamp for when a claimer is eligible for claiming NFTs again.
            function getClaimTimestamp(uint256 _conditionId, address _claimer)
                public
                view
                returns (uint256 lastClaimTimestamp, uint256 nextValidClaimTimestamp)
            {
                lastClaimTimestamp = claimCondition.limitLastClaimTimestamp[_conditionId][_claimer];
                unchecked {
                    nextValidClaimTimestamp =
                        lastClaimTimestamp +
                        claimCondition.phases[_conditionId].waitTimeInSecondsBetweenClaims;
                    if (nextValidClaimTimestamp < lastClaimTimestamp) {
                        nextValidClaimTimestamp = type(uint256).max;
                    }
                }
            }
            /// @dev Returns the claim condition at the given uid.
            function getClaimConditionById(uint256 _conditionId) external view returns (ClaimCondition memory condition) {
                condition = claimCondition.phases[_conditionId];
            }
            /// @dev Returns the amount of stored baseURIs
            function getBaseURICount() external view returns (uint256) {
                return baseURIIndices.length;
            }
            /*///////////////////////////////////////////////////////////////
                                Setter functions
            //////////////////////////////////////////////////////////////*/
            /// @dev Lets a contract admin set a claim count for a wallet.
            function setWalletClaimCount(address _claimer, uint256 _count) external onlyRole(DEFAULT_ADMIN_ROLE) {
                walletClaimCount[_claimer] = _count;
                emit WalletClaimCountUpdated(_claimer, _count);
            }
            /// @dev Lets a contract admin set a maximum number of NFTs that can be claimed by any wallet.
            function setMaxWalletClaimCount(uint256 _count) external onlyRole(DEFAULT_ADMIN_ROLE) {
                maxWalletClaimCount = _count;
                emit MaxWalletClaimCountUpdated(_count);
            }
            /// @dev Lets a contract admin set the global maximum supply for collection's NFTs.
            function setMaxTotalSupply(uint256 _maxTotalSupply) external onlyRole(DEFAULT_ADMIN_ROLE) {
                require(_maxTotalSupply < nextTokenIdToMint, "already minted more than desired max supply");
                maxTotalSupply = _maxTotalSupply;
                emit MaxTotalSupplyUpdated(_maxTotalSupply);
            }
            /// @dev Lets a contract admin set the recipient for all primary sales.
            function setPrimarySaleRecipient(address _saleRecipient) external onlyRole(DEFAULT_ADMIN_ROLE) {
                primarySaleRecipient = _saleRecipient;
                emit PrimarySaleRecipientUpdated(_saleRecipient);
            }
            /// @dev Lets a contract admin update the default royalty recipient and bps.
            function setDefaultRoyaltyInfo(address _royaltyRecipient, uint256 _royaltyBps)
                external
                onlyRole(DEFAULT_ADMIN_ROLE)
            {
                require(_royaltyBps <= MAX_BPS, "exceed royalty bps");
                royaltyRecipient = _royaltyRecipient;
                royaltyBps = uint128(_royaltyBps);
                emit DefaultRoyalty(_royaltyRecipient, _royaltyBps);
            }
            /// @dev Lets a contract admin set the royalty recipient and bps for a particular token Id.
            function setRoyaltyInfoForToken(
                uint256 _tokenId,
                address _recipient,
                uint256 _bps
            ) external onlyRole(DEFAULT_ADMIN_ROLE) {
                require(_bps <= MAX_BPS, "exceed royalty bps");
                royaltyInfoForToken[_tokenId] = RoyaltyInfo({ recipient: _recipient, bps: _bps });
                emit RoyaltyForToken(_tokenId, _recipient, _bps);
            }
            /// @dev Lets a contract admin update the platform fee recipient and bps
            function setPlatformFeeInfo(address _platformFeeRecipient, uint256 _platformFeeBps)
                external
                onlyRole(DEFAULT_ADMIN_ROLE)
            {
                require(_platformFeeBps <= MAX_BPS, "bps <= 10000.");
                platformFeeBps = uint64(_platformFeeBps);
                platformFeeRecipient = _platformFeeRecipient;
                emit PlatformFeeInfoUpdated(_platformFeeRecipient, _platformFeeBps);
            }
            /// @dev Lets a contract admin set a new owner for the contract. The new owner must be a contract admin.
            function setOwner(address _newOwner) external onlyRole(DEFAULT_ADMIN_ROLE) {
                require(hasRole(DEFAULT_ADMIN_ROLE, _newOwner), "new owner not contract admin.");
                address _prevOwner = _owner;
                _owner = _newOwner;
                emit OwnerUpdated(_prevOwner, _newOwner);
            }
            /// @dev Lets a contract admin set the URI for contract-level metadata.
            function setContractURI(string calldata _uri) external onlyRole(DEFAULT_ADMIN_ROLE) {
                contractURI = _uri;
            }
            /*///////////////////////////////////////////////////////////////
                                Miscellaneous
            //////////////////////////////////////////////////////////////*/
            /// @dev Burns `tokenId`. See {ERC721-_burn}.
            function burn(uint256 tokenId) public virtual {
                //solhint-disable-next-line max-line-length
                require(_isApprovedOrOwner(_msgSender(), tokenId), "caller is not owner nor approved");
                _burn(tokenId);
            }
            /// @dev See {ERC721-_beforeTokenTransfer}.
            function _beforeTokenTransfer(
                address from,
                address to,
                uint256 tokenId
            ) internal virtual override(ERC721EnumerableUpgradeable) {
                super._beforeTokenTransfer(from, to, tokenId);
                // if transfer is restricted on the contract, we still want to allow burning and minting
                if (!hasRole(TRANSFER_ROLE, address(0)) && from != address(0) && to != address(0)) {
                    require(hasRole(TRANSFER_ROLE, from) || hasRole(TRANSFER_ROLE, to), "restricted to TRANSFER_ROLE holders");
                }
            }
            function _msgSender()
                internal
                view
                virtual
                override(ContextUpgradeable, ERC2771ContextUpgradeable)
                returns (address sender)
            {
                return ERC2771ContextUpgradeable._msgSender();
            }
            function _msgData()
                internal
                view
                virtual
                override(ContextUpgradeable, ERC2771ContextUpgradeable)
                returns (bytes calldata)
            {
                return ERC2771ContextUpgradeable._msgData();
            }
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts v4.4.1 (token/ERC721/extensions/ERC721Enumerable.sol)
        pragma solidity ^0.8.0;
        import "../ERC721Upgradeable.sol";
        import "./IERC721EnumerableUpgradeable.sol";
        import "../../../proxy/utils/Initializable.sol";
        /**
         * @dev This implements an optional extension of {ERC721} defined in the EIP that adds
         * enumerability of all the token ids in the contract as well as all token ids owned by each
         * account.
         */
        abstract contract ERC721EnumerableUpgradeable is Initializable, ERC721Upgradeable, IERC721EnumerableUpgradeable {
            function __ERC721Enumerable_init() internal onlyInitializing {
            }
            function __ERC721Enumerable_init_unchained() internal onlyInitializing {
            }
            // Mapping from owner to list of owned token IDs
            mapping(address => mapping(uint256 => uint256)) private _ownedTokens;
            // Mapping from token ID to index of the owner tokens list
            mapping(uint256 => uint256) private _ownedTokensIndex;
            // Array with all token ids, used for enumeration
            uint256[] private _allTokens;
            // Mapping from token id to position in the allTokens array
            mapping(uint256 => uint256) private _allTokensIndex;
            /**
             * @dev See {IERC165-supportsInterface}.
             */
            function supportsInterface(bytes4 interfaceId) public view virtual override(IERC165Upgradeable, ERC721Upgradeable) returns (bool) {
                return interfaceId == type(IERC721EnumerableUpgradeable).interfaceId || super.supportsInterface(interfaceId);
            }
            /**
             * @dev See {IERC721Enumerable-tokenOfOwnerByIndex}.
             */
            function tokenOfOwnerByIndex(address owner, uint256 index) public view virtual override returns (uint256) {
                require(index < ERC721Upgradeable.balanceOf(owner), "ERC721Enumerable: owner index out of bounds");
                return _ownedTokens[owner][index];
            }
            /**
             * @dev See {IERC721Enumerable-totalSupply}.
             */
            function totalSupply() public view virtual override returns (uint256) {
                return _allTokens.length;
            }
            /**
             * @dev See {IERC721Enumerable-tokenByIndex}.
             */
            function tokenByIndex(uint256 index) public view virtual override returns (uint256) {
                require(index < ERC721EnumerableUpgradeable.totalSupply(), "ERC721Enumerable: global index out of bounds");
                return _allTokens[index];
            }
            /**
             * @dev Hook that is called before any token transfer. This includes minting
             * and burning.
             *
             * Calling conditions:
             *
             * - When `from` and `to` are both non-zero, ``from``'s `tokenId` will be
             * transferred to `to`.
             * - When `from` is zero, `tokenId` will be minted for `to`.
             * - When `to` is zero, ``from``'s `tokenId` will be burned.
             * - `from` cannot be the zero address.
             * - `to` cannot be the zero address.
             *
             * To learn more about hooks, head to xref:ROOT:extending-contracts.adoc#using-hooks[Using Hooks].
             */
            function _beforeTokenTransfer(
                address from,
                address to,
                uint256 tokenId
            ) internal virtual override {
                super._beforeTokenTransfer(from, to, tokenId);
                if (from == address(0)) {
                    _addTokenToAllTokensEnumeration(tokenId);
                } else if (from != to) {
                    _removeTokenFromOwnerEnumeration(from, tokenId);
                }
                if (to == address(0)) {
                    _removeTokenFromAllTokensEnumeration(tokenId);
                } else if (to != from) {
                    _addTokenToOwnerEnumeration(to, tokenId);
                }
            }
            /**
             * @dev Private function to add a token to this extension's ownership-tracking data structures.
             * @param to address representing the new owner of the given token ID
             * @param tokenId uint256 ID of the token to be added to the tokens list of the given address
             */
            function _addTokenToOwnerEnumeration(address to, uint256 tokenId) private {
                uint256 length = ERC721Upgradeable.balanceOf(to);
                _ownedTokens[to][length] = tokenId;
                _ownedTokensIndex[tokenId] = length;
            }
            /**
             * @dev Private function to add a token to this extension's token tracking data structures.
             * @param tokenId uint256 ID of the token to be added to the tokens list
             */
            function _addTokenToAllTokensEnumeration(uint256 tokenId) private {
                _allTokensIndex[tokenId] = _allTokens.length;
                _allTokens.push(tokenId);
            }
            /**
             * @dev Private function to remove a token from this extension's ownership-tracking data structures. Note that
             * while the token is not assigned a new owner, the `_ownedTokensIndex` mapping is _not_ updated: this allows for
             * gas optimizations e.g. when performing a transfer operation (avoiding double writes).
             * This has O(1) time complexity, but alters the order of the _ownedTokens array.
             * @param from address representing the previous owner of the given token ID
             * @param tokenId uint256 ID of the token to be removed from the tokens list of the given address
             */
            function _removeTokenFromOwnerEnumeration(address from, uint256 tokenId) private {
                // To prevent a gap in from's tokens array, we store the last token in the index of the token to delete, and
                // then delete the last slot (swap and pop).
                uint256 lastTokenIndex = ERC721Upgradeable.balanceOf(from) - 1;
                uint256 tokenIndex = _ownedTokensIndex[tokenId];
                // When the token to delete is the last token, the swap operation is unnecessary
                if (tokenIndex != lastTokenIndex) {
                    uint256 lastTokenId = _ownedTokens[from][lastTokenIndex];
                    _ownedTokens[from][tokenIndex] = lastTokenId; // Move the last token to the slot of the to-delete token
                    _ownedTokensIndex[lastTokenId] = tokenIndex; // Update the moved token's index
                }
                // This also deletes the contents at the last position of the array
                delete _ownedTokensIndex[tokenId];
                delete _ownedTokens[from][lastTokenIndex];
            }
            /**
             * @dev Private function to remove a token from this extension's token tracking data structures.
             * This has O(1) time complexity, but alters the order of the _allTokens array.
             * @param tokenId uint256 ID of the token to be removed from the tokens list
             */
            function _removeTokenFromAllTokensEnumeration(uint256 tokenId) private {
                // To prevent a gap in the tokens array, we store the last token in the index of the token to delete, and
                // then delete the last slot (swap and pop).
                uint256 lastTokenIndex = _allTokens.length - 1;
                uint256 tokenIndex = _allTokensIndex[tokenId];
                // When the token to delete is the last token, the swap operation is unnecessary. However, since this occurs so
                // rarely (when the last minted token is burnt) that we still do the swap here to avoid the gas cost of adding
                // an 'if' statement (like in _removeTokenFromOwnerEnumeration)
                uint256 lastTokenId = _allTokens[lastTokenIndex];
                _allTokens[tokenIndex] = lastTokenId; // Move the last token to the slot of the to-delete token
                _allTokensIndex[lastTokenId] = tokenIndex; // Update the moved token's index
                // This also deletes the contents at the last position of the array
                delete _allTokensIndex[tokenId];
                _allTokens.pop();
            }
            /**
             * @dev This empty reserved space is put in place to allow future versions to add new
             * variables without shifting down storage in the inheritance chain.
             * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps
             */
            uint256[46] private __gap;
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v4.5.0) (interfaces/IERC2981.sol)
        pragma solidity ^0.8.0;
        import "./IERC165Upgradeable.sol";
        /**
         * @dev Interface for the NFT Royalty Standard.
         *
         * A standardized way to retrieve royalty payment information for non-fungible tokens (NFTs) to enable universal
         * support for royalty payments across all NFT marketplaces and ecosystem participants.
         *
         * _Available since v4.5._
         */
        interface IERC2981Upgradeable is IERC165Upgradeable {
            /**
             * @dev Returns how much royalty is owed and to whom, based on a sale price that may be denominated in any unit of
             * exchange. The royalty amount is denominated and should be payed in that same unit of exchange.
             */
            function royaltyInfo(uint256 tokenId, uint256 salePrice)
                external
                view
                returns (address receiver, uint256 royaltyAmount);
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v4.5.0) (access/AccessControlEnumerable.sol)
        pragma solidity ^0.8.0;
        import "./IAccessControlEnumerableUpgradeable.sol";
        import "./AccessControlUpgradeable.sol";
        import "../utils/structs/EnumerableSetUpgradeable.sol";
        import "../proxy/utils/Initializable.sol";
        /**
         * @dev Extension of {AccessControl} that allows enumerating the members of each role.
         */
        abstract contract AccessControlEnumerableUpgradeable is Initializable, IAccessControlEnumerableUpgradeable, AccessControlUpgradeable {
            function __AccessControlEnumerable_init() internal onlyInitializing {
            }
            function __AccessControlEnumerable_init_unchained() internal onlyInitializing {
            }
            using EnumerableSetUpgradeable for EnumerableSetUpgradeable.AddressSet;
            mapping(bytes32 => EnumerableSetUpgradeable.AddressSet) private _roleMembers;
            /**
             * @dev See {IERC165-supportsInterface}.
             */
            function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) {
                return interfaceId == type(IAccessControlEnumerableUpgradeable).interfaceId || super.supportsInterface(interfaceId);
            }
            /**
             * @dev Returns one of the accounts that have `role`. `index` must be a
             * value between 0 and {getRoleMemberCount}, non-inclusive.
             *
             * Role bearers are not sorted in any particular way, and their ordering may
             * change at any point.
             *
             * WARNING: When using {getRoleMember} and {getRoleMemberCount}, make sure
             * you perform all queries on the same block. See the following
             * https://forum.openzeppelin.com/t/iterating-over-elements-on-enumerableset-in-openzeppelin-contracts/2296[forum post]
             * for more information.
             */
            function getRoleMember(bytes32 role, uint256 index) public view virtual override returns (address) {
                return _roleMembers[role].at(index);
            }
            /**
             * @dev Returns the number of accounts that have `role`. Can be used
             * together with {getRoleMember} to enumerate all bearers of a role.
             */
            function getRoleMemberCount(bytes32 role) public view virtual override returns (uint256) {
                return _roleMembers[role].length();
            }
            /**
             * @dev Overload {_grantRole} to track enumerable memberships
             */
            function _grantRole(bytes32 role, address account) internal virtual override {
                super._grantRole(role, account);
                _roleMembers[role].add(account);
            }
            /**
             * @dev Overload {_revokeRole} to track enumerable memberships
             */
            function _revokeRole(bytes32 role, address account) internal virtual override {
                super._revokeRole(role, account);
                _roleMembers[role].remove(account);
            }
            /**
             * @dev This empty reserved space is put in place to allow future versions to add new
             * variables without shifting down storage in the inheritance chain.
             * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps
             */
            uint256[49] private __gap;
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts v4.4.1 (security/ReentrancyGuard.sol)
        pragma solidity ^0.8.0;
        import "../proxy/utils/Initializable.sol";
        /**
         * @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 ReentrancyGuardUpgradeable is Initializable {
            // 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;
            function __ReentrancyGuard_init() internal onlyInitializing {
                __ReentrancyGuard_init_unchained();
            }
            function __ReentrancyGuard_init_unchained() internal onlyInitializing {
                _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;
            }
            /**
             * @dev This empty reserved space is put in place to allow future versions to add new
             * variables without shifting down storage in the inheritance chain.
             * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps
             */
            uint256[49] private __gap;
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts v4.4.1 (utils/structs/BitMaps.sol)
        pragma solidity ^0.8.0;
        /**
         * @dev Library for managing uint256 to bool mapping in a compact and efficient way, providing the keys are sequential.
         * Largelly inspired by Uniswap's https://github.com/Uniswap/merkle-distributor/blob/master/contracts/MerkleDistributor.sol[merkle-distributor].
         */
        library BitMapsUpgradeable {
            struct BitMap {
                mapping(uint256 => uint256) _data;
            }
            /**
             * @dev Returns whether the bit at `index` is set.
             */
            function get(BitMap storage bitmap, uint256 index) internal view returns (bool) {
                uint256 bucket = index >> 8;
                uint256 mask = 1 << (index & 0xff);
                return bitmap._data[bucket] & mask != 0;
            }
            /**
             * @dev Sets the bit at `index` to the boolean `value`.
             */
            function setTo(
                BitMap storage bitmap,
                uint256 index,
                bool value
            ) internal {
                if (value) {
                    set(bitmap, index);
                } else {
                    unset(bitmap, index);
                }
            }
            /**
             * @dev Sets the bit at `index`.
             */
            function set(BitMap storage bitmap, uint256 index) internal {
                uint256 bucket = index >> 8;
                uint256 mask = 1 << (index & 0xff);
                bitmap._data[bucket] |= mask;
            }
            /**
             * @dev Unsets the bit at `index`.
             */
            function unset(BitMap storage bitmap, uint256 index) internal {
                uint256 bucket = index >> 8;
                uint256 mask = 1 << (index & 0xff);
                bitmap._data[bucket] &= ~mask;
            }
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v4.5.0) (utils/Multicall.sol)
        pragma solidity ^0.8.0;
        import "./AddressUpgradeable.sol";
        import "../proxy/utils/Initializable.sol";
        /**
         * @dev Provides a function to batch together multiple calls in a single external call.
         *
         * _Available since v4.1._
         */
        abstract contract MulticallUpgradeable is Initializable {
            function __Multicall_init() internal onlyInitializing {
            }
            function __Multicall_init_unchained() internal onlyInitializing {
            }
            /**
             * @dev Receives and executes a batch of function calls on this contract.
             */
            function multicall(bytes[] calldata data) external virtual returns (bytes[] memory results) {
                results = new bytes[](data.length);
                for (uint256 i = 0; i < data.length; i++) {
                    results[i] = _functionDelegateCall(address(this), data[i]);
                }
                return results;
            }
            /**
             * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],
             * but performing a delegate call.
             *
             * _Available since v3.4._
             */
            function _functionDelegateCall(address target, bytes memory data) private returns (bytes memory) {
                require(AddressUpgradeable.isContract(target), "Address: delegate call to non-contract");
                // solhint-disable-next-line avoid-low-level-calls
                (bool success, bytes memory returndata) = target.delegatecall(data);
                return AddressUpgradeable.verifyCallResult(success, returndata, "Address: low-level delegate call failed");
            }
            /**
             * @dev This empty reserved space is put in place to allow future versions to add new
             * variables without shifting down storage in the inheritance chain.
             * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps
             */
            uint256[50] private __gap;
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts v4.4.1 (utils/Strings.sol)
        pragma solidity ^0.8.0;
        /**
         * @dev String operations.
         */
        library StringsUpgradeable {
            bytes16 private constant _HEX_SYMBOLS = "0123456789abcdef";
            /**
             * @dev Converts a `uint256` to its ASCII `string` decimal representation.
             */
            function toString(uint256 value) internal pure returns (string memory) {
                // Inspired by OraclizeAPI's implementation - MIT licence
                // https://github.com/oraclize/ethereum-api/blob/b42146b063c7d6ee1358846c198246239e9360e8/oraclizeAPI_0.4.25.sol
                if (value == 0) {
                    return "0";
                }
                uint256 temp = value;
                uint256 digits;
                while (temp != 0) {
                    digits++;
                    temp /= 10;
                }
                bytes memory buffer = new bytes(digits);
                while (value != 0) {
                    digits -= 1;
                    buffer[digits] = bytes1(uint8(48 + uint256(value % 10)));
                    value /= 10;
                }
                return string(buffer);
            }
            /**
             * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation.
             */
            function toHexString(uint256 value) internal pure returns (string memory) {
                if (value == 0) {
                    return "0x00";
                }
                uint256 temp = value;
                uint256 length = 0;
                while (temp != 0) {
                    length++;
                    temp >>= 8;
                }
                return toHexString(value, length);
            }
            /**
             * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation with fixed length.
             */
            function toHexString(uint256 value, uint256 length) internal pure returns (string memory) {
                bytes memory buffer = new bytes(2 * length + 2);
                buffer[0] = "0";
                buffer[1] = "x";
                for (uint256 i = 2 * length + 1; i > 1; --i) {
                    buffer[i] = _HEX_SYMBOLS[value & 0xf];
                    value >>= 4;
                }
                require(value == 0, "Strings: hex length insufficient");
                return string(buffer);
            }
        }
        // SPDX-License-Identifier: Apache-2.0
        pragma solidity ^0.8.11;
        import "@openzeppelin/contracts-upgradeable/token/ERC721/IERC721Upgradeable.sol";
        import "../IThirdwebContract.sol";
        import "../IThirdwebPlatformFee.sol";
        import "../IThirdwebPrimarySale.sol";
        import "../IThirdwebRoyalty.sol";
        import "../IThirdwebOwnable.sol";
        import "./IDropClaimCondition.sol";
        /**
         *  Thirdweb's 'Drop' contracts are distribution mechanisms for tokens. The
         *  `DropERC721` contract is a distribution mechanism for ERC721 tokens.
         *
         *  A minter wallet (i.e. holder of `MINTER_ROLE`) can (lazy)mint 'n' tokens
         *  at once by providing a single base URI for all tokens being lazy minted.
         *  The URI for each of the 'n' tokens lazy minted is the provided base URI +
         *  `{tokenId}` of the respective token. (e.g. "ipsf://Qmece.../1").
         *
         *  A minter can choose to lazy mint 'delayed-reveal' tokens. More on 'delayed-reveal'
         *  tokens in [this article](https://blog.thirdweb.com/delayed-reveal-nfts).
         *
         *  A contract admin (i.e. holder of `DEFAULT_ADMIN_ROLE`) can create claim conditions
         *  with non-overlapping time windows, and accounts can claim the tokens according to
         *  restrictions defined in the claim condition that is active at the time of the transaction.
         */
        interface IDropERC721 is
            IThirdwebContract,
            IThirdwebOwnable,
            IThirdwebRoyalty,
            IThirdwebPrimarySale,
            IThirdwebPlatformFee,
            IERC721Upgradeable,
            IDropClaimCondition
        {
            /// @dev Emitted when tokens are claimed.
            event TokensClaimed(
                uint256 indexed claimConditionIndex,
                address indexed claimer,
                address indexed receiver,
                uint256 startTokenId,
                uint256 quantityClaimed
            );
            /// @dev Emitted when tokens are lazy minted.
            event TokensLazyMinted(uint256 startTokenId, uint256 endTokenId, string baseURI, bytes encryptedBaseURI);
            /// @dev Emitted when the URI for a batch of 'delayed-reveal' NFTs is revealed.
            event NFTRevealed(uint256 endTokenId, string revealedURI);
            /// @dev Emitted when new claim conditions are set.
            event ClaimConditionsUpdated(ClaimCondition[] claimConditions);
            /// @dev Emitted when a new primary sale recipient is set.
            event PrimarySaleRecipientUpdated(address indexed recipient);
            /// @dev Emitted when fee platform fee recipient or bps is updated.
            event PlatformFeeInfoUpdated(address platformFeeRecipient, uint256 platformFeeBps);
            /// @dev Emitted when a new owner is set.
            event OwnerUpdated(address prevOwner, address newOwner);
            /// @dev Emitted when the global max supply of tokens is updated.
            event MaxTotalSupplyUpdated(uint256 maxTotalSupply);
            /// @dev Emitted when the wallet claim count for an address is updated.
            event WalletClaimCountUpdated(address indexed wallet, uint256 count);
            /// @dev Emitted when the global max wallet claim count is updated.
            event MaxWalletClaimCountUpdated(uint256 count);
            /**
             *  @notice Lets an account with `MINTER_ROLE` lazy mint 'n' NFTs.
             *          The URIs for each token is the provided `_baseURIForTokens` + `{tokenId}`.
             *
             *  @param amount           The amount of NFTs to lazy mint.
             *  @param baseURIForTokens The URI for the NFTs to lazy mint. If lazy minting
             *                           'delayed-reveal' NFTs, the is a URI for NFTs in the
             *                           un-revealed state.
             *  @param encryptedBaseURI If lazy minting 'delayed-reveal' NFTs, this is the
             *                           result of encrypting the URI of the NFTs in the revealed
             *                           state.
             */
            function lazyMint(
                uint256 amount,
                string calldata baseURIForTokens,
                bytes calldata encryptedBaseURI
            ) external;
            /**
             *  @notice Lets an account claim a given quantity of NFTs.
             *
             *  @param receiver                       The receiver of the NFTs to claim.
             *  @param quantity                       The quantity of NFTs to claim.
             *  @param currency                       The currency in which to pay for the claim.
             *  @param pricePerToken                  The price per token to pay for the claim.
             *  @param proofs                         The proof of the claimer's inclusion in the merkle root allowlist
             *                                        of the claim conditions that apply.
             *  @param proofMaxQuantityPerTransaction (Optional) The maximum number of NFTs an address included in an
             *                                        allowlist can claim.
             */
            function claim(
                address receiver,
                uint256 quantity,
                address currency,
                uint256 pricePerToken,
                bytes32[] calldata proofs,
                uint256 proofMaxQuantityPerTransaction
            ) external payable;
            /**
             *  @notice Lets a contract admin (account with `DEFAULT_ADMIN_ROLE`) set claim conditions.
             *
             *  @param phases                Claim conditions in ascending order by `startTimestamp`.
             *  @param resetClaimEligibility Whether to reset `limitLastClaimTimestamp` and
             *                               `limitMerkleProofClaim` values when setting new
             *                               claim conditions.
             */
            function setClaimConditions(ClaimCondition[] calldata phases, bool resetClaimEligibility) external;
        }
        // SPDX-License-Identifier: Apache-2.0
        pragma solidity ^0.8.11;
        interface ITWFee {
            function getFeeInfo(address _proxy, uint256 _type) external view returns (address recipient, uint256 bps);
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts v4.4.0 (metatx/ERC2771Context.sol)
        pragma solidity ^0.8.11;
        import "@openzeppelin/contracts-upgradeable/utils/ContextUpgradeable.sol";
        import "@openzeppelin/contracts-upgradeable/proxy/utils/Initializable.sol";
        /**
         * @dev Context variant with ERC2771 support.
         */
        abstract contract ERC2771ContextUpgradeable is Initializable, ContextUpgradeable {
            mapping(address => bool) private _trustedForwarder;
            function __ERC2771Context_init(address[] memory trustedForwarder) internal onlyInitializing {
                __Context_init_unchained();
                __ERC2771Context_init_unchained(trustedForwarder);
            }
            function __ERC2771Context_init_unchained(address[] memory trustedForwarder) internal onlyInitializing {
                for (uint256 i = 0; i < trustedForwarder.length; i++) {
                    _trustedForwarder[trustedForwarder[i]] = true;
                }
            }
            function isTrustedForwarder(address forwarder) public view virtual returns (bool) {
                return _trustedForwarder[forwarder];
            }
            function _msgSender() internal view virtual override returns (address sender) {
                if (isTrustedForwarder(msg.sender)) {
                    // The assembly code is more direct than the Solidity version using `abi.decode`.
                    assembly {
                        sender := shr(96, calldataload(sub(calldatasize(), 20)))
                    }
                } else {
                    return super._msgSender();
                }
            }
            function _msgData() internal view virtual override returns (bytes calldata) {
                if (isTrustedForwarder(msg.sender)) {
                    return msg.data[:msg.data.length - 20];
                } else {
                    return super._msgData();
                }
            }
            uint256[49] private __gap;
        }
        // SPDX-License-Identifier: Apache-2.0
        pragma solidity ^0.8.11;
        // Helper interfaces
        import { IWETH } from "../interfaces/IWETH.sol";
        import "@openzeppelin/contracts-upgradeable/token/ERC20/IERC20Upgradeable.sol";
        library CurrencyTransferLib {
            /// @dev The address interpreted as native token of the chain.
            address public constant NATIVE_TOKEN = 0xEeeeeEeeeEeEeeEeEeEeeEEEeeeeEeeeeeeeEEeE;
            /// @dev Transfers a given amount of currency.
            function transferCurrency(
                address _currency,
                address _from,
                address _to,
                uint256 _amount
            ) internal {
                if (_amount == 0) {
                    return;
                }
                if (_currency == NATIVE_TOKEN) {
                    safeTransferNativeToken(_to, _amount);
                } else {
                    safeTransferERC20(_currency, _from, _to, _amount);
                }
            }
            /// @dev Transfers a given amount of currency. (With native token wrapping)
            function transferCurrencyWithWrapperAndBalanceCheck(
                address _currency,
                address _from,
                address _to,
                uint256 _amount,
                address _nativeTokenWrapper
            ) internal {
                if (_amount == 0) {
                    return;
                }
                if (_currency == NATIVE_TOKEN) {
                    if (_from == address(this)) {
                        // withdraw from weth then transfer withdrawn native token to recipient
                        IWETH(_nativeTokenWrapper).withdraw(_amount);
                        safeTransferNativeTokenWithWrapper(_to, _amount, _nativeTokenWrapper);
                    } else if (_to == address(this)) {
                        // store native currency in weth
                        require(_amount == msg.value, "msg.value != amount");
                        IWETH(_nativeTokenWrapper).deposit{ value: _amount }();
                    } else {
                        safeTransferNativeTokenWithWrapper(_to, _amount, _nativeTokenWrapper);
                    }
                } else {
                    safeTransferERC20WithBalanceCheck(_currency, _from, _to, _amount);
                }
            }
            /// @dev Transfer `amount` of ERC20 token from `from` to `to`.
            function safeTransferERC20(
                address _currency,
                address _from,
                address _to,
                uint256 _amount
            ) internal {
                if (_from == _to) {
                    return;
                }
                bool success = _from == address(this)
                    ? IERC20Upgradeable(_currency).transfer(_to, _amount)
                    : IERC20Upgradeable(_currency).transferFrom(_from, _to, _amount);
                require(success, "currency transfer failed.");
            }
            /// @dev Transfer `amount` of ERC20 token from `from` to `to`.
            function safeTransferERC20WithBalanceCheck(
                address _currency,
                address _from,
                address _to,
                uint256 _amount
            ) internal {
                if (_from == _to) {
                    return;
                }
                uint256 balBefore = IERC20Upgradeable(_currency).balanceOf(_to);
                bool success = _from == address(this)
                    ? IERC20Upgradeable(_currency).transfer(_to, _amount)
                    : IERC20Upgradeable(_currency).transferFrom(_from, _to, _amount);
                uint256 balAfter = IERC20Upgradeable(_currency).balanceOf(_to);
                require(success && (balAfter == balBefore + _amount), "currency transfer failed.");
            }
            /// @dev Transfers `amount` of native token to `to`.
            function safeTransferNativeToken(address to, uint256 value) internal {
                // solhint-disable avoid-low-level-calls
                // slither-disable-next-line low-level-calls
                (bool success, ) = to.call{ value: value }("");
                require(success, "native token transfer failed");
            }
            /// @dev Transfers `amount` of native token to `to`. (With native token wrapping)
            function safeTransferNativeTokenWithWrapper(
                address to,
                uint256 value,
                address _nativeTokenWrapper
            ) internal {
                // solhint-disable avoid-low-level-calls
                // slither-disable-next-line low-level-calls
                (bool success, ) = to.call{ value: value }("");
                if (!success) {
                    IWETH(_nativeTokenWrapper).deposit{ value: value }();
                    require(IERC20Upgradeable(_nativeTokenWrapper).transfer(to, value), "transfer failed");
                }
            }
        }
        // SPDX-License-Identifier: Apache-2.0
        pragma solidity ^0.8.11;
        library FeeType {
            uint256 internal constant PRIMARY_SALE = 0;
            uint256 internal constant MARKET_SALE = 1;
            uint256 internal constant SPLIT = 2;
        }
        // SPDX-License-Identifier: MIT
        // Modified from https://github.com/OpenZeppelin/openzeppelin-contracts/blob/v4.3.0/contracts/utils/cryptography/MerkleProof.sol
        // Copied from https://github.com/ensdomains/governance/blob/master/contracts/MerkleProof.sol
        pragma solidity ^0.8.11;
        /**
         * @dev These functions deal with verification of Merkle Trees proofs.
         *
         * The proofs can be generated using the JavaScript library
         * https://github.com/miguelmota/merkletreejs[merkletreejs].
         * Note: the hashing algorithm should be keccak256 and pair sorting should be enabled.
         *
         * See `test/utils/cryptography/MerkleProof.test.js` for some examples.
         *
         * Source: https://github.com/ensdomains/governance/blob/master/contracts/MerkleProof.sol
         */
        library MerkleProof {
            /**
             * @dev Returns true if a `leaf` can be proved to be a part of a Merkle tree
             * defined by `root`. For this, a `proof` must be provided, containing
             * sibling hashes on the branch from the leaf to the root of the tree. Each
             * pair of leaves and each pair of pre-images are assumed to be sorted.
             */
            function verify(
                bytes32[] memory proof,
                bytes32 root,
                bytes32 leaf
            ) internal pure returns (bool, uint256) {
                bytes32 computedHash = leaf;
                uint256 index = 0;
                for (uint256 i = 0; i < proof.length; i++) {
                    index *= 2;
                    bytes32 proofElement = proof[i];
                    if (computedHash <= proofElement) {
                        // Hash(current computed hash + current element of the proof)
                        computedHash = keccak256(abi.encodePacked(computedHash, proofElement));
                    } else {
                        // Hash(current element of the proof + current computed hash)
                        computedHash = keccak256(abi.encodePacked(proofElement, computedHash));
                        index += 1;
                    }
                }
                // Check if the computed hash (root) is equal to the provided root
                return (computedHash == root, index);
            }
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v4.5.0) (token/ERC721/ERC721.sol)
        pragma solidity ^0.8.0;
        import "./IERC721Upgradeable.sol";
        import "./IERC721ReceiverUpgradeable.sol";
        import "./extensions/IERC721MetadataUpgradeable.sol";
        import "../../utils/AddressUpgradeable.sol";
        import "../../utils/ContextUpgradeable.sol";
        import "../../utils/StringsUpgradeable.sol";
        import "../../utils/introspection/ERC165Upgradeable.sol";
        import "../../proxy/utils/Initializable.sol";
        /**
         * @dev Implementation of https://eips.ethereum.org/EIPS/eip-721[ERC721] Non-Fungible Token Standard, including
         * the Metadata extension, but not including the Enumerable extension, which is available separately as
         * {ERC721Enumerable}.
         */
        contract ERC721Upgradeable is Initializable, ContextUpgradeable, ERC165Upgradeable, IERC721Upgradeable, IERC721MetadataUpgradeable {
            using AddressUpgradeable for address;
            using StringsUpgradeable for uint256;
            // Token name
            string private _name;
            // Token symbol
            string private _symbol;
            // Mapping from token ID to owner address
            mapping(uint256 => address) private _owners;
            // Mapping owner address to token count
            mapping(address => uint256) private _balances;
            // Mapping from token ID to approved address
            mapping(uint256 => address) private _tokenApprovals;
            // Mapping from owner to operator approvals
            mapping(address => mapping(address => bool)) private _operatorApprovals;
            /**
             * @dev Initializes the contract by setting a `name` and a `symbol` to the token collection.
             */
            function __ERC721_init(string memory name_, string memory symbol_) internal onlyInitializing {
                __ERC721_init_unchained(name_, symbol_);
            }
            function __ERC721_init_unchained(string memory name_, string memory symbol_) internal onlyInitializing {
                _name = name_;
                _symbol = symbol_;
            }
            /**
             * @dev See {IERC165-supportsInterface}.
             */
            function supportsInterface(bytes4 interfaceId) public view virtual override(ERC165Upgradeable, IERC165Upgradeable) returns (bool) {
                return
                    interfaceId == type(IERC721Upgradeable).interfaceId ||
                    interfaceId == type(IERC721MetadataUpgradeable).interfaceId ||
                    super.supportsInterface(interfaceId);
            }
            /**
             * @dev See {IERC721-balanceOf}.
             */
            function balanceOf(address owner) public view virtual override returns (uint256) {
                require(owner != address(0), "ERC721: balance query for the zero address");
                return _balances[owner];
            }
            /**
             * @dev See {IERC721-ownerOf}.
             */
            function ownerOf(uint256 tokenId) public view virtual override returns (address) {
                address owner = _owners[tokenId];
                require(owner != address(0), "ERC721: owner query for nonexistent token");
                return owner;
            }
            /**
             * @dev See {IERC721Metadata-name}.
             */
            function name() public view virtual override returns (string memory) {
                return _name;
            }
            /**
             * @dev See {IERC721Metadata-symbol}.
             */
            function symbol() public view virtual override returns (string memory) {
                return _symbol;
            }
            /**
             * @dev See {IERC721Metadata-tokenURI}.
             */
            function tokenURI(uint256 tokenId) public view virtual override returns (string memory) {
                require(_exists(tokenId), "ERC721Metadata: URI query for nonexistent token");
                string memory baseURI = _baseURI();
                return bytes(baseURI).length > 0 ? string(abi.encodePacked(baseURI, tokenId.toString())) : "";
            }
            /**
             * @dev Base URI for computing {tokenURI}. If set, the resulting URI for each
             * token will be the concatenation of the `baseURI` and the `tokenId`. Empty
             * by default, can be overriden in child contracts.
             */
            function _baseURI() internal view virtual returns (string memory) {
                return "";
            }
            /**
             * @dev See {IERC721-approve}.
             */
            function approve(address to, uint256 tokenId) public virtual override {
                address owner = ERC721Upgradeable.ownerOf(tokenId);
                require(to != owner, "ERC721: approval to current owner");
                require(
                    _msgSender() == owner || isApprovedForAll(owner, _msgSender()),
                    "ERC721: approve caller is not owner nor approved for all"
                );
                _approve(to, tokenId);
            }
            /**
             * @dev See {IERC721-getApproved}.
             */
            function getApproved(uint256 tokenId) public view virtual override returns (address) {
                require(_exists(tokenId), "ERC721: approved query for nonexistent token");
                return _tokenApprovals[tokenId];
            }
            /**
             * @dev See {IERC721-setApprovalForAll}.
             */
            function setApprovalForAll(address operator, bool approved) public virtual override {
                _setApprovalForAll(_msgSender(), operator, approved);
            }
            /**
             * @dev See {IERC721-isApprovedForAll}.
             */
            function isApprovedForAll(address owner, address operator) public view virtual override returns (bool) {
                return _operatorApprovals[owner][operator];
            }
            /**
             * @dev See {IERC721-transferFrom}.
             */
            function transferFrom(
                address from,
                address to,
                uint256 tokenId
            ) public virtual override {
                //solhint-disable-next-line max-line-length
                require(_isApprovedOrOwner(_msgSender(), tokenId), "ERC721: transfer caller is not owner nor approved");
                _transfer(from, to, tokenId);
            }
            /**
             * @dev See {IERC721-safeTransferFrom}.
             */
            function safeTransferFrom(
                address from,
                address to,
                uint256 tokenId
            ) public virtual override {
                safeTransferFrom(from, to, tokenId, "");
            }
            /**
             * @dev See {IERC721-safeTransferFrom}.
             */
            function safeTransferFrom(
                address from,
                address to,
                uint256 tokenId,
                bytes memory _data
            ) public virtual override {
                require(_isApprovedOrOwner(_msgSender(), tokenId), "ERC721: transfer caller is not owner nor approved");
                _safeTransfer(from, to, tokenId, _data);
            }
            /**
             * @dev Safely transfers `tokenId` token from `from` to `to`, checking first that contract recipients
             * are aware of the ERC721 protocol to prevent tokens from being forever locked.
             *
             * `_data` is additional data, it has no specified format and it is sent in call to `to`.
             *
             * This internal function is equivalent to {safeTransferFrom}, and can be used to e.g.
             * implement alternative mechanisms to perform token transfer, such as signature-based.
             *
             * Requirements:
             *
             * - `from` cannot be the zero address.
             * - `to` cannot be the zero address.
             * - `tokenId` token must exist and be owned by `from`.
             * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer.
             *
             * Emits a {Transfer} event.
             */
            function _safeTransfer(
                address from,
                address to,
                uint256 tokenId,
                bytes memory _data
            ) internal virtual {
                _transfer(from, to, tokenId);
                require(_checkOnERC721Received(from, to, tokenId, _data), "ERC721: transfer to non ERC721Receiver implementer");
            }
            /**
             * @dev Returns whether `tokenId` exists.
             *
             * Tokens can be managed by their owner or approved accounts via {approve} or {setApprovalForAll}.
             *
             * Tokens start existing when they are minted (`_mint`),
             * and stop existing when they are burned (`_burn`).
             */
            function _exists(uint256 tokenId) internal view virtual returns (bool) {
                return _owners[tokenId] != address(0);
            }
            /**
             * @dev Returns whether `spender` is allowed to manage `tokenId`.
             *
             * Requirements:
             *
             * - `tokenId` must exist.
             */
            function _isApprovedOrOwner(address spender, uint256 tokenId) internal view virtual returns (bool) {
                require(_exists(tokenId), "ERC721: operator query for nonexistent token");
                address owner = ERC721Upgradeable.ownerOf(tokenId);
                return (spender == owner || getApproved(tokenId) == spender || isApprovedForAll(owner, spender));
            }
            /**
             * @dev Safely mints `tokenId` and transfers it to `to`.
             *
             * Requirements:
             *
             * - `tokenId` must not exist.
             * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer.
             *
             * Emits a {Transfer} event.
             */
            function _safeMint(address to, uint256 tokenId) internal virtual {
                _safeMint(to, tokenId, "");
            }
            /**
             * @dev Same as {xref-ERC721-_safeMint-address-uint256-}[`_safeMint`], with an additional `data` parameter which is
             * forwarded in {IERC721Receiver-onERC721Received} to contract recipients.
             */
            function _safeMint(
                address to,
                uint256 tokenId,
                bytes memory _data
            ) internal virtual {
                _mint(to, tokenId);
                require(
                    _checkOnERC721Received(address(0), to, tokenId, _data),
                    "ERC721: transfer to non ERC721Receiver implementer"
                );
            }
            /**
             * @dev Mints `tokenId` and transfers it to `to`.
             *
             * WARNING: Usage of this method is discouraged, use {_safeMint} whenever possible
             *
             * Requirements:
             *
             * - `tokenId` must not exist.
             * - `to` cannot be the zero address.
             *
             * Emits a {Transfer} event.
             */
            function _mint(address to, uint256 tokenId) internal virtual {
                require(to != address(0), "ERC721: mint to the zero address");
                require(!_exists(tokenId), "ERC721: token already minted");
                _beforeTokenTransfer(address(0), to, tokenId);
                _balances[to] += 1;
                _owners[tokenId] = to;
                emit Transfer(address(0), to, tokenId);
                _afterTokenTransfer(address(0), to, tokenId);
            }
            /**
             * @dev Destroys `tokenId`.
             * The approval is cleared when the token is burned.
             *
             * Requirements:
             *
             * - `tokenId` must exist.
             *
             * Emits a {Transfer} event.
             */
            function _burn(uint256 tokenId) internal virtual {
                address owner = ERC721Upgradeable.ownerOf(tokenId);
                _beforeTokenTransfer(owner, address(0), tokenId);
                // Clear approvals
                _approve(address(0), tokenId);
                _balances[owner] -= 1;
                delete _owners[tokenId];
                emit Transfer(owner, address(0), tokenId);
                _afterTokenTransfer(owner, address(0), tokenId);
            }
            /**
             * @dev Transfers `tokenId` from `from` to `to`.
             *  As opposed to {transferFrom}, this imposes no restrictions on msg.sender.
             *
             * Requirements:
             *
             * - `to` cannot be the zero address.
             * - `tokenId` token must be owned by `from`.
             *
             * Emits a {Transfer} event.
             */
            function _transfer(
                address from,
                address to,
                uint256 tokenId
            ) internal virtual {
                require(ERC721Upgradeable.ownerOf(tokenId) == from, "ERC721: transfer from incorrect owner");
                require(to != address(0), "ERC721: transfer to the zero address");
                _beforeTokenTransfer(from, to, tokenId);
                // Clear approvals from the previous owner
                _approve(address(0), tokenId);
                _balances[from] -= 1;
                _balances[to] += 1;
                _owners[tokenId] = to;
                emit Transfer(from, to, tokenId);
                _afterTokenTransfer(from, to, tokenId);
            }
            /**
             * @dev Approve `to` to operate on `tokenId`
             *
             * Emits a {Approval} event.
             */
            function _approve(address to, uint256 tokenId) internal virtual {
                _tokenApprovals[tokenId] = to;
                emit Approval(ERC721Upgradeable.ownerOf(tokenId), to, tokenId);
            }
            /**
             * @dev Approve `operator` to operate on all of `owner` tokens
             *
             * Emits a {ApprovalForAll} event.
             */
            function _setApprovalForAll(
                address owner,
                address operator,
                bool approved
            ) internal virtual {
                require(owner != operator, "ERC721: approve to caller");
                _operatorApprovals[owner][operator] = approved;
                emit ApprovalForAll(owner, operator, approved);
            }
            /**
             * @dev Internal function to invoke {IERC721Receiver-onERC721Received} on a target address.
             * The call is not executed if the target address is not a contract.
             *
             * @param from address representing the previous owner of the given token ID
             * @param to target address that will receive the tokens
             * @param tokenId uint256 ID of the token to be transferred
             * @param _data bytes optional data to send along with the call
             * @return bool whether the call correctly returned the expected magic value
             */
            function _checkOnERC721Received(
                address from,
                address to,
                uint256 tokenId,
                bytes memory _data
            ) private returns (bool) {
                if (to.isContract()) {
                    try IERC721ReceiverUpgradeable(to).onERC721Received(_msgSender(), from, tokenId, _data) returns (bytes4 retval) {
                        return retval == IERC721ReceiverUpgradeable.onERC721Received.selector;
                    } catch (bytes memory reason) {
                        if (reason.length == 0) {
                            revert("ERC721: transfer to non ERC721Receiver implementer");
                        } else {
                            assembly {
                                revert(add(32, reason), mload(reason))
                            }
                        }
                    }
                } else {
                    return true;
                }
            }
            /**
             * @dev Hook that is called before any token transfer. This includes minting
             * and burning.
             *
             * Calling conditions:
             *
             * - When `from` and `to` are both non-zero, ``from``'s `tokenId` will be
             * transferred to `to`.
             * - When `from` is zero, `tokenId` will be minted for `to`.
             * - When `to` is zero, ``from``'s `tokenId` will be burned.
             * - `from` and `to` are never both zero.
             *
             * To learn more about hooks, head to xref:ROOT:extending-contracts.adoc#using-hooks[Using Hooks].
             */
            function _beforeTokenTransfer(
                address from,
                address to,
                uint256 tokenId
            ) internal virtual {}
            /**
             * @dev Hook that is called after any transfer of tokens. This includes
             * minting and burning.
             *
             * Calling conditions:
             *
             * - when `from` and `to` are both non-zero.
             * - `from` and `to` are never both zero.
             *
             * To learn more about hooks, head to xref:ROOT:extending-contracts.adoc#using-hooks[Using Hooks].
             */
            function _afterTokenTransfer(
                address from,
                address to,
                uint256 tokenId
            ) internal virtual {}
            /**
             * @dev This empty reserved space is put in place to allow future versions to add new
             * variables without shifting down storage in the inheritance chain.
             * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps
             */
            uint256[44] private __gap;
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v4.5.0) (token/ERC721/extensions/IERC721Enumerable.sol)
        pragma solidity ^0.8.0;
        import "../IERC721Upgradeable.sol";
        /**
         * @title ERC-721 Non-Fungible Token Standard, optional enumeration extension
         * @dev See https://eips.ethereum.org/EIPS/eip-721
         */
        interface IERC721EnumerableUpgradeable is IERC721Upgradeable {
            /**
             * @dev Returns the total amount of tokens stored by the contract.
             */
            function totalSupply() external view returns (uint256);
            /**
             * @dev Returns a token ID owned by `owner` at a given `index` of its token list.
             * Use along with {balanceOf} to enumerate all of ``owner``'s tokens.
             */
            function tokenOfOwnerByIndex(address owner, uint256 index) external view returns (uint256);
            /**
             * @dev Returns a token ID at a given `index` of all the tokens stored by the contract.
             * Use along with {totalSupply} to enumerate all tokens.
             */
            function tokenByIndex(uint256 index) external view returns (uint256);
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v4.5.0) (proxy/utils/Initializable.sol)
        pragma solidity ^0.8.0;
        import "../../utils/AddressUpgradeable.sol";
        /**
         * @dev This is a base contract to aid in writing upgradeable contracts, or any kind of contract that will be deployed
         * behind a proxy. Since proxied contracts do not make use of a constructor, it's common to move constructor logic to an
         * external initializer function, usually called `initialize`. It then becomes necessary to protect this initializer
         * function so it can only be called once. The {initializer} modifier provided by this contract will have this effect.
         *
         * TIP: To avoid leaving the proxy in an uninitialized state, the initializer function should be called as early as
         * possible by providing the encoded function call as the `_data` argument to {ERC1967Proxy-constructor}.
         *
         * CAUTION: When used with inheritance, manual care must be taken to not invoke a parent initializer twice, or to ensure
         * that all initializers are idempotent. This is not verified automatically as constructors are by Solidity.
         *
         * [CAUTION]
         * ====
         * Avoid leaving a contract uninitialized.
         *
         * An uninitialized contract can be taken over by an attacker. This applies to both a proxy and its implementation
         * contract, which may impact the proxy. To initialize the implementation contract, you can either invoke the
         * initializer manually, or you can include a constructor to automatically mark it as initialized when it is deployed:
         *
         * [.hljs-theme-light.nopadding]
         * ```
         * /// @custom:oz-upgrades-unsafe-allow constructor
         * constructor() initializer {}
         * ```
         * ====
         */
        abstract contract Initializable {
            /**
             * @dev Indicates that the contract has been initialized.
             */
            bool private _initialized;
            /**
             * @dev Indicates that the contract is in the process of being initialized.
             */
            bool private _initializing;
            /**
             * @dev Modifier to protect an initializer function from being invoked twice.
             */
            modifier initializer() {
                // If the contract is initializing we ignore whether _initialized is set in order to support multiple
                // inheritance patterns, but we only do this in the context of a constructor, because in other contexts the
                // contract may have been reentered.
                require(_initializing ? _isConstructor() : !_initialized, "Initializable: contract is already initialized");
                bool isTopLevelCall = !_initializing;
                if (isTopLevelCall) {
                    _initializing = true;
                    _initialized = true;
                }
                _;
                if (isTopLevelCall) {
                    _initializing = false;
                }
            }
            /**
             * @dev Modifier to protect an initialization function so that it can only be invoked by functions with the
             * {initializer} modifier, directly or indirectly.
             */
            modifier onlyInitializing() {
                require(_initializing, "Initializable: contract is not initializing");
                _;
            }
            function _isConstructor() private view returns (bool) {
                return !AddressUpgradeable.isContract(address(this));
            }
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts v4.4.1 (token/ERC721/IERC721.sol)
        pragma solidity ^0.8.0;
        import "../../utils/introspection/IERC165Upgradeable.sol";
        /**
         * @dev Required interface of an ERC721 compliant contract.
         */
        interface IERC721Upgradeable is IERC165Upgradeable {
            /**
             * @dev Emitted when `tokenId` token is transferred from `from` to `to`.
             */
            event Transfer(address indexed from, address indexed to, uint256 indexed tokenId);
            /**
             * @dev Emitted when `owner` enables `approved` to manage the `tokenId` token.
             */
            event Approval(address indexed owner, address indexed approved, uint256 indexed tokenId);
            /**
             * @dev Emitted when `owner` enables or disables (`approved`) `operator` to manage all of its assets.
             */
            event ApprovalForAll(address indexed owner, address indexed operator, bool approved);
            /**
             * @dev Returns the number of tokens in ``owner``'s account.
             */
            function balanceOf(address owner) external view returns (uint256 balance);
            /**
             * @dev Returns the owner of the `tokenId` token.
             *
             * Requirements:
             *
             * - `tokenId` must exist.
             */
            function ownerOf(uint256 tokenId) external view returns (address owner);
            /**
             * @dev Safely transfers `tokenId` token from `from` to `to`, checking first that contract recipients
             * are aware of the ERC721 protocol to prevent tokens from being forever locked.
             *
             * Requirements:
             *
             * - `from` cannot be the zero address.
             * - `to` cannot be the zero address.
             * - `tokenId` token must exist and be owned by `from`.
             * - If the caller is not `from`, it must be have been allowed to move this token by either {approve} or {setApprovalForAll}.
             * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer.
             *
             * Emits a {Transfer} event.
             */
            function safeTransferFrom(
                address from,
                address to,
                uint256 tokenId
            ) external;
            /**
             * @dev Transfers `tokenId` token from `from` to `to`.
             *
             * WARNING: Usage of this method is discouraged, use {safeTransferFrom} whenever possible.
             *
             * Requirements:
             *
             * - `from` cannot be the zero address.
             * - `to` cannot be the zero address.
             * - `tokenId` token must be owned by `from`.
             * - If the caller is not `from`, it must be approved to move this token by either {approve} or {setApprovalForAll}.
             *
             * Emits a {Transfer} event.
             */
            function transferFrom(
                address from,
                address to,
                uint256 tokenId
            ) external;
            /**
             * @dev Gives permission to `to` to transfer `tokenId` token to another account.
             * The approval is cleared when the token is transferred.
             *
             * Only a single account can be approved at a time, so approving the zero address clears previous approvals.
             *
             * Requirements:
             *
             * - The caller must own the token or be an approved operator.
             * - `tokenId` must exist.
             *
             * Emits an {Approval} event.
             */
            function approve(address to, uint256 tokenId) external;
            /**
             * @dev Returns the account approved for `tokenId` token.
             *
             * Requirements:
             *
             * - `tokenId` must exist.
             */
            function getApproved(uint256 tokenId) external view returns (address operator);
            /**
             * @dev Approve or remove `operator` as an operator for the caller.
             * Operators can call {transferFrom} or {safeTransferFrom} for any token owned by the caller.
             *
             * Requirements:
             *
             * - The `operator` cannot be the caller.
             *
             * Emits an {ApprovalForAll} event.
             */
            function setApprovalForAll(address operator, bool _approved) external;
            /**
             * @dev Returns if the `operator` is allowed to manage all of the assets of `owner`.
             *
             * See {setApprovalForAll}
             */
            function isApprovedForAll(address owner, address operator) external view returns (bool);
            /**
             * @dev Safely transfers `tokenId` token from `from` to `to`.
             *
             * Requirements:
             *
             * - `from` cannot be the zero address.
             * - `to` cannot be the zero address.
             * - `tokenId` token must exist and be owned by `from`.
             * - If the caller is not `from`, it must be approved to move this token by either {approve} or {setApprovalForAll}.
             * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer.
             *
             * Emits a {Transfer} event.
             */
            function safeTransferFrom(
                address from,
                address to,
                uint256 tokenId,
                bytes calldata data
            ) external;
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts v4.4.1 (token/ERC721/IERC721Receiver.sol)
        pragma solidity ^0.8.0;
        /**
         * @title ERC721 token receiver interface
         * @dev Interface for any contract that wants to support safeTransfers
         * from ERC721 asset contracts.
         */
        interface IERC721ReceiverUpgradeable {
            /**
             * @dev Whenever an {IERC721} `tokenId` token is transferred to this contract via {IERC721-safeTransferFrom}
             * by `operator` from `from`, this function is called.
             *
             * It must return its Solidity selector to confirm the token transfer.
             * If any other value is returned or the interface is not implemented by the recipient, the transfer will be reverted.
             *
             * The selector can be obtained in Solidity with `IERC721.onERC721Received.selector`.
             */
            function onERC721Received(
                address operator,
                address from,
                uint256 tokenId,
                bytes calldata data
            ) external returns (bytes4);
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts v4.4.1 (token/ERC721/extensions/IERC721Metadata.sol)
        pragma solidity ^0.8.0;
        import "../IERC721Upgradeable.sol";
        /**
         * @title ERC-721 Non-Fungible Token Standard, optional metadata extension
         * @dev See https://eips.ethereum.org/EIPS/eip-721
         */
        interface IERC721MetadataUpgradeable is IERC721Upgradeable {
            /**
             * @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);
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v4.5.0) (utils/Address.sol)
        pragma solidity ^0.8.1;
        /**
         * @dev Collection of functions related to the address type
         */
        library AddressUpgradeable {
            /**
             * @dev Returns true if `account` is a contract.
             *
             * [IMPORTANT]
             * ====
             * It is unsafe to assume that an address for which this function returns
             * false is an externally-owned account (EOA) and not a contract.
             *
             * Among others, `isContract` will return false for the following
             * types of addresses:
             *
             *  - an externally-owned account
             *  - a contract in construction
             *  - an address where a contract will be created
             *  - an address where a contract lived, but was destroyed
             * ====
             *
             * [IMPORTANT]
             * ====
             * You shouldn't rely on `isContract` to protect against flash loan attacks!
             *
             * Preventing calls from contracts is highly discouraged. It breaks composability, breaks support for smart wallets
             * like Gnosis Safe, and does not provide security since it can be circumvented by calling from a contract
             * constructor.
             * ====
             */
            function isContract(address account) internal view returns (bool) {
                // This method relies on extcodesize/address.code.length, which returns 0
                // for contracts in construction, since the code is only stored at the end
                // of the constructor execution.
                return account.code.length > 0;
            }
            /**
             * @dev Replacement for Solidity's `transfer`: sends `amount` wei to
             * `recipient`, forwarding all available gas and reverting on errors.
             *
             * https://eips.ethereum.org/EIPS/eip-1884[EIP1884] increases the gas cost
             * of certain opcodes, possibly making contracts go over the 2300 gas limit
             * imposed by `transfer`, making them unable to receive funds via
             * `transfer`. {sendValue} removes this limitation.
             *
             * https://diligence.consensys.net/posts/2019/09/stop-using-soliditys-transfer-now/[Learn more].
             *
             * IMPORTANT: because control is transferred to `recipient`, care must be
             * taken to not create reentrancy vulnerabilities. Consider using
             * {ReentrancyGuard} or the
             * https://solidity.readthedocs.io/en/v0.5.11/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern].
             */
            function sendValue(address payable recipient, uint256 amount) internal {
                require(address(this).balance >= amount, "Address: insufficient balance");
                (bool success, ) = recipient.call{value: amount}("");
                require(success, "Address: unable to send value, recipient may have reverted");
            }
            /**
             * @dev Performs a Solidity function call using a low level `call`. A
             * plain `call` is an unsafe replacement for a function call: use this
             * function instead.
             *
             * If `target` reverts with a revert reason, it is bubbled up by this
             * function (like regular Solidity function calls).
             *
             * Returns the raw returned data. To convert to the expected return value,
             * use https://solidity.readthedocs.io/en/latest/units-and-global-variables.html?highlight=abi.decode#abi-encoding-and-decoding-functions[`abi.decode`].
             *
             * Requirements:
             *
             * - `target` must be a contract.
             * - calling `target` with `data` must not revert.
             *
             * _Available since v3.1._
             */
            function functionCall(address target, bytes memory data) internal returns (bytes memory) {
                return functionCall(target, data, "Address: low-level call failed");
            }
            /**
             * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], but with
             * `errorMessage` as a fallback revert reason when `target` reverts.
             *
             * _Available since v3.1._
             */
            function functionCall(
                address target,
                bytes memory data,
                string memory errorMessage
            ) internal returns (bytes memory) {
                return functionCallWithValue(target, data, 0, errorMessage);
            }
            /**
             * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
             * but also transferring `value` wei to `target`.
             *
             * Requirements:
             *
             * - the calling contract must have an ETH balance of at least `value`.
             * - the called Solidity function must be `payable`.
             *
             * _Available since v3.1._
             */
            function functionCallWithValue(
                address target,
                bytes memory data,
                uint256 value
            ) internal returns (bytes memory) {
                return functionCallWithValue(target, data, value, "Address: low-level call with value failed");
            }
            /**
             * @dev Same as {xref-Address-functionCallWithValue-address-bytes-uint256-}[`functionCallWithValue`], but
             * with `errorMessage` as a fallback revert reason when `target` reverts.
             *
             * _Available since v3.1._
             */
            function functionCallWithValue(
                address target,
                bytes memory data,
                uint256 value,
                string memory errorMessage
            ) internal returns (bytes memory) {
                require(address(this).balance >= value, "Address: insufficient balance for call");
                require(isContract(target), "Address: call to non-contract");
                (bool success, bytes memory returndata) = target.call{value: value}(data);
                return verifyCallResult(success, returndata, errorMessage);
            }
            /**
             * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
             * but performing a static call.
             *
             * _Available since v3.3._
             */
            function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) {
                return functionStaticCall(target, data, "Address: low-level static call failed");
            }
            /**
             * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],
             * but performing a static call.
             *
             * _Available since v3.3._
             */
            function functionStaticCall(
                address target,
                bytes memory data,
                string memory errorMessage
            ) internal view returns (bytes memory) {
                require(isContract(target), "Address: static call to non-contract");
                (bool success, bytes memory returndata) = target.staticcall(data);
                return verifyCallResult(success, returndata, errorMessage);
            }
            /**
             * @dev Tool to verifies that a low level call was successful, and revert if it wasn't, either by bubbling the
             * revert reason using the provided one.
             *
             * _Available since v4.3._
             */
            function verifyCallResult(
                bool success,
                bytes memory returndata,
                string memory errorMessage
            ) internal pure returns (bytes memory) {
                if (success) {
                    return returndata;
                } else {
                    // Look for revert reason and bubble it up if present
                    if (returndata.length > 0) {
                        // The easiest way to bubble the revert reason is using memory via assembly
                        assembly {
                            let returndata_size := mload(returndata)
                            revert(add(32, returndata), returndata_size)
                        }
                    } else {
                        revert(errorMessage);
                    }
                }
            }
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts v4.4.1 (utils/Context.sol)
        pragma solidity ^0.8.0;
        import "../proxy/utils/Initializable.sol";
        /**
         * @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 ContextUpgradeable is Initializable {
            function __Context_init() internal onlyInitializing {
            }
            function __Context_init_unchained() internal onlyInitializing {
            }
            function _msgSender() internal view virtual returns (address) {
                return msg.sender;
            }
            function _msgData() internal view virtual returns (bytes calldata) {
                return msg.data;
            }
            /**
             * @dev This empty reserved space is put in place to allow future versions to add new
             * variables without shifting down storage in the inheritance chain.
             * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps
             */
            uint256[50] private __gap;
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts v4.4.1 (utils/introspection/ERC165.sol)
        pragma solidity ^0.8.0;
        import "./IERC165Upgradeable.sol";
        import "../../proxy/utils/Initializable.sol";
        /**
         * @dev Implementation of the {IERC165} interface.
         *
         * Contracts that want to implement ERC165 should inherit from this contract and override {supportsInterface} to check
         * for the additional interface id that will be supported. For example:
         *
         * ```solidity
         * function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) {
         *     return interfaceId == type(MyInterface).interfaceId || super.supportsInterface(interfaceId);
         * }
         * ```
         *
         * Alternatively, {ERC165Storage} provides an easier to use but more expensive implementation.
         */
        abstract contract ERC165Upgradeable is Initializable, IERC165Upgradeable {
            function __ERC165_init() internal onlyInitializing {
            }
            function __ERC165_init_unchained() internal onlyInitializing {
            }
            /**
             * @dev See {IERC165-supportsInterface}.
             */
            function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) {
                return interfaceId == type(IERC165Upgradeable).interfaceId;
            }
            /**
             * @dev This empty reserved space is put in place to allow future versions to add new
             * variables without shifting down storage in the inheritance chain.
             * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps
             */
            uint256[50] private __gap;
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts v4.4.1 (utils/introspection/IERC165.sol)
        pragma solidity ^0.8.0;
        /**
         * @dev Interface of the ERC165 standard, as defined in the
         * https://eips.ethereum.org/EIPS/eip-165[EIP].
         *
         * Implementers can declare support of contract interfaces, which can then be
         * queried by others ({ERC165Checker}).
         *
         * For an implementation, see {ERC165}.
         */
        interface IERC165Upgradeable {
            /**
             * @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);
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts v4.4.1 (interfaces/IERC165.sol)
        pragma solidity ^0.8.0;
        import "../utils/introspection/IERC165Upgradeable.sol";
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts v4.4.1 (access/IAccessControlEnumerable.sol)
        pragma solidity ^0.8.0;
        import "./IAccessControlUpgradeable.sol";
        /**
         * @dev External interface of AccessControlEnumerable declared to support ERC165 detection.
         */
        interface IAccessControlEnumerableUpgradeable is IAccessControlUpgradeable {
            /**
             * @dev Returns one of the accounts that have `role`. `index` must be a
             * value between 0 and {getRoleMemberCount}, non-inclusive.
             *
             * Role bearers are not sorted in any particular way, and their ordering may
             * change at any point.
             *
             * WARNING: When using {getRoleMember} and {getRoleMemberCount}, make sure
             * you perform all queries on the same block. See the following
             * https://forum.openzeppelin.com/t/iterating-over-elements-on-enumerableset-in-openzeppelin-contracts/2296[forum post]
             * for more information.
             */
            function getRoleMember(bytes32 role, uint256 index) external view returns (address);
            /**
             * @dev Returns the number of accounts that have `role`. Can be used
             * together with {getRoleMember} to enumerate all bearers of a role.
             */
            function getRoleMemberCount(bytes32 role) external view returns (uint256);
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v4.5.0) (access/AccessControl.sol)
        pragma solidity ^0.8.0;
        import "./IAccessControlUpgradeable.sol";
        import "../utils/ContextUpgradeable.sol";
        import "../utils/StringsUpgradeable.sol";
        import "../utils/introspection/ERC165Upgradeable.sol";
        import "../proxy/utils/Initializable.sol";
        /**
         * @dev Contract module that allows children to implement role-based access
         * control mechanisms. This is a lightweight version that doesn't allow enumerating role
         * members except through off-chain means by accessing the contract event logs. Some
         * applications may benefit from on-chain enumerability, for those cases see
         * {AccessControlEnumerable}.
         *
         * Roles are referred to by their `bytes32` identifier. These should be exposed
         * in the external API and be unique. The best way to achieve this is by
         * using `public constant` hash digests:
         *
         * ```
         * bytes32 public constant MY_ROLE = keccak256("MY_ROLE");
         * ```
         *
         * Roles can be used to represent a set of permissions. To restrict access to a
         * function call, use {hasRole}:
         *
         * ```
         * function foo() public {
         *     require(hasRole(MY_ROLE, msg.sender));
         *     ...
         * }
         * ```
         *
         * Roles can be granted and revoked dynamically via the {grantRole} and
         * {revokeRole} functions. Each role has an associated admin role, and only
         * accounts that have a role's admin role can call {grantRole} and {revokeRole}.
         *
         * By default, the admin role for all roles is `DEFAULT_ADMIN_ROLE`, which means
         * that only accounts with this role will be able to grant or revoke other
         * roles. More complex role relationships can be created by using
         * {_setRoleAdmin}.
         *
         * WARNING: The `DEFAULT_ADMIN_ROLE` is also its own admin: it has permission to
         * grant and revoke this role. Extra precautions should be taken to secure
         * accounts that have been granted it.
         */
        abstract contract AccessControlUpgradeable is Initializable, ContextUpgradeable, IAccessControlUpgradeable, ERC165Upgradeable {
            function __AccessControl_init() internal onlyInitializing {
            }
            function __AccessControl_init_unchained() internal onlyInitializing {
            }
            struct RoleData {
                mapping(address => bool) members;
                bytes32 adminRole;
            }
            mapping(bytes32 => RoleData) private _roles;
            bytes32 public constant DEFAULT_ADMIN_ROLE = 0x00;
            /**
             * @dev Modifier that checks that an account has a specific role. Reverts
             * with a standardized message including the required role.
             *
             * The format of the revert reason is given by the following regular expression:
             *
             *  /^AccessControl: account (0x[0-9a-f]{40}) is missing role (0x[0-9a-f]{64})$/
             *
             * _Available since v4.1._
             */
            modifier onlyRole(bytes32 role) {
                _checkRole(role, _msgSender());
                _;
            }
            /**
             * @dev See {IERC165-supportsInterface}.
             */
            function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) {
                return interfaceId == type(IAccessControlUpgradeable).interfaceId || super.supportsInterface(interfaceId);
            }
            /**
             * @dev Returns `true` if `account` has been granted `role`.
             */
            function hasRole(bytes32 role, address account) public view virtual override returns (bool) {
                return _roles[role].members[account];
            }
            /**
             * @dev Revert with a standard message if `account` is missing `role`.
             *
             * The format of the revert reason is given by the following regular expression:
             *
             *  /^AccessControl: account (0x[0-9a-f]{40}) is missing role (0x[0-9a-f]{64})$/
             */
            function _checkRole(bytes32 role, address account) internal view virtual {
                if (!hasRole(role, account)) {
                    revert(
                        string(
                            abi.encodePacked(
                                "AccessControl: account ",
                                StringsUpgradeable.toHexString(uint160(account), 20),
                                " is missing role ",
                                StringsUpgradeable.toHexString(uint256(role), 32)
                            )
                        )
                    );
                }
            }
            /**
             * @dev Returns the admin role that controls `role`. See {grantRole} and
             * {revokeRole}.
             *
             * To change a role's admin, use {_setRoleAdmin}.
             */
            function getRoleAdmin(bytes32 role) public view virtual override returns (bytes32) {
                return _roles[role].adminRole;
            }
            /**
             * @dev Grants `role` to `account`.
             *
             * If `account` had not been already granted `role`, emits a {RoleGranted}
             * event.
             *
             * Requirements:
             *
             * - the caller must have ``role``'s admin role.
             */
            function grantRole(bytes32 role, address account) public virtual override onlyRole(getRoleAdmin(role)) {
                _grantRole(role, account);
            }
            /**
             * @dev Revokes `role` from `account`.
             *
             * If `account` had been granted `role`, emits a {RoleRevoked} event.
             *
             * Requirements:
             *
             * - the caller must have ``role``'s admin role.
             */
            function revokeRole(bytes32 role, address account) public virtual override onlyRole(getRoleAdmin(role)) {
                _revokeRole(role, account);
            }
            /**
             * @dev Revokes `role` from the calling account.
             *
             * Roles are often managed via {grantRole} and {revokeRole}: this function's
             * purpose is to provide a mechanism for accounts to lose their privileges
             * if they are compromised (such as when a trusted device is misplaced).
             *
             * If the calling account had been revoked `role`, emits a {RoleRevoked}
             * event.
             *
             * Requirements:
             *
             * - the caller must be `account`.
             */
            function renounceRole(bytes32 role, address account) public virtual override {
                require(account == _msgSender(), "AccessControl: can only renounce roles for self");
                _revokeRole(role, account);
            }
            /**
             * @dev Grants `role` to `account`.
             *
             * If `account` had not been already granted `role`, emits a {RoleGranted}
             * event. Note that unlike {grantRole}, this function doesn't perform any
             * checks on the calling account.
             *
             * [WARNING]
             * ====
             * This function should only be called from the constructor when setting
             * up the initial roles for the system.
             *
             * Using this function in any other way is effectively circumventing the admin
             * system imposed by {AccessControl}.
             * ====
             *
             * NOTE: This function is deprecated in favor of {_grantRole}.
             */
            function _setupRole(bytes32 role, address account) internal virtual {
                _grantRole(role, account);
            }
            /**
             * @dev Sets `adminRole` as ``role``'s admin role.
             *
             * Emits a {RoleAdminChanged} event.
             */
            function _setRoleAdmin(bytes32 role, bytes32 adminRole) internal virtual {
                bytes32 previousAdminRole = getRoleAdmin(role);
                _roles[role].adminRole = adminRole;
                emit RoleAdminChanged(role, previousAdminRole, adminRole);
            }
            /**
             * @dev Grants `role` to `account`.
             *
             * Internal function without access restriction.
             */
            function _grantRole(bytes32 role, address account) internal virtual {
                if (!hasRole(role, account)) {
                    _roles[role].members[account] = true;
                    emit RoleGranted(role, account, _msgSender());
                }
            }
            /**
             * @dev Revokes `role` from `account`.
             *
             * Internal function without access restriction.
             */
            function _revokeRole(bytes32 role, address account) internal virtual {
                if (hasRole(role, account)) {
                    _roles[role].members[account] = false;
                    emit RoleRevoked(role, account, _msgSender());
                }
            }
            /**
             * @dev This empty reserved space is put in place to allow future versions to add new
             * variables without shifting down storage in the inheritance chain.
             * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps
             */
            uint256[49] private __gap;
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts v4.4.1 (utils/structs/EnumerableSet.sol)
        pragma solidity ^0.8.0;
        /**
         * @dev Library for managing
         * https://en.wikipedia.org/wiki/Set_(abstract_data_type)[sets] of primitive
         * types.
         *
         * Sets have the following properties:
         *
         * - Elements are added, removed, and checked for existence in constant time
         * (O(1)).
         * - Elements are enumerated in O(n). No guarantees are made on the ordering.
         *
         * ```
         * contract Example {
         *     // Add the library methods
         *     using EnumerableSet for EnumerableSet.AddressSet;
         *
         *     // Declare a set state variable
         *     EnumerableSet.AddressSet private mySet;
         * }
         * ```
         *
         * As of v3.3.0, sets of type `bytes32` (`Bytes32Set`), `address` (`AddressSet`)
         * and `uint256` (`UintSet`) are supported.
         */
        library EnumerableSetUpgradeable {
            // To implement this library for multiple types with as little code
            // repetition as possible, we write it in terms of a generic Set type with
            // bytes32 values.
            // The Set implementation uses private functions, and user-facing
            // implementations (such as AddressSet) are just wrappers around the
            // underlying Set.
            // This means that we can only create new EnumerableSets for types that fit
            // in bytes32.
            struct Set {
                // Storage of set values
                bytes32[] _values;
                // Position of the value in the `values` array, plus 1 because index 0
                // means a value is not in the set.
                mapping(bytes32 => uint256) _indexes;
            }
            /**
             * @dev Add a value to a set. O(1).
             *
             * Returns true if the value was added to the set, that is if it was not
             * already present.
             */
            function _add(Set storage set, bytes32 value) private returns (bool) {
                if (!_contains(set, value)) {
                    set._values.push(value);
                    // The value is stored at length-1, but we add 1 to all indexes
                    // and use 0 as a sentinel value
                    set._indexes[value] = set._values.length;
                    return true;
                } else {
                    return false;
                }
            }
            /**
             * @dev Removes a value from a set. O(1).
             *
             * Returns true if the value was removed from the set, that is if it was
             * present.
             */
            function _remove(Set storage set, bytes32 value) private returns (bool) {
                // We read and store the value's index to prevent multiple reads from the same storage slot
                uint256 valueIndex = set._indexes[value];
                if (valueIndex != 0) {
                    // Equivalent to contains(set, value)
                    // To delete an element from the _values array in O(1), we swap the element to delete with the last one in
                    // the array, and then remove the last element (sometimes called as 'swap and pop').
                    // This modifies the order of the array, as noted in {at}.
                    uint256 toDeleteIndex = valueIndex - 1;
                    uint256 lastIndex = set._values.length - 1;
                    if (lastIndex != toDeleteIndex) {
                        bytes32 lastvalue = set._values[lastIndex];
                        // Move the last value to the index where the value to delete is
                        set._values[toDeleteIndex] = lastvalue;
                        // Update the index for the moved value
                        set._indexes[lastvalue] = valueIndex; // Replace lastvalue's index to valueIndex
                    }
                    // Delete the slot where the moved value was stored
                    set._values.pop();
                    // Delete the index for the deleted slot
                    delete set._indexes[value];
                    return true;
                } else {
                    return false;
                }
            }
            /**
             * @dev Returns true if the value is in the set. O(1).
             */
            function _contains(Set storage set, bytes32 value) private view returns (bool) {
                return set._indexes[value] != 0;
            }
            /**
             * @dev Returns the number of values on the set. O(1).
             */
            function _length(Set storage set) private view returns (uint256) {
                return set._values.length;
            }
            /**
             * @dev Returns the value stored at position `index` in the set. O(1).
             *
             * Note that there are no guarantees on the ordering of values inside the
             * array, and it may change when more values are added or removed.
             *
             * Requirements:
             *
             * - `index` must be strictly less than {length}.
             */
            function _at(Set storage set, uint256 index) private view returns (bytes32) {
                return set._values[index];
            }
            /**
             * @dev Return the entire set in an array
             *
             * WARNING: This operation will copy the entire storage to memory, which can be quite expensive. This is designed
             * to mostly be used by view accessors that are queried without any gas fees. Developers should keep in mind that
             * this function has an unbounded cost, and using it as part of a state-changing function may render the function
             * uncallable if the set grows to a point where copying to memory consumes too much gas to fit in a block.
             */
            function _values(Set storage set) private view returns (bytes32[] memory) {
                return set._values;
            }
            // Bytes32Set
            struct Bytes32Set {
                Set _inner;
            }
            /**
             * @dev Add a value to a set. O(1).
             *
             * Returns true if the value was added to the set, that is if it was not
             * already present.
             */
            function add(Bytes32Set storage set, bytes32 value) internal returns (bool) {
                return _add(set._inner, value);
            }
            /**
             * @dev Removes a value from a set. O(1).
             *
             * Returns true if the value was removed from the set, that is if it was
             * present.
             */
            function remove(Bytes32Set storage set, bytes32 value) internal returns (bool) {
                return _remove(set._inner, value);
            }
            /**
             * @dev Returns true if the value is in the set. O(1).
             */
            function contains(Bytes32Set storage set, bytes32 value) internal view returns (bool) {
                return _contains(set._inner, value);
            }
            /**
             * @dev Returns the number of values in the set. O(1).
             */
            function length(Bytes32Set storage set) internal view returns (uint256) {
                return _length(set._inner);
            }
            /**
             * @dev Returns the value stored at position `index` in the set. O(1).
             *
             * Note that there are no guarantees on the ordering of values inside the
             * array, and it may change when more values are added or removed.
             *
             * Requirements:
             *
             * - `index` must be strictly less than {length}.
             */
            function at(Bytes32Set storage set, uint256 index) internal view returns (bytes32) {
                return _at(set._inner, index);
            }
            /**
             * @dev Return the entire set in an array
             *
             * WARNING: This operation will copy the entire storage to memory, which can be quite expensive. This is designed
             * to mostly be used by view accessors that are queried without any gas fees. Developers should keep in mind that
             * this function has an unbounded cost, and using it as part of a state-changing function may render the function
             * uncallable if the set grows to a point where copying to memory consumes too much gas to fit in a block.
             */
            function values(Bytes32Set storage set) internal view returns (bytes32[] memory) {
                return _values(set._inner);
            }
            // AddressSet
            struct AddressSet {
                Set _inner;
            }
            /**
             * @dev Add a value to a set. O(1).
             *
             * Returns true if the value was added to the set, that is if it was not
             * already present.
             */
            function add(AddressSet storage set, address value) internal returns (bool) {
                return _add(set._inner, bytes32(uint256(uint160(value))));
            }
            /**
             * @dev Removes a value from a set. O(1).
             *
             * Returns true if the value was removed from the set, that is if it was
             * present.
             */
            function remove(AddressSet storage set, address value) internal returns (bool) {
                return _remove(set._inner, bytes32(uint256(uint160(value))));
            }
            /**
             * @dev Returns true if the value is in the set. O(1).
             */
            function contains(AddressSet storage set, address value) internal view returns (bool) {
                return _contains(set._inner, bytes32(uint256(uint160(value))));
            }
            /**
             * @dev Returns the number of values in the set. O(1).
             */
            function length(AddressSet storage set) internal view returns (uint256) {
                return _length(set._inner);
            }
            /**
             * @dev Returns the value stored at position `index` in the set. O(1).
             *
             * Note that there are no guarantees on the ordering of values inside the
             * array, and it may change when more values are added or removed.
             *
             * Requirements:
             *
             * - `index` must be strictly less than {length}.
             */
            function at(AddressSet storage set, uint256 index) internal view returns (address) {
                return address(uint160(uint256(_at(set._inner, index))));
            }
            /**
             * @dev Return the entire set in an array
             *
             * WARNING: This operation will copy the entire storage to memory, which can be quite expensive. This is designed
             * to mostly be used by view accessors that are queried without any gas fees. Developers should keep in mind that
             * this function has an unbounded cost, and using it as part of a state-changing function may render the function
             * uncallable if the set grows to a point where copying to memory consumes too much gas to fit in a block.
             */
            function values(AddressSet storage set) internal view returns (address[] memory) {
                bytes32[] memory store = _values(set._inner);
                address[] memory result;
                assembly {
                    result := store
                }
                return result;
            }
            // UintSet
            struct UintSet {
                Set _inner;
            }
            /**
             * @dev Add a value to a set. O(1).
             *
             * Returns true if the value was added to the set, that is if it was not
             * already present.
             */
            function add(UintSet storage set, uint256 value) internal returns (bool) {
                return _add(set._inner, bytes32(value));
            }
            /**
             * @dev Removes a value from a set. O(1).
             *
             * Returns true if the value was removed from the set, that is if it was
             * present.
             */
            function remove(UintSet storage set, uint256 value) internal returns (bool) {
                return _remove(set._inner, bytes32(value));
            }
            /**
             * @dev Returns true if the value is in the set. O(1).
             */
            function contains(UintSet storage set, uint256 value) internal view returns (bool) {
                return _contains(set._inner, bytes32(value));
            }
            /**
             * @dev Returns the number of values on the set. O(1).
             */
            function length(UintSet storage set) internal view returns (uint256) {
                return _length(set._inner);
            }
            /**
             * @dev Returns the value stored at position `index` in the set. O(1).
             *
             * Note that there are no guarantees on the ordering of values inside the
             * array, and it may change when more values are added or removed.
             *
             * Requirements:
             *
             * - `index` must be strictly less than {length}.
             */
            function at(UintSet storage set, uint256 index) internal view returns (uint256) {
                return uint256(_at(set._inner, index));
            }
            /**
             * @dev Return the entire set in an array
             *
             * WARNING: This operation will copy the entire storage to memory, which can be quite expensive. This is designed
             * to mostly be used by view accessors that are queried without any gas fees. Developers should keep in mind that
             * this function has an unbounded cost, and using it as part of a state-changing function may render the function
             * uncallable if the set grows to a point where copying to memory consumes too much gas to fit in a block.
             */
            function values(UintSet storage set) internal view returns (uint256[] memory) {
                bytes32[] memory store = _values(set._inner);
                uint256[] memory result;
                assembly {
                    result := store
                }
                return result;
            }
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts v4.4.1 (access/IAccessControl.sol)
        pragma solidity ^0.8.0;
        /**
         * @dev External interface of AccessControl declared to support ERC165 detection.
         */
        interface IAccessControlUpgradeable {
            /**
             * @dev Emitted when `newAdminRole` is set as ``role``'s admin role, replacing `previousAdminRole`
             *
             * `DEFAULT_ADMIN_ROLE` is the starting admin for all roles, despite
             * {RoleAdminChanged} not being emitted signaling this.
             *
             * _Available since v3.1._
             */
            event RoleAdminChanged(bytes32 indexed role, bytes32 indexed previousAdminRole, bytes32 indexed newAdminRole);
            /**
             * @dev Emitted when `account` is granted `role`.
             *
             * `sender` is the account that originated the contract call, an admin role
             * bearer except when using {AccessControl-_setupRole}.
             */
            event RoleGranted(bytes32 indexed role, address indexed account, address indexed sender);
            /**
             * @dev Emitted when `account` is revoked `role`.
             *
             * `sender` is the account that originated the contract call:
             *   - if using `revokeRole`, it is the admin role bearer
             *   - if using `renounceRole`, it is the role bearer (i.e. `account`)
             */
            event RoleRevoked(bytes32 indexed role, address indexed account, address indexed sender);
            /**
             * @dev Returns `true` if `account` has been granted `role`.
             */
            function hasRole(bytes32 role, address account) external view returns (bool);
            /**
             * @dev Returns the admin role that controls `role`. See {grantRole} and
             * {revokeRole}.
             *
             * To change a role's admin, use {AccessControl-_setRoleAdmin}.
             */
            function getRoleAdmin(bytes32 role) external view returns (bytes32);
            /**
             * @dev Grants `role` to `account`.
             *
             * If `account` had not been already granted `role`, emits a {RoleGranted}
             * event.
             *
             * Requirements:
             *
             * - the caller must have ``role``'s admin role.
             */
            function grantRole(bytes32 role, address account) external;
            /**
             * @dev Revokes `role` from `account`.
             *
             * If `account` had been granted `role`, emits a {RoleRevoked} event.
             *
             * Requirements:
             *
             * - the caller must have ``role``'s admin role.
             */
            function revokeRole(bytes32 role, address account) external;
            /**
             * @dev Revokes `role` from the calling account.
             *
             * Roles are often managed via {grantRole} and {revokeRole}: this function's
             * purpose is to provide a mechanism for accounts to lose their privileges
             * if they are compromised (such as when a trusted device is misplaced).
             *
             * If the calling account had been granted `role`, emits a {RoleRevoked}
             * event.
             *
             * Requirements:
             *
             * - the caller must be `account`.
             */
            function renounceRole(bytes32 role, address account) external;
        }
        // SPDX-License-Identifier: Apache-2.0
        pragma solidity ^0.8.11;
        interface IThirdwebContract {
            /// @dev Returns the module type of the contract.
            function contractType() external pure returns (bytes32);
            /// @dev Returns the version of the contract.
            function contractVersion() external pure returns (uint8);
            /// @dev Returns the metadata URI of the contract.
            function contractURI() external view returns (string memory);
            /**
             *  @dev Sets contract URI for the storefront-level metadata of the contract.
             *       Only module admin can call this function.
             */
            function setContractURI(string calldata _uri) external;
        }
        // SPDX-License-Identifier: Apache-2.0
        pragma solidity ^0.8.11;
        interface IThirdwebPlatformFee {
            /// @dev Returns the platform fee bps and recipient.
            function getPlatformFeeInfo() external view returns (address platformFeeRecipient, uint16 platformFeeBps);
            /// @dev Lets a module admin update the fees on primary sales.
            function setPlatformFeeInfo(address _platformFeeRecipient, uint256 _platformFeeBps) external;
        }
        // SPDX-License-Identifier: Apache-2.0
        pragma solidity ^0.8.11;
        interface IThirdwebPrimarySale {
            /// @dev The adress that receives all primary sales value.
            function primarySaleRecipient() external view returns (address);
            /// @dev Lets a module admin set the default recipient of all primary sales.
            function setPrimarySaleRecipient(address _saleRecipient) external;
        }
        // SPDX-License-Identifier: Apache-2.0
        pragma solidity ^0.8.11;
        import "@openzeppelin/contracts-upgradeable/interfaces/IERC2981Upgradeable.sol";
        interface IThirdwebRoyalty is IERC2981Upgradeable {
            struct RoyaltyInfo {
                address recipient;
                uint256 bps;
            }
            /// @dev Returns the royalty recipient and fee bps.
            function getDefaultRoyaltyInfo() external view returns (address, uint16);
            /// @dev Lets a module admin update the royalty bps and recipient.
            function setDefaultRoyaltyInfo(address _royaltyRecipient, uint256 _royaltyBps) external;
            /// @dev Lets a module admin set the royalty recipient for a particular token Id.
            function setRoyaltyInfoForToken(
                uint256 tokenId,
                address recipient,
                uint256 bps
            ) external;
            /// @dev Returns the royalty recipient for a particular token Id.
            function getRoyaltyInfoForToken(uint256 tokenId) external view returns (address, uint16);
            /// @dev Emitted when royalty info is updated.
            event DefaultRoyalty(address newRoyaltyRecipient, uint256 newRoyaltyBps);
            /// @dev Emitted when royalty recipient for tokenId is set
            event RoyaltyForToken(uint256 indexed tokenId, address royaltyRecipient, uint256 royaltyBps);
        }
        // SPDX-License-Identifier: Apache-2.0
        pragma solidity ^0.8.11;
        interface IThirdwebOwnable {
            /// @dev Returns the owner of the contract.
            function owner() external view returns (address);
            /// @dev Lets a module admin set a new owner for the contract. The new owner must be a module admin.
            function setOwner(address _newOwner) external;
        }
        // SPDX-License-Identifier: Apache-2.0
        pragma solidity ^0.8.11;
        import "@openzeppelin/contracts-upgradeable/utils/structs/BitMapsUpgradeable.sol";
        /**
         *  Thirdweb's 'Drop' contracts are distribution mechanisms for tokens.
         *
         *  A contract admin (i.e. a holder of `DEFAULT_ADMIN_ROLE`) can set a series of claim conditions,
         *  ordered by their respective `startTimestamp`. A claim condition defines criteria under which
         *  accounts can mint tokens. Claim conditions can be overwritten or added to by the contract admin.
         *  At any moment, there is only one active claim condition.
         */
        interface IDropClaimCondition {
            /**
             *  @notice The criteria that make up a claim condition.
             *
             *  @param startTimestamp                 The unix timestamp after which the claim condition applies.
             *                                        The same claim condition applies until the `startTimestamp`
             *                                        of the next claim condition.
             *
             *  @param maxClaimableSupply             The maximum total number of tokens that can be claimed under
             *                                        the claim condition.
             *
             *  @param supplyClaimed                  At any given point, the number of tokens that have been claimed
             *                                        under the claim condition.
             *
             *  @param quantityLimitPerTransaction    The maximum number of tokens that can be claimed in a single
             *                                        transaction.
             *
             *  @param waitTimeInSecondsBetweenClaims The least number of seconds an account must wait after claiming
             *                                        tokens, to be able to claim tokens again.
             *
             *  @param merkleRoot                     The allowlist of addresses that can claim tokens under the claim
             *                                        condition.
             *
             *  @param pricePerToken                  The price required to pay per token claimed.
             *
             *  @param currency                       The currency in which the `pricePerToken` must be paid.
             */
            struct ClaimCondition {
                uint256 startTimestamp;
                uint256 maxClaimableSupply;
                uint256 supplyClaimed;
                uint256 quantityLimitPerTransaction;
                uint256 waitTimeInSecondsBetweenClaims;
                bytes32 merkleRoot;
                uint256 pricePerToken;
                address currency;
            }
            /**
             *  @notice The set of all claim conditions, at any given moment.
             *  Claim Phase ID = [currentStartId, currentStartId + length - 1];
             *
             *  @param currentStartId           Acts as the uid for each claim condition. Incremented
             *                                  by one every time a claim condition is created.
             *
             *  @param count                    The total number of phases / claim conditions in the list
             *                                  of claim conditions.
             *
             *  @param phases                   The claim conditions at a given uid. Claim conditions
             *                                  are ordered in an ascending order by their `startTimestamp`.
             *
             *  @param limitLastClaimTimestamp  Map from an account and uid for a claim condition, to the last timestamp
             *                                  at which the account claimed tokens under that claim condition.
             *
             *  @param limitMerkleProofClaim    Map from a claim condition uid to whether an address in an allowlist
             *                                  has already claimed tokens i.e. used their place in the allowlist.
             */
            struct ClaimConditionList {
                uint256 currentStartId;
                uint256 count;
                mapping(uint256 => ClaimCondition) phases;
                mapping(uint256 => mapping(address => uint256)) limitLastClaimTimestamp;
                mapping(uint256 => BitMapsUpgradeable.BitMap) limitMerkleProofClaim;
            }
        }
        // SPDX-License-Identifier: Apache-2.0
        pragma solidity ^0.8.11;
        interface IWETH {
            function deposit() external payable;
            function withdraw(uint256 amount) external;
            function transfer(address to, uint256 value) external returns (bool);
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v4.5.0) (token/ERC20/IERC20.sol)
        pragma solidity ^0.8.0;
        /**
         * @dev Interface of the ERC20 standard as defined in the EIP.
         */
        interface IERC20Upgradeable {
            /**
             * @dev Returns the amount of tokens in existence.
             */
            function totalSupply() external view returns (uint256);
            /**
             * @dev Returns the amount of tokens owned by `account`.
             */
            function balanceOf(address account) external view returns (uint256);
            /**
             * @dev Moves `amount` tokens from the caller's account to `to`.
             *
             * Returns a boolean value indicating whether the operation succeeded.
             *
             * Emits a {Transfer} event.
             */
            function transfer(address to, uint256 amount) external returns (bool);
            /**
             * @dev Returns the remaining number of tokens that `spender` will be
             * allowed to spend on behalf of `owner` through {transferFrom}. This is
             * zero by default.
             *
             * This value changes when {approve} or {transferFrom} are called.
             */
            function allowance(address owner, address spender) external view returns (uint256);
            /**
             * @dev Sets `amount` as the allowance of `spender` over the caller's tokens.
             *
             * Returns a boolean value indicating whether the operation succeeded.
             *
             * IMPORTANT: Beware that changing an allowance with this method brings the risk
             * that someone may use both the old and the new allowance by unfortunate
             * transaction ordering. One possible solution to mitigate this race
             * condition is to first reduce the spender's allowance to 0 and set the
             * desired value afterwards:
             * https://github.com/ethereum/EIPs/issues/20#issuecomment-263524729
             *
             * Emits an {Approval} event.
             */
            function approve(address spender, uint256 amount) external returns (bool);
            /**
             * @dev Moves `amount` tokens from `from` to `to` using the
             * allowance mechanism. `amount` is then deducted from the caller's
             * allowance.
             *
             * Returns a boolean value indicating whether the operation succeeded.
             *
             * Emits a {Transfer} event.
             */
            function transferFrom(
                address from,
                address to,
                uint256 amount
            ) external returns (bool);
            /**
             * @dev Emitted when `value` tokens are moved from one account (`from`) to
             * another (`to`).
             *
             * Note that `value` may be zero.
             */
            event Transfer(address indexed from, address indexed to, uint256 value);
            /**
             * @dev Emitted when the allowance of a `spender` for an `owner` is set by
             * a call to {approve}. `value` is the new allowance.
             */
            event Approval(address indexed owner, address indexed spender, uint256 value);
        }
        

        File 3 of 3: DropERC721
        // SPDX-License-Identifier: Apache-2.0
        pragma solidity ^0.8.11;
        //  ==========  External imports    ==========
        import "@openzeppelin/contracts-upgradeable/token/ERC721/extensions/ERC721EnumerableUpgradeable.sol";
        import "@openzeppelin/contracts-upgradeable/interfaces/IERC2981Upgradeable.sol";
        import "@openzeppelin/contracts-upgradeable/access/AccessControlEnumerableUpgradeable.sol";
        import "@openzeppelin/contracts-upgradeable/security/ReentrancyGuardUpgradeable.sol";
        import "@openzeppelin/contracts-upgradeable/utils/structs/BitMapsUpgradeable.sol";
        import "@openzeppelin/contracts-upgradeable/utils/MulticallUpgradeable.sol";
        import "@openzeppelin/contracts-upgradeable/utils/StringsUpgradeable.sol";
        //  ==========  Internal imports    ==========
        import { IDropERC721 } from "../interfaces/drop/IDropERC721.sol";
        import { ITWFee } from "../interfaces/ITWFee.sol";
        import "../openzeppelin-presets/metatx/ERC2771ContextUpgradeable.sol";
        import "../lib/CurrencyTransferLib.sol";
        import "../lib/FeeType.sol";
        import "../lib/MerkleProof.sol";
        contract DropERC721 is
            Initializable,
            ReentrancyGuardUpgradeable,
            ERC2771ContextUpgradeable,
            MulticallUpgradeable,
            AccessControlEnumerableUpgradeable,
            ERC721EnumerableUpgradeable,
            IDropERC721
        {
            using BitMapsUpgradeable for BitMapsUpgradeable.BitMap;
            using StringsUpgradeable for uint256;
            /*///////////////////////////////////////////////////////////////
                                    State variables
            //////////////////////////////////////////////////////////////*/
            bytes32 private constant MODULE_TYPE = bytes32("DropERC721");
            uint256 private constant VERSION = 1;
            /// @dev Only transfers to or from TRANSFER_ROLE holders are valid, when transfers are restricted.
            bytes32 private constant TRANSFER_ROLE = keccak256("TRANSFER_ROLE");
            /// @dev Only MINTER_ROLE holders can lazy mint NFTs.
            bytes32 private constant MINTER_ROLE = keccak256("MINTER_ROLE");
            /// @dev Max bps in the thirdweb system.
            uint256 private constant MAX_BPS = 10_000;
            /// @dev The thirdweb contract with fee related information.
            ITWFee public immutable thirdwebFee;
            /// @dev Owner of the contract (purpose: OpenSea compatibility)
            address private _owner;
            /// @dev The next token ID of the NFT to "lazy mint".
            uint256 public nextTokenIdToMint;
            /// @dev The next token ID of the NFT that can be claimed.
            uint256 public nextTokenIdToClaim;
            /// @dev The address that receives all primary sales value.
            address public primarySaleRecipient;
            /// @dev The max number of NFTs a wallet can claim.
            uint256 public maxWalletClaimCount;
            /// @dev Global max total supply of NFTs.
            uint256 public maxTotalSupply;
            /// @dev The address that receives all platform fees from all sales.
            address private platformFeeRecipient;
            /// @dev The (default) address that receives all royalty value.
            address private royaltyRecipient;
            /// @dev The (default) % of a sale to take as royalty (in basis points).
            uint128 private royaltyBps;
            /// @dev The % of primary sales collected as platform fees.
            uint128 private platformFeeBps;
            /// @dev Contract level metadata.
            string public contractURI;
            /// @dev Largest tokenId of each batch of tokens with the same baseURI
            uint256[] public baseURIIndices;
            /// @dev The set of all claim conditions, at any given moment.
            ClaimConditionList public claimCondition;
            /*///////////////////////////////////////////////////////////////
                                        Mappings
            //////////////////////////////////////////////////////////////*/
            /**
             *  @dev Mapping from 'Largest tokenId of a batch of tokens with the same baseURI'
             *       to base URI for the respective batch of tokens.
             **/
            mapping(uint256 => string) private baseURI;
            /**
             *  @dev Mapping from 'Largest tokenId of a batch of 'delayed-reveal' tokens with
             *       the same baseURI' to encrypted base URI for the respective batch of tokens.
             **/
            mapping(uint256 => bytes) public encryptedBaseURI;
            /// @dev Mapping from address => total number of NFTs a wallet has claimed.
            mapping(address => uint256) public walletClaimCount;
            /// @dev Token ID => royalty recipient and bps for token
            mapping(uint256 => RoyaltyInfo) private royaltyInfoForToken;
            /*///////////////////////////////////////////////////////////////
                            Constructor + initializer logic
            //////////////////////////////////////////////////////////////*/
            constructor(address _thirdwebFee) initializer {
                thirdwebFee = ITWFee(_thirdwebFee);
            }
            /// @dev Initiliazes the contract, like a constructor.
            function initialize(
                address _defaultAdmin,
                string memory _name,
                string memory _symbol,
                string memory _contractURI,
                address[] memory _trustedForwarders,
                address _saleRecipient,
                address _royaltyRecipient,
                uint128 _royaltyBps,
                uint128 _platformFeeBps,
                address _platformFeeRecipient
            ) external initializer {
                // Initialize inherited contracts, most base-like -> most derived.
                __ReentrancyGuard_init();
                __ERC2771Context_init(_trustedForwarders);
                __ERC721_init(_name, _symbol);
                // Initialize this contract's state.
                royaltyRecipient = _royaltyRecipient;
                royaltyBps = _royaltyBps;
                platformFeeRecipient = _platformFeeRecipient;
                platformFeeBps = _platformFeeBps;
                primarySaleRecipient = _saleRecipient;
                contractURI = _contractURI;
                _owner = _defaultAdmin;
                _setupRole(DEFAULT_ADMIN_ROLE, _defaultAdmin);
                _setupRole(MINTER_ROLE, _defaultAdmin);
                _setupRole(TRANSFER_ROLE, _defaultAdmin);
                _setupRole(TRANSFER_ROLE, address(0));
            }
            /*///////////////////////////////////////////////////////////////
                                Generic contract logic
            //////////////////////////////////////////////////////////////*/
            /// @dev Returns the type of the contract.
            function contractType() external pure returns (bytes32) {
                return MODULE_TYPE;
            }
            /// @dev Returns the version of the contract.
            function contractVersion() external pure returns (uint8) {
                return uint8(VERSION);
            }
            /**
             * @dev Returns the address of the current owner.
             */
            function owner() public view returns (address) {
                return hasRole(DEFAULT_ADMIN_ROLE, _owner) ? _owner : address(0);
            }
            /*///////////////////////////////////////////////////////////////
                                ERC 165 / 721 / 2981 logic
            //////////////////////////////////////////////////////////////*/
            /// @dev Returns the URI for a given tokenId.
            function tokenURI(uint256 _tokenId) public view override returns (string memory) {
                for (uint256 i = 0; i < baseURIIndices.length; i += 1) {
                    if (_tokenId < baseURIIndices[i]) {
                        if (encryptedBaseURI[baseURIIndices[i]].length != 0) {
                            return string(abi.encodePacked(baseURI[baseURIIndices[i]], "0"));
                        } else {
                            return string(abi.encodePacked(baseURI[baseURIIndices[i]], _tokenId.toString()));
                        }
                    }
                }
                return "";
            }
            /// @dev See ERC 165
            function supportsInterface(bytes4 interfaceId)
                public
                view
                virtual
                override(ERC721EnumerableUpgradeable, AccessControlEnumerableUpgradeable, IERC165Upgradeable)
                returns (bool)
            {
                return super.supportsInterface(interfaceId) || type(IERC2981Upgradeable).interfaceId == interfaceId;
            }
            /// @dev Returns the royalty recipient and amount, given a tokenId and sale price.
            function royaltyInfo(uint256 tokenId, uint256 salePrice)
                external
                view
                virtual
                returns (address receiver, uint256 royaltyAmount)
            {
                (address recipient, uint256 bps) = getRoyaltyInfoForToken(tokenId);
                receiver = recipient;
                royaltyAmount = (salePrice * bps) / MAX_BPS;
            }
            /*///////////////////////////////////////////////////////////////
                            Minting + delayed-reveal logic
            //////////////////////////////////////////////////////////////*/
            /**
             *  @dev Lets an account with `MINTER_ROLE` lazy mint 'n' NFTs.
             *       The URIs for each token is the provided `_baseURIForTokens` + `{tokenId}`.
             */
            function lazyMint(
                uint256 _amount,
                string calldata _baseURIForTokens,
                bytes calldata _encryptedBaseURI
            ) external onlyRole(MINTER_ROLE) {
                uint256 startId = nextTokenIdToMint;
                uint256 baseURIIndex = startId + _amount;
                nextTokenIdToMint = baseURIIndex;
                baseURI[baseURIIndex] = _baseURIForTokens;
                baseURIIndices.push(baseURIIndex);
                if (_encryptedBaseURI.length != 0) {
                    encryptedBaseURI[baseURIIndex] = _encryptedBaseURI;
                }
                emit TokensLazyMinted(startId, startId + _amount - 1, _baseURIForTokens, _encryptedBaseURI);
            }
            /// @dev Lets an account with `MINTER_ROLE` reveal the URI for a batch of 'delayed-reveal' NFTs.
            function reveal(uint256 index, bytes calldata _key)
                external
                onlyRole(MINTER_ROLE)
                returns (string memory revealedURI)
            {
                require(index < baseURIIndices.length, "invalid index.");
                uint256 _index = baseURIIndices[index];
                bytes memory encryptedURI = encryptedBaseURI[_index];
                require(encryptedURI.length != 0, "nothing to reveal.");
                revealedURI = string(encryptDecrypt(encryptedURI, _key));
                baseURI[_index] = revealedURI;
                delete encryptedBaseURI[_index];
                emit NFTRevealed(_index, revealedURI);
                return revealedURI;
            }
            /// @dev See: https://ethereum.stackexchange.com/questions/69825/decrypt-message-on-chain
            function encryptDecrypt(bytes memory data, bytes calldata key) public pure returns (bytes memory result) {
                // Store data length on stack for later use
                uint256 length = data.length;
                // solhint-disable-next-line no-inline-assembly
                assembly {
                    // Set result to free memory pointer
                    result := mload(0x40)
                    // Increase free memory pointer by lenght + 32
                    mstore(0x40, add(add(result, length), 32))
                    // Set result length
                    mstore(result, length)
                }
                // Iterate over the data stepping by 32 bytes
                for (uint256 i = 0; i < length; i += 32) {
                    // Generate hash of the key and offset
                    bytes32 hash = keccak256(abi.encodePacked(key, i));
                    bytes32 chunk;
                    // solhint-disable-next-line no-inline-assembly
                    assembly {
                        // Read 32-bytes data chunk
                        chunk := mload(add(data, add(i, 32)))
                    }
                    // XOR the chunk with hash
                    chunk ^= hash;
                    // solhint-disable-next-line no-inline-assembly
                    assembly {
                        // Write 32-byte encrypted chunk
                        mstore(add(result, add(i, 32)), chunk)
                    }
                }
            }
            /*///////////////////////////////////////////////////////////////
                                    Claim logic
            //////////////////////////////////////////////////////////////*/
            /// @dev Lets an account claim NFTs.
            function claim(
                address _receiver,
                uint256 _quantity,
                address _currency,
                uint256 _pricePerToken,
                bytes32[] calldata _proofs,
                uint256 _proofMaxQuantityPerTransaction
            ) external payable nonReentrant {
                uint256 tokenIdToClaim = nextTokenIdToClaim;
                // Get the claim conditions.
                uint256 activeConditionId = getActiveClaimConditionId();
                // Verify claim validity. If not valid, revert.
                verifyClaim(activeConditionId, _msgSender(), _quantity, _currency, _pricePerToken);
                // Verify inclusion in allowlist
                (bool validMerkleProof, uint256 merkleProofIndex) = verifyClaimMerkleProof(
                    activeConditionId,
                    _msgSender(),
                    _quantity,
                    _proofs,
                    _proofMaxQuantityPerTransaction
                );
                if (validMerkleProof && _proofMaxQuantityPerTransaction > 0) {
                    // Mark the claimer's use of their position in the allowlist.
                    claimCondition.limitMerkleProofClaim[activeConditionId].set(merkleProofIndex);
                }
                // If there's a price, collect price.
                collectClaimPrice(_quantity, _currency, _pricePerToken);
                // Mint the relevant NFTs to claimer.
                transferClaimedTokens(_receiver, activeConditionId, _quantity);
                emit TokensClaimed(activeConditionId, _msgSender(), _receiver, tokenIdToClaim, _quantity);
            }
            /// @dev Lets a contract admin (account with `DEFAULT_ADMIN_ROLE`) set claim conditions.
            function setClaimConditions(ClaimCondition[] calldata _phases, bool _resetClaimEligibility)
                external
                onlyRole(DEFAULT_ADMIN_ROLE)
            {
                uint256 existingStartIndex = claimCondition.currentStartId;
                uint256 existingPhaseCount = claimCondition.count;
                /**
                 *  `limitLastClaimTimestamp` and `limitMerkleProofClaim` are mappings that use a
                 *  claim condition's UID as a key.
                 *
                 *  If `_resetClaimEligibility == true`, we assign completely new UIDs to the claim
                 *  conditions in `_phases`, effectively resetting the restrictions on claims expressed
                 *  by `limitLastClaimTimestamp` and `limitMerkleProofClaim`.
                 */
                uint256 newStartIndex = existingStartIndex;
                if (_resetClaimEligibility) {
                    newStartIndex = existingStartIndex + existingPhaseCount;
                }
                claimCondition.count = _phases.length;
                claimCondition.currentStartId = newStartIndex;
                uint256 lastConditionStartTimestamp;
                for (uint256 i = 0; i < _phases.length; i++) {
                    require(
                        i == 0 || lastConditionStartTimestamp < _phases[i].startTimestamp,
                        "startTimestamp must be in ascending order."
                    );
                    claimCondition.phases[newStartIndex + i] = _phases[i];
                    claimCondition.phases[newStartIndex + i].supplyClaimed = 0;
                    lastConditionStartTimestamp = _phases[i].startTimestamp;
                }
                /**
                 *  Gas refunds (as much as possible)
                 *
                 *  If `_resetClaimEligibility == true`, we assign completely new UIDs to the claim
                 *  conditions in `_phases`. So, we delete claim conditions with UID < `newStartIndex`.
                 *
                 *  If `_resetClaimEligibility == false`, and there are more existing claim conditions
                 *  than in `_phases`, we delete the existing claim conditions that don't get replaced
                 *  by the conditions in `_phases`.
                 */
                if (_resetClaimEligibility) {
                    for (uint256 i = existingStartIndex; i < newStartIndex; i++) {
                        delete claimCondition.phases[i];
                        delete claimCondition.limitMerkleProofClaim[i];
                    }
                } else {
                    if (existingPhaseCount > _phases.length) {
                        for (uint256 i = _phases.length; i < existingPhaseCount; i++) {
                            delete claimCondition.phases[newStartIndex + i];
                            delete claimCondition.limitMerkleProofClaim[newStartIndex + i];
                        }
                    }
                }
                emit ClaimConditionsUpdated(_phases);
            }
            /// @dev Collects and distributes the primary sale value of NFTs being claimed.
            function collectClaimPrice(
                uint256 _quantityToClaim,
                address _currency,
                uint256 _pricePerToken
            ) internal {
                if (_pricePerToken == 0) {
                    return;
                }
                uint256 totalPrice = _quantityToClaim * _pricePerToken;
                uint256 platformFees = (totalPrice * platformFeeBps) / MAX_BPS;
                (address twFeeRecipient, uint256 twFeeBps) = thirdwebFee.getFeeInfo(address(this), FeeType.PRIMARY_SALE);
                uint256 twFee = (totalPrice * twFeeBps) / MAX_BPS;
                if (_currency == CurrencyTransferLib.NATIVE_TOKEN) {
                    require(msg.value == totalPrice, "must send total price.");
                }
                CurrencyTransferLib.transferCurrency(_currency, _msgSender(), platformFeeRecipient, platformFees);
                CurrencyTransferLib.transferCurrency(_currency, _msgSender(), twFeeRecipient, twFee);
                CurrencyTransferLib.transferCurrency(
                    _currency,
                    _msgSender(),
                    primarySaleRecipient,
                    totalPrice - platformFees - twFee
                );
            }
            /// @dev Transfers the NFTs being claimed.
            function transferClaimedTokens(
                address _to,
                uint256 _conditionId,
                uint256 _quantityBeingClaimed
            ) internal {
                // Update the supply minted under mint condition.
                claimCondition.phases[_conditionId].supplyClaimed += _quantityBeingClaimed;
                // if transfer claimed tokens is called when `to != msg.sender`, it'd use msg.sender's limits.
                // behavior would be similar to `msg.sender` mint for itself, then transfer to `_to`.
                claimCondition.limitLastClaimTimestamp[_conditionId][_msgSender()] = block.timestamp;
                walletClaimCount[_msgSender()] += _quantityBeingClaimed;
                uint256 tokenIdToClaim = nextTokenIdToClaim;
                for (uint256 i = 0; i < _quantityBeingClaimed; i += 1) {
                    _mint(_to, tokenIdToClaim);
                    tokenIdToClaim += 1;
                }
                nextTokenIdToClaim = tokenIdToClaim;
            }
            /// @dev Checks a request to claim NFTs against the active claim condition's criteria.
            function verifyClaim(
                uint256 _conditionId,
                address _claimer,
                uint256 _quantity,
                address _currency,
                uint256 _pricePerToken
            ) public view {
                ClaimCondition memory currentClaimPhase = claimCondition.phases[_conditionId];
                require(
                    _currency == currentClaimPhase.currency && _pricePerToken == currentClaimPhase.pricePerToken,
                    "invalid currency or price specified."
                );
                require(
                    _quantity > 0 && _quantity <= currentClaimPhase.quantityLimitPerTransaction,
                    "invalid quantity claimed."
                );
                require(
                    currentClaimPhase.supplyClaimed + _quantity <= currentClaimPhase.maxClaimableSupply,
                    "exceed max mint supply."
                );
                require(nextTokenIdToClaim + _quantity <= nextTokenIdToMint, "not enough minted tokens.");
                require(maxTotalSupply == 0 || nextTokenIdToClaim + _quantity <= maxTotalSupply, "exceed max total supply.");
                require(
                    maxWalletClaimCount == 0 || walletClaimCount[_claimer] + _quantity <= maxWalletClaimCount,
                    "exceed claim limit for wallet"
                );
                (uint256 lastClaimTimestamp, uint256 nextValidClaimTimestamp) = getClaimTimestamp(_conditionId, _claimer);
                require(lastClaimTimestamp == 0 || block.timestamp >= nextValidClaimTimestamp, "cannot claim yet.");
            }
            /// @dev Checks whether a claimer meets the claim condition's allowlist criteria.
            function verifyClaimMerkleProof(
                uint256 _conditionId,
                address _claimer,
                uint256 _quantity,
                bytes32[] calldata _proofs,
                uint256 _proofMaxQuantityPerTransaction
            ) public view returns (bool validMerkleProof, uint256 merkleProofIndex) {
                ClaimCondition memory currentClaimPhase = claimCondition.phases[_conditionId];
                if (currentClaimPhase.merkleRoot != bytes32(0)) {
                    (validMerkleProof, merkleProofIndex) = MerkleProof.verify(
                        _proofs,
                        currentClaimPhase.merkleRoot,
                        keccak256(abi.encodePacked(_claimer, _proofMaxQuantityPerTransaction))
                    );
                    require(validMerkleProof, "not in whitelist.");
                    require(!claimCondition.limitMerkleProofClaim[_conditionId].get(merkleProofIndex), "proof claimed.");
                    require(
                        _proofMaxQuantityPerTransaction == 0 || _quantity <= _proofMaxQuantityPerTransaction,
                        "invalid quantity proof."
                    );
                }
            }
            /*///////////////////////////////////////////////////////////////
                                Getter functions
            //////////////////////////////////////////////////////////////*/
            /// @dev At any given moment, returns the uid for the active claim condition.
            function getActiveClaimConditionId() public view returns (uint256) {
                for (uint256 i = claimCondition.currentStartId + claimCondition.count; i > claimCondition.currentStartId; i--) {
                    if (block.timestamp >= claimCondition.phases[i - 1].startTimestamp) {
                        return i - 1;
                    }
                }
                revert("no active mint condition.");
            }
            /// @dev Returns the royalty recipient and bps for a particular token Id.
            function getRoyaltyInfoForToken(uint256 _tokenId) public view returns (address, uint16) {
                RoyaltyInfo memory royaltyForToken = royaltyInfoForToken[_tokenId];
                return
                    royaltyForToken.recipient == address(0)
                        ? (royaltyRecipient, uint16(royaltyBps))
                        : (royaltyForToken.recipient, uint16(royaltyForToken.bps));
            }
            /// @dev Returns the platform fee recipient and bps.
            function getPlatformFeeInfo() external view returns (address, uint16) {
                return (platformFeeRecipient, uint16(platformFeeBps));
            }
            /// @dev Returns the default royalty recipient and bps.
            function getDefaultRoyaltyInfo() external view returns (address, uint16) {
                return (royaltyRecipient, uint16(royaltyBps));
            }
            /// @dev Returns the timestamp for when a claimer is eligible for claiming NFTs again.
            function getClaimTimestamp(uint256 _conditionId, address _claimer)
                public
                view
                returns (uint256 lastClaimTimestamp, uint256 nextValidClaimTimestamp)
            {
                lastClaimTimestamp = claimCondition.limitLastClaimTimestamp[_conditionId][_claimer];
                unchecked {
                    nextValidClaimTimestamp =
                        lastClaimTimestamp +
                        claimCondition.phases[_conditionId].waitTimeInSecondsBetweenClaims;
                    if (nextValidClaimTimestamp < lastClaimTimestamp) {
                        nextValidClaimTimestamp = type(uint256).max;
                    }
                }
            }
            /// @dev Returns the claim condition at the given uid.
            function getClaimConditionById(uint256 _conditionId) external view returns (ClaimCondition memory condition) {
                condition = claimCondition.phases[_conditionId];
            }
            /// @dev Returns the amount of stored baseURIs
            function getBaseURICount() external view returns (uint256) {
                return baseURIIndices.length;
            }
            /*///////////////////////////////////////////////////////////////
                                Setter functions
            //////////////////////////////////////////////////////////////*/
            /// @dev Lets a contract admin set a claim count for a wallet.
            function setWalletClaimCount(address _claimer, uint256 _count) external onlyRole(DEFAULT_ADMIN_ROLE) {
                walletClaimCount[_claimer] = _count;
                emit WalletClaimCountUpdated(_claimer, _count);
            }
            /// @dev Lets a contract admin set a maximum number of NFTs that can be claimed by any wallet.
            function setMaxWalletClaimCount(uint256 _count) external onlyRole(DEFAULT_ADMIN_ROLE) {
                maxWalletClaimCount = _count;
                emit MaxWalletClaimCountUpdated(_count);
            }
            /// @dev Lets a contract admin set the global maximum supply for collection's NFTs.
            function setMaxTotalSupply(uint256 _maxTotalSupply) external onlyRole(DEFAULT_ADMIN_ROLE) {
                require(_maxTotalSupply < nextTokenIdToMint, "already minted more than desired max supply");
                maxTotalSupply = _maxTotalSupply;
                emit MaxTotalSupplyUpdated(_maxTotalSupply);
            }
            /// @dev Lets a contract admin set the recipient for all primary sales.
            function setPrimarySaleRecipient(address _saleRecipient) external onlyRole(DEFAULT_ADMIN_ROLE) {
                primarySaleRecipient = _saleRecipient;
                emit PrimarySaleRecipientUpdated(_saleRecipient);
            }
            /// @dev Lets a contract admin update the default royalty recipient and bps.
            function setDefaultRoyaltyInfo(address _royaltyRecipient, uint256 _royaltyBps)
                external
                onlyRole(DEFAULT_ADMIN_ROLE)
            {
                require(_royaltyBps <= MAX_BPS, "exceed royalty bps");
                royaltyRecipient = _royaltyRecipient;
                royaltyBps = uint128(_royaltyBps);
                emit DefaultRoyalty(_royaltyRecipient, _royaltyBps);
            }
            /// @dev Lets a contract admin set the royalty recipient and bps for a particular token Id.
            function setRoyaltyInfoForToken(
                uint256 _tokenId,
                address _recipient,
                uint256 _bps
            ) external onlyRole(DEFAULT_ADMIN_ROLE) {
                require(_bps <= MAX_BPS, "exceed royalty bps");
                royaltyInfoForToken[_tokenId] = RoyaltyInfo({ recipient: _recipient, bps: _bps });
                emit RoyaltyForToken(_tokenId, _recipient, _bps);
            }
            /// @dev Lets a contract admin update the platform fee recipient and bps
            function setPlatformFeeInfo(address _platformFeeRecipient, uint256 _platformFeeBps)
                external
                onlyRole(DEFAULT_ADMIN_ROLE)
            {
                require(_platformFeeBps <= MAX_BPS, "bps <= 10000.");
                platformFeeBps = uint64(_platformFeeBps);
                platformFeeRecipient = _platformFeeRecipient;
                emit PlatformFeeInfoUpdated(_platformFeeRecipient, _platformFeeBps);
            }
            /// @dev Lets a contract admin set a new owner for the contract. The new owner must be a contract admin.
            function setOwner(address _newOwner) external onlyRole(DEFAULT_ADMIN_ROLE) {
                require(hasRole(DEFAULT_ADMIN_ROLE, _newOwner), "new owner not contract admin.");
                address _prevOwner = _owner;
                _owner = _newOwner;
                emit OwnerUpdated(_prevOwner, _newOwner);
            }
            /// @dev Lets a contract admin set the URI for contract-level metadata.
            function setContractURI(string calldata _uri) external onlyRole(DEFAULT_ADMIN_ROLE) {
                contractURI = _uri;
            }
            /*///////////////////////////////////////////////////////////////
                                Miscellaneous
            //////////////////////////////////////////////////////////////*/
            /// @dev Burns `tokenId`. See {ERC721-_burn}.
            function burn(uint256 tokenId) public virtual {
                //solhint-disable-next-line max-line-length
                require(_isApprovedOrOwner(_msgSender(), tokenId), "caller is not owner nor approved");
                _burn(tokenId);
            }
            /// @dev See {ERC721-_beforeTokenTransfer}.
            function _beforeTokenTransfer(
                address from,
                address to,
                uint256 tokenId
            ) internal virtual override(ERC721EnumerableUpgradeable) {
                super._beforeTokenTransfer(from, to, tokenId);
                // if transfer is restricted on the contract, we still want to allow burning and minting
                if (!hasRole(TRANSFER_ROLE, address(0)) && from != address(0) && to != address(0)) {
                    require(hasRole(TRANSFER_ROLE, from) || hasRole(TRANSFER_ROLE, to), "restricted to TRANSFER_ROLE holders");
                }
            }
            function _msgSender()
                internal
                view
                virtual
                override(ContextUpgradeable, ERC2771ContextUpgradeable)
                returns (address sender)
            {
                return ERC2771ContextUpgradeable._msgSender();
            }
            function _msgData()
                internal
                view
                virtual
                override(ContextUpgradeable, ERC2771ContextUpgradeable)
                returns (bytes calldata)
            {
                return ERC2771ContextUpgradeable._msgData();
            }
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts v4.4.1 (token/ERC721/extensions/ERC721Enumerable.sol)
        pragma solidity ^0.8.0;
        import "../ERC721Upgradeable.sol";
        import "./IERC721EnumerableUpgradeable.sol";
        import "../../../proxy/utils/Initializable.sol";
        /**
         * @dev This implements an optional extension of {ERC721} defined in the EIP that adds
         * enumerability of all the token ids in the contract as well as all token ids owned by each
         * account.
         */
        abstract contract ERC721EnumerableUpgradeable is Initializable, ERC721Upgradeable, IERC721EnumerableUpgradeable {
            function __ERC721Enumerable_init() internal onlyInitializing {
            }
            function __ERC721Enumerable_init_unchained() internal onlyInitializing {
            }
            // Mapping from owner to list of owned token IDs
            mapping(address => mapping(uint256 => uint256)) private _ownedTokens;
            // Mapping from token ID to index of the owner tokens list
            mapping(uint256 => uint256) private _ownedTokensIndex;
            // Array with all token ids, used for enumeration
            uint256[] private _allTokens;
            // Mapping from token id to position in the allTokens array
            mapping(uint256 => uint256) private _allTokensIndex;
            /**
             * @dev See {IERC165-supportsInterface}.
             */
            function supportsInterface(bytes4 interfaceId) public view virtual override(IERC165Upgradeable, ERC721Upgradeable) returns (bool) {
                return interfaceId == type(IERC721EnumerableUpgradeable).interfaceId || super.supportsInterface(interfaceId);
            }
            /**
             * @dev See {IERC721Enumerable-tokenOfOwnerByIndex}.
             */
            function tokenOfOwnerByIndex(address owner, uint256 index) public view virtual override returns (uint256) {
                require(index < ERC721Upgradeable.balanceOf(owner), "ERC721Enumerable: owner index out of bounds");
                return _ownedTokens[owner][index];
            }
            /**
             * @dev See {IERC721Enumerable-totalSupply}.
             */
            function totalSupply() public view virtual override returns (uint256) {
                return _allTokens.length;
            }
            /**
             * @dev See {IERC721Enumerable-tokenByIndex}.
             */
            function tokenByIndex(uint256 index) public view virtual override returns (uint256) {
                require(index < ERC721EnumerableUpgradeable.totalSupply(), "ERC721Enumerable: global index out of bounds");
                return _allTokens[index];
            }
            /**
             * @dev Hook that is called before any token transfer. This includes minting
             * and burning.
             *
             * Calling conditions:
             *
             * - When `from` and `to` are both non-zero, ``from``'s `tokenId` will be
             * transferred to `to`.
             * - When `from` is zero, `tokenId` will be minted for `to`.
             * - When `to` is zero, ``from``'s `tokenId` will be burned.
             * - `from` cannot be the zero address.
             * - `to` cannot be the zero address.
             *
             * To learn more about hooks, head to xref:ROOT:extending-contracts.adoc#using-hooks[Using Hooks].
             */
            function _beforeTokenTransfer(
                address from,
                address to,
                uint256 tokenId
            ) internal virtual override {
                super._beforeTokenTransfer(from, to, tokenId);
                if (from == address(0)) {
                    _addTokenToAllTokensEnumeration(tokenId);
                } else if (from != to) {
                    _removeTokenFromOwnerEnumeration(from, tokenId);
                }
                if (to == address(0)) {
                    _removeTokenFromAllTokensEnumeration(tokenId);
                } else if (to != from) {
                    _addTokenToOwnerEnumeration(to, tokenId);
                }
            }
            /**
             * @dev Private function to add a token to this extension's ownership-tracking data structures.
             * @param to address representing the new owner of the given token ID
             * @param tokenId uint256 ID of the token to be added to the tokens list of the given address
             */
            function _addTokenToOwnerEnumeration(address to, uint256 tokenId) private {
                uint256 length = ERC721Upgradeable.balanceOf(to);
                _ownedTokens[to][length] = tokenId;
                _ownedTokensIndex[tokenId] = length;
            }
            /**
             * @dev Private function to add a token to this extension's token tracking data structures.
             * @param tokenId uint256 ID of the token to be added to the tokens list
             */
            function _addTokenToAllTokensEnumeration(uint256 tokenId) private {
                _allTokensIndex[tokenId] = _allTokens.length;
                _allTokens.push(tokenId);
            }
            /**
             * @dev Private function to remove a token from this extension's ownership-tracking data structures. Note that
             * while the token is not assigned a new owner, the `_ownedTokensIndex` mapping is _not_ updated: this allows for
             * gas optimizations e.g. when performing a transfer operation (avoiding double writes).
             * This has O(1) time complexity, but alters the order of the _ownedTokens array.
             * @param from address representing the previous owner of the given token ID
             * @param tokenId uint256 ID of the token to be removed from the tokens list of the given address
             */
            function _removeTokenFromOwnerEnumeration(address from, uint256 tokenId) private {
                // To prevent a gap in from's tokens array, we store the last token in the index of the token to delete, and
                // then delete the last slot (swap and pop).
                uint256 lastTokenIndex = ERC721Upgradeable.balanceOf(from) - 1;
                uint256 tokenIndex = _ownedTokensIndex[tokenId];
                // When the token to delete is the last token, the swap operation is unnecessary
                if (tokenIndex != lastTokenIndex) {
                    uint256 lastTokenId = _ownedTokens[from][lastTokenIndex];
                    _ownedTokens[from][tokenIndex] = lastTokenId; // Move the last token to the slot of the to-delete token
                    _ownedTokensIndex[lastTokenId] = tokenIndex; // Update the moved token's index
                }
                // This also deletes the contents at the last position of the array
                delete _ownedTokensIndex[tokenId];
                delete _ownedTokens[from][lastTokenIndex];
            }
            /**
             * @dev Private function to remove a token from this extension's token tracking data structures.
             * This has O(1) time complexity, but alters the order of the _allTokens array.
             * @param tokenId uint256 ID of the token to be removed from the tokens list
             */
            function _removeTokenFromAllTokensEnumeration(uint256 tokenId) private {
                // To prevent a gap in the tokens array, we store the last token in the index of the token to delete, and
                // then delete the last slot (swap and pop).
                uint256 lastTokenIndex = _allTokens.length - 1;
                uint256 tokenIndex = _allTokensIndex[tokenId];
                // When the token to delete is the last token, the swap operation is unnecessary. However, since this occurs so
                // rarely (when the last minted token is burnt) that we still do the swap here to avoid the gas cost of adding
                // an 'if' statement (like in _removeTokenFromOwnerEnumeration)
                uint256 lastTokenId = _allTokens[lastTokenIndex];
                _allTokens[tokenIndex] = lastTokenId; // Move the last token to the slot of the to-delete token
                _allTokensIndex[lastTokenId] = tokenIndex; // Update the moved token's index
                // This also deletes the contents at the last position of the array
                delete _allTokensIndex[tokenId];
                _allTokens.pop();
            }
            /**
             * @dev This empty reserved space is put in place to allow future versions to add new
             * variables without shifting down storage in the inheritance chain.
             * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps
             */
            uint256[46] private __gap;
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v4.5.0) (interfaces/IERC2981.sol)
        pragma solidity ^0.8.0;
        import "./IERC165Upgradeable.sol";
        /**
         * @dev Interface for the NFT Royalty Standard.
         *
         * A standardized way to retrieve royalty payment information for non-fungible tokens (NFTs) to enable universal
         * support for royalty payments across all NFT marketplaces and ecosystem participants.
         *
         * _Available since v4.5._
         */
        interface IERC2981Upgradeable is IERC165Upgradeable {
            /**
             * @dev Returns how much royalty is owed and to whom, based on a sale price that may be denominated in any unit of
             * exchange. The royalty amount is denominated and should be payed in that same unit of exchange.
             */
            function royaltyInfo(uint256 tokenId, uint256 salePrice)
                external
                view
                returns (address receiver, uint256 royaltyAmount);
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v4.5.0) (access/AccessControlEnumerable.sol)
        pragma solidity ^0.8.0;
        import "./IAccessControlEnumerableUpgradeable.sol";
        import "./AccessControlUpgradeable.sol";
        import "../utils/structs/EnumerableSetUpgradeable.sol";
        import "../proxy/utils/Initializable.sol";
        /**
         * @dev Extension of {AccessControl} that allows enumerating the members of each role.
         */
        abstract contract AccessControlEnumerableUpgradeable is Initializable, IAccessControlEnumerableUpgradeable, AccessControlUpgradeable {
            function __AccessControlEnumerable_init() internal onlyInitializing {
            }
            function __AccessControlEnumerable_init_unchained() internal onlyInitializing {
            }
            using EnumerableSetUpgradeable for EnumerableSetUpgradeable.AddressSet;
            mapping(bytes32 => EnumerableSetUpgradeable.AddressSet) private _roleMembers;
            /**
             * @dev See {IERC165-supportsInterface}.
             */
            function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) {
                return interfaceId == type(IAccessControlEnumerableUpgradeable).interfaceId || super.supportsInterface(interfaceId);
            }
            /**
             * @dev Returns one of the accounts that have `role`. `index` must be a
             * value between 0 and {getRoleMemberCount}, non-inclusive.
             *
             * Role bearers are not sorted in any particular way, and their ordering may
             * change at any point.
             *
             * WARNING: When using {getRoleMember} and {getRoleMemberCount}, make sure
             * you perform all queries on the same block. See the following
             * https://forum.openzeppelin.com/t/iterating-over-elements-on-enumerableset-in-openzeppelin-contracts/2296[forum post]
             * for more information.
             */
            function getRoleMember(bytes32 role, uint256 index) public view virtual override returns (address) {
                return _roleMembers[role].at(index);
            }
            /**
             * @dev Returns the number of accounts that have `role`. Can be used
             * together with {getRoleMember} to enumerate all bearers of a role.
             */
            function getRoleMemberCount(bytes32 role) public view virtual override returns (uint256) {
                return _roleMembers[role].length();
            }
            /**
             * @dev Overload {_grantRole} to track enumerable memberships
             */
            function _grantRole(bytes32 role, address account) internal virtual override {
                super._grantRole(role, account);
                _roleMembers[role].add(account);
            }
            /**
             * @dev Overload {_revokeRole} to track enumerable memberships
             */
            function _revokeRole(bytes32 role, address account) internal virtual override {
                super._revokeRole(role, account);
                _roleMembers[role].remove(account);
            }
            /**
             * @dev This empty reserved space is put in place to allow future versions to add new
             * variables without shifting down storage in the inheritance chain.
             * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps
             */
            uint256[49] private __gap;
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts v4.4.1 (security/ReentrancyGuard.sol)
        pragma solidity ^0.8.0;
        import "../proxy/utils/Initializable.sol";
        /**
         * @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 ReentrancyGuardUpgradeable is Initializable {
            // 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;
            function __ReentrancyGuard_init() internal onlyInitializing {
                __ReentrancyGuard_init_unchained();
            }
            function __ReentrancyGuard_init_unchained() internal onlyInitializing {
                _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;
            }
            /**
             * @dev This empty reserved space is put in place to allow future versions to add new
             * variables without shifting down storage in the inheritance chain.
             * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps
             */
            uint256[49] private __gap;
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts v4.4.1 (utils/structs/BitMaps.sol)
        pragma solidity ^0.8.0;
        /**
         * @dev Library for managing uint256 to bool mapping in a compact and efficient way, providing the keys are sequential.
         * Largelly inspired by Uniswap's https://github.com/Uniswap/merkle-distributor/blob/master/contracts/MerkleDistributor.sol[merkle-distributor].
         */
        library BitMapsUpgradeable {
            struct BitMap {
                mapping(uint256 => uint256) _data;
            }
            /**
             * @dev Returns whether the bit at `index` is set.
             */
            function get(BitMap storage bitmap, uint256 index) internal view returns (bool) {
                uint256 bucket = index >> 8;
                uint256 mask = 1 << (index & 0xff);
                return bitmap._data[bucket] & mask != 0;
            }
            /**
             * @dev Sets the bit at `index` to the boolean `value`.
             */
            function setTo(
                BitMap storage bitmap,
                uint256 index,
                bool value
            ) internal {
                if (value) {
                    set(bitmap, index);
                } else {
                    unset(bitmap, index);
                }
            }
            /**
             * @dev Sets the bit at `index`.
             */
            function set(BitMap storage bitmap, uint256 index) internal {
                uint256 bucket = index >> 8;
                uint256 mask = 1 << (index & 0xff);
                bitmap._data[bucket] |= mask;
            }
            /**
             * @dev Unsets the bit at `index`.
             */
            function unset(BitMap storage bitmap, uint256 index) internal {
                uint256 bucket = index >> 8;
                uint256 mask = 1 << (index & 0xff);
                bitmap._data[bucket] &= ~mask;
            }
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v4.5.0) (utils/Multicall.sol)
        pragma solidity ^0.8.0;
        import "./AddressUpgradeable.sol";
        import "../proxy/utils/Initializable.sol";
        /**
         * @dev Provides a function to batch together multiple calls in a single external call.
         *
         * _Available since v4.1._
         */
        abstract contract MulticallUpgradeable is Initializable {
            function __Multicall_init() internal onlyInitializing {
            }
            function __Multicall_init_unchained() internal onlyInitializing {
            }
            /**
             * @dev Receives and executes a batch of function calls on this contract.
             */
            function multicall(bytes[] calldata data) external virtual returns (bytes[] memory results) {
                results = new bytes[](data.length);
                for (uint256 i = 0; i < data.length; i++) {
                    results[i] = _functionDelegateCall(address(this), data[i]);
                }
                return results;
            }
            /**
             * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],
             * but performing a delegate call.
             *
             * _Available since v3.4._
             */
            function _functionDelegateCall(address target, bytes memory data) private returns (bytes memory) {
                require(AddressUpgradeable.isContract(target), "Address: delegate call to non-contract");
                // solhint-disable-next-line avoid-low-level-calls
                (bool success, bytes memory returndata) = target.delegatecall(data);
                return AddressUpgradeable.verifyCallResult(success, returndata, "Address: low-level delegate call failed");
            }
            /**
             * @dev This empty reserved space is put in place to allow future versions to add new
             * variables without shifting down storage in the inheritance chain.
             * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps
             */
            uint256[50] private __gap;
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts v4.4.1 (utils/Strings.sol)
        pragma solidity ^0.8.0;
        /**
         * @dev String operations.
         */
        library StringsUpgradeable {
            bytes16 private constant _HEX_SYMBOLS = "0123456789abcdef";
            /**
             * @dev Converts a `uint256` to its ASCII `string` decimal representation.
             */
            function toString(uint256 value) internal pure returns (string memory) {
                // Inspired by OraclizeAPI's implementation - MIT licence
                // https://github.com/oraclize/ethereum-api/blob/b42146b063c7d6ee1358846c198246239e9360e8/oraclizeAPI_0.4.25.sol
                if (value == 0) {
                    return "0";
                }
                uint256 temp = value;
                uint256 digits;
                while (temp != 0) {
                    digits++;
                    temp /= 10;
                }
                bytes memory buffer = new bytes(digits);
                while (value != 0) {
                    digits -= 1;
                    buffer[digits] = bytes1(uint8(48 + uint256(value % 10)));
                    value /= 10;
                }
                return string(buffer);
            }
            /**
             * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation.
             */
            function toHexString(uint256 value) internal pure returns (string memory) {
                if (value == 0) {
                    return "0x00";
                }
                uint256 temp = value;
                uint256 length = 0;
                while (temp != 0) {
                    length++;
                    temp >>= 8;
                }
                return toHexString(value, length);
            }
            /**
             * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation with fixed length.
             */
            function toHexString(uint256 value, uint256 length) internal pure returns (string memory) {
                bytes memory buffer = new bytes(2 * length + 2);
                buffer[0] = "0";
                buffer[1] = "x";
                for (uint256 i = 2 * length + 1; i > 1; --i) {
                    buffer[i] = _HEX_SYMBOLS[value & 0xf];
                    value >>= 4;
                }
                require(value == 0, "Strings: hex length insufficient");
                return string(buffer);
            }
        }
        // SPDX-License-Identifier: Apache-2.0
        pragma solidity ^0.8.11;
        import "@openzeppelin/contracts-upgradeable/token/ERC721/IERC721Upgradeable.sol";
        import "../IThirdwebContract.sol";
        import "../IThirdwebPlatformFee.sol";
        import "../IThirdwebPrimarySale.sol";
        import "../IThirdwebRoyalty.sol";
        import "../IThirdwebOwnable.sol";
        import "./IDropClaimCondition.sol";
        /**
         *  Thirdweb's 'Drop' contracts are distribution mechanisms for tokens. The
         *  `DropERC721` contract is a distribution mechanism for ERC721 tokens.
         *
         *  A minter wallet (i.e. holder of `MINTER_ROLE`) can (lazy)mint 'n' tokens
         *  at once by providing a single base URI for all tokens being lazy minted.
         *  The URI for each of the 'n' tokens lazy minted is the provided base URI +
         *  `{tokenId}` of the respective token. (e.g. "ipsf://Qmece.../1").
         *
         *  A minter can choose to lazy mint 'delayed-reveal' tokens. More on 'delayed-reveal'
         *  tokens in [this article](https://blog.thirdweb.com/delayed-reveal-nfts).
         *
         *  A contract admin (i.e. holder of `DEFAULT_ADMIN_ROLE`) can create claim conditions
         *  with non-overlapping time windows, and accounts can claim the tokens according to
         *  restrictions defined in the claim condition that is active at the time of the transaction.
         */
        interface IDropERC721 is
            IThirdwebContract,
            IThirdwebOwnable,
            IThirdwebRoyalty,
            IThirdwebPrimarySale,
            IThirdwebPlatformFee,
            IERC721Upgradeable,
            IDropClaimCondition
        {
            /// @dev Emitted when tokens are claimed.
            event TokensClaimed(
                uint256 indexed claimConditionIndex,
                address indexed claimer,
                address indexed receiver,
                uint256 startTokenId,
                uint256 quantityClaimed
            );
            /// @dev Emitted when tokens are lazy minted.
            event TokensLazyMinted(uint256 startTokenId, uint256 endTokenId, string baseURI, bytes encryptedBaseURI);
            /// @dev Emitted when the URI for a batch of 'delayed-reveal' NFTs is revealed.
            event NFTRevealed(uint256 endTokenId, string revealedURI);
            /// @dev Emitted when new claim conditions are set.
            event ClaimConditionsUpdated(ClaimCondition[] claimConditions);
            /// @dev Emitted when a new primary sale recipient is set.
            event PrimarySaleRecipientUpdated(address indexed recipient);
            /// @dev Emitted when fee platform fee recipient or bps is updated.
            event PlatformFeeInfoUpdated(address platformFeeRecipient, uint256 platformFeeBps);
            /// @dev Emitted when a new owner is set.
            event OwnerUpdated(address prevOwner, address newOwner);
            /// @dev Emitted when the global max supply of tokens is updated.
            event MaxTotalSupplyUpdated(uint256 maxTotalSupply);
            /// @dev Emitted when the wallet claim count for an address is updated.
            event WalletClaimCountUpdated(address indexed wallet, uint256 count);
            /// @dev Emitted when the global max wallet claim count is updated.
            event MaxWalletClaimCountUpdated(uint256 count);
            /**
             *  @notice Lets an account with `MINTER_ROLE` lazy mint 'n' NFTs.
             *          The URIs for each token is the provided `_baseURIForTokens` + `{tokenId}`.
             *
             *  @param amount           The amount of NFTs to lazy mint.
             *  @param baseURIForTokens The URI for the NFTs to lazy mint. If lazy minting
             *                           'delayed-reveal' NFTs, the is a URI for NFTs in the
             *                           un-revealed state.
             *  @param encryptedBaseURI If lazy minting 'delayed-reveal' NFTs, this is the
             *                           result of encrypting the URI of the NFTs in the revealed
             *                           state.
             */
            function lazyMint(
                uint256 amount,
                string calldata baseURIForTokens,
                bytes calldata encryptedBaseURI
            ) external;
            /**
             *  @notice Lets an account claim a given quantity of NFTs.
             *
             *  @param receiver                       The receiver of the NFTs to claim.
             *  @param quantity                       The quantity of NFTs to claim.
             *  @param currency                       The currency in which to pay for the claim.
             *  @param pricePerToken                  The price per token to pay for the claim.
             *  @param proofs                         The proof of the claimer's inclusion in the merkle root allowlist
             *                                        of the claim conditions that apply.
             *  @param proofMaxQuantityPerTransaction (Optional) The maximum number of NFTs an address included in an
             *                                        allowlist can claim.
             */
            function claim(
                address receiver,
                uint256 quantity,
                address currency,
                uint256 pricePerToken,
                bytes32[] calldata proofs,
                uint256 proofMaxQuantityPerTransaction
            ) external payable;
            /**
             *  @notice Lets a contract admin (account with `DEFAULT_ADMIN_ROLE`) set claim conditions.
             *
             *  @param phases                Claim conditions in ascending order by `startTimestamp`.
             *  @param resetClaimEligibility Whether to reset `limitLastClaimTimestamp` and
             *                               `limitMerkleProofClaim` values when setting new
             *                               claim conditions.
             */
            function setClaimConditions(ClaimCondition[] calldata phases, bool resetClaimEligibility) external;
        }
        // SPDX-License-Identifier: Apache-2.0
        pragma solidity ^0.8.11;
        interface ITWFee {
            function getFeeInfo(address _proxy, uint256 _type) external view returns (address recipient, uint256 bps);
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts v4.4.0 (metatx/ERC2771Context.sol)
        pragma solidity ^0.8.11;
        import "@openzeppelin/contracts-upgradeable/utils/ContextUpgradeable.sol";
        import "@openzeppelin/contracts-upgradeable/proxy/utils/Initializable.sol";
        /**
         * @dev Context variant with ERC2771 support.
         */
        abstract contract ERC2771ContextUpgradeable is Initializable, ContextUpgradeable {
            mapping(address => bool) private _trustedForwarder;
            function __ERC2771Context_init(address[] memory trustedForwarder) internal onlyInitializing {
                __Context_init_unchained();
                __ERC2771Context_init_unchained(trustedForwarder);
            }
            function __ERC2771Context_init_unchained(address[] memory trustedForwarder) internal onlyInitializing {
                for (uint256 i = 0; i < trustedForwarder.length; i++) {
                    _trustedForwarder[trustedForwarder[i]] = true;
                }
            }
            function isTrustedForwarder(address forwarder) public view virtual returns (bool) {
                return _trustedForwarder[forwarder];
            }
            function _msgSender() internal view virtual override returns (address sender) {
                if (isTrustedForwarder(msg.sender)) {
                    // The assembly code is more direct than the Solidity version using `abi.decode`.
                    assembly {
                        sender := shr(96, calldataload(sub(calldatasize(), 20)))
                    }
                } else {
                    return super._msgSender();
                }
            }
            function _msgData() internal view virtual override returns (bytes calldata) {
                if (isTrustedForwarder(msg.sender)) {
                    return msg.data[:msg.data.length - 20];
                } else {
                    return super._msgData();
                }
            }
            uint256[49] private __gap;
        }
        // SPDX-License-Identifier: Apache-2.0
        pragma solidity ^0.8.11;
        // Helper interfaces
        import { IWETH } from "../interfaces/IWETH.sol";
        import "@openzeppelin/contracts-upgradeable/token/ERC20/IERC20Upgradeable.sol";
        library CurrencyTransferLib {
            /// @dev The address interpreted as native token of the chain.
            address public constant NATIVE_TOKEN = 0xEeeeeEeeeEeEeeEeEeEeeEEEeeeeEeeeeeeeEEeE;
            /// @dev Transfers a given amount of currency.
            function transferCurrency(
                address _currency,
                address _from,
                address _to,
                uint256 _amount
            ) internal {
                if (_amount == 0) {
                    return;
                }
                if (_currency == NATIVE_TOKEN) {
                    safeTransferNativeToken(_to, _amount);
                } else {
                    safeTransferERC20(_currency, _from, _to, _amount);
                }
            }
            /// @dev Transfers a given amount of currency. (With native token wrapping)
            function transferCurrencyWithWrapperAndBalanceCheck(
                address _currency,
                address _from,
                address _to,
                uint256 _amount,
                address _nativeTokenWrapper
            ) internal {
                if (_amount == 0) {
                    return;
                }
                if (_currency == NATIVE_TOKEN) {
                    if (_from == address(this)) {
                        // withdraw from weth then transfer withdrawn native token to recipient
                        IWETH(_nativeTokenWrapper).withdraw(_amount);
                        safeTransferNativeTokenWithWrapper(_to, _amount, _nativeTokenWrapper);
                    } else if (_to == address(this)) {
                        // store native currency in weth
                        require(_amount == msg.value, "msg.value != amount");
                        IWETH(_nativeTokenWrapper).deposit{ value: _amount }();
                    } else {
                        safeTransferNativeTokenWithWrapper(_to, _amount, _nativeTokenWrapper);
                    }
                } else {
                    safeTransferERC20WithBalanceCheck(_currency, _from, _to, _amount);
                }
            }
            /// @dev Transfer `amount` of ERC20 token from `from` to `to`.
            function safeTransferERC20(
                address _currency,
                address _from,
                address _to,
                uint256 _amount
            ) internal {
                if (_from == _to) {
                    return;
                }
                bool success = _from == address(this)
                    ? IERC20Upgradeable(_currency).transfer(_to, _amount)
                    : IERC20Upgradeable(_currency).transferFrom(_from, _to, _amount);
                require(success, "currency transfer failed.");
            }
            /// @dev Transfer `amount` of ERC20 token from `from` to `to`.
            function safeTransferERC20WithBalanceCheck(
                address _currency,
                address _from,
                address _to,
                uint256 _amount
            ) internal {
                if (_from == _to) {
                    return;
                }
                uint256 balBefore = IERC20Upgradeable(_currency).balanceOf(_to);
                bool success = _from == address(this)
                    ? IERC20Upgradeable(_currency).transfer(_to, _amount)
                    : IERC20Upgradeable(_currency).transferFrom(_from, _to, _amount);
                uint256 balAfter = IERC20Upgradeable(_currency).balanceOf(_to);
                require(success && (balAfter == balBefore + _amount), "currency transfer failed.");
            }
            /// @dev Transfers `amount` of native token to `to`.
            function safeTransferNativeToken(address to, uint256 value) internal {
                // solhint-disable avoid-low-level-calls
                // slither-disable-next-line low-level-calls
                (bool success, ) = to.call{ value: value }("");
                require(success, "native token transfer failed");
            }
            /// @dev Transfers `amount` of native token to `to`. (With native token wrapping)
            function safeTransferNativeTokenWithWrapper(
                address to,
                uint256 value,
                address _nativeTokenWrapper
            ) internal {
                // solhint-disable avoid-low-level-calls
                // slither-disable-next-line low-level-calls
                (bool success, ) = to.call{ value: value }("");
                if (!success) {
                    IWETH(_nativeTokenWrapper).deposit{ value: value }();
                    require(IERC20Upgradeable(_nativeTokenWrapper).transfer(to, value), "transfer failed");
                }
            }
        }
        // SPDX-License-Identifier: Apache-2.0
        pragma solidity ^0.8.11;
        library FeeType {
            uint256 internal constant PRIMARY_SALE = 0;
            uint256 internal constant MARKET_SALE = 1;
            uint256 internal constant SPLIT = 2;
        }
        // SPDX-License-Identifier: MIT
        // Modified from https://github.com/OpenZeppelin/openzeppelin-contracts/blob/v4.3.0/contracts/utils/cryptography/MerkleProof.sol
        // Copied from https://github.com/ensdomains/governance/blob/master/contracts/MerkleProof.sol
        pragma solidity ^0.8.11;
        /**
         * @dev These functions deal with verification of Merkle Trees proofs.
         *
         * The proofs can be generated using the JavaScript library
         * https://github.com/miguelmota/merkletreejs[merkletreejs].
         * Note: the hashing algorithm should be keccak256 and pair sorting should be enabled.
         *
         * See `test/utils/cryptography/MerkleProof.test.js` for some examples.
         *
         * Source: https://github.com/ensdomains/governance/blob/master/contracts/MerkleProof.sol
         */
        library MerkleProof {
            /**
             * @dev Returns true if a `leaf` can be proved to be a part of a Merkle tree
             * defined by `root`. For this, a `proof` must be provided, containing
             * sibling hashes on the branch from the leaf to the root of the tree. Each
             * pair of leaves and each pair of pre-images are assumed to be sorted.
             */
            function verify(
                bytes32[] memory proof,
                bytes32 root,
                bytes32 leaf
            ) internal pure returns (bool, uint256) {
                bytes32 computedHash = leaf;
                uint256 index = 0;
                for (uint256 i = 0; i < proof.length; i++) {
                    index *= 2;
                    bytes32 proofElement = proof[i];
                    if (computedHash <= proofElement) {
                        // Hash(current computed hash + current element of the proof)
                        computedHash = keccak256(abi.encodePacked(computedHash, proofElement));
                    } else {
                        // Hash(current element of the proof + current computed hash)
                        computedHash = keccak256(abi.encodePacked(proofElement, computedHash));
                        index += 1;
                    }
                }
                // Check if the computed hash (root) is equal to the provided root
                return (computedHash == root, index);
            }
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v4.5.0) (token/ERC721/ERC721.sol)
        pragma solidity ^0.8.0;
        import "./IERC721Upgradeable.sol";
        import "./IERC721ReceiverUpgradeable.sol";
        import "./extensions/IERC721MetadataUpgradeable.sol";
        import "../../utils/AddressUpgradeable.sol";
        import "../../utils/ContextUpgradeable.sol";
        import "../../utils/StringsUpgradeable.sol";
        import "../../utils/introspection/ERC165Upgradeable.sol";
        import "../../proxy/utils/Initializable.sol";
        /**
         * @dev Implementation of https://eips.ethereum.org/EIPS/eip-721[ERC721] Non-Fungible Token Standard, including
         * the Metadata extension, but not including the Enumerable extension, which is available separately as
         * {ERC721Enumerable}.
         */
        contract ERC721Upgradeable is Initializable, ContextUpgradeable, ERC165Upgradeable, IERC721Upgradeable, IERC721MetadataUpgradeable {
            using AddressUpgradeable for address;
            using StringsUpgradeable for uint256;
            // Token name
            string private _name;
            // Token symbol
            string private _symbol;
            // Mapping from token ID to owner address
            mapping(uint256 => address) private _owners;
            // Mapping owner address to token count
            mapping(address => uint256) private _balances;
            // Mapping from token ID to approved address
            mapping(uint256 => address) private _tokenApprovals;
            // Mapping from owner to operator approvals
            mapping(address => mapping(address => bool)) private _operatorApprovals;
            /**
             * @dev Initializes the contract by setting a `name` and a `symbol` to the token collection.
             */
            function __ERC721_init(string memory name_, string memory symbol_) internal onlyInitializing {
                __ERC721_init_unchained(name_, symbol_);
            }
            function __ERC721_init_unchained(string memory name_, string memory symbol_) internal onlyInitializing {
                _name = name_;
                _symbol = symbol_;
            }
            /**
             * @dev See {IERC165-supportsInterface}.
             */
            function supportsInterface(bytes4 interfaceId) public view virtual override(ERC165Upgradeable, IERC165Upgradeable) returns (bool) {
                return
                    interfaceId == type(IERC721Upgradeable).interfaceId ||
                    interfaceId == type(IERC721MetadataUpgradeable).interfaceId ||
                    super.supportsInterface(interfaceId);
            }
            /**
             * @dev See {IERC721-balanceOf}.
             */
            function balanceOf(address owner) public view virtual override returns (uint256) {
                require(owner != address(0), "ERC721: balance query for the zero address");
                return _balances[owner];
            }
            /**
             * @dev See {IERC721-ownerOf}.
             */
            function ownerOf(uint256 tokenId) public view virtual override returns (address) {
                address owner = _owners[tokenId];
                require(owner != address(0), "ERC721: owner query for nonexistent token");
                return owner;
            }
            /**
             * @dev See {IERC721Metadata-name}.
             */
            function name() public view virtual override returns (string memory) {
                return _name;
            }
            /**
             * @dev See {IERC721Metadata-symbol}.
             */
            function symbol() public view virtual override returns (string memory) {
                return _symbol;
            }
            /**
             * @dev See {IERC721Metadata-tokenURI}.
             */
            function tokenURI(uint256 tokenId) public view virtual override returns (string memory) {
                require(_exists(tokenId), "ERC721Metadata: URI query for nonexistent token");
                string memory baseURI = _baseURI();
                return bytes(baseURI).length > 0 ? string(abi.encodePacked(baseURI, tokenId.toString())) : "";
            }
            /**
             * @dev Base URI for computing {tokenURI}. If set, the resulting URI for each
             * token will be the concatenation of the `baseURI` and the `tokenId`. Empty
             * by default, can be overriden in child contracts.
             */
            function _baseURI() internal view virtual returns (string memory) {
                return "";
            }
            /**
             * @dev See {IERC721-approve}.
             */
            function approve(address to, uint256 tokenId) public virtual override {
                address owner = ERC721Upgradeable.ownerOf(tokenId);
                require(to != owner, "ERC721: approval to current owner");
                require(
                    _msgSender() == owner || isApprovedForAll(owner, _msgSender()),
                    "ERC721: approve caller is not owner nor approved for all"
                );
                _approve(to, tokenId);
            }
            /**
             * @dev See {IERC721-getApproved}.
             */
            function getApproved(uint256 tokenId) public view virtual override returns (address) {
                require(_exists(tokenId), "ERC721: approved query for nonexistent token");
                return _tokenApprovals[tokenId];
            }
            /**
             * @dev See {IERC721-setApprovalForAll}.
             */
            function setApprovalForAll(address operator, bool approved) public virtual override {
                _setApprovalForAll(_msgSender(), operator, approved);
            }
            /**
             * @dev See {IERC721-isApprovedForAll}.
             */
            function isApprovedForAll(address owner, address operator) public view virtual override returns (bool) {
                return _operatorApprovals[owner][operator];
            }
            /**
             * @dev See {IERC721-transferFrom}.
             */
            function transferFrom(
                address from,
                address to,
                uint256 tokenId
            ) public virtual override {
                //solhint-disable-next-line max-line-length
                require(_isApprovedOrOwner(_msgSender(), tokenId), "ERC721: transfer caller is not owner nor approved");
                _transfer(from, to, tokenId);
            }
            /**
             * @dev See {IERC721-safeTransferFrom}.
             */
            function safeTransferFrom(
                address from,
                address to,
                uint256 tokenId
            ) public virtual override {
                safeTransferFrom(from, to, tokenId, "");
            }
            /**
             * @dev See {IERC721-safeTransferFrom}.
             */
            function safeTransferFrom(
                address from,
                address to,
                uint256 tokenId,
                bytes memory _data
            ) public virtual override {
                require(_isApprovedOrOwner(_msgSender(), tokenId), "ERC721: transfer caller is not owner nor approved");
                _safeTransfer(from, to, tokenId, _data);
            }
            /**
             * @dev Safely transfers `tokenId` token from `from` to `to`, checking first that contract recipients
             * are aware of the ERC721 protocol to prevent tokens from being forever locked.
             *
             * `_data` is additional data, it has no specified format and it is sent in call to `to`.
             *
             * This internal function is equivalent to {safeTransferFrom}, and can be used to e.g.
             * implement alternative mechanisms to perform token transfer, such as signature-based.
             *
             * Requirements:
             *
             * - `from` cannot be the zero address.
             * - `to` cannot be the zero address.
             * - `tokenId` token must exist and be owned by `from`.
             * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer.
             *
             * Emits a {Transfer} event.
             */
            function _safeTransfer(
                address from,
                address to,
                uint256 tokenId,
                bytes memory _data
            ) internal virtual {
                _transfer(from, to, tokenId);
                require(_checkOnERC721Received(from, to, tokenId, _data), "ERC721: transfer to non ERC721Receiver implementer");
            }
            /**
             * @dev Returns whether `tokenId` exists.
             *
             * Tokens can be managed by their owner or approved accounts via {approve} or {setApprovalForAll}.
             *
             * Tokens start existing when they are minted (`_mint`),
             * and stop existing when they are burned (`_burn`).
             */
            function _exists(uint256 tokenId) internal view virtual returns (bool) {
                return _owners[tokenId] != address(0);
            }
            /**
             * @dev Returns whether `spender` is allowed to manage `tokenId`.
             *
             * Requirements:
             *
             * - `tokenId` must exist.
             */
            function _isApprovedOrOwner(address spender, uint256 tokenId) internal view virtual returns (bool) {
                require(_exists(tokenId), "ERC721: operator query for nonexistent token");
                address owner = ERC721Upgradeable.ownerOf(tokenId);
                return (spender == owner || getApproved(tokenId) == spender || isApprovedForAll(owner, spender));
            }
            /**
             * @dev Safely mints `tokenId` and transfers it to `to`.
             *
             * Requirements:
             *
             * - `tokenId` must not exist.
             * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer.
             *
             * Emits a {Transfer} event.
             */
            function _safeMint(address to, uint256 tokenId) internal virtual {
                _safeMint(to, tokenId, "");
            }
            /**
             * @dev Same as {xref-ERC721-_safeMint-address-uint256-}[`_safeMint`], with an additional `data` parameter which is
             * forwarded in {IERC721Receiver-onERC721Received} to contract recipients.
             */
            function _safeMint(
                address to,
                uint256 tokenId,
                bytes memory _data
            ) internal virtual {
                _mint(to, tokenId);
                require(
                    _checkOnERC721Received(address(0), to, tokenId, _data),
                    "ERC721: transfer to non ERC721Receiver implementer"
                );
            }
            /**
             * @dev Mints `tokenId` and transfers it to `to`.
             *
             * WARNING: Usage of this method is discouraged, use {_safeMint} whenever possible
             *
             * Requirements:
             *
             * - `tokenId` must not exist.
             * - `to` cannot be the zero address.
             *
             * Emits a {Transfer} event.
             */
            function _mint(address to, uint256 tokenId) internal virtual {
                require(to != address(0), "ERC721: mint to the zero address");
                require(!_exists(tokenId), "ERC721: token already minted");
                _beforeTokenTransfer(address(0), to, tokenId);
                _balances[to] += 1;
                _owners[tokenId] = to;
                emit Transfer(address(0), to, tokenId);
                _afterTokenTransfer(address(0), to, tokenId);
            }
            /**
             * @dev Destroys `tokenId`.
             * The approval is cleared when the token is burned.
             *
             * Requirements:
             *
             * - `tokenId` must exist.
             *
             * Emits a {Transfer} event.
             */
            function _burn(uint256 tokenId) internal virtual {
                address owner = ERC721Upgradeable.ownerOf(tokenId);
                _beforeTokenTransfer(owner, address(0), tokenId);
                // Clear approvals
                _approve(address(0), tokenId);
                _balances[owner] -= 1;
                delete _owners[tokenId];
                emit Transfer(owner, address(0), tokenId);
                _afterTokenTransfer(owner, address(0), tokenId);
            }
            /**
             * @dev Transfers `tokenId` from `from` to `to`.
             *  As opposed to {transferFrom}, this imposes no restrictions on msg.sender.
             *
             * Requirements:
             *
             * - `to` cannot be the zero address.
             * - `tokenId` token must be owned by `from`.
             *
             * Emits a {Transfer} event.
             */
            function _transfer(
                address from,
                address to,
                uint256 tokenId
            ) internal virtual {
                require(ERC721Upgradeable.ownerOf(tokenId) == from, "ERC721: transfer from incorrect owner");
                require(to != address(0), "ERC721: transfer to the zero address");
                _beforeTokenTransfer(from, to, tokenId);
                // Clear approvals from the previous owner
                _approve(address(0), tokenId);
                _balances[from] -= 1;
                _balances[to] += 1;
                _owners[tokenId] = to;
                emit Transfer(from, to, tokenId);
                _afterTokenTransfer(from, to, tokenId);
            }
            /**
             * @dev Approve `to` to operate on `tokenId`
             *
             * Emits a {Approval} event.
             */
            function _approve(address to, uint256 tokenId) internal virtual {
                _tokenApprovals[tokenId] = to;
                emit Approval(ERC721Upgradeable.ownerOf(tokenId), to, tokenId);
            }
            /**
             * @dev Approve `operator` to operate on all of `owner` tokens
             *
             * Emits a {ApprovalForAll} event.
             */
            function _setApprovalForAll(
                address owner,
                address operator,
                bool approved
            ) internal virtual {
                require(owner != operator, "ERC721: approve to caller");
                _operatorApprovals[owner][operator] = approved;
                emit ApprovalForAll(owner, operator, approved);
            }
            /**
             * @dev Internal function to invoke {IERC721Receiver-onERC721Received} on a target address.
             * The call is not executed if the target address is not a contract.
             *
             * @param from address representing the previous owner of the given token ID
             * @param to target address that will receive the tokens
             * @param tokenId uint256 ID of the token to be transferred
             * @param _data bytes optional data to send along with the call
             * @return bool whether the call correctly returned the expected magic value
             */
            function _checkOnERC721Received(
                address from,
                address to,
                uint256 tokenId,
                bytes memory _data
            ) private returns (bool) {
                if (to.isContract()) {
                    try IERC721ReceiverUpgradeable(to).onERC721Received(_msgSender(), from, tokenId, _data) returns (bytes4 retval) {
                        return retval == IERC721ReceiverUpgradeable.onERC721Received.selector;
                    } catch (bytes memory reason) {
                        if (reason.length == 0) {
                            revert("ERC721: transfer to non ERC721Receiver implementer");
                        } else {
                            assembly {
                                revert(add(32, reason), mload(reason))
                            }
                        }
                    }
                } else {
                    return true;
                }
            }
            /**
             * @dev Hook that is called before any token transfer. This includes minting
             * and burning.
             *
             * Calling conditions:
             *
             * - When `from` and `to` are both non-zero, ``from``'s `tokenId` will be
             * transferred to `to`.
             * - When `from` is zero, `tokenId` will be minted for `to`.
             * - When `to` is zero, ``from``'s `tokenId` will be burned.
             * - `from` and `to` are never both zero.
             *
             * To learn more about hooks, head to xref:ROOT:extending-contracts.adoc#using-hooks[Using Hooks].
             */
            function _beforeTokenTransfer(
                address from,
                address to,
                uint256 tokenId
            ) internal virtual {}
            /**
             * @dev Hook that is called after any transfer of tokens. This includes
             * minting and burning.
             *
             * Calling conditions:
             *
             * - when `from` and `to` are both non-zero.
             * - `from` and `to` are never both zero.
             *
             * To learn more about hooks, head to xref:ROOT:extending-contracts.adoc#using-hooks[Using Hooks].
             */
            function _afterTokenTransfer(
                address from,
                address to,
                uint256 tokenId
            ) internal virtual {}
            /**
             * @dev This empty reserved space is put in place to allow future versions to add new
             * variables without shifting down storage in the inheritance chain.
             * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps
             */
            uint256[44] private __gap;
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v4.5.0) (token/ERC721/extensions/IERC721Enumerable.sol)
        pragma solidity ^0.8.0;
        import "../IERC721Upgradeable.sol";
        /**
         * @title ERC-721 Non-Fungible Token Standard, optional enumeration extension
         * @dev See https://eips.ethereum.org/EIPS/eip-721
         */
        interface IERC721EnumerableUpgradeable is IERC721Upgradeable {
            /**
             * @dev Returns the total amount of tokens stored by the contract.
             */
            function totalSupply() external view returns (uint256);
            /**
             * @dev Returns a token ID owned by `owner` at a given `index` of its token list.
             * Use along with {balanceOf} to enumerate all of ``owner``'s tokens.
             */
            function tokenOfOwnerByIndex(address owner, uint256 index) external view returns (uint256);
            /**
             * @dev Returns a token ID at a given `index` of all the tokens stored by the contract.
             * Use along with {totalSupply} to enumerate all tokens.
             */
            function tokenByIndex(uint256 index) external view returns (uint256);
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v4.5.0) (proxy/utils/Initializable.sol)
        pragma solidity ^0.8.0;
        import "../../utils/AddressUpgradeable.sol";
        /**
         * @dev This is a base contract to aid in writing upgradeable contracts, or any kind of contract that will be deployed
         * behind a proxy. Since proxied contracts do not make use of a constructor, it's common to move constructor logic to an
         * external initializer function, usually called `initialize`. It then becomes necessary to protect this initializer
         * function so it can only be called once. The {initializer} modifier provided by this contract will have this effect.
         *
         * TIP: To avoid leaving the proxy in an uninitialized state, the initializer function should be called as early as
         * possible by providing the encoded function call as the `_data` argument to {ERC1967Proxy-constructor}.
         *
         * CAUTION: When used with inheritance, manual care must be taken to not invoke a parent initializer twice, or to ensure
         * that all initializers are idempotent. This is not verified automatically as constructors are by Solidity.
         *
         * [CAUTION]
         * ====
         * Avoid leaving a contract uninitialized.
         *
         * An uninitialized contract can be taken over by an attacker. This applies to both a proxy and its implementation
         * contract, which may impact the proxy. To initialize the implementation contract, you can either invoke the
         * initializer manually, or you can include a constructor to automatically mark it as initialized when it is deployed:
         *
         * [.hljs-theme-light.nopadding]
         * ```
         * /// @custom:oz-upgrades-unsafe-allow constructor
         * constructor() initializer {}
         * ```
         * ====
         */
        abstract contract Initializable {
            /**
             * @dev Indicates that the contract has been initialized.
             */
            bool private _initialized;
            /**
             * @dev Indicates that the contract is in the process of being initialized.
             */
            bool private _initializing;
            /**
             * @dev Modifier to protect an initializer function from being invoked twice.
             */
            modifier initializer() {
                // If the contract is initializing we ignore whether _initialized is set in order to support multiple
                // inheritance patterns, but we only do this in the context of a constructor, because in other contexts the
                // contract may have been reentered.
                require(_initializing ? _isConstructor() : !_initialized, "Initializable: contract is already initialized");
                bool isTopLevelCall = !_initializing;
                if (isTopLevelCall) {
                    _initializing = true;
                    _initialized = true;
                }
                _;
                if (isTopLevelCall) {
                    _initializing = false;
                }
            }
            /**
             * @dev Modifier to protect an initialization function so that it can only be invoked by functions with the
             * {initializer} modifier, directly or indirectly.
             */
            modifier onlyInitializing() {
                require(_initializing, "Initializable: contract is not initializing");
                _;
            }
            function _isConstructor() private view returns (bool) {
                return !AddressUpgradeable.isContract(address(this));
            }
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts v4.4.1 (token/ERC721/IERC721.sol)
        pragma solidity ^0.8.0;
        import "../../utils/introspection/IERC165Upgradeable.sol";
        /**
         * @dev Required interface of an ERC721 compliant contract.
         */
        interface IERC721Upgradeable is IERC165Upgradeable {
            /**
             * @dev Emitted when `tokenId` token is transferred from `from` to `to`.
             */
            event Transfer(address indexed from, address indexed to, uint256 indexed tokenId);
            /**
             * @dev Emitted when `owner` enables `approved` to manage the `tokenId` token.
             */
            event Approval(address indexed owner, address indexed approved, uint256 indexed tokenId);
            /**
             * @dev Emitted when `owner` enables or disables (`approved`) `operator` to manage all of its assets.
             */
            event ApprovalForAll(address indexed owner, address indexed operator, bool approved);
            /**
             * @dev Returns the number of tokens in ``owner``'s account.
             */
            function balanceOf(address owner) external view returns (uint256 balance);
            /**
             * @dev Returns the owner of the `tokenId` token.
             *
             * Requirements:
             *
             * - `tokenId` must exist.
             */
            function ownerOf(uint256 tokenId) external view returns (address owner);
            /**
             * @dev Safely transfers `tokenId` token from `from` to `to`, checking first that contract recipients
             * are aware of the ERC721 protocol to prevent tokens from being forever locked.
             *
             * Requirements:
             *
             * - `from` cannot be the zero address.
             * - `to` cannot be the zero address.
             * - `tokenId` token must exist and be owned by `from`.
             * - If the caller is not `from`, it must be have been allowed to move this token by either {approve} or {setApprovalForAll}.
             * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer.
             *
             * Emits a {Transfer} event.
             */
            function safeTransferFrom(
                address from,
                address to,
                uint256 tokenId
            ) external;
            /**
             * @dev Transfers `tokenId` token from `from` to `to`.
             *
             * WARNING: Usage of this method is discouraged, use {safeTransferFrom} whenever possible.
             *
             * Requirements:
             *
             * - `from` cannot be the zero address.
             * - `to` cannot be the zero address.
             * - `tokenId` token must be owned by `from`.
             * - If the caller is not `from`, it must be approved to move this token by either {approve} or {setApprovalForAll}.
             *
             * Emits a {Transfer} event.
             */
            function transferFrom(
                address from,
                address to,
                uint256 tokenId
            ) external;
            /**
             * @dev Gives permission to `to` to transfer `tokenId` token to another account.
             * The approval is cleared when the token is transferred.
             *
             * Only a single account can be approved at a time, so approving the zero address clears previous approvals.
             *
             * Requirements:
             *
             * - The caller must own the token or be an approved operator.
             * - `tokenId` must exist.
             *
             * Emits an {Approval} event.
             */
            function approve(address to, uint256 tokenId) external;
            /**
             * @dev Returns the account approved for `tokenId` token.
             *
             * Requirements:
             *
             * - `tokenId` must exist.
             */
            function getApproved(uint256 tokenId) external view returns (address operator);
            /**
             * @dev Approve or remove `operator` as an operator for the caller.
             * Operators can call {transferFrom} or {safeTransferFrom} for any token owned by the caller.
             *
             * Requirements:
             *
             * - The `operator` cannot be the caller.
             *
             * Emits an {ApprovalForAll} event.
             */
            function setApprovalForAll(address operator, bool _approved) external;
            /**
             * @dev Returns if the `operator` is allowed to manage all of the assets of `owner`.
             *
             * See {setApprovalForAll}
             */
            function isApprovedForAll(address owner, address operator) external view returns (bool);
            /**
             * @dev Safely transfers `tokenId` token from `from` to `to`.
             *
             * Requirements:
             *
             * - `from` cannot be the zero address.
             * - `to` cannot be the zero address.
             * - `tokenId` token must exist and be owned by `from`.
             * - If the caller is not `from`, it must be approved to move this token by either {approve} or {setApprovalForAll}.
             * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer.
             *
             * Emits a {Transfer} event.
             */
            function safeTransferFrom(
                address from,
                address to,
                uint256 tokenId,
                bytes calldata data
            ) external;
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts v4.4.1 (token/ERC721/IERC721Receiver.sol)
        pragma solidity ^0.8.0;
        /**
         * @title ERC721 token receiver interface
         * @dev Interface for any contract that wants to support safeTransfers
         * from ERC721 asset contracts.
         */
        interface IERC721ReceiverUpgradeable {
            /**
             * @dev Whenever an {IERC721} `tokenId` token is transferred to this contract via {IERC721-safeTransferFrom}
             * by `operator` from `from`, this function is called.
             *
             * It must return its Solidity selector to confirm the token transfer.
             * If any other value is returned or the interface is not implemented by the recipient, the transfer will be reverted.
             *
             * The selector can be obtained in Solidity with `IERC721.onERC721Received.selector`.
             */
            function onERC721Received(
                address operator,
                address from,
                uint256 tokenId,
                bytes calldata data
            ) external returns (bytes4);
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts v4.4.1 (token/ERC721/extensions/IERC721Metadata.sol)
        pragma solidity ^0.8.0;
        import "../IERC721Upgradeable.sol";
        /**
         * @title ERC-721 Non-Fungible Token Standard, optional metadata extension
         * @dev See https://eips.ethereum.org/EIPS/eip-721
         */
        interface IERC721MetadataUpgradeable is IERC721Upgradeable {
            /**
             * @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);
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v4.5.0) (utils/Address.sol)
        pragma solidity ^0.8.1;
        /**
         * @dev Collection of functions related to the address type
         */
        library AddressUpgradeable {
            /**
             * @dev Returns true if `account` is a contract.
             *
             * [IMPORTANT]
             * ====
             * It is unsafe to assume that an address for which this function returns
             * false is an externally-owned account (EOA) and not a contract.
             *
             * Among others, `isContract` will return false for the following
             * types of addresses:
             *
             *  - an externally-owned account
             *  - a contract in construction
             *  - an address where a contract will be created
             *  - an address where a contract lived, but was destroyed
             * ====
             *
             * [IMPORTANT]
             * ====
             * You shouldn't rely on `isContract` to protect against flash loan attacks!
             *
             * Preventing calls from contracts is highly discouraged. It breaks composability, breaks support for smart wallets
             * like Gnosis Safe, and does not provide security since it can be circumvented by calling from a contract
             * constructor.
             * ====
             */
            function isContract(address account) internal view returns (bool) {
                // This method relies on extcodesize/address.code.length, which returns 0
                // for contracts in construction, since the code is only stored at the end
                // of the constructor execution.
                return account.code.length > 0;
            }
            /**
             * @dev Replacement for Solidity's `transfer`: sends `amount` wei to
             * `recipient`, forwarding all available gas and reverting on errors.
             *
             * https://eips.ethereum.org/EIPS/eip-1884[EIP1884] increases the gas cost
             * of certain opcodes, possibly making contracts go over the 2300 gas limit
             * imposed by `transfer`, making them unable to receive funds via
             * `transfer`. {sendValue} removes this limitation.
             *
             * https://diligence.consensys.net/posts/2019/09/stop-using-soliditys-transfer-now/[Learn more].
             *
             * IMPORTANT: because control is transferred to `recipient`, care must be
             * taken to not create reentrancy vulnerabilities. Consider using
             * {ReentrancyGuard} or the
             * https://solidity.readthedocs.io/en/v0.5.11/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern].
             */
            function sendValue(address payable recipient, uint256 amount) internal {
                require(address(this).balance >= amount, "Address: insufficient balance");
                (bool success, ) = recipient.call{value: amount}("");
                require(success, "Address: unable to send value, recipient may have reverted");
            }
            /**
             * @dev Performs a Solidity function call using a low level `call`. A
             * plain `call` is an unsafe replacement for a function call: use this
             * function instead.
             *
             * If `target` reverts with a revert reason, it is bubbled up by this
             * function (like regular Solidity function calls).
             *
             * Returns the raw returned data. To convert to the expected return value,
             * use https://solidity.readthedocs.io/en/latest/units-and-global-variables.html?highlight=abi.decode#abi-encoding-and-decoding-functions[`abi.decode`].
             *
             * Requirements:
             *
             * - `target` must be a contract.
             * - calling `target` with `data` must not revert.
             *
             * _Available since v3.1._
             */
            function functionCall(address target, bytes memory data) internal returns (bytes memory) {
                return functionCall(target, data, "Address: low-level call failed");
            }
            /**
             * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], but with
             * `errorMessage` as a fallback revert reason when `target` reverts.
             *
             * _Available since v3.1._
             */
            function functionCall(
                address target,
                bytes memory data,
                string memory errorMessage
            ) internal returns (bytes memory) {
                return functionCallWithValue(target, data, 0, errorMessage);
            }
            /**
             * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
             * but also transferring `value` wei to `target`.
             *
             * Requirements:
             *
             * - the calling contract must have an ETH balance of at least `value`.
             * - the called Solidity function must be `payable`.
             *
             * _Available since v3.1._
             */
            function functionCallWithValue(
                address target,
                bytes memory data,
                uint256 value
            ) internal returns (bytes memory) {
                return functionCallWithValue(target, data, value, "Address: low-level call with value failed");
            }
            /**
             * @dev Same as {xref-Address-functionCallWithValue-address-bytes-uint256-}[`functionCallWithValue`], but
             * with `errorMessage` as a fallback revert reason when `target` reverts.
             *
             * _Available since v3.1._
             */
            function functionCallWithValue(
                address target,
                bytes memory data,
                uint256 value,
                string memory errorMessage
            ) internal returns (bytes memory) {
                require(address(this).balance >= value, "Address: insufficient balance for call");
                require(isContract(target), "Address: call to non-contract");
                (bool success, bytes memory returndata) = target.call{value: value}(data);
                return verifyCallResult(success, returndata, errorMessage);
            }
            /**
             * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
             * but performing a static call.
             *
             * _Available since v3.3._
             */
            function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) {
                return functionStaticCall(target, data, "Address: low-level static call failed");
            }
            /**
             * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],
             * but performing a static call.
             *
             * _Available since v3.3._
             */
            function functionStaticCall(
                address target,
                bytes memory data,
                string memory errorMessage
            ) internal view returns (bytes memory) {
                require(isContract(target), "Address: static call to non-contract");
                (bool success, bytes memory returndata) = target.staticcall(data);
                return verifyCallResult(success, returndata, errorMessage);
            }
            /**
             * @dev Tool to verifies that a low level call was successful, and revert if it wasn't, either by bubbling the
             * revert reason using the provided one.
             *
             * _Available since v4.3._
             */
            function verifyCallResult(
                bool success,
                bytes memory returndata,
                string memory errorMessage
            ) internal pure returns (bytes memory) {
                if (success) {
                    return returndata;
                } else {
                    // Look for revert reason and bubble it up if present
                    if (returndata.length > 0) {
                        // The easiest way to bubble the revert reason is using memory via assembly
                        assembly {
                            let returndata_size := mload(returndata)
                            revert(add(32, returndata), returndata_size)
                        }
                    } else {
                        revert(errorMessage);
                    }
                }
            }
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts v4.4.1 (utils/Context.sol)
        pragma solidity ^0.8.0;
        import "../proxy/utils/Initializable.sol";
        /**
         * @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 ContextUpgradeable is Initializable {
            function __Context_init() internal onlyInitializing {
            }
            function __Context_init_unchained() internal onlyInitializing {
            }
            function _msgSender() internal view virtual returns (address) {
                return msg.sender;
            }
            function _msgData() internal view virtual returns (bytes calldata) {
                return msg.data;
            }
            /**
             * @dev This empty reserved space is put in place to allow future versions to add new
             * variables without shifting down storage in the inheritance chain.
             * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps
             */
            uint256[50] private __gap;
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts v4.4.1 (utils/introspection/ERC165.sol)
        pragma solidity ^0.8.0;
        import "./IERC165Upgradeable.sol";
        import "../../proxy/utils/Initializable.sol";
        /**
         * @dev Implementation of the {IERC165} interface.
         *
         * Contracts that want to implement ERC165 should inherit from this contract and override {supportsInterface} to check
         * for the additional interface id that will be supported. For example:
         *
         * ```solidity
         * function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) {
         *     return interfaceId == type(MyInterface).interfaceId || super.supportsInterface(interfaceId);
         * }
         * ```
         *
         * Alternatively, {ERC165Storage} provides an easier to use but more expensive implementation.
         */
        abstract contract ERC165Upgradeable is Initializable, IERC165Upgradeable {
            function __ERC165_init() internal onlyInitializing {
            }
            function __ERC165_init_unchained() internal onlyInitializing {
            }
            /**
             * @dev See {IERC165-supportsInterface}.
             */
            function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) {
                return interfaceId == type(IERC165Upgradeable).interfaceId;
            }
            /**
             * @dev This empty reserved space is put in place to allow future versions to add new
             * variables without shifting down storage in the inheritance chain.
             * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps
             */
            uint256[50] private __gap;
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts v4.4.1 (utils/introspection/IERC165.sol)
        pragma solidity ^0.8.0;
        /**
         * @dev Interface of the ERC165 standard, as defined in the
         * https://eips.ethereum.org/EIPS/eip-165[EIP].
         *
         * Implementers can declare support of contract interfaces, which can then be
         * queried by others ({ERC165Checker}).
         *
         * For an implementation, see {ERC165}.
         */
        interface IERC165Upgradeable {
            /**
             * @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);
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts v4.4.1 (interfaces/IERC165.sol)
        pragma solidity ^0.8.0;
        import "../utils/introspection/IERC165Upgradeable.sol";
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts v4.4.1 (access/IAccessControlEnumerable.sol)
        pragma solidity ^0.8.0;
        import "./IAccessControlUpgradeable.sol";
        /**
         * @dev External interface of AccessControlEnumerable declared to support ERC165 detection.
         */
        interface IAccessControlEnumerableUpgradeable is IAccessControlUpgradeable {
            /**
             * @dev Returns one of the accounts that have `role`. `index` must be a
             * value between 0 and {getRoleMemberCount}, non-inclusive.
             *
             * Role bearers are not sorted in any particular way, and their ordering may
             * change at any point.
             *
             * WARNING: When using {getRoleMember} and {getRoleMemberCount}, make sure
             * you perform all queries on the same block. See the following
             * https://forum.openzeppelin.com/t/iterating-over-elements-on-enumerableset-in-openzeppelin-contracts/2296[forum post]
             * for more information.
             */
            function getRoleMember(bytes32 role, uint256 index) external view returns (address);
            /**
             * @dev Returns the number of accounts that have `role`. Can be used
             * together with {getRoleMember} to enumerate all bearers of a role.
             */
            function getRoleMemberCount(bytes32 role) external view returns (uint256);
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v4.5.0) (access/AccessControl.sol)
        pragma solidity ^0.8.0;
        import "./IAccessControlUpgradeable.sol";
        import "../utils/ContextUpgradeable.sol";
        import "../utils/StringsUpgradeable.sol";
        import "../utils/introspection/ERC165Upgradeable.sol";
        import "../proxy/utils/Initializable.sol";
        /**
         * @dev Contract module that allows children to implement role-based access
         * control mechanisms. This is a lightweight version that doesn't allow enumerating role
         * members except through off-chain means by accessing the contract event logs. Some
         * applications may benefit from on-chain enumerability, for those cases see
         * {AccessControlEnumerable}.
         *
         * Roles are referred to by their `bytes32` identifier. These should be exposed
         * in the external API and be unique. The best way to achieve this is by
         * using `public constant` hash digests:
         *
         * ```
         * bytes32 public constant MY_ROLE = keccak256("MY_ROLE");
         * ```
         *
         * Roles can be used to represent a set of permissions. To restrict access to a
         * function call, use {hasRole}:
         *
         * ```
         * function foo() public {
         *     require(hasRole(MY_ROLE, msg.sender));
         *     ...
         * }
         * ```
         *
         * Roles can be granted and revoked dynamically via the {grantRole} and
         * {revokeRole} functions. Each role has an associated admin role, and only
         * accounts that have a role's admin role can call {grantRole} and {revokeRole}.
         *
         * By default, the admin role for all roles is `DEFAULT_ADMIN_ROLE`, which means
         * that only accounts with this role will be able to grant or revoke other
         * roles. More complex role relationships can be created by using
         * {_setRoleAdmin}.
         *
         * WARNING: The `DEFAULT_ADMIN_ROLE` is also its own admin: it has permission to
         * grant and revoke this role. Extra precautions should be taken to secure
         * accounts that have been granted it.
         */
        abstract contract AccessControlUpgradeable is Initializable, ContextUpgradeable, IAccessControlUpgradeable, ERC165Upgradeable {
            function __AccessControl_init() internal onlyInitializing {
            }
            function __AccessControl_init_unchained() internal onlyInitializing {
            }
            struct RoleData {
                mapping(address => bool) members;
                bytes32 adminRole;
            }
            mapping(bytes32 => RoleData) private _roles;
            bytes32 public constant DEFAULT_ADMIN_ROLE = 0x00;
            /**
             * @dev Modifier that checks that an account has a specific role. Reverts
             * with a standardized message including the required role.
             *
             * The format of the revert reason is given by the following regular expression:
             *
             *  /^AccessControl: account (0x[0-9a-f]{40}) is missing role (0x[0-9a-f]{64})$/
             *
             * _Available since v4.1._
             */
            modifier onlyRole(bytes32 role) {
                _checkRole(role, _msgSender());
                _;
            }
            /**
             * @dev See {IERC165-supportsInterface}.
             */
            function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) {
                return interfaceId == type(IAccessControlUpgradeable).interfaceId || super.supportsInterface(interfaceId);
            }
            /**
             * @dev Returns `true` if `account` has been granted `role`.
             */
            function hasRole(bytes32 role, address account) public view virtual override returns (bool) {
                return _roles[role].members[account];
            }
            /**
             * @dev Revert with a standard message if `account` is missing `role`.
             *
             * The format of the revert reason is given by the following regular expression:
             *
             *  /^AccessControl: account (0x[0-9a-f]{40}) is missing role (0x[0-9a-f]{64})$/
             */
            function _checkRole(bytes32 role, address account) internal view virtual {
                if (!hasRole(role, account)) {
                    revert(
                        string(
                            abi.encodePacked(
                                "AccessControl: account ",
                                StringsUpgradeable.toHexString(uint160(account), 20),
                                " is missing role ",
                                StringsUpgradeable.toHexString(uint256(role), 32)
                            )
                        )
                    );
                }
            }
            /**
             * @dev Returns the admin role that controls `role`. See {grantRole} and
             * {revokeRole}.
             *
             * To change a role's admin, use {_setRoleAdmin}.
             */
            function getRoleAdmin(bytes32 role) public view virtual override returns (bytes32) {
                return _roles[role].adminRole;
            }
            /**
             * @dev Grants `role` to `account`.
             *
             * If `account` had not been already granted `role`, emits a {RoleGranted}
             * event.
             *
             * Requirements:
             *
             * - the caller must have ``role``'s admin role.
             */
            function grantRole(bytes32 role, address account) public virtual override onlyRole(getRoleAdmin(role)) {
                _grantRole(role, account);
            }
            /**
             * @dev Revokes `role` from `account`.
             *
             * If `account` had been granted `role`, emits a {RoleRevoked} event.
             *
             * Requirements:
             *
             * - the caller must have ``role``'s admin role.
             */
            function revokeRole(bytes32 role, address account) public virtual override onlyRole(getRoleAdmin(role)) {
                _revokeRole(role, account);
            }
            /**
             * @dev Revokes `role` from the calling account.
             *
             * Roles are often managed via {grantRole} and {revokeRole}: this function's
             * purpose is to provide a mechanism for accounts to lose their privileges
             * if they are compromised (such as when a trusted device is misplaced).
             *
             * If the calling account had been revoked `role`, emits a {RoleRevoked}
             * event.
             *
             * Requirements:
             *
             * - the caller must be `account`.
             */
            function renounceRole(bytes32 role, address account) public virtual override {
                require(account == _msgSender(), "AccessControl: can only renounce roles for self");
                _revokeRole(role, account);
            }
            /**
             * @dev Grants `role` to `account`.
             *
             * If `account` had not been already granted `role`, emits a {RoleGranted}
             * event. Note that unlike {grantRole}, this function doesn't perform any
             * checks on the calling account.
             *
             * [WARNING]
             * ====
             * This function should only be called from the constructor when setting
             * up the initial roles for the system.
             *
             * Using this function in any other way is effectively circumventing the admin
             * system imposed by {AccessControl}.
             * ====
             *
             * NOTE: This function is deprecated in favor of {_grantRole}.
             */
            function _setupRole(bytes32 role, address account) internal virtual {
                _grantRole(role, account);
            }
            /**
             * @dev Sets `adminRole` as ``role``'s admin role.
             *
             * Emits a {RoleAdminChanged} event.
             */
            function _setRoleAdmin(bytes32 role, bytes32 adminRole) internal virtual {
                bytes32 previousAdminRole = getRoleAdmin(role);
                _roles[role].adminRole = adminRole;
                emit RoleAdminChanged(role, previousAdminRole, adminRole);
            }
            /**
             * @dev Grants `role` to `account`.
             *
             * Internal function without access restriction.
             */
            function _grantRole(bytes32 role, address account) internal virtual {
                if (!hasRole(role, account)) {
                    _roles[role].members[account] = true;
                    emit RoleGranted(role, account, _msgSender());
                }
            }
            /**
             * @dev Revokes `role` from `account`.
             *
             * Internal function without access restriction.
             */
            function _revokeRole(bytes32 role, address account) internal virtual {
                if (hasRole(role, account)) {
                    _roles[role].members[account] = false;
                    emit RoleRevoked(role, account, _msgSender());
                }
            }
            /**
             * @dev This empty reserved space is put in place to allow future versions to add new
             * variables without shifting down storage in the inheritance chain.
             * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps
             */
            uint256[49] private __gap;
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts v4.4.1 (utils/structs/EnumerableSet.sol)
        pragma solidity ^0.8.0;
        /**
         * @dev Library for managing
         * https://en.wikipedia.org/wiki/Set_(abstract_data_type)[sets] of primitive
         * types.
         *
         * Sets have the following properties:
         *
         * - Elements are added, removed, and checked for existence in constant time
         * (O(1)).
         * - Elements are enumerated in O(n). No guarantees are made on the ordering.
         *
         * ```
         * contract Example {
         *     // Add the library methods
         *     using EnumerableSet for EnumerableSet.AddressSet;
         *
         *     // Declare a set state variable
         *     EnumerableSet.AddressSet private mySet;
         * }
         * ```
         *
         * As of v3.3.0, sets of type `bytes32` (`Bytes32Set`), `address` (`AddressSet`)
         * and `uint256` (`UintSet`) are supported.
         */
        library EnumerableSetUpgradeable {
            // To implement this library for multiple types with as little code
            // repetition as possible, we write it in terms of a generic Set type with
            // bytes32 values.
            // The Set implementation uses private functions, and user-facing
            // implementations (such as AddressSet) are just wrappers around the
            // underlying Set.
            // This means that we can only create new EnumerableSets for types that fit
            // in bytes32.
            struct Set {
                // Storage of set values
                bytes32[] _values;
                // Position of the value in the `values` array, plus 1 because index 0
                // means a value is not in the set.
                mapping(bytes32 => uint256) _indexes;
            }
            /**
             * @dev Add a value to a set. O(1).
             *
             * Returns true if the value was added to the set, that is if it was not
             * already present.
             */
            function _add(Set storage set, bytes32 value) private returns (bool) {
                if (!_contains(set, value)) {
                    set._values.push(value);
                    // The value is stored at length-1, but we add 1 to all indexes
                    // and use 0 as a sentinel value
                    set._indexes[value] = set._values.length;
                    return true;
                } else {
                    return false;
                }
            }
            /**
             * @dev Removes a value from a set. O(1).
             *
             * Returns true if the value was removed from the set, that is if it was
             * present.
             */
            function _remove(Set storage set, bytes32 value) private returns (bool) {
                // We read and store the value's index to prevent multiple reads from the same storage slot
                uint256 valueIndex = set._indexes[value];
                if (valueIndex != 0) {
                    // Equivalent to contains(set, value)
                    // To delete an element from the _values array in O(1), we swap the element to delete with the last one in
                    // the array, and then remove the last element (sometimes called as 'swap and pop').
                    // This modifies the order of the array, as noted in {at}.
                    uint256 toDeleteIndex = valueIndex - 1;
                    uint256 lastIndex = set._values.length - 1;
                    if (lastIndex != toDeleteIndex) {
                        bytes32 lastvalue = set._values[lastIndex];
                        // Move the last value to the index where the value to delete is
                        set._values[toDeleteIndex] = lastvalue;
                        // Update the index for the moved value
                        set._indexes[lastvalue] = valueIndex; // Replace lastvalue's index to valueIndex
                    }
                    // Delete the slot where the moved value was stored
                    set._values.pop();
                    // Delete the index for the deleted slot
                    delete set._indexes[value];
                    return true;
                } else {
                    return false;
                }
            }
            /**
             * @dev Returns true if the value is in the set. O(1).
             */
            function _contains(Set storage set, bytes32 value) private view returns (bool) {
                return set._indexes[value] != 0;
            }
            /**
             * @dev Returns the number of values on the set. O(1).
             */
            function _length(Set storage set) private view returns (uint256) {
                return set._values.length;
            }
            /**
             * @dev Returns the value stored at position `index` in the set. O(1).
             *
             * Note that there are no guarantees on the ordering of values inside the
             * array, and it may change when more values are added or removed.
             *
             * Requirements:
             *
             * - `index` must be strictly less than {length}.
             */
            function _at(Set storage set, uint256 index) private view returns (bytes32) {
                return set._values[index];
            }
            /**
             * @dev Return the entire set in an array
             *
             * WARNING: This operation will copy the entire storage to memory, which can be quite expensive. This is designed
             * to mostly be used by view accessors that are queried without any gas fees. Developers should keep in mind that
             * this function has an unbounded cost, and using it as part of a state-changing function may render the function
             * uncallable if the set grows to a point where copying to memory consumes too much gas to fit in a block.
             */
            function _values(Set storage set) private view returns (bytes32[] memory) {
                return set._values;
            }
            // Bytes32Set
            struct Bytes32Set {
                Set _inner;
            }
            /**
             * @dev Add a value to a set. O(1).
             *
             * Returns true if the value was added to the set, that is if it was not
             * already present.
             */
            function add(Bytes32Set storage set, bytes32 value) internal returns (bool) {
                return _add(set._inner, value);
            }
            /**
             * @dev Removes a value from a set. O(1).
             *
             * Returns true if the value was removed from the set, that is if it was
             * present.
             */
            function remove(Bytes32Set storage set, bytes32 value) internal returns (bool) {
                return _remove(set._inner, value);
            }
            /**
             * @dev Returns true if the value is in the set. O(1).
             */
            function contains(Bytes32Set storage set, bytes32 value) internal view returns (bool) {
                return _contains(set._inner, value);
            }
            /**
             * @dev Returns the number of values in the set. O(1).
             */
            function length(Bytes32Set storage set) internal view returns (uint256) {
                return _length(set._inner);
            }
            /**
             * @dev Returns the value stored at position `index` in the set. O(1).
             *
             * Note that there are no guarantees on the ordering of values inside the
             * array, and it may change when more values are added or removed.
             *
             * Requirements:
             *
             * - `index` must be strictly less than {length}.
             */
            function at(Bytes32Set storage set, uint256 index) internal view returns (bytes32) {
                return _at(set._inner, index);
            }
            /**
             * @dev Return the entire set in an array
             *
             * WARNING: This operation will copy the entire storage to memory, which can be quite expensive. This is designed
             * to mostly be used by view accessors that are queried without any gas fees. Developers should keep in mind that
             * this function has an unbounded cost, and using it as part of a state-changing function may render the function
             * uncallable if the set grows to a point where copying to memory consumes too much gas to fit in a block.
             */
            function values(Bytes32Set storage set) internal view returns (bytes32[] memory) {
                return _values(set._inner);
            }
            // AddressSet
            struct AddressSet {
                Set _inner;
            }
            /**
             * @dev Add a value to a set. O(1).
             *
             * Returns true if the value was added to the set, that is if it was not
             * already present.
             */
            function add(AddressSet storage set, address value) internal returns (bool) {
                return _add(set._inner, bytes32(uint256(uint160(value))));
            }
            /**
             * @dev Removes a value from a set. O(1).
             *
             * Returns true if the value was removed from the set, that is if it was
             * present.
             */
            function remove(AddressSet storage set, address value) internal returns (bool) {
                return _remove(set._inner, bytes32(uint256(uint160(value))));
            }
            /**
             * @dev Returns true if the value is in the set. O(1).
             */
            function contains(AddressSet storage set, address value) internal view returns (bool) {
                return _contains(set._inner, bytes32(uint256(uint160(value))));
            }
            /**
             * @dev Returns the number of values in the set. O(1).
             */
            function length(AddressSet storage set) internal view returns (uint256) {
                return _length(set._inner);
            }
            /**
             * @dev Returns the value stored at position `index` in the set. O(1).
             *
             * Note that there are no guarantees on the ordering of values inside the
             * array, and it may change when more values are added or removed.
             *
             * Requirements:
             *
             * - `index` must be strictly less than {length}.
             */
            function at(AddressSet storage set, uint256 index) internal view returns (address) {
                return address(uint160(uint256(_at(set._inner, index))));
            }
            /**
             * @dev Return the entire set in an array
             *
             * WARNING: This operation will copy the entire storage to memory, which can be quite expensive. This is designed
             * to mostly be used by view accessors that are queried without any gas fees. Developers should keep in mind that
             * this function has an unbounded cost, and using it as part of a state-changing function may render the function
             * uncallable if the set grows to a point where copying to memory consumes too much gas to fit in a block.
             */
            function values(AddressSet storage set) internal view returns (address[] memory) {
                bytes32[] memory store = _values(set._inner);
                address[] memory result;
                assembly {
                    result := store
                }
                return result;
            }
            // UintSet
            struct UintSet {
                Set _inner;
            }
            /**
             * @dev Add a value to a set. O(1).
             *
             * Returns true if the value was added to the set, that is if it was not
             * already present.
             */
            function add(UintSet storage set, uint256 value) internal returns (bool) {
                return _add(set._inner, bytes32(value));
            }
            /**
             * @dev Removes a value from a set. O(1).
             *
             * Returns true if the value was removed from the set, that is if it was
             * present.
             */
            function remove(UintSet storage set, uint256 value) internal returns (bool) {
                return _remove(set._inner, bytes32(value));
            }
            /**
             * @dev Returns true if the value is in the set. O(1).
             */
            function contains(UintSet storage set, uint256 value) internal view returns (bool) {
                return _contains(set._inner, bytes32(value));
            }
            /**
             * @dev Returns the number of values on the set. O(1).
             */
            function length(UintSet storage set) internal view returns (uint256) {
                return _length(set._inner);
            }
            /**
             * @dev Returns the value stored at position `index` in the set. O(1).
             *
             * Note that there are no guarantees on the ordering of values inside the
             * array, and it may change when more values are added or removed.
             *
             * Requirements:
             *
             * - `index` must be strictly less than {length}.
             */
            function at(UintSet storage set, uint256 index) internal view returns (uint256) {
                return uint256(_at(set._inner, index));
            }
            /**
             * @dev Return the entire set in an array
             *
             * WARNING: This operation will copy the entire storage to memory, which can be quite expensive. This is designed
             * to mostly be used by view accessors that are queried without any gas fees. Developers should keep in mind that
             * this function has an unbounded cost, and using it as part of a state-changing function may render the function
             * uncallable if the set grows to a point where copying to memory consumes too much gas to fit in a block.
             */
            function values(UintSet storage set) internal view returns (uint256[] memory) {
                bytes32[] memory store = _values(set._inner);
                uint256[] memory result;
                assembly {
                    result := store
                }
                return result;
            }
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts v4.4.1 (access/IAccessControl.sol)
        pragma solidity ^0.8.0;
        /**
         * @dev External interface of AccessControl declared to support ERC165 detection.
         */
        interface IAccessControlUpgradeable {
            /**
             * @dev Emitted when `newAdminRole` is set as ``role``'s admin role, replacing `previousAdminRole`
             *
             * `DEFAULT_ADMIN_ROLE` is the starting admin for all roles, despite
             * {RoleAdminChanged} not being emitted signaling this.
             *
             * _Available since v3.1._
             */
            event RoleAdminChanged(bytes32 indexed role, bytes32 indexed previousAdminRole, bytes32 indexed newAdminRole);
            /**
             * @dev Emitted when `account` is granted `role`.
             *
             * `sender` is the account that originated the contract call, an admin role
             * bearer except when using {AccessControl-_setupRole}.
             */
            event RoleGranted(bytes32 indexed role, address indexed account, address indexed sender);
            /**
             * @dev Emitted when `account` is revoked `role`.
             *
             * `sender` is the account that originated the contract call:
             *   - if using `revokeRole`, it is the admin role bearer
             *   - if using `renounceRole`, it is the role bearer (i.e. `account`)
             */
            event RoleRevoked(bytes32 indexed role, address indexed account, address indexed sender);
            /**
             * @dev Returns `true` if `account` has been granted `role`.
             */
            function hasRole(bytes32 role, address account) external view returns (bool);
            /**
             * @dev Returns the admin role that controls `role`. See {grantRole} and
             * {revokeRole}.
             *
             * To change a role's admin, use {AccessControl-_setRoleAdmin}.
             */
            function getRoleAdmin(bytes32 role) external view returns (bytes32);
            /**
             * @dev Grants `role` to `account`.
             *
             * If `account` had not been already granted `role`, emits a {RoleGranted}
             * event.
             *
             * Requirements:
             *
             * - the caller must have ``role``'s admin role.
             */
            function grantRole(bytes32 role, address account) external;
            /**
             * @dev Revokes `role` from `account`.
             *
             * If `account` had been granted `role`, emits a {RoleRevoked} event.
             *
             * Requirements:
             *
             * - the caller must have ``role``'s admin role.
             */
            function revokeRole(bytes32 role, address account) external;
            /**
             * @dev Revokes `role` from the calling account.
             *
             * Roles are often managed via {grantRole} and {revokeRole}: this function's
             * purpose is to provide a mechanism for accounts to lose their privileges
             * if they are compromised (such as when a trusted device is misplaced).
             *
             * If the calling account had been granted `role`, emits a {RoleRevoked}
             * event.
             *
             * Requirements:
             *
             * - the caller must be `account`.
             */
            function renounceRole(bytes32 role, address account) external;
        }
        // SPDX-License-Identifier: Apache-2.0
        pragma solidity ^0.8.11;
        interface IThirdwebContract {
            /// @dev Returns the module type of the contract.
            function contractType() external pure returns (bytes32);
            /// @dev Returns the version of the contract.
            function contractVersion() external pure returns (uint8);
            /// @dev Returns the metadata URI of the contract.
            function contractURI() external view returns (string memory);
            /**
             *  @dev Sets contract URI for the storefront-level metadata of the contract.
             *       Only module admin can call this function.
             */
            function setContractURI(string calldata _uri) external;
        }
        // SPDX-License-Identifier: Apache-2.0
        pragma solidity ^0.8.11;
        interface IThirdwebPlatformFee {
            /// @dev Returns the platform fee bps and recipient.
            function getPlatformFeeInfo() external view returns (address platformFeeRecipient, uint16 platformFeeBps);
            /// @dev Lets a module admin update the fees on primary sales.
            function setPlatformFeeInfo(address _platformFeeRecipient, uint256 _platformFeeBps) external;
        }
        // SPDX-License-Identifier: Apache-2.0
        pragma solidity ^0.8.11;
        interface IThirdwebPrimarySale {
            /// @dev The adress that receives all primary sales value.
            function primarySaleRecipient() external view returns (address);
            /// @dev Lets a module admin set the default recipient of all primary sales.
            function setPrimarySaleRecipient(address _saleRecipient) external;
        }
        // SPDX-License-Identifier: Apache-2.0
        pragma solidity ^0.8.11;
        import "@openzeppelin/contracts-upgradeable/interfaces/IERC2981Upgradeable.sol";
        interface IThirdwebRoyalty is IERC2981Upgradeable {
            struct RoyaltyInfo {
                address recipient;
                uint256 bps;
            }
            /// @dev Returns the royalty recipient and fee bps.
            function getDefaultRoyaltyInfo() external view returns (address, uint16);
            /// @dev Lets a module admin update the royalty bps and recipient.
            function setDefaultRoyaltyInfo(address _royaltyRecipient, uint256 _royaltyBps) external;
            /// @dev Lets a module admin set the royalty recipient for a particular token Id.
            function setRoyaltyInfoForToken(
                uint256 tokenId,
                address recipient,
                uint256 bps
            ) external;
            /// @dev Returns the royalty recipient for a particular token Id.
            function getRoyaltyInfoForToken(uint256 tokenId) external view returns (address, uint16);
            /// @dev Emitted when royalty info is updated.
            event DefaultRoyalty(address newRoyaltyRecipient, uint256 newRoyaltyBps);
            /// @dev Emitted when royalty recipient for tokenId is set
            event RoyaltyForToken(uint256 indexed tokenId, address royaltyRecipient, uint256 royaltyBps);
        }
        // SPDX-License-Identifier: Apache-2.0
        pragma solidity ^0.8.11;
        interface IThirdwebOwnable {
            /// @dev Returns the owner of the contract.
            function owner() external view returns (address);
            /// @dev Lets a module admin set a new owner for the contract. The new owner must be a module admin.
            function setOwner(address _newOwner) external;
        }
        // SPDX-License-Identifier: Apache-2.0
        pragma solidity ^0.8.11;
        import "@openzeppelin/contracts-upgradeable/utils/structs/BitMapsUpgradeable.sol";
        /**
         *  Thirdweb's 'Drop' contracts are distribution mechanisms for tokens.
         *
         *  A contract admin (i.e. a holder of `DEFAULT_ADMIN_ROLE`) can set a series of claim conditions,
         *  ordered by their respective `startTimestamp`. A claim condition defines criteria under which
         *  accounts can mint tokens. Claim conditions can be overwritten or added to by the contract admin.
         *  At any moment, there is only one active claim condition.
         */
        interface IDropClaimCondition {
            /**
             *  @notice The criteria that make up a claim condition.
             *
             *  @param startTimestamp                 The unix timestamp after which the claim condition applies.
             *                                        The same claim condition applies until the `startTimestamp`
             *                                        of the next claim condition.
             *
             *  @param maxClaimableSupply             The maximum total number of tokens that can be claimed under
             *                                        the claim condition.
             *
             *  @param supplyClaimed                  At any given point, the number of tokens that have been claimed
             *                                        under the claim condition.
             *
             *  @param quantityLimitPerTransaction    The maximum number of tokens that can be claimed in a single
             *                                        transaction.
             *
             *  @param waitTimeInSecondsBetweenClaims The least number of seconds an account must wait after claiming
             *                                        tokens, to be able to claim tokens again.
             *
             *  @param merkleRoot                     The allowlist of addresses that can claim tokens under the claim
             *                                        condition.
             *
             *  @param pricePerToken                  The price required to pay per token claimed.
             *
             *  @param currency                       The currency in which the `pricePerToken` must be paid.
             */
            struct ClaimCondition {
                uint256 startTimestamp;
                uint256 maxClaimableSupply;
                uint256 supplyClaimed;
                uint256 quantityLimitPerTransaction;
                uint256 waitTimeInSecondsBetweenClaims;
                bytes32 merkleRoot;
                uint256 pricePerToken;
                address currency;
            }
            /**
             *  @notice The set of all claim conditions, at any given moment.
             *  Claim Phase ID = [currentStartId, currentStartId + length - 1];
             *
             *  @param currentStartId           Acts as the uid for each claim condition. Incremented
             *                                  by one every time a claim condition is created.
             *
             *  @param count                    The total number of phases / claim conditions in the list
             *                                  of claim conditions.
             *
             *  @param phases                   The claim conditions at a given uid. Claim conditions
             *                                  are ordered in an ascending order by their `startTimestamp`.
             *
             *  @param limitLastClaimTimestamp  Map from an account and uid for a claim condition, to the last timestamp
             *                                  at which the account claimed tokens under that claim condition.
             *
             *  @param limitMerkleProofClaim    Map from a claim condition uid to whether an address in an allowlist
             *                                  has already claimed tokens i.e. used their place in the allowlist.
             */
            struct ClaimConditionList {
                uint256 currentStartId;
                uint256 count;
                mapping(uint256 => ClaimCondition) phases;
                mapping(uint256 => mapping(address => uint256)) limitLastClaimTimestamp;
                mapping(uint256 => BitMapsUpgradeable.BitMap) limitMerkleProofClaim;
            }
        }
        // SPDX-License-Identifier: Apache-2.0
        pragma solidity ^0.8.11;
        interface IWETH {
            function deposit() external payable;
            function withdraw(uint256 amount) external;
            function transfer(address to, uint256 value) external returns (bool);
        }
        // SPDX-License-Identifier: MIT
        // OpenZeppelin Contracts (last updated v4.5.0) (token/ERC20/IERC20.sol)
        pragma solidity ^0.8.0;
        /**
         * @dev Interface of the ERC20 standard as defined in the EIP.
         */
        interface IERC20Upgradeable {
            /**
             * @dev Returns the amount of tokens in existence.
             */
            function totalSupply() external view returns (uint256);
            /**
             * @dev Returns the amount of tokens owned by `account`.
             */
            function balanceOf(address account) external view returns (uint256);
            /**
             * @dev Moves `amount` tokens from the caller's account to `to`.
             *
             * Returns a boolean value indicating whether the operation succeeded.
             *
             * Emits a {Transfer} event.
             */
            function transfer(address to, uint256 amount) external returns (bool);
            /**
             * @dev Returns the remaining number of tokens that `spender` will be
             * allowed to spend on behalf of `owner` through {transferFrom}. This is
             * zero by default.
             *
             * This value changes when {approve} or {transferFrom} are called.
             */
            function allowance(address owner, address spender) external view returns (uint256);
            /**
             * @dev Sets `amount` as the allowance of `spender` over the caller's tokens.
             *
             * Returns a boolean value indicating whether the operation succeeded.
             *
             * IMPORTANT: Beware that changing an allowance with this method brings the risk
             * that someone may use both the old and the new allowance by unfortunate
             * transaction ordering. One possible solution to mitigate this race
             * condition is to first reduce the spender's allowance to 0 and set the
             * desired value afterwards:
             * https://github.com/ethereum/EIPs/issues/20#issuecomment-263524729
             *
             * Emits an {Approval} event.
             */
            function approve(address spender, uint256 amount) external returns (bool);
            /**
             * @dev Moves `amount` tokens from `from` to `to` using the
             * allowance mechanism. `amount` is then deducted from the caller's
             * allowance.
             *
             * Returns a boolean value indicating whether the operation succeeded.
             *
             * Emits a {Transfer} event.
             */
            function transferFrom(
                address from,
                address to,
                uint256 amount
            ) external returns (bool);
            /**
             * @dev Emitted when `value` tokens are moved from one account (`from`) to
             * another (`to`).
             *
             * Note that `value` may be zero.
             */
            event Transfer(address indexed from, address indexed to, uint256 value);
            /**
             * @dev Emitted when the allowance of a `spender` for an `owner` is set by
             * a call to {approve}. `value` is the new allowance.
             */
            event Approval(address indexed owner, address indexed spender, uint256 value);
        }