diff --git a/deployments/rinkeby/DropCreator.json b/deployments/rinkeby/DropCreator.json index c1f35de5..5d7374de 100644 --- a/deployments/rinkeby/DropCreator.json +++ b/deployments/rinkeby/DropCreator.json @@ -1,5 +1,5 @@ { - "address": "0x3a27Aff3289b969bA4e0Bbb0eBeA2Da1bc5E336b", + "address": "0x9e1649e7F3985dddC8E03370710092718e21DAB6", "abi": [ { "inputs": [ @@ -110,29 +110,29 @@ "type": "function" } ], - "transactionHash": "0x22793fac4035861d831a424fa888ae37d1599b1076fab0eb314b88ac9023e388", + "transactionHash": "0x721cd43f7622767f10b174e19eebe2fdb4c7504170934aab86cf4d758e3bdab1", "receipt": { "to": null, "from": "0xaD1fcD83DE77518d3D1b769F22B0A169eD55A919", - "contractAddress": "0x3a27Aff3289b969bA4e0Bbb0eBeA2Da1bc5E336b", - "transactionIndex": 5, + "contractAddress": "0x9e1649e7F3985dddC8E03370710092718e21DAB6", + "transactionIndex": 7, "gasUsed": "385197", "logsBloom": "0x00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000", - "blockHash": "0x199c679f9bc79fc15f6a25eb51aa1ed069b8a6162be13c0cee57af4d23d91d03", - "transactionHash": "0x22793fac4035861d831a424fa888ae37d1599b1076fab0eb314b88ac9023e388", + "blockHash": "0x7dc7544ff41830568623fdbb7a887647ffb6e0dfdcb762cbfb6214ae0717c828", + "transactionHash": "0x721cd43f7622767f10b174e19eebe2fdb4c7504170934aab86cf4d758e3bdab1", "logs": [], - "blockNumber": 11091890, - "cumulativeGasUsed": "802774", + "blockNumber": 11092053, + "cumulativeGasUsed": "782072", "status": 1, "byzantium": true }, "args": [ - "0x856e6c62c5564e9013053b3988E7E4256F3887dC" + "0xfddbEb53227C2CB1729C3C35c6480c79603EC373" ], - "solcInputHash": "f22aeb84b70c3f3bdbb59fc583a836f8", - "metadata": "{\"compiler\":{\"version\":\"0.8.15+commit.e14f2714\"},\"language\":\"Solidity\",\"output\":{\"abi\":[{\"inputs\":[{\"internalType\":\"address\",\"name\":\"_implementation\",\"type\":\"address\"}],\"stateMutability\":\"nonpayable\",\"type\":\"constructor\"},{\"anonymous\":false,\"inputs\":[{\"indexed\":true,\"internalType\":\"uint256\",\"name\":\"dropId\",\"type\":\"uint256\"},{\"indexed\":true,\"internalType\":\"address\",\"name\":\"creator\",\"type\":\"address\"},{\"indexed\":false,\"internalType\":\"uint256\",\"name\":\"dropSize\",\"type\":\"uint256\"},{\"indexed\":false,\"internalType\":\"address\",\"name\":\"dropContractAddress\",\"type\":\"address\"}],\"name\":\"CreatedDrop\",\"type\":\"event\"},{\"inputs\":[{\"internalType\":\"address\",\"name\":\"_artistWallet\",\"type\":\"address\"},{\"internalType\":\"string\",\"name\":\"_name\",\"type\":\"string\"},{\"internalType\":\"string\",\"name\":\"_symbol\",\"type\":\"string\"},{\"internalType\":\"uint256\",\"name\":\"_dropSize\",\"type\":\"uint256\"}],\"name\":\"createDrop\",\"outputs\":[{\"internalType\":\"uint256\",\"name\":\"\",\"type\":\"uint256\"}],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"uint256\",\"name\":\"dropId\",\"type\":\"uint256\"}],\"name\":\"getDropAtId\",\"outputs\":[{\"internalType\":\"contract ExpandedNFT\",\"name\":\"\",\"type\":\"address\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[],\"name\":\"implementation\",\"outputs\":[{\"internalType\":\"address\",\"name\":\"\",\"type\":\"address\"}],\"stateMutability\":\"view\",\"type\":\"function\"}],\"devdoc\":{\"events\":{\"CreatedDrop(uint256,address,uint256,address)\":{\"params\":{\"dropId\":\"ID of newly created drop\"}}},\"kind\":\"dev\",\"methods\":{\"constructor\":{\"params\":{\"_implementation\":\"ExpandedNFT logic implementation contract to clone\"}},\"createDrop(address,string,string,uint256)\":{\"params\":{\"_artistWallet\":\"User that created the drop\",\"_name\":\"Name of the drop contract\",\"_symbol\":\"Symbol of the drop contract\"}},\"getDropAtId(uint256)\":{\"params\":{\"dropId\":\"id of drop to get contract for\"},\"returns\":{\"_0\":\"ExpandedNFT Drop NFT contract\"}}},\"version\":1},\"userdoc\":{\"events\":{\"CreatedDrop(uint256,address,uint256,address)\":{\"notice\":\"Emitted when a drop is created reserving the corresponding token IDs.\"}},\"kind\":\"user\",\"methods\":{\"constructor\":{\"notice\":\"Initializes factory with address of implementation logic\"},\"createDrop(address,string,string,uint256)\":{\"notice\":\"Creates a new drop contract as a factory with a deterministic address Important: None of these fields (except the Url fields with the same hash) can be changed after calling\"},\"getDropAtId(uint256)\":{\"notice\":\"Get drop given the created ID\"},\"implementation()\":{\"notice\":\"Address for implementation of ExpandedNFT to clone\"}},\"version\":1}},\"settings\":{\"compilationTarget\":{\"contracts/DropCreator.sol\":\"DropCreator\"},\"evmVersion\":\"london\",\"libraries\":{},\"metadata\":{\"bytecodeHash\":\"ipfs\",\"useLiteralContent\":true},\"optimizer\":{\"enabled\":true,\"runs\":100},\"remappings\":[]},\"sources\":{\"@openzeppelin/contracts-upgradeable/access/OwnableUpgradeable.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n// OpenZeppelin Contracts (last updated v4.7.0) (access/Ownable.sol)\\n\\npragma solidity ^0.8.0;\\n\\nimport \\\"../utils/ContextUpgradeable.sol\\\";\\nimport \\\"../proxy/utils/Initializable.sol\\\";\\n\\n/**\\n * @dev Contract module which provides a basic access control mechanism, where\\n * there is an account (an owner) that can be granted exclusive access to\\n * specific functions.\\n *\\n * By default, the owner account will be the one that deploys the contract. This\\n * can later be changed with {transferOwnership}.\\n *\\n * This module is used through inheritance. It will make available the modifier\\n * `onlyOwner`, which can be applied to your functions to restrict their use to\\n * the owner.\\n */\\nabstract contract OwnableUpgradeable is Initializable, ContextUpgradeable {\\n address private _owner;\\n\\n event OwnershipTransferred(address indexed previousOwner, address indexed newOwner);\\n\\n /**\\n * @dev Initializes the contract setting the deployer as the initial owner.\\n */\\n function __Ownable_init() internal onlyInitializing {\\n __Ownable_init_unchained();\\n }\\n\\n function __Ownable_init_unchained() internal onlyInitializing {\\n _transferOwnership(_msgSender());\\n }\\n\\n /**\\n * @dev Throws if called by any account other than the owner.\\n */\\n modifier onlyOwner() {\\n _checkOwner();\\n _;\\n }\\n\\n /**\\n * @dev Returns the address of the current owner.\\n */\\n function owner() public view virtual returns (address) {\\n return _owner;\\n }\\n\\n /**\\n * @dev Throws if the sender is not the owner.\\n */\\n function _checkOwner() internal view virtual {\\n require(owner() == _msgSender(), \\\"Ownable: caller is not the owner\\\");\\n }\\n\\n /**\\n * @dev Leaves the contract without owner. It will not be possible to call\\n * `onlyOwner` functions anymore. Can only be called by the current owner.\\n *\\n * NOTE: Renouncing ownership will leave the contract without an owner,\\n * thereby removing any functionality that is only available to the owner.\\n */\\n function renounceOwnership() public virtual onlyOwner {\\n _transferOwnership(address(0));\\n }\\n\\n /**\\n * @dev Transfers ownership of the contract to a new account (`newOwner`).\\n * Can only be called by the current owner.\\n */\\n function transferOwnership(address newOwner) public virtual onlyOwner {\\n require(newOwner != address(0), \\\"Ownable: new owner is the zero address\\\");\\n _transferOwnership(newOwner);\\n }\\n\\n /**\\n * @dev Transfers ownership of the contract to a new account (`newOwner`).\\n * Internal function without access restriction.\\n */\\n function _transferOwnership(address newOwner) internal virtual {\\n address oldOwner = _owner;\\n _owner = newOwner;\\n emit OwnershipTransferred(oldOwner, newOwner);\\n }\\n\\n /**\\n * @dev This empty reserved space is put in place to allow future versions to add new\\n * variables without shifting down storage in the inheritance chain.\\n * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps\\n */\\n uint256[49] private __gap;\\n}\\n\",\"keccak256\":\"0x247c62047745915c0af6b955470a72d1696ebad4352d7d3011aef1a2463cd888\",\"license\":\"MIT\"},\"@openzeppelin/contracts-upgradeable/interfaces/IERC2981Upgradeable.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n// OpenZeppelin Contracts (last updated v4.6.0) (interfaces/IERC2981.sol)\\n\\npragma solidity ^0.8.0;\\n\\nimport \\\"../utils/introspection/IERC165Upgradeable.sol\\\";\\n\\n/**\\n * @dev Interface for the NFT Royalty Standard.\\n *\\n * A standardized way to retrieve royalty payment information for non-fungible tokens (NFTs) to enable universal\\n * support for royalty payments across all NFT marketplaces and ecosystem participants.\\n *\\n * _Available since v4.5._\\n */\\ninterface IERC2981Upgradeable is IERC165Upgradeable {\\n /**\\n * @dev Returns how much royalty is owed and to whom, based on a sale price that may be denominated in any unit of\\n * exchange. The royalty amount is denominated and should be paid in that same unit of exchange.\\n */\\n function royaltyInfo(uint256 tokenId, uint256 salePrice)\\n external\\n view\\n returns (address receiver, uint256 royaltyAmount);\\n}\\n\",\"keccak256\":\"0xa8ff557539dcfed5706eddde2aa929e06bb1764e71aa8c1048a78970bf3ca37d\",\"license\":\"MIT\"},\"@openzeppelin/contracts-upgradeable/proxy/ClonesUpgradeable.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n// OpenZeppelin Contracts (last updated v4.7.0) (proxy/Clones.sol)\\n\\npragma solidity ^0.8.0;\\n\\n/**\\n * @dev https://eips.ethereum.org/EIPS/eip-1167[EIP 1167] is a standard for\\n * deploying minimal proxy contracts, also known as \\\"clones\\\".\\n *\\n * > To simply and cheaply clone contract functionality in an immutable way, this standard specifies\\n * > a minimal bytecode implementation that delegates all calls to a known, fixed address.\\n *\\n * The library includes functions to deploy a proxy using either `create` (traditional deployment) or `create2`\\n * (salted deterministic deployment). It also includes functions to predict the addresses of clones deployed using the\\n * deterministic method.\\n *\\n * _Available since v3.4._\\n */\\nlibrary ClonesUpgradeable {\\n /**\\n * @dev Deploys and returns the address of a clone that mimics the behaviour of `implementation`.\\n *\\n * This function uses the create opcode, which should never revert.\\n */\\n function clone(address implementation) internal returns (address instance) {\\n /// @solidity memory-safe-assembly\\n assembly {\\n let ptr := mload(0x40)\\n mstore(ptr, 0x3d602d80600a3d3981f3363d3d373d3d3d363d73000000000000000000000000)\\n mstore(add(ptr, 0x14), shl(0x60, implementation))\\n mstore(add(ptr, 0x28), 0x5af43d82803e903d91602b57fd5bf30000000000000000000000000000000000)\\n instance := create(0, ptr, 0x37)\\n }\\n require(instance != address(0), \\\"ERC1167: create failed\\\");\\n }\\n\\n /**\\n * @dev Deploys and returns the address of a clone that mimics the behaviour of `implementation`.\\n *\\n * This function uses the create2 opcode and a `salt` to deterministically deploy\\n * the clone. Using the same `implementation` and `salt` multiple time will revert, since\\n * the clones cannot be deployed twice at the same address.\\n */\\n function cloneDeterministic(address implementation, bytes32 salt) internal returns (address instance) {\\n /// @solidity memory-safe-assembly\\n assembly {\\n let ptr := mload(0x40)\\n mstore(ptr, 0x3d602d80600a3d3981f3363d3d373d3d3d363d73000000000000000000000000)\\n mstore(add(ptr, 0x14), shl(0x60, implementation))\\n mstore(add(ptr, 0x28), 0x5af43d82803e903d91602b57fd5bf30000000000000000000000000000000000)\\n instance := create2(0, ptr, 0x37, salt)\\n }\\n require(instance != address(0), \\\"ERC1167: create2 failed\\\");\\n }\\n\\n /**\\n * @dev Computes the address of a clone deployed using {Clones-cloneDeterministic}.\\n */\\n function predictDeterministicAddress(\\n address implementation,\\n bytes32 salt,\\n address deployer\\n ) internal pure returns (address predicted) {\\n /// @solidity memory-safe-assembly\\n assembly {\\n let ptr := mload(0x40)\\n mstore(ptr, 0x3d602d80600a3d3981f3363d3d373d3d3d363d73000000000000000000000000)\\n mstore(add(ptr, 0x14), shl(0x60, implementation))\\n mstore(add(ptr, 0x28), 0x5af43d82803e903d91602b57fd5bf3ff00000000000000000000000000000000)\\n mstore(add(ptr, 0x38), shl(0x60, deployer))\\n mstore(add(ptr, 0x4c), salt)\\n mstore(add(ptr, 0x6c), keccak256(ptr, 0x37))\\n predicted := keccak256(add(ptr, 0x37), 0x55)\\n }\\n }\\n\\n /**\\n * @dev Computes the address of a clone deployed using {Clones-cloneDeterministic}.\\n */\\n function predictDeterministicAddress(address implementation, bytes32 salt)\\n internal\\n view\\n returns (address predicted)\\n {\\n return predictDeterministicAddress(implementation, salt, address(this));\\n }\\n}\\n\",\"keccak256\":\"0x3734e36dc4de32780bfd344a94b85b6aab985fd6e42672983ed251ba7754a2e0\",\"license\":\"MIT\"},\"@openzeppelin/contracts-upgradeable/proxy/utils/Initializable.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n// OpenZeppelin Contracts (last updated v4.7.0) (proxy/utils/Initializable.sol)\\n\\npragma solidity ^0.8.2;\\n\\nimport \\\"../../utils/AddressUpgradeable.sol\\\";\\n\\n/**\\n * @dev This is a base contract to aid in writing upgradeable contracts, or any kind of contract that will be deployed\\n * behind a proxy. Since proxied contracts do not make use of a constructor, it's common to move constructor logic to an\\n * external initializer function, usually called `initialize`. It then becomes necessary to protect this initializer\\n * function so it can only be called once. The {initializer} modifier provided by this contract will have this effect.\\n *\\n * The initialization functions use a version number. Once a version number is used, it is consumed and cannot be\\n * reused. This mechanism prevents re-execution of each \\\"step\\\" but allows the creation of new initialization steps in\\n * case an upgrade adds a module that needs to be initialized.\\n *\\n * For example:\\n *\\n * [.hljs-theme-light.nopadding]\\n * ```\\n * contract MyToken is ERC20Upgradeable {\\n * function initialize() initializer public {\\n * __ERC20_init(\\\"MyToken\\\", \\\"MTK\\\");\\n * }\\n * }\\n * contract MyTokenV2 is MyToken, ERC20PermitUpgradeable {\\n * function initializeV2() reinitializer(2) public {\\n * __ERC20Permit_init(\\\"MyToken\\\");\\n * }\\n * }\\n * ```\\n *\\n * TIP: To avoid leaving the proxy in an uninitialized state, the initializer function should be called as early as\\n * possible by providing the encoded function call as the `_data` argument to {ERC1967Proxy-constructor}.\\n *\\n * CAUTION: When used with inheritance, manual care must be taken to not invoke a parent initializer twice, or to ensure\\n * that all initializers are idempotent. This is not verified automatically as constructors are by Solidity.\\n *\\n * [CAUTION]\\n * ====\\n * Avoid leaving a contract uninitialized.\\n *\\n * An uninitialized contract can be taken over by an attacker. This applies to both a proxy and its implementation\\n * contract, which may impact the proxy. To prevent the implementation contract from being used, you should invoke\\n * the {_disableInitializers} function in the constructor to automatically lock it when it is deployed:\\n *\\n * [.hljs-theme-light.nopadding]\\n * ```\\n * /// @custom:oz-upgrades-unsafe-allow constructor\\n * constructor() {\\n * _disableInitializers();\\n * }\\n * ```\\n * ====\\n */\\nabstract contract Initializable {\\n /**\\n * @dev Indicates that the contract has been initialized.\\n * @custom:oz-retyped-from bool\\n */\\n uint8 private _initialized;\\n\\n /**\\n * @dev Indicates that the contract is in the process of being initialized.\\n */\\n bool private _initializing;\\n\\n /**\\n * @dev Triggered when the contract has been initialized or reinitialized.\\n */\\n event Initialized(uint8 version);\\n\\n /**\\n * @dev A modifier that defines a protected initializer function that can be invoked at most once. In its scope,\\n * `onlyInitializing` functions can be used to initialize parent contracts. Equivalent to `reinitializer(1)`.\\n */\\n modifier initializer() {\\n bool isTopLevelCall = !_initializing;\\n require(\\n (isTopLevelCall && _initialized < 1) || (!AddressUpgradeable.isContract(address(this)) && _initialized == 1),\\n \\\"Initializable: contract is already initialized\\\"\\n );\\n _initialized = 1;\\n if (isTopLevelCall) {\\n _initializing = true;\\n }\\n _;\\n if (isTopLevelCall) {\\n _initializing = false;\\n emit Initialized(1);\\n }\\n }\\n\\n /**\\n * @dev A modifier that defines a protected reinitializer function that can be invoked at most once, and only if the\\n * contract hasn't been initialized to a greater version before. In its scope, `onlyInitializing` functions can be\\n * used to initialize parent contracts.\\n *\\n * `initializer` is equivalent to `reinitializer(1)`, so a reinitializer may be used after the original\\n * initialization step. This is essential to configure modules that are added through upgrades and that require\\n * initialization.\\n *\\n * Note that versions can jump in increments greater than 1; this implies that if multiple reinitializers coexist in\\n * a contract, executing them in the right order is up to the developer or operator.\\n */\\n modifier reinitializer(uint8 version) {\\n require(!_initializing && _initialized < version, \\\"Initializable: contract is already initialized\\\");\\n _initialized = version;\\n _initializing = true;\\n _;\\n _initializing = false;\\n emit Initialized(version);\\n }\\n\\n /**\\n * @dev Modifier to protect an initialization function so that it can only be invoked by functions with the\\n * {initializer} and {reinitializer} modifiers, directly or indirectly.\\n */\\n modifier onlyInitializing() {\\n require(_initializing, \\\"Initializable: contract is not initializing\\\");\\n _;\\n }\\n\\n /**\\n * @dev Locks the contract, preventing any future reinitialization. This cannot be part of an initializer call.\\n * Calling this in the constructor of a contract will prevent that contract from being initialized or reinitialized\\n * to any version. It is recommended to use this to lock implementation contracts that are designed to be called\\n * through proxies.\\n */\\n function _disableInitializers() internal virtual {\\n require(!_initializing, \\\"Initializable: contract is initializing\\\");\\n if (_initialized < type(uint8).max) {\\n _initialized = type(uint8).max;\\n emit Initialized(type(uint8).max);\\n }\\n }\\n}\\n\",\"keccak256\":\"0x0203dcadc5737d9ef2c211d6fa15d18ebc3b30dfa51903b64870b01a062b0b4e\",\"license\":\"MIT\"},\"@openzeppelin/contracts-upgradeable/token/ERC20/IERC20Upgradeable.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n// OpenZeppelin Contracts (last updated v4.6.0) (token/ERC20/IERC20.sol)\\n\\npragma solidity ^0.8.0;\\n\\n/**\\n * @dev Interface of the ERC20 standard as defined in the EIP.\\n */\\ninterface IERC20Upgradeable {\\n /**\\n * @dev Emitted when `value` tokens are moved from one account (`from`) to\\n * another (`to`).\\n *\\n * Note that `value` may be zero.\\n */\\n event Transfer(address indexed from, address indexed to, uint256 value);\\n\\n /**\\n * @dev Emitted when the allowance of a `spender` for an `owner` is set by\\n * a call to {approve}. `value` is the new allowance.\\n */\\n event Approval(address indexed owner, address indexed spender, uint256 value);\\n\\n /**\\n * @dev Returns the amount of tokens in existence.\\n */\\n function totalSupply() external view returns (uint256);\\n\\n /**\\n * @dev Returns the amount of tokens owned by `account`.\\n */\\n function balanceOf(address account) external view returns (uint256);\\n\\n /**\\n * @dev Moves `amount` tokens from the caller's account to `to`.\\n *\\n * Returns a boolean value indicating whether the operation succeeded.\\n *\\n * Emits a {Transfer} event.\\n */\\n function transfer(address to, uint256 amount) external returns (bool);\\n\\n /**\\n * @dev Returns the remaining number of tokens that `spender` will be\\n * allowed to spend on behalf of `owner` through {transferFrom}. This is\\n * zero by default.\\n *\\n * This value changes when {approve} or {transferFrom} are called.\\n */\\n function allowance(address owner, address spender) external view returns (uint256);\\n\\n /**\\n * @dev Sets `amount` as the allowance of `spender` over the caller's tokens.\\n *\\n * Returns a boolean value indicating whether the operation succeeded.\\n *\\n * IMPORTANT: Beware that changing an allowance with this method brings the risk\\n * that someone may use both the old and the new allowance by unfortunate\\n * transaction ordering. One possible solution to mitigate this race\\n * condition is to first reduce the spender's allowance to 0 and set the\\n * desired value afterwards:\\n * https://github.com/ethereum/EIPs/issues/20#issuecomment-263524729\\n *\\n * Emits an {Approval} event.\\n */\\n function approve(address spender, uint256 amount) external returns (bool);\\n\\n /**\\n * @dev Moves `amount` tokens from `from` to `to` using the\\n * allowance mechanism. `amount` is then deducted from the caller's\\n * allowance.\\n *\\n * Returns a boolean value indicating whether the operation succeeded.\\n *\\n * Emits a {Transfer} event.\\n */\\n function transferFrom(\\n address from,\\n address to,\\n uint256 amount\\n ) external returns (bool);\\n}\\n\",\"keccak256\":\"0x4e733d3164f73f461eaf9d8087a7ad1ea180bdc8ba0d3d61b0e1ae16d8e63dff\",\"license\":\"MIT\"},\"@openzeppelin/contracts-upgradeable/token/ERC721/ERC721Upgradeable.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n// OpenZeppelin Contracts (last updated v4.7.0) (token/ERC721/ERC721.sol)\\n\\npragma solidity ^0.8.0;\\n\\nimport \\\"./IERC721Upgradeable.sol\\\";\\nimport \\\"./IERC721ReceiverUpgradeable.sol\\\";\\nimport \\\"./extensions/IERC721MetadataUpgradeable.sol\\\";\\nimport \\\"../../utils/AddressUpgradeable.sol\\\";\\nimport \\\"../../utils/ContextUpgradeable.sol\\\";\\nimport \\\"../../utils/StringsUpgradeable.sol\\\";\\nimport \\\"../../utils/introspection/ERC165Upgradeable.sol\\\";\\nimport \\\"../../proxy/utils/Initializable.sol\\\";\\n\\n/**\\n * @dev Implementation of https://eips.ethereum.org/EIPS/eip-721[ERC721] Non-Fungible Token Standard, including\\n * the Metadata extension, but not including the Enumerable extension, which is available separately as\\n * {ERC721Enumerable}.\\n */\\ncontract ERC721Upgradeable is Initializable, ContextUpgradeable, ERC165Upgradeable, IERC721Upgradeable, IERC721MetadataUpgradeable {\\n using AddressUpgradeable for address;\\n using StringsUpgradeable for uint256;\\n\\n // Token name\\n string private _name;\\n\\n // Token symbol\\n string private _symbol;\\n\\n // Mapping from token ID to owner address\\n mapping(uint256 => address) private _owners;\\n\\n // Mapping owner address to token count\\n mapping(address => uint256) private _balances;\\n\\n // Mapping from token ID to approved address\\n mapping(uint256 => address) private _tokenApprovals;\\n\\n // Mapping from owner to operator approvals\\n mapping(address => mapping(address => bool)) private _operatorApprovals;\\n\\n /**\\n * @dev Initializes the contract by setting a `name` and a `symbol` to the token collection.\\n */\\n function __ERC721_init(string memory name_, string memory symbol_) internal onlyInitializing {\\n __ERC721_init_unchained(name_, symbol_);\\n }\\n\\n function __ERC721_init_unchained(string memory name_, string memory symbol_) internal onlyInitializing {\\n _name = name_;\\n _symbol = symbol_;\\n }\\n\\n /**\\n * @dev See {IERC165-supportsInterface}.\\n */\\n function supportsInterface(bytes4 interfaceId) public view virtual override(ERC165Upgradeable, IERC165Upgradeable) returns (bool) {\\n return\\n interfaceId == type(IERC721Upgradeable).interfaceId ||\\n interfaceId == type(IERC721MetadataUpgradeable).interfaceId ||\\n super.supportsInterface(interfaceId);\\n }\\n\\n /**\\n * @dev See {IERC721-balanceOf}.\\n */\\n function balanceOf(address owner) public view virtual override returns (uint256) {\\n require(owner != address(0), \\\"ERC721: address zero is not a valid owner\\\");\\n return _balances[owner];\\n }\\n\\n /**\\n * @dev See {IERC721-ownerOf}.\\n */\\n function ownerOf(uint256 tokenId) public view virtual override returns (address) {\\n address owner = _owners[tokenId];\\n require(owner != address(0), \\\"ERC721: invalid token ID\\\");\\n return owner;\\n }\\n\\n /**\\n * @dev See {IERC721Metadata-name}.\\n */\\n function name() public view virtual override returns (string memory) {\\n return _name;\\n }\\n\\n /**\\n * @dev See {IERC721Metadata-symbol}.\\n */\\n function symbol() public view virtual override returns (string memory) {\\n return _symbol;\\n }\\n\\n /**\\n * @dev See {IERC721Metadata-tokenURI}.\\n */\\n function tokenURI(uint256 tokenId) public view virtual override returns (string memory) {\\n _requireMinted(tokenId);\\n\\n string memory baseURI = _baseURI();\\n return bytes(baseURI).length > 0 ? string(abi.encodePacked(baseURI, tokenId.toString())) : \\\"\\\";\\n }\\n\\n /**\\n * @dev Base URI for computing {tokenURI}. If set, the resulting URI for each\\n * token will be the concatenation of the `baseURI` and the `tokenId`. Empty\\n * by default, can be overridden in child contracts.\\n */\\n function _baseURI() internal view virtual returns (string memory) {\\n return \\\"\\\";\\n }\\n\\n /**\\n * @dev See {IERC721-approve}.\\n */\\n function approve(address to, uint256 tokenId) public virtual override {\\n address owner = ERC721Upgradeable.ownerOf(tokenId);\\n require(to != owner, \\\"ERC721: approval to current owner\\\");\\n\\n require(\\n _msgSender() == owner || isApprovedForAll(owner, _msgSender()),\\n \\\"ERC721: approve caller is not token owner nor approved for all\\\"\\n );\\n\\n _approve(to, tokenId);\\n }\\n\\n /**\\n * @dev See {IERC721-getApproved}.\\n */\\n function getApproved(uint256 tokenId) public view virtual override returns (address) {\\n _requireMinted(tokenId);\\n\\n return _tokenApprovals[tokenId];\\n }\\n\\n /**\\n * @dev See {IERC721-setApprovalForAll}.\\n */\\n function setApprovalForAll(address operator, bool approved) public virtual override {\\n _setApprovalForAll(_msgSender(), operator, approved);\\n }\\n\\n /**\\n * @dev See {IERC721-isApprovedForAll}.\\n */\\n function isApprovedForAll(address owner, address operator) public view virtual override returns (bool) {\\n return _operatorApprovals[owner][operator];\\n }\\n\\n /**\\n * @dev See {IERC721-transferFrom}.\\n */\\n function transferFrom(\\n address from,\\n address to,\\n uint256 tokenId\\n ) public virtual override {\\n //solhint-disable-next-line max-line-length\\n require(_isApprovedOrOwner(_msgSender(), tokenId), \\\"ERC721: caller is not token owner nor approved\\\");\\n\\n _transfer(from, to, tokenId);\\n }\\n\\n /**\\n * @dev See {IERC721-safeTransferFrom}.\\n */\\n function safeTransferFrom(\\n address from,\\n address to,\\n uint256 tokenId\\n ) public virtual override {\\n safeTransferFrom(from, to, tokenId, \\\"\\\");\\n }\\n\\n /**\\n * @dev See {IERC721-safeTransferFrom}.\\n */\\n function safeTransferFrom(\\n address from,\\n address to,\\n uint256 tokenId,\\n bytes memory data\\n ) public virtual override {\\n require(_isApprovedOrOwner(_msgSender(), tokenId), \\\"ERC721: caller is not token owner nor approved\\\");\\n _safeTransfer(from, to, tokenId, data);\\n }\\n\\n /**\\n * @dev Safely transfers `tokenId` token from `from` to `to`, checking first that contract recipients\\n * are aware of the ERC721 protocol to prevent tokens from being forever locked.\\n *\\n * `data` is additional data, it has no specified format and it is sent in call to `to`.\\n *\\n * This internal function is equivalent to {safeTransferFrom}, and can be used to e.g.\\n * implement alternative mechanisms to perform token transfer, such as signature-based.\\n *\\n * Requirements:\\n *\\n * - `from` cannot be the zero address.\\n * - `to` cannot be the zero address.\\n * - `tokenId` token must exist and be owned by `from`.\\n * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer.\\n *\\n * Emits a {Transfer} event.\\n */\\n function _safeTransfer(\\n address from,\\n address to,\\n uint256 tokenId,\\n bytes memory data\\n ) internal virtual {\\n _transfer(from, to, tokenId);\\n require(_checkOnERC721Received(from, to, tokenId, data), \\\"ERC721: transfer to non ERC721Receiver implementer\\\");\\n }\\n\\n /**\\n * @dev Returns whether `tokenId` exists.\\n *\\n * Tokens can be managed by their owner or approved accounts via {approve} or {setApprovalForAll}.\\n *\\n * Tokens start existing when they are minted (`_mint`),\\n * and stop existing when they are burned (`_burn`).\\n */\\n function _exists(uint256 tokenId) internal view virtual returns (bool) {\\n return _owners[tokenId] != address(0);\\n }\\n\\n /**\\n * @dev Returns whether `spender` is allowed to manage `tokenId`.\\n *\\n * Requirements:\\n *\\n * - `tokenId` must exist.\\n */\\n function _isApprovedOrOwner(address spender, uint256 tokenId) internal view virtual returns (bool) {\\n address owner = ERC721Upgradeable.ownerOf(tokenId);\\n return (spender == owner || isApprovedForAll(owner, spender) || getApproved(tokenId) == spender);\\n }\\n\\n /**\\n * @dev Safely mints `tokenId` and transfers it to `to`.\\n *\\n * Requirements:\\n *\\n * - `tokenId` must not exist.\\n * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer.\\n *\\n * Emits a {Transfer} event.\\n */\\n function _safeMint(address to, uint256 tokenId) internal virtual {\\n _safeMint(to, tokenId, \\\"\\\");\\n }\\n\\n /**\\n * @dev Same as {xref-ERC721-_safeMint-address-uint256-}[`_safeMint`], with an additional `data` parameter which is\\n * forwarded in {IERC721Receiver-onERC721Received} to contract recipients.\\n */\\n function _safeMint(\\n address to,\\n uint256 tokenId,\\n bytes memory data\\n ) internal virtual {\\n _mint(to, tokenId);\\n require(\\n _checkOnERC721Received(address(0), to, tokenId, data),\\n \\\"ERC721: transfer to non ERC721Receiver implementer\\\"\\n );\\n }\\n\\n /**\\n * @dev Mints `tokenId` and transfers it to `to`.\\n *\\n * WARNING: Usage of this method is discouraged, use {_safeMint} whenever possible\\n *\\n * Requirements:\\n *\\n * - `tokenId` must not exist.\\n * - `to` cannot be the zero address.\\n *\\n * Emits a {Transfer} event.\\n */\\n function _mint(address to, uint256 tokenId) internal virtual {\\n require(to != address(0), \\\"ERC721: mint to the zero address\\\");\\n require(!_exists(tokenId), \\\"ERC721: token already minted\\\");\\n\\n _beforeTokenTransfer(address(0), to, tokenId);\\n\\n _balances[to] += 1;\\n _owners[tokenId] = to;\\n\\n emit Transfer(address(0), to, tokenId);\\n\\n _afterTokenTransfer(address(0), to, tokenId);\\n }\\n\\n /**\\n * @dev Destroys `tokenId`.\\n * The approval is cleared when the token is burned.\\n *\\n * Requirements:\\n *\\n * - `tokenId` must exist.\\n *\\n * Emits a {Transfer} event.\\n */\\n function _burn(uint256 tokenId) internal virtual {\\n address owner = ERC721Upgradeable.ownerOf(tokenId);\\n\\n _beforeTokenTransfer(owner, address(0), tokenId);\\n\\n // Clear approvals\\n _approve(address(0), tokenId);\\n\\n _balances[owner] -= 1;\\n delete _owners[tokenId];\\n\\n emit Transfer(owner, address(0), tokenId);\\n\\n _afterTokenTransfer(owner, address(0), tokenId);\\n }\\n\\n /**\\n * @dev Transfers `tokenId` from `from` to `to`.\\n * As opposed to {transferFrom}, this imposes no restrictions on msg.sender.\\n *\\n * Requirements:\\n *\\n * - `to` cannot be the zero address.\\n * - `tokenId` token must be owned by `from`.\\n *\\n * Emits a {Transfer} event.\\n */\\n function _transfer(\\n address from,\\n address to,\\n uint256 tokenId\\n ) internal virtual {\\n require(ERC721Upgradeable.ownerOf(tokenId) == from, \\\"ERC721: transfer from incorrect owner\\\");\\n require(to != address(0), \\\"ERC721: transfer to the zero address\\\");\\n\\n _beforeTokenTransfer(from, to, tokenId);\\n\\n // Clear approvals from the previous owner\\n _approve(address(0), tokenId);\\n\\n _balances[from] -= 1;\\n _balances[to] += 1;\\n _owners[tokenId] = to;\\n\\n emit Transfer(from, to, tokenId);\\n\\n _afterTokenTransfer(from, to, tokenId);\\n }\\n\\n /**\\n * @dev Approve `to` to operate on `tokenId`\\n *\\n * Emits an {Approval} event.\\n */\\n function _approve(address to, uint256 tokenId) internal virtual {\\n _tokenApprovals[tokenId] = to;\\n emit Approval(ERC721Upgradeable.ownerOf(tokenId), to, tokenId);\\n }\\n\\n /**\\n * @dev Approve `operator` to operate on all of `owner` tokens\\n *\\n * Emits an {ApprovalForAll} event.\\n */\\n function _setApprovalForAll(\\n address owner,\\n address operator,\\n bool approved\\n ) internal virtual {\\n require(owner != operator, \\\"ERC721: approve to caller\\\");\\n _operatorApprovals[owner][operator] = approved;\\n emit ApprovalForAll(owner, operator, approved);\\n }\\n\\n /**\\n * @dev Reverts if the `tokenId` has not been minted yet.\\n */\\n function _requireMinted(uint256 tokenId) internal view virtual {\\n require(_exists(tokenId), \\\"ERC721: invalid token ID\\\");\\n }\\n\\n /**\\n * @dev Internal function to invoke {IERC721Receiver-onERC721Received} on a target address.\\n * The call is not executed if the target address is not a contract.\\n *\\n * @param from address representing the previous owner of the given token ID\\n * @param to target address that will receive the tokens\\n * @param tokenId uint256 ID of the token to be transferred\\n * @param data bytes optional data to send along with the call\\n * @return bool whether the call correctly returned the expected magic value\\n */\\n function _checkOnERC721Received(\\n address from,\\n address to,\\n uint256 tokenId,\\n bytes memory data\\n ) private returns (bool) {\\n if (to.isContract()) {\\n try IERC721ReceiverUpgradeable(to).onERC721Received(_msgSender(), from, tokenId, data) returns (bytes4 retval) {\\n return retval == IERC721ReceiverUpgradeable.onERC721Received.selector;\\n } catch (bytes memory reason) {\\n if (reason.length == 0) {\\n revert(\\\"ERC721: transfer to non ERC721Receiver implementer\\\");\\n } else {\\n /// @solidity memory-safe-assembly\\n assembly {\\n revert(add(32, reason), mload(reason))\\n }\\n }\\n }\\n } else {\\n return true;\\n }\\n }\\n\\n /**\\n * @dev Hook that is called before any token transfer. This includes minting\\n * and burning.\\n *\\n * Calling conditions:\\n *\\n * - When `from` and `to` are both non-zero, ``from``'s `tokenId` will be\\n * transferred to `to`.\\n * - When `from` is zero, `tokenId` will be minted for `to`.\\n * - When `to` is zero, ``from``'s `tokenId` will be burned.\\n * - `from` and `to` are never both zero.\\n *\\n * To learn more about hooks, head to xref:ROOT:extending-contracts.adoc#using-hooks[Using Hooks].\\n */\\n function _beforeTokenTransfer(\\n address from,\\n address to,\\n uint256 tokenId\\n ) internal virtual {}\\n\\n /**\\n * @dev Hook that is called after any transfer of tokens. This includes\\n * minting and burning.\\n *\\n * Calling conditions:\\n *\\n * - when `from` and `to` are both non-zero.\\n * - `from` and `to` are never both zero.\\n *\\n * To learn more about hooks, head to xref:ROOT:extending-contracts.adoc#using-hooks[Using Hooks].\\n */\\n function _afterTokenTransfer(\\n address from,\\n address to,\\n uint256 tokenId\\n ) internal virtual {}\\n\\n /**\\n * @dev This empty reserved space is put in place to allow future versions to add new\\n * variables without shifting down storage in the inheritance chain.\\n * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps\\n */\\n uint256[44] private __gap;\\n}\\n\",\"keccak256\":\"0x5331c8909221d9f9f3851cfadd5959d0873413a2c27e30e0f2fa234158c1c6cf\",\"license\":\"MIT\"},\"@openzeppelin/contracts-upgradeable/token/ERC721/IERC721ReceiverUpgradeable.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n// OpenZeppelin Contracts (last updated v4.6.0) (token/ERC721/IERC721Receiver.sol)\\n\\npragma solidity ^0.8.0;\\n\\n/**\\n * @title ERC721 token receiver interface\\n * @dev Interface for any contract that wants to support safeTransfers\\n * from ERC721 asset contracts.\\n */\\ninterface IERC721ReceiverUpgradeable {\\n /**\\n * @dev Whenever an {IERC721} `tokenId` token is transferred to this contract via {IERC721-safeTransferFrom}\\n * by `operator` from `from`, this function is called.\\n *\\n * It must return its Solidity selector to confirm the token transfer.\\n * If any other value is returned or the interface is not implemented by the recipient, the transfer will be reverted.\\n *\\n * The selector can be obtained in Solidity with `IERC721Receiver.onERC721Received.selector`.\\n */\\n function onERC721Received(\\n address operator,\\n address from,\\n uint256 tokenId,\\n bytes calldata data\\n ) external returns (bytes4);\\n}\\n\",\"keccak256\":\"0xbb2ed8106d94aeae6858e2551a1e7174df73994b77b13ebd120ccaaef80155f5\",\"license\":\"MIT\"},\"@openzeppelin/contracts-upgradeable/token/ERC721/IERC721Upgradeable.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n// OpenZeppelin Contracts (last updated v4.7.0) (token/ERC721/IERC721.sol)\\n\\npragma solidity ^0.8.0;\\n\\nimport \\\"../../utils/introspection/IERC165Upgradeable.sol\\\";\\n\\n/**\\n * @dev Required interface of an ERC721 compliant contract.\\n */\\ninterface IERC721Upgradeable is IERC165Upgradeable {\\n /**\\n * @dev Emitted when `tokenId` token is transferred from `from` to `to`.\\n */\\n event Transfer(address indexed from, address indexed to, uint256 indexed tokenId);\\n\\n /**\\n * @dev Emitted when `owner` enables `approved` to manage the `tokenId` token.\\n */\\n event Approval(address indexed owner, address indexed approved, uint256 indexed tokenId);\\n\\n /**\\n * @dev Emitted when `owner` enables or disables (`approved`) `operator` to manage all of its assets.\\n */\\n event ApprovalForAll(address indexed owner, address indexed operator, bool approved);\\n\\n /**\\n * @dev Returns the number of tokens in ``owner``'s account.\\n */\\n function balanceOf(address owner) external view returns (uint256 balance);\\n\\n /**\\n * @dev Returns the owner of the `tokenId` token.\\n *\\n * Requirements:\\n *\\n * - `tokenId` must exist.\\n */\\n function ownerOf(uint256 tokenId) external view returns (address owner);\\n\\n /**\\n * @dev Safely transfers `tokenId` token from `from` to `to`.\\n *\\n * Requirements:\\n *\\n * - `from` cannot be the zero address.\\n * - `to` cannot be the zero address.\\n * - `tokenId` token must exist and be owned by `from`.\\n * - If the caller is not `from`, it must be approved to move this token by either {approve} or {setApprovalForAll}.\\n * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer.\\n *\\n * Emits a {Transfer} event.\\n */\\n function safeTransferFrom(\\n address from,\\n address to,\\n uint256 tokenId,\\n bytes calldata data\\n ) external;\\n\\n /**\\n * @dev Safely transfers `tokenId` token from `from` to `to`, checking first that contract recipients\\n * are aware of the ERC721 protocol to prevent tokens from being forever locked.\\n *\\n * Requirements:\\n *\\n * - `from` cannot be the zero address.\\n * - `to` cannot be the zero address.\\n * - `tokenId` token must exist and be owned by `from`.\\n * - If the caller is not `from`, it must have been allowed to move this token by either {approve} or {setApprovalForAll}.\\n * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer.\\n *\\n * Emits a {Transfer} event.\\n */\\n function safeTransferFrom(\\n address from,\\n address to,\\n uint256 tokenId\\n ) external;\\n\\n /**\\n * @dev Transfers `tokenId` token from `from` to `to`.\\n *\\n * WARNING: Usage of this method is discouraged, use {safeTransferFrom} whenever possible.\\n *\\n * Requirements:\\n *\\n * - `from` cannot be the zero address.\\n * - `to` cannot be the zero address.\\n * - `tokenId` token must be owned by `from`.\\n * - If the caller is not `from`, it must be approved to move this token by either {approve} or {setApprovalForAll}.\\n *\\n * Emits a {Transfer} event.\\n */\\n function transferFrom(\\n address from,\\n address to,\\n uint256 tokenId\\n ) external;\\n\\n /**\\n * @dev Gives permission to `to` to transfer `tokenId` token to another account.\\n * The approval is cleared when the token is transferred.\\n *\\n * Only a single account can be approved at a time, so approving the zero address clears previous approvals.\\n *\\n * Requirements:\\n *\\n * - The caller must own the token or be an approved operator.\\n * - `tokenId` must exist.\\n *\\n * Emits an {Approval} event.\\n */\\n function approve(address to, uint256 tokenId) external;\\n\\n /**\\n * @dev Approve or remove `operator` as an operator for the caller.\\n * Operators can call {transferFrom} or {safeTransferFrom} for any token owned by the caller.\\n *\\n * Requirements:\\n *\\n * - The `operator` cannot be the caller.\\n *\\n * Emits an {ApprovalForAll} event.\\n */\\n function setApprovalForAll(address operator, bool _approved) external;\\n\\n /**\\n * @dev Returns the account approved for `tokenId` token.\\n *\\n * Requirements:\\n *\\n * - `tokenId` must exist.\\n */\\n function getApproved(uint256 tokenId) external view returns (address operator);\\n\\n /**\\n * @dev Returns if the `operator` is allowed to manage all of the assets of `owner`.\\n *\\n * See {setApprovalForAll}\\n */\\n function isApprovedForAll(address owner, address operator) external view returns (bool);\\n}\\n\",\"keccak256\":\"0x016298e66a5810253c6c905e61966bb31c8775c3f3517bf946ff56ee31d6c005\",\"license\":\"MIT\"},\"@openzeppelin/contracts-upgradeable/token/ERC721/extensions/IERC721MetadataUpgradeable.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n// OpenZeppelin Contracts v4.4.1 (token/ERC721/extensions/IERC721Metadata.sol)\\n\\npragma solidity ^0.8.0;\\n\\nimport \\\"../IERC721Upgradeable.sol\\\";\\n\\n/**\\n * @title ERC-721 Non-Fungible Token Standard, optional metadata extension\\n * @dev See https://eips.ethereum.org/EIPS/eip-721\\n */\\ninterface IERC721MetadataUpgradeable is IERC721Upgradeable {\\n /**\\n * @dev Returns the token collection name.\\n */\\n function name() external view returns (string memory);\\n\\n /**\\n * @dev Returns the token collection symbol.\\n */\\n function symbol() external view returns (string memory);\\n\\n /**\\n * @dev Returns the Uniform Resource Identifier (URI) for `tokenId` token.\\n */\\n function tokenURI(uint256 tokenId) external view returns (string memory);\\n}\\n\",\"keccak256\":\"0x95a471796eb5f030fdc438660bebec121ad5d063763e64d92376ffb4b5ce8b70\",\"license\":\"MIT\"},\"@openzeppelin/contracts-upgradeable/utils/AddressUpgradeable.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n// OpenZeppelin Contracts (last updated v4.7.0) (utils/Address.sol)\\n\\npragma solidity ^0.8.1;\\n\\n/**\\n * @dev Collection of functions related to the address type\\n */\\nlibrary AddressUpgradeable {\\n /**\\n * @dev Returns true if `account` is a contract.\\n *\\n * [IMPORTANT]\\n * ====\\n * It is unsafe to assume that an address for which this function returns\\n * false is an externally-owned account (EOA) and not a contract.\\n *\\n * Among others, `isContract` will return false for the following\\n * types of addresses:\\n *\\n * - an externally-owned account\\n * - a contract in construction\\n * - an address where a contract will be created\\n * - an address where a contract lived, but was destroyed\\n * ====\\n *\\n * [IMPORTANT]\\n * ====\\n * You shouldn't rely on `isContract` to protect against flash loan attacks!\\n *\\n * Preventing calls from contracts is highly discouraged. It breaks composability, breaks support for smart wallets\\n * like Gnosis Safe, and does not provide security since it can be circumvented by calling from a contract\\n * constructor.\\n * ====\\n */\\n function isContract(address account) internal view returns (bool) {\\n // This method relies on extcodesize/address.code.length, which returns 0\\n // for contracts in construction, since the code is only stored at the end\\n // of the constructor execution.\\n\\n return account.code.length > 0;\\n }\\n\\n /**\\n * @dev Replacement for Solidity's `transfer`: sends `amount` wei to\\n * `recipient`, forwarding all available gas and reverting on errors.\\n *\\n * https://eips.ethereum.org/EIPS/eip-1884[EIP1884] increases the gas cost\\n * of certain opcodes, possibly making contracts go over the 2300 gas limit\\n * imposed by `transfer`, making them unable to receive funds via\\n * `transfer`. {sendValue} removes this limitation.\\n *\\n * https://diligence.consensys.net/posts/2019/09/stop-using-soliditys-transfer-now/[Learn more].\\n *\\n * IMPORTANT: because control is transferred to `recipient`, care must be\\n * taken to not create reentrancy vulnerabilities. Consider using\\n * {ReentrancyGuard} or the\\n * https://solidity.readthedocs.io/en/v0.5.11/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern].\\n */\\n function sendValue(address payable recipient, uint256 amount) internal {\\n require(address(this).balance >= amount, \\\"Address: insufficient balance\\\");\\n\\n (bool success, ) = recipient.call{value: amount}(\\\"\\\");\\n require(success, \\\"Address: unable to send value, recipient may have reverted\\\");\\n }\\n\\n /**\\n * @dev Performs a Solidity function call using a low level `call`. A\\n * plain `call` is an unsafe replacement for a function call: use this\\n * function instead.\\n *\\n * If `target` reverts with a revert reason, it is bubbled up by this\\n * function (like regular Solidity function calls).\\n *\\n * Returns the raw returned data. To convert to the expected return value,\\n * use https://solidity.readthedocs.io/en/latest/units-and-global-variables.html?highlight=abi.decode#abi-encoding-and-decoding-functions[`abi.decode`].\\n *\\n * Requirements:\\n *\\n * - `target` must be a contract.\\n * - calling `target` with `data` must not revert.\\n *\\n * _Available since v3.1._\\n */\\n function functionCall(address target, bytes memory data) internal returns (bytes memory) {\\n return functionCall(target, data, \\\"Address: low-level call failed\\\");\\n }\\n\\n /**\\n * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], but with\\n * `errorMessage` as a fallback revert reason when `target` reverts.\\n *\\n * _Available since v3.1._\\n */\\n function functionCall(\\n address target,\\n bytes memory data,\\n string memory errorMessage\\n ) internal returns (bytes memory) {\\n return functionCallWithValue(target, data, 0, errorMessage);\\n }\\n\\n /**\\n * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],\\n * but also transferring `value` wei to `target`.\\n *\\n * Requirements:\\n *\\n * - the calling contract must have an ETH balance of at least `value`.\\n * - the called Solidity function must be `payable`.\\n *\\n * _Available since v3.1._\\n */\\n function functionCallWithValue(\\n address target,\\n bytes memory data,\\n uint256 value\\n ) internal returns (bytes memory) {\\n return functionCallWithValue(target, data, value, \\\"Address: low-level call with value failed\\\");\\n }\\n\\n /**\\n * @dev Same as {xref-Address-functionCallWithValue-address-bytes-uint256-}[`functionCallWithValue`], but\\n * with `errorMessage` as a fallback revert reason when `target` reverts.\\n *\\n * _Available since v3.1._\\n */\\n function functionCallWithValue(\\n address target,\\n bytes memory data,\\n uint256 value,\\n string memory errorMessage\\n ) internal returns (bytes memory) {\\n require(address(this).balance >= value, \\\"Address: insufficient balance for call\\\");\\n require(isContract(target), \\\"Address: call to non-contract\\\");\\n\\n (bool success, bytes memory returndata) = target.call{value: value}(data);\\n return verifyCallResult(success, returndata, errorMessage);\\n }\\n\\n /**\\n * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],\\n * but performing a static call.\\n *\\n * _Available since v3.3._\\n */\\n function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) {\\n return functionStaticCall(target, data, \\\"Address: low-level static call failed\\\");\\n }\\n\\n /**\\n * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],\\n * but performing a static call.\\n *\\n * _Available since v3.3._\\n */\\n function functionStaticCall(\\n address target,\\n bytes memory data,\\n string memory errorMessage\\n ) internal view returns (bytes memory) {\\n require(isContract(target), \\\"Address: static call to non-contract\\\");\\n\\n (bool success, bytes memory returndata) = target.staticcall(data);\\n return verifyCallResult(success, returndata, errorMessage);\\n }\\n\\n /**\\n * @dev Tool to verifies that a low level call was successful, and revert if it wasn't, either by bubbling the\\n * revert reason using the provided one.\\n *\\n * _Available since v4.3._\\n */\\n function verifyCallResult(\\n bool success,\\n bytes memory returndata,\\n string memory errorMessage\\n ) internal pure returns (bytes memory) {\\n if (success) {\\n return returndata;\\n } else {\\n // Look for revert reason and bubble it up if present\\n if (returndata.length > 0) {\\n // The easiest way to bubble the revert reason is using memory via assembly\\n /// @solidity memory-safe-assembly\\n assembly {\\n let returndata_size := mload(returndata)\\n revert(add(32, returndata), returndata_size)\\n }\\n } else {\\n revert(errorMessage);\\n }\\n }\\n }\\n}\\n\",\"keccak256\":\"0x611aa3f23e59cfdd1863c536776407b3e33d695152a266fa7cfb34440a29a8a3\",\"license\":\"MIT\"},\"@openzeppelin/contracts-upgradeable/utils/ContextUpgradeable.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n// OpenZeppelin Contracts v4.4.1 (utils/Context.sol)\\n\\npragma solidity ^0.8.0;\\nimport \\\"../proxy/utils/Initializable.sol\\\";\\n\\n/**\\n * @dev Provides information about the current execution context, including the\\n * sender of the transaction and its data. While these are generally available\\n * via msg.sender and msg.data, they should not be accessed in such a direct\\n * manner, since when dealing with meta-transactions the account sending and\\n * paying for execution may not be the actual sender (as far as an application\\n * is concerned).\\n *\\n * This contract is only required for intermediate, library-like contracts.\\n */\\nabstract contract ContextUpgradeable is Initializable {\\n function __Context_init() internal onlyInitializing {\\n }\\n\\n function __Context_init_unchained() internal onlyInitializing {\\n }\\n function _msgSender() internal view virtual returns (address) {\\n return msg.sender;\\n }\\n\\n function _msgData() internal view virtual returns (bytes calldata) {\\n return msg.data;\\n }\\n\\n /**\\n * @dev This empty reserved space is put in place to allow future versions to add new\\n * variables without shifting down storage in the inheritance chain.\\n * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps\\n */\\n uint256[50] private __gap;\\n}\\n\",\"keccak256\":\"0x963ea7f0b48b032eef72fe3a7582edf78408d6f834115b9feadd673a4d5bd149\",\"license\":\"MIT\"},\"@openzeppelin/contracts-upgradeable/utils/CountersUpgradeable.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n// OpenZeppelin Contracts v4.4.1 (utils/Counters.sol)\\n\\npragma solidity ^0.8.0;\\n\\n/**\\n * @title Counters\\n * @author Matt Condon (@shrugs)\\n * @dev Provides counters that can only be incremented, decremented or reset. This can be used e.g. to track the number\\n * of elements in a mapping, issuing ERC721 ids, or counting request ids.\\n *\\n * Include with `using Counters for Counters.Counter;`\\n */\\nlibrary CountersUpgradeable {\\n struct Counter {\\n // This variable should never be directly accessed by users of the library: interactions must be restricted to\\n // the library's function. As of Solidity v0.5.2, this cannot be enforced, though there is a proposal to add\\n // this feature: see https://github.com/ethereum/solidity/issues/4637\\n uint256 _value; // default: 0\\n }\\n\\n function current(Counter storage counter) internal view returns (uint256) {\\n return counter._value;\\n }\\n\\n function increment(Counter storage counter) internal {\\n unchecked {\\n counter._value += 1;\\n }\\n }\\n\\n function decrement(Counter storage counter) internal {\\n uint256 value = counter._value;\\n require(value > 0, \\\"Counter: decrement overflow\\\");\\n unchecked {\\n counter._value = value - 1;\\n }\\n }\\n\\n function reset(Counter storage counter) internal {\\n counter._value = 0;\\n }\\n}\\n\",\"keccak256\":\"0x798741e231b22b81e2dd2eddaaf8832dee4baf5cd8e2dbaa5c1dd12a1c053c4d\",\"license\":\"MIT\"},\"@openzeppelin/contracts-upgradeable/utils/StringsUpgradeable.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n// OpenZeppelin Contracts (last updated v4.7.0) (utils/Strings.sol)\\n\\npragma solidity ^0.8.0;\\n\\n/**\\n * @dev String operations.\\n */\\nlibrary StringsUpgradeable {\\n bytes16 private constant _HEX_SYMBOLS = \\\"0123456789abcdef\\\";\\n uint8 private constant _ADDRESS_LENGTH = 20;\\n\\n /**\\n * @dev Converts a `uint256` to its ASCII `string` decimal representation.\\n */\\n function toString(uint256 value) internal pure returns (string memory) {\\n // Inspired by OraclizeAPI's implementation - MIT licence\\n // https://github.com/oraclize/ethereum-api/blob/b42146b063c7d6ee1358846c198246239e9360e8/oraclizeAPI_0.4.25.sol\\n\\n if (value == 0) {\\n return \\\"0\\\";\\n }\\n uint256 temp = value;\\n uint256 digits;\\n while (temp != 0) {\\n digits++;\\n temp /= 10;\\n }\\n bytes memory buffer = new bytes(digits);\\n while (value != 0) {\\n digits -= 1;\\n buffer[digits] = bytes1(uint8(48 + uint256(value % 10)));\\n value /= 10;\\n }\\n return string(buffer);\\n }\\n\\n /**\\n * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation.\\n */\\n function toHexString(uint256 value) internal pure returns (string memory) {\\n if (value == 0) {\\n return \\\"0x00\\\";\\n }\\n uint256 temp = value;\\n uint256 length = 0;\\n while (temp != 0) {\\n length++;\\n temp >>= 8;\\n }\\n return toHexString(value, length);\\n }\\n\\n /**\\n * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation with fixed length.\\n */\\n function toHexString(uint256 value, uint256 length) internal pure returns (string memory) {\\n bytes memory buffer = new bytes(2 * length + 2);\\n buffer[0] = \\\"0\\\";\\n buffer[1] = \\\"x\\\";\\n for (uint256 i = 2 * length + 1; i > 1; --i) {\\n buffer[i] = _HEX_SYMBOLS[value & 0xf];\\n value >>= 4;\\n }\\n require(value == 0, \\\"Strings: hex length insufficient\\\");\\n return string(buffer);\\n }\\n\\n /**\\n * @dev Converts an `address` with fixed length of 20 bytes to its not checksummed ASCII `string` hexadecimal representation.\\n */\\n function toHexString(address addr) internal pure returns (string memory) {\\n return toHexString(uint256(uint160(addr)), _ADDRESS_LENGTH);\\n }\\n}\\n\",\"keccak256\":\"0xea5339a7fff0ed42b45be56a88efdd0b2ddde9fa480dc99fef9a6a4c5b776863\",\"license\":\"MIT\"},\"@openzeppelin/contracts-upgradeable/utils/introspection/ERC165Upgradeable.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n// OpenZeppelin Contracts v4.4.1 (utils/introspection/ERC165.sol)\\n\\npragma solidity ^0.8.0;\\n\\nimport \\\"./IERC165Upgradeable.sol\\\";\\nimport \\\"../../proxy/utils/Initializable.sol\\\";\\n\\n/**\\n * @dev Implementation of the {IERC165} interface.\\n *\\n * Contracts that want to implement ERC165 should inherit from this contract and override {supportsInterface} to check\\n * for the additional interface id that will be supported. For example:\\n *\\n * ```solidity\\n * function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) {\\n * return interfaceId == type(MyInterface).interfaceId || super.supportsInterface(interfaceId);\\n * }\\n * ```\\n *\\n * Alternatively, {ERC165Storage} provides an easier to use but more expensive implementation.\\n */\\nabstract contract ERC165Upgradeable is Initializable, IERC165Upgradeable {\\n function __ERC165_init() internal onlyInitializing {\\n }\\n\\n function __ERC165_init_unchained() internal onlyInitializing {\\n }\\n /**\\n * @dev See {IERC165-supportsInterface}.\\n */\\n function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) {\\n return interfaceId == type(IERC165Upgradeable).interfaceId;\\n }\\n\\n /**\\n * @dev This empty reserved space is put in place to allow future versions to add new\\n * variables without shifting down storage in the inheritance chain.\\n * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps\\n */\\n uint256[50] private __gap;\\n}\\n\",\"keccak256\":\"0x9a3b990bd56d139df3e454a9edf1c64668530b5a77fc32eb063bc206f958274a\",\"license\":\"MIT\"},\"@openzeppelin/contracts-upgradeable/utils/introspection/IERC165Upgradeable.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n// OpenZeppelin Contracts v4.4.1 (utils/introspection/IERC165.sol)\\n\\npragma solidity ^0.8.0;\\n\\n/**\\n * @dev Interface of the ERC165 standard, as defined in the\\n * https://eips.ethereum.org/EIPS/eip-165[EIP].\\n *\\n * Implementers can declare support of contract interfaces, which can then be\\n * queried by others ({ERC165Checker}).\\n *\\n * For an implementation, see {ERC165}.\\n */\\ninterface IERC165Upgradeable {\\n /**\\n * @dev Returns true if this contract implements the interface defined by\\n * `interfaceId`. See the corresponding\\n * https://eips.ethereum.org/EIPS/eip-165#how-interfaces-are-identified[EIP section]\\n * to learn more about how these ids are created.\\n *\\n * This function call must use less than 30 000 gas.\\n */\\n function supportsInterface(bytes4 interfaceId) external view returns (bool);\\n}\\n\",\"keccak256\":\"0xc6cef87559d0aeffdf0a99803de655938a7779ec0a3cd5d4383483ad85565a09\",\"license\":\"MIT\"},\"base64-sol/base64.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n\\npragma solidity >=0.6.0;\\n\\n/// @title Base64\\n/// @author Brecht Devos - \\n/// @notice Provides functions for encoding/decoding base64\\nlibrary Base64 {\\n string internal constant TABLE_ENCODE = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/';\\n bytes internal constant TABLE_DECODE = hex\\\"0000000000000000000000000000000000000000000000000000000000000000\\\"\\n hex\\\"00000000000000000000003e0000003f3435363738393a3b3c3d000000000000\\\"\\n hex\\\"00000102030405060708090a0b0c0d0e0f101112131415161718190000000000\\\"\\n hex\\\"001a1b1c1d1e1f202122232425262728292a2b2c2d2e2f303132330000000000\\\";\\n\\n function encode(bytes memory data) internal pure returns (string memory) {\\n if (data.length == 0) return '';\\n\\n // load the table into memory\\n string memory table = TABLE_ENCODE;\\n\\n // multiply by 4/3 rounded up\\n uint256 encodedLen = 4 * ((data.length + 2) / 3);\\n\\n // add some extra buffer at the end required for the writing\\n string memory result = new string(encodedLen + 32);\\n\\n assembly {\\n // set the actual output length\\n mstore(result, encodedLen)\\n\\n // prepare the lookup table\\n let tablePtr := add(table, 1)\\n\\n // input ptr\\n let dataPtr := data\\n let endPtr := add(dataPtr, mload(data))\\n\\n // result ptr, jump over length\\n let resultPtr := add(result, 32)\\n\\n // run over the input, 3 bytes at a time\\n for {} lt(dataPtr, endPtr) {}\\n {\\n // read 3 bytes\\n dataPtr := add(dataPtr, 3)\\n let input := mload(dataPtr)\\n\\n // write 4 characters\\n mstore8(resultPtr, mload(add(tablePtr, and(shr(18, input), 0x3F))))\\n resultPtr := add(resultPtr, 1)\\n mstore8(resultPtr, mload(add(tablePtr, and(shr(12, input), 0x3F))))\\n resultPtr := add(resultPtr, 1)\\n mstore8(resultPtr, mload(add(tablePtr, and(shr( 6, input), 0x3F))))\\n resultPtr := add(resultPtr, 1)\\n mstore8(resultPtr, mload(add(tablePtr, and( input, 0x3F))))\\n resultPtr := add(resultPtr, 1)\\n }\\n\\n // padding with '='\\n switch mod(mload(data), 3)\\n case 1 { mstore(sub(resultPtr, 2), shl(240, 0x3d3d)) }\\n case 2 { mstore(sub(resultPtr, 1), shl(248, 0x3d)) }\\n }\\n\\n return result;\\n }\\n\\n function decode(string memory _data) internal pure returns (bytes memory) {\\n bytes memory data = bytes(_data);\\n\\n if (data.length == 0) return new bytes(0);\\n require(data.length % 4 == 0, \\\"invalid base64 decoder input\\\");\\n\\n // load the table into memory\\n bytes memory table = TABLE_DECODE;\\n\\n // every 4 characters represent 3 bytes\\n uint256 decodedLen = (data.length / 4) * 3;\\n\\n // add some extra buffer at the end required for the writing\\n bytes memory result = new bytes(decodedLen + 32);\\n\\n assembly {\\n // padding with '='\\n let lastBytes := mload(add(data, mload(data)))\\n if eq(and(lastBytes, 0xFF), 0x3d) {\\n decodedLen := sub(decodedLen, 1)\\n if eq(and(lastBytes, 0xFFFF), 0x3d3d) {\\n decodedLen := sub(decodedLen, 1)\\n }\\n }\\n\\n // set the actual output length\\n mstore(result, decodedLen)\\n\\n // prepare the lookup table\\n let tablePtr := add(table, 1)\\n\\n // input ptr\\n let dataPtr := data\\n let endPtr := add(dataPtr, mload(data))\\n\\n // result ptr, jump over length\\n let resultPtr := add(result, 32)\\n\\n // run over the input, 4 characters at a time\\n for {} lt(dataPtr, endPtr) {}\\n {\\n // read 4 characters\\n dataPtr := add(dataPtr, 4)\\n let input := mload(dataPtr)\\n\\n // write 3 bytes\\n let output := add(\\n add(\\n shl(18, and(mload(add(tablePtr, and(shr(24, input), 0xFF))), 0xFF)),\\n shl(12, and(mload(add(tablePtr, and(shr(16, input), 0xFF))), 0xFF))),\\n add(\\n shl( 6, and(mload(add(tablePtr, and(shr( 8, input), 0xFF))), 0xFF)),\\n and(mload(add(tablePtr, and( input , 0xFF))), 0xFF)\\n )\\n )\\n mstore(resultPtr, shl(232, output))\\n resultPtr := add(resultPtr, 3)\\n }\\n }\\n\\n return result;\\n }\\n}\\n\",\"keccak256\":\"0xa73959e6ef0b693e4423a562e612370160b934a75e618361ddd8c9c4b8ddbaaf\",\"license\":\"MIT\"},\"contracts/DropCreator.sol\":{\"content\":\"// SPDX-License-Identifier: GPL-3.0\\n\\n/**\\n\\n ExpandedNFTs\\n \\n */\\n\\npragma solidity ^0.8.15;\\n\\nimport {ClonesUpgradeable} from \\\"@openzeppelin/contracts-upgradeable/proxy/ClonesUpgradeable.sol\\\";\\nimport {CountersUpgradeable} from \\\"@openzeppelin/contracts-upgradeable/utils/CountersUpgradeable.sol\\\";\\n\\nimport \\\"./ExpandedNFT.sol\\\";\\n\\ncontract DropCreator {\\n using CountersUpgradeable for CountersUpgradeable.Counter;\\n\\n /// Counter for current contract id upgraded\\n CountersUpgradeable.Counter private _atContract;\\n\\n /// Address for implementation of ExpandedNFT to clone\\n address public implementation;\\n\\n /// Initializes factory with address of implementation logic\\n /// @param _implementation ExpandedNFT logic implementation contract to clone\\n constructor(address _implementation) {\\n implementation = _implementation;\\n }\\n\\n /// Creates a new drop contract as a factory with a deterministic address\\n /// Important: None of these fields (except the Url fields with the same hash) can be changed after calling\\n /// @param _artistWallet User that created the drop\\n /// @param _name Name of the drop contract\\n /// @param _symbol Symbol of the drop contract\\n function createDrop(\\n address _artistWallet,\\n string memory _name,\\n string memory _symbol,\\n uint256 _dropSize\\n ) external returns (uint256) {\\n require(_dropSize > 0, \\\"Drop size must be > 0\\\");\\n\\n address newContract = ClonesUpgradeable.cloneDeterministic(\\n implementation,\\n bytes32(abi.encodePacked(_atContract.current()))\\n );\\n\\n ExpandedNFT(newContract).initialize(\\n msg.sender,\\n _artistWallet,\\n _name,\\n _symbol,\\n _dropSize\\n );\\n\\n uint256 newId = _atContract.current(); \\n emit CreatedDrop(newId, msg.sender, _dropSize, newContract);\\n // Returns the ID of the recently created minting contract\\n // Also increments for the next contract creation call\\n _atContract.increment();\\n return newId;\\n }\\n\\n /// Get drop given the created ID\\n /// @param dropId id of drop to get contract for\\n /// @return ExpandedNFT Drop NFT contract\\n function getDropAtId(uint256 dropId)\\n external\\n view\\n returns (ExpandedNFT)\\n {\\n return\\n ExpandedNFT(\\n ClonesUpgradeable.predictDeterministicAddress(\\n implementation,\\n bytes32(abi.encodePacked(dropId)),\\n address(this)\\n )\\n );\\n }\\n\\n /// Emitted when a drop is created reserving the corresponding token IDs.\\n /// @param dropId ID of newly created drop\\n event CreatedDrop(\\n uint256 indexed dropId,\\n address indexed creator,\\n uint256 dropSize,\\n address dropContractAddress\\n );\\n}\\n\",\"keccak256\":\"0xd6a5ada49de66390c0a846b5221a6ddcca1941ec61eaa168dbabd030437dbe45\",\"license\":\"GPL-3.0\"},\"contracts/ExpandedNFT.sol\":{\"content\":\"// SPDX-License-Identifier: GPL-3.0\\n\\n/**\\n\\n ExpandedNFTs\\n\\n */\\n\\npragma solidity ^0.8.15;\\n\\nimport {ERC721Upgradeable} from \\\"@openzeppelin/contracts-upgradeable/token/ERC721/ERC721Upgradeable.sol\\\";\\nimport {IERC2981Upgradeable, IERC165Upgradeable} from \\\"@openzeppelin/contracts-upgradeable/interfaces/IERC2981Upgradeable.sol\\\";\\nimport {OwnableUpgradeable} from \\\"@openzeppelin/contracts-upgradeable/access/OwnableUpgradeable.sol\\\";\\nimport {AddressUpgradeable} from \\\"@openzeppelin/contracts-upgradeable/utils/AddressUpgradeable.sol\\\";\\nimport {IERC20Upgradeable} from \\\"@openzeppelin/contracts-upgradeable/token/ERC20/IERC20Upgradeable.sol\\\";\\n\\nimport {SharedNFTLogic} from \\\"./SharedNFTLogic.sol\\\";\\nimport {IExpandedNFT} from \\\"./IExpandedNFT.sol\\\";\\n/**\\n This is a smart contract for handling dynamic contract minting.\\n\\n @dev This allows creators to mint a unique serial drop of an expanded NFT within a custom contract\\n @author Zien\\n Repository: https://github.com/joinzien/expanded-nft\\n*/\\ncontract ExpandedNFT is\\n ERC721Upgradeable,\\n IExpandedNFT,\\n IERC2981Upgradeable,\\n OwnableUpgradeable\\n{\\n enum WhoCanMint{ ONLY_OWNER, VIPS, MEMBERS, ANYONE }\\n\\n enum ExpandedNFTStates{ UNMINTED, MINTED, REDEEM_STARTED, SET_OFFER_TERMS, ACCEPTED_OFFER, PRODUCTION_COMPLETE, REDEEMED }\\n \\n event PriceChanged(uint256 amount);\\n event EditionSold(uint256 price, address owner);\\n event WhoCanMintChanged(WhoCanMint minters);\\n\\n // State change events\\n event RedeemStarted(uint256 tokenId, address owner);\\n event RedeemAborted(uint256 tokenId, address owner); \\n event OfferTermsSet(uint256 tokenId);\\n event OfferAccepted(uint256 tokenId);\\n event OfferRejected(uint256 tokenId);\\n event ProductionComplete(uint256 tokenId);\\n event DeliveryAccepted(uint256 tokenId);\\n\\n struct PerToken { \\n // Hashmap of the Edition ID to the current \\n ExpandedNFTStates editionState;\\n\\n // Redemption price\\n uint256 editionFee; \\n\\n // Edition description\\n string description;\\n\\n // Minted\\n\\n // animation_url field in the metadata\\n string animationUrl;\\n // Hash for the associated animation\\n bytes32 animationHash;\\n // Image in the metadata\\n string imageUrl;\\n // Hash for the associated image\\n bytes32 imageHash;\\n\\n // Redeemed\\n\\n // animation_url field in the metadata\\n string redeemedAnimationUrl;\\n // Hash for the associated animation\\n bytes32 redeemedAnimationHash;\\n // Image in the metadata\\n string redeemedImageUrl;\\n // Hash for the associated image\\n bytes32 redeemedImageHash;\\n // Condition report in the metadata\\n string conditionReportUrl;\\n // Hash for the condition report\\n bytes32 conditionReportHash;\\n }\\n\\n struct Pricing { \\n // Royalty amount in bps\\n uint256 royaltyBPS;\\n\\n // Split amount to the platforms. the artist in bps\\n uint256 splitBPS;\\n\\n // Price for VIP sales\\n uint256 vipSalePrice;\\n\\n // Price for member sales\\n uint256 membersSalePrice; \\n\\n // Price for VIP sales\\n uint256 vipMintLimit;\\n\\n // Price for member sales\\n uint256 membersMintLimit;\\n\\n // Price for general sales\\n uint256 generalMintLimit; \\n\\n // Addresses allowed to mint edition\\n mapping(address => bool) allowedMinters;\\n // VIP Addresses allowed to mint edition\\n mapping(address => bool) vipAllowedMinters;\\n\\n // Who can currently mint\\n WhoCanMint whoCanMint;\\n\\n // Mint counts for each address\\n mapping(address => uint256) mintCounts; \\n }\\n\\n // metadata\\n string public description;\\n\\n // Artists wallet address\\n address private _artistWallet;\\n\\n // Per Token data\\n mapping(uint256 => PerToken) private _perTokenMetadata;\\n\\n // Total size of the drop that can be minted\\n uint256 public dropSize;\\n\\n uint256 private _loadedMetadata;\\n\\n // reservation list\\n uint256 private _reserveCount;\\n mapping(uint256 => address) private _reserveAddress;\\n mapping(uint256 => uint256) private _reserveTokenId;\\n\\n mapping(uint256 => bool) private _tokenClaimed; \\n uint256 private _claimCount; \\n uint256 private _currentIndex;\\n\\n Pricing private _pricing;\\n\\n // Price for general sales\\n uint256 public salePrice;\\n\\n // ERC20 interface for the payment token\\n IERC20Upgradeable private _paymentTokenERC20;\\n\\n // NFT rendering logic contract\\n SharedNFTLogic private immutable _sharedNFTLogic;\\n\\n // Global constructor for factory\\n constructor(SharedNFTLogic sharedNFTLogic) {\\n _sharedNFTLogic = sharedNFTLogic;\\n _pricing.whoCanMint = WhoCanMint.ONLY_OWNER;\\n }\\n\\n /**\\n @param _owner wallet addres for the user that owns and can mint the drop, gets royalty and sales payouts and can update the base url if needed.\\n @param artistWallet wallet address for thr User that created the drop\\n @param _name Name of drop, used in the title as \\\"$NAME NUMBER/TOTAL\\\"\\n @param _symbol Symbol of the new token contract\\n @param _dropSize Number of editions that can be minted in total. \\n @dev Function to create a new drop. Can only be called by the allowed creator\\n Sets the only allowed minter to the address that creates/owns the drop.\\n This can be re-assigned or updated later\\n */\\n function initialize(\\n address _owner,\\n address artistWallet,\\n string memory _name,\\n string memory _symbol,\\n uint256 _dropSize\\n ) public initializer {\\n require(_dropSize > 0, \\\"Drop size must be > 0\\\");\\n\\n __ERC721_init(_name, _symbol);\\n __Ownable_init();\\n\\n // Set ownership to original sender of contract call\\n transferOwnership(_owner);\\n\\n _artistWallet = artistWallet;\\n dropSize = _dropSize;\\n\\n // Set edition id start to be 1 not 0\\n _claimCount = 0; \\n _currentIndex = 1;\\n\\n // Set the metadata\\n description = _name;\\n _loadedMetadata = 0; \\n }\\n\\n /**\\n @param _description Description of the edition, used in the description field of the NFT\\n @param imageUrl Image URL of the the edition. Strongly encouraged to be used, if necessary, only animation URL can be used. One of animation and image url need to exist in a drop to render the NFT.\\n @param imageHash SHA256 of the given image in bytes32 format (0xHASH). If no image is included, the hash can be zero.\\n @param animationUrl Animation URL of the edition. Not required, but if omitted image URL needs to be included. This follows the opensea spec for NFTs\\n @param animationHash The associated hash of the animation in sha-256 bytes32 format. If animation is omitted the hash can be zero.\\n @dev Function to create a new drop. Can only be called by the allowed creator\\n Sets the only allowed minter to the address that creates/owns the drop.\\n This can be re-assigned or updated later\\n */\\n function loadMetadataChunk(\\n uint256 startOffset,\\n uint256 count,\\n string[] memory _description,\\n string[] memory imageUrl,\\n bytes32[] memory imageHash,\\n string[] memory animationUrl,\\n bytes32[] memory animationHash\\n ) public {\\n require(_description.length == count, \\\"Data size mismatch\\\");\\n require(animationUrl.length == count, \\\"Data size mismatch\\\");\\n require(animationHash.length == count, \\\"Data size mismatch\\\");\\n require(imageUrl.length == count, \\\"Data size mismatch\\\");\\n require(imageHash.length == count, \\\"Data size mismatch\\\");\\n\\n for (uint i = 0; i < count; i++) {\\n uint index = startOffset + i + 1;\\n \\n _perTokenMetadata[index].description = _description[i];\\n\\n _perTokenMetadata[index].animationUrl = animationUrl[i];\\n _perTokenMetadata[index].animationHash = animationHash[i];\\n _perTokenMetadata[index].imageUrl = imageUrl[i];\\n _perTokenMetadata[index].imageHash = imageHash[i];\\n }\\n\\n _loadedMetadata += count;\\n }\\n\\n function metadataloaded() public view returns (bool){\\n return (_loadedMetadata >= dropSize);\\n }\\n\\n /// @dev returns the number of minted tokens within the drop\\n function totalSupply() public view returns (uint256) {\\n return _claimCount;\\n }\\n\\n /// @dev returns the royalty BPS\\n function getRoyaltyBPS() public view returns (uint256) {\\n return _pricing.royaltyBPS;\\n }\\n\\n /// @dev returns the split BPS\\n function getSplitBPS() public view returns (uint256) {\\n return _pricing.splitBPS;\\n }\\n\\n /// @dev returns the VIP sale price\\n function getVIPSalePrice() public view returns (uint256) {\\n return _pricing.vipSalePrice;\\n }\\n\\n /// @dev returns the member sale price\\n function getMembersSalePrice() public view returns (uint256) {\\n return _pricing.membersSalePrice;\\n }\\n\\n /// @dev returns the VIP mint limit\\n function getVIPMintLimit() public view returns (uint256) {\\n return _pricing.vipMintLimit;\\n }\\n\\n /// @dev returns the member mint limit\\n function getMembersMintLimit() public view returns (uint256) {\\n return _pricing.membersMintLimit;\\n }\\n\\n /// @dev returns the general mint limit\\n function getGeneralMintLimit() public view returns (uint256) {\\n return salePrice;\\n }\\n\\n /// @dev returns who can mint\\n function getWhoCanMint() public view returns (uint256) {\\n return uint256(_pricing.whoCanMint);\\n }\\n\\n /**\\n Simple eth-based sales function\\n More complex sales functions can be implemented through IExpandedNFT interface\\n */\\n\\n /**\\n @dev This allows the user to purchase an edition\\n at the given price in the contract.\\n */\\n\\n function purchase() external payable returns (uint256) {\\n uint256 currentPrice = _currentSalesPrice();\\n emit EditionSold(currentPrice, msg.sender);\\n\\n address[] memory toMint = new address[](1);\\n toMint[0] = msg.sender;\\n\\n return _mintEditionsBody(toMint); \\n }\\n\\n /**\\n @param to address to send the newly minted edition to\\n @dev This mints one edition to the given address by an allowed minter on the edition instance.\\n */\\n function mintEdition(address to) external payable override returns (uint256) {\\n address[] memory toMint = new address[](1);\\n toMint[0] = to;\\n\\n return _mintEditionsBody(toMint); \\n }\\n\\n /**\\n @param recipients list of addresses to send the newly minted editions to\\n @dev This mints multiple editions to the given list of addresses.\\n */\\n function mintEditions(address[] memory recipients)\\n external payable override returns (uint256)\\n {\\n return _mintEditionsBody(recipients);\\n } \\n\\n /**\\n @param recipients list of addresses to send the newly minted editions to\\n @dev This mints multiple editions to the given list of addresses.\\n */\\n function _mintEditionsBody(address[] memory recipients)\\n internal returns (uint256)\\n {\\n require(_loadedMetadata >= dropSize, \\\"Not all metadata loaded\\\");\\n\\n require(_isAllowedToMint(), \\\"Needs to be an allowed minter\\\");\\n\\n uint256 currentPrice = _currentSalesPrice();\\n require(currentPrice > 0, \\\"Not for sale\\\");\\n require(msg.value == (currentPrice * recipients.length), \\\"Wrong price\\\");\\n\\n require((_pricing.mintCounts[msg.sender] + recipients.length - 1) < _currentMintLimit(), \\\"Exceeded mint limit\\\");\\n\\n require(_claimCount + recipients.length <= dropSize, \\\"Over drop size\\\");\\n\\n if (_pricing.whoCanMint == WhoCanMint.VIPS) {\\n return _vipMintEditions(recipients);\\n }\\n\\n return _mintEditions(recipients);\\n } \\n\\n /**\\n @dev Private function to mint without any access checks.\\n Called by the public edition minting functions.\\n */\\n function _vipMintEditions(address[] memory recipients)\\n internal\\n returns (uint256)\\n {\\n address currentMinter = msg.sender;\\n\\n uint256 unclaimed = 0;\\n uint256 firstUnclaimed = _reserveCount;\\n\\n for (uint256 r = 0; r < _reserveCount; r++) {\\n if (_reserveAddress[r] == currentMinter) {\\n uint256 id = _reserveTokenId[r];\\n\\n if (_tokenClaimed[id] != true) {\\n if (r < firstUnclaimed) {\\n firstUnclaimed = r; \\n }\\n\\n unclaimed++;\\n }\\n }\\n }\\n\\n require(unclaimed >= recipients.length, \\\"Can not mint all editions\\\");\\n\\n uint256 idToMint = 1;\\n\\n uint256 reservationCounter = firstUnclaimed;\\n for (uint256 i = 0; i < recipients.length; i++) {\\n while (_reserveAddress[reservationCounter] != currentMinter) {\\n reservationCounter++;\\n } \\n\\n idToMint = _reserveTokenId[reservationCounter];\\n\\n _mint(recipients[i], idToMint);\\n\\n _perTokenMetadata[idToMint].editionState = ExpandedNFTStates.MINTED;\\n _tokenClaimed[idToMint] = true;\\n _pricing.mintCounts[currentMinter]++;\\n _claimCount++;\\n\\n reservationCounter++;\\n }\\n\\n return idToMint; \\n } \\n\\n /**\\n @dev Private function to mint without any access checks.\\n Called by the public edition minting functions.\\n */\\n function _mintEditions(address[] memory recipients)\\n internal\\n returns (uint256)\\n {\\n address currentMinter = msg.sender;\\n \\n for (uint256 i = 0; i < recipients.length; i++) {\\n while (_tokenClaimed[_currentIndex] == true) {\\n _currentIndex++;\\n } \\n\\n _mint(recipients[i], _currentIndex);\\n\\n _perTokenMetadata[_currentIndex].editionState = ExpandedNFTStates.MINTED;\\n _tokenClaimed[_currentIndex] = true;\\n _pricing.mintCounts[currentMinter]++;\\n _claimCount++;\\n }\\n\\n return _currentIndex; \\n } \\n\\n /**\\n @param _royaltyBPS BPS of the royalty set on the contract. Can be 0 for no royalty.\\n @param _splitBPS BPS of the royalty set on the contract. Can be 0 for no royalty. \\n @param _vipSalePrice Sale price for VIPs\\n @param _membersSalePrice SalePrice for Members \\n @param _generalSalePrice SalePrice for the general public \\n @param _vipMintLimit Mint limit for VIPs\\n @param _membersMintLimit Mint limit for Members \\n @param _generalMintLimit Mint limit for the general public \\n @dev Set various pricing related values\\n */\\n function setPricing (\\n uint256 _royaltyBPS,\\n uint256 _splitBPS,\\n uint256 _vipSalePrice,\\n uint256 _membersSalePrice, \\n uint256 _generalSalePrice,\\n uint256 _vipMintLimit,\\n uint256 _membersMintLimit,\\n uint256 _generalMintLimit \\n ) external onlyOwner { \\n _pricing.royaltyBPS = _royaltyBPS;\\n _pricing.splitBPS = _splitBPS;\\n\\n _pricing.vipSalePrice = _vipSalePrice;\\n _pricing.membersSalePrice = _membersSalePrice;\\n salePrice = _generalSalePrice;\\n\\n _pricing.vipMintLimit = _vipMintLimit;\\n _pricing.membersMintLimit = _membersMintLimit;\\n _pricing.generalMintLimit = _generalMintLimit;\\n\\n emit PriceChanged(salePrice);\\n }\\n\\n /**\\n @dev returns the current ETH sales price\\n based on who can currently mint.\\n */\\n function _currentSalesPrice() internal view returns (uint256){\\n if (_pricing.whoCanMint == WhoCanMint.VIPS) {\\n return _pricing.vipSalePrice;\\n } else if (_pricing.whoCanMint == WhoCanMint.MEMBERS) {\\n return _pricing.membersSalePrice;\\n } else if (_pricing.whoCanMint == WhoCanMint.ANYONE) {\\n return salePrice;\\n } \\n \\n return 0; \\n }\\n\\n /**\\n @param wallets A list of wallets\\n @param tokenIDs A list of tokenId to reserve \\n @dev Set various pricing related values\\n */\\n function reserve (address[] calldata wallets, uint256[] calldata tokenIDs) \\n external onlyOwner { \\n for (uint256 i = 0; i < wallets.length; i++) {\\n _reserveAddress[_reserveCount] = wallets[i]; \\n _reserveTokenId[_reserveCount] = tokenIDs[i]; \\n _reserveCount++;\\n }\\n }\\n\\n /**\\n @dev returns the current loimit on edition that \\n can be minted by one wallet\\n */\\n function _currentMintLimit() internal view returns (uint256){\\n if (_pricing.whoCanMint == WhoCanMint.VIPS) {\\n return _pricing.vipMintLimit;\\n } else if (_pricing.whoCanMint == WhoCanMint.MEMBERS) {\\n return _pricing.membersMintLimit;\\n } else if (_pricing.whoCanMint == WhoCanMint.ANYONE) {\\n return _pricing.generalMintLimit;\\n } \\n \\n return 0; \\n }\\n\\n /**\\n @param _salePrice if sale price is 0 sale is stopped, otherwise that amount \\n of ETH is needed to start the sale.\\n @dev This sets a simple ETH sales price\\n Setting a sales price allows users to mint the drop until it sells out.\\n For more granular sales, use an external sales contract.\\n */\\n function setSalePrice(uint256 _salePrice) external onlyOwner {\\n salePrice = _salePrice;\\n\\n _pricing.whoCanMint = WhoCanMint.ANYONE;\\n\\n emit WhoCanMintChanged(_pricing.whoCanMint);\\n emit PriceChanged(salePrice);\\n }\\n\\n /**\\n @param _salePrice if sale price is 0 sale is stopped, otherwise that amount \\n of ETH is needed to start the sale.\\n @dev This sets the VIP ETH sales price\\n Setting a sales price allows users to mint the drop until it sells out.\\n For more granular sales, use an external sales contract.\\n */\\n function setVIPSalePrice(uint256 _salePrice) external onlyOwner {\\n _pricing.vipSalePrice = _salePrice;\\n\\n _pricing.whoCanMint = WhoCanMint.VIPS;\\n\\n emit WhoCanMintChanged(_pricing.whoCanMint);\\n emit PriceChanged(salePrice);\\n }\\n\\n /**\\n @param _salePrice if sale price is 0 sale is stopped, otherwise that amount \\n of ETH is needed to start the sale.\\n @dev This sets the members ETH sales price\\n Setting a sales price allows users to mint the drop until it sells out.\\n For more granular sales, use an external sales contract.\\n */\\n function setMembersSalePrice(uint256 _salePrice) external onlyOwner {\\n _pricing.membersSalePrice = _salePrice;\\n\\n _pricing.whoCanMint = WhoCanMint.MEMBERS;\\n\\n emit WhoCanMintChanged(_pricing.whoCanMint);\\n emit PriceChanged(salePrice);\\n } \\n\\n\\n /**\\n @param vipSalePrice if sale price is 0 sale is stopped, otherwise that amount \\n of ETH is needed to start the sale.\\n @param membersSalePrice if sale price is 0 sale is stopped, otherwise that amount \\n of ETH is needed to start the sale.\\n @param generalSalePrice if sale price is 0 sale is stopped, otherwise that amount \\n of ETH is needed to start the sale. \\n @dev This sets the members ETH sales price\\n Setting a sales price allows users to mint the drop until it sells out.\\n For more granular sales, use an external sales contract.\\n */\\n function setSalePrices(uint256 vipSalePrice, uint256 membersSalePrice, uint256 generalSalePrice) external onlyOwner {\\n _pricing.vipSalePrice = vipSalePrice;\\n _pricing.membersSalePrice = membersSalePrice;\\n salePrice = generalSalePrice; \\n\\n emit PriceChanged(salePrice);\\n } \\n\\n /**\\n @dev This withdraws ETH from the contract to the contract owner.\\n */\\n function withdraw() external onlyOwner {\\n uint256 currentBalance = address(this).balance;\\n if (currentBalance > 0) {\\n uint256 platformFee = (currentBalance * _pricing.splitBPS) / 10000;\\n uint256 artistFee = currentBalance - platformFee;\\n\\n AddressUpgradeable.sendValue(payable(owner()), platformFee);\\n AddressUpgradeable.sendValue(payable(_artistWallet), artistFee);\\n }\\n\\n if (address(_paymentTokenERC20) != address(0x0)) {\\n uint256 currentBalanceERC20 = _paymentTokenERC20.balanceOf(address(this));\\n if (currentBalanceERC20 > 0) {\\n uint256 platformFee = (currentBalanceERC20 * _pricing.splitBPS) / 10000;\\n uint256 artistFee = currentBalanceERC20 - platformFee;\\n\\n _paymentTokenERC20.transfer(owner(), platformFee);\\n _paymentTokenERC20.transfer(_artistWallet, artistFee);\\n }\\n }\\n }\\n\\n /**\\n @dev This helper function checks if the msg.sender is allowed to mint the\\n given edition id.\\n */\\n function _isAllowedToMint() internal view returns (bool) {\\n if (_pricing.whoCanMint == WhoCanMint.ANYONE) {\\n return true;\\n }\\n\\n if (_pricing.whoCanMint == WhoCanMint.MEMBERS) {\\n if (_pricing.vipAllowedMinters[msg.sender]) {\\n return true;\\n } \\n\\n if (_pricing.allowedMinters[msg.sender]) {\\n return true;\\n } \\n }\\n\\n if (_pricing.whoCanMint == WhoCanMint.VIPS) {\\n if (_pricing.vipAllowedMinters[msg.sender]) {\\n return true;\\n } \\n }\\n\\n if (owner() == msg.sender) {\\n return true;\\n }\\n\\n return false;\\n }\\n\\n /**\\n Simple override for owner interface.\\n */\\n function owner()\\n public\\n view\\n override(OwnableUpgradeable, IExpandedNFT)\\n returns (address)\\n {\\n return super.owner();\\n }\\n\\n /**\\n return the artists wallet address\\n */\\n function getArtistWallet()\\n public\\n view\\n returns (address)\\n {\\n return _artistWallet;\\n }\\n\\n /**\\n set the artists wallet address\\n */\\n function setArtistWallet(address wallet)\\n public\\n onlyOwner\\n {\\n _artistWallet = wallet;\\n } \\n\\n /**\\n return the payment tokens address\\n */\\n function getPaymentToken()\\n public\\n view\\n returns (address)\\n {\\n return address(_paymentTokenERC20);\\n }\\n\\n /**\\n set a new payment token address\\n */\\n function setPaymentToken(address paymentToken)\\n public\\n onlyOwner\\n {\\n if (address(_paymentTokenERC20) != address(0x0)) {\\n require(_paymentTokenERC20.balanceOf(address(this)) == 0, \\\"token must have 0 balance\\\");\\n }\\n\\n _paymentTokenERC20 = IERC20Upgradeable(paymentToken);\\n } \\n\\n /**\\n @dev Sets the types of users who is allowed to mint.\\n */\\n function getAllowedMinter() public view returns (WhoCanMint){\\n return _pricing.whoCanMint;\\n }\\n\\n /**\\n @param minters WhoCanMint enum of minter types\\n @dev Sets the types of users who is allowed to mint.\\n */\\n function setAllowedMinter(WhoCanMint minters) public onlyOwner {\\n require(((minters >= WhoCanMint.ONLY_OWNER) && (minters <= WhoCanMint.ANYONE)), \\\"Needs to be a valid minter type\\\");\\n\\n _pricing.whoCanMint = minters;\\n emit WhoCanMintChanged(minters);\\n }\\n\\n /**\\n @param minter address to set approved minting status for\\n @param allowed boolean if that address is allowed to mint\\n @dev Sets the approved minting status of the given address.\\n This requires that msg.sender is the owner of the given edition id.\\n If the ZeroAddress (address(0x0)) is set as a minter,\\n anyone will be allowed to mint.\\n This setup is similar to setApprovalForAll in the ERC721 spec.\\n */\\n function setApprovedMinters(uint256 count, address[] calldata minter, bool[] calldata allowed) public onlyOwner {\\n for (uint256 i = 0; i < count; i++) {\\n _pricing.allowedMinters[minter[i]] = allowed[i];\\n }\\n }\\n\\n /**\\n @param minter address to set approved minting status for\\n @param allowed boolean if that address is allowed to mint\\n @dev Sets the approved minting status of the given address.\\n This requires that msg.sender is the owner of the given edition id.\\n If the ZeroAddress (address(0x0)) is set as a minter,\\n anyone will be allowed to mint.\\n This setup is similar to setApprovalForAll in the ERC721 spec.\\n */\\n function setApprovedVIPMinters(uint256 count, address[] calldata minter, bool[] calldata allowed) public onlyOwner {\\n for (uint256 i = 0; i < count; i++) {\\n _pricing.vipAllowedMinters[minter[i]] = allowed[i];\\n }\\n }\\n\\n /**\\n @dev Allows for updates of edition urls by the owner of the edition.\\n Only URLs can be updated (data-uris are supported), hashes cannot be updated.\\n */\\n function updateEditionURLs(\\n uint256 tokenId,\\n string memory imageUrl,\\n string memory animationUrl\\n ) public onlyOwner {\\n _perTokenMetadata[tokenId].imageUrl = imageUrl;\\n _perTokenMetadata[tokenId].animationUrl = animationUrl;\\n }\\n\\n /// Returns the number of editions allowed to mint\\n function numberCanMint() public view override returns (uint256) {\\n return dropSize - _claimCount;\\n }\\n\\n /**\\n @param tokenId Token ID to burn\\n User burn function for token id \\n */\\n function burn(uint256 tokenId) public {\\n require(_isApprovedOrOwner(_msgSender(), tokenId), \\\"Not approved\\\");\\n _burn(tokenId);\\n }\\n\\n function redeem(uint256 tokenId) public {\\n require(_exists(tokenId), \\\"No token\\\");\\n require(_isApprovedOrOwner(_msgSender(), tokenId), \\\"Not approved\\\");\\n\\n require((_perTokenMetadata[tokenId].editionState == ExpandedNFTStates.MINTED), \\\"You currently can not redeem\\\");\\n\\n _perTokenMetadata[tokenId].editionState = ExpandedNFTStates.REDEEM_STARTED;\\n emit RedeemStarted(tokenId, _msgSender());\\n }\\n\\n function abortRedemption(uint256 tokenId) public {\\n require(_exists(tokenId), \\\"No token\\\");\\n require(_isApprovedOrOwner(_msgSender(), tokenId), \\\"Not approved\\\");\\n\\n require((_perTokenMetadata[tokenId].editionState == ExpandedNFTStates.REDEEM_STARTED), \\\"You currently can not redeem\\\");\\n\\n _perTokenMetadata[tokenId].editionState = ExpandedNFTStates.MINTED;\\n emit RedeemAborted(tokenId, _msgSender());\\n }\\n\\n function setOfferTerms(uint256 tokenId, uint256 fee) public onlyOwner {\\n require(_exists(tokenId), \\\"No token\\\"); \\n require((_perTokenMetadata[tokenId].editionState == ExpandedNFTStates.REDEEM_STARTED), \\\"Wrong state\\\");\\n\\n _perTokenMetadata[tokenId].editionState = ExpandedNFTStates.SET_OFFER_TERMS;\\n _perTokenMetadata[tokenId].editionFee = fee;\\n\\n emit OfferTermsSet(tokenId);\\n }\\n\\n function rejectOfferTerms(uint256 tokenId) public {\\n require(_exists(tokenId), \\\"No token\\\"); \\n require(_isApprovedOrOwner(_msgSender(), tokenId), \\\"Not approved\\\");\\n\\n require((_perTokenMetadata[tokenId].editionState == ExpandedNFTStates.SET_OFFER_TERMS), \\\"You currently can not redeem\\\");\\n\\n _perTokenMetadata[tokenId].editionState = ExpandedNFTStates.MINTED;\\n\\n emit OfferRejected(tokenId);\\n }\\n\\n function acceptOfferTerms(uint256 tokenId, uint256 paymentAmount) external {\\n require(_exists(tokenId), \\\"No token\\\"); \\n require(_isApprovedOrOwner(_msgSender(), tokenId), \\\"Not approved\\\");\\n\\n require((_perTokenMetadata[tokenId].editionState == ExpandedNFTStates.SET_OFFER_TERMS), \\\"You currently can not redeem\\\");\\n\\n require(paymentAmount >= _perTokenMetadata[tokenId].editionFee, \\\"Wrong price\\\");\\n require(_paymentTokenERC20.allowance(_msgSender(), address(this)) >= _perTokenMetadata[tokenId].editionFee, \\\"Insufficient allowance\\\");\\n\\n bool success = _paymentTokenERC20.transferFrom(_msgSender(), address(this), _perTokenMetadata[tokenId].editionFee);\\n require(success, \\\"Could not transfer token\\\");\\n\\n _perTokenMetadata[tokenId].editionState = ExpandedNFTStates.ACCEPTED_OFFER; \\n\\n emit OfferAccepted(tokenId);\\n }\\n\\n function productionComplete(\\n uint256 tokenId,\\n string memory _description,\\n string memory animationUrl,\\n bytes32 animationHash,\\n string memory imageUrl,\\n bytes32 imageHash, \\n string memory conditionReportUrl,\\n bytes32 conditionReportHash \\n ) public onlyOwner {\\n require(_exists(tokenId), \\\"No token\\\"); \\n require((_perTokenMetadata[tokenId].editionState == ExpandedNFTStates.ACCEPTED_OFFER), \\\"You currently can not redeem\\\");\\n\\n // Set the NFT to display as redeemed\\n _perTokenMetadata[tokenId].description = _description;\\n _perTokenMetadata[tokenId].redeemedAnimationUrl = animationUrl;\\n _perTokenMetadata[tokenId].redeemedAnimationHash = animationHash;\\n _perTokenMetadata[tokenId].redeemedImageUrl = imageUrl;\\n _perTokenMetadata[tokenId].redeemedImageHash = imageHash;\\n _perTokenMetadata[tokenId].conditionReportUrl = conditionReportUrl;\\n _perTokenMetadata[tokenId].conditionReportHash = conditionReportHash;\\n\\n _perTokenMetadata[tokenId].editionState = ExpandedNFTStates.PRODUCTION_COMPLETE;\\n\\n emit ProductionComplete(tokenId);\\n }\\n\\n function acceptDelivery(uint256 tokenId) public {\\n require(_exists(tokenId), \\\"No token\\\"); \\n require(_isApprovedOrOwner(_msgSender(), tokenId), \\\"Not approved\\\");\\n\\n require((_perTokenMetadata[tokenId].editionState == ExpandedNFTStates.PRODUCTION_COMPLETE), \\\"You currently can not redeem\\\");\\n\\n _perTokenMetadata[tokenId].editionState = ExpandedNFTStates.REDEEMED;\\n\\n emit OfferRejected(tokenId);\\n }\\n\\n /**\\n @dev Get URIs for the condition report\\n @return _imageUrl, _imageHash\\n */\\n function getConditionReport(uint256 tokenId)\\n public\\n view\\n returns (\\n string memory,\\n bytes32\\n )\\n {\\n return (_perTokenMetadata[tokenId].conditionReportUrl, _perTokenMetadata[tokenId].conditionReportHash);\\n }\\n\\n /**\\n @dev Get royalty information for token\\n @param _salePrice Sale price for the token\\n */\\n function royaltyInfo(uint256, uint256 _salePrice)\\n external\\n view\\n override\\n returns (address receiver, uint256 royaltyAmount)\\n {\\n if (owner() == address(0x0)) {\\n return (owner(), 0);\\n }\\n return (owner(), (_salePrice * _pricing.royaltyBPS) / 10_000);\\n }\\n\\n /**\\n @dev Get URIs for edition NFT\\n @return _imageUrl, _imageHash, _animationUrl, _animationHash\\n */\\n function getURIs(uint256 tokenId)\\n public\\n view\\n returns (\\n string memory,\\n bytes32,\\n string memory,\\n bytes32\\n )\\n {\\n if (_perTokenMetadata[tokenId].editionState == ExpandedNFTStates.REDEEMED) { \\n return (_perTokenMetadata[tokenId].redeemedImageUrl, _perTokenMetadata[tokenId].redeemedImageHash,\\n _perTokenMetadata[tokenId].redeemedAnimationUrl, _perTokenMetadata[tokenId].redeemedAnimationHash);\\n }\\n\\n return (_perTokenMetadata[tokenId].imageUrl, _perTokenMetadata[tokenId].imageHash,\\n _perTokenMetadata[tokenId].animationUrl, _perTokenMetadata[tokenId].animationHash);\\n }\\n\\n /**\\n @dev Get URI for given token id\\n @param tokenId token id to get uri for\\n @return base64-encoded json metadata object\\n */\\n function tokenURI(uint256 tokenId)\\n public\\n view\\n override\\n returns (string memory)\\n {\\n require(_exists(tokenId), \\\"No token\\\");\\n\\n if (_perTokenMetadata[tokenId].editionState == ExpandedNFTStates.REDEEMED) {\\n return\\n _sharedNFTLogic.createMetadataEdition(\\n name(),\\n _perTokenMetadata[tokenId].description,\\n _perTokenMetadata[tokenId].redeemedImageUrl,\\n _perTokenMetadata[tokenId].redeemedAnimationUrl,\\n tokenId,\\n dropSize\\n );\\n }\\n\\n return\\n _sharedNFTLogic.createMetadataEdition(\\n name(),\\n _perTokenMetadata[tokenId].description,\\n _perTokenMetadata[tokenId].imageUrl,\\n _perTokenMetadata[tokenId].animationUrl,\\n tokenId,\\n dropSize\\n );\\n }\\n\\n function supportsInterface(bytes4 interfaceId)\\n public\\n view\\n override(ERC721Upgradeable, IERC165Upgradeable)\\n returns (bool)\\n {\\n return\\n type(IERC2981Upgradeable).interfaceId == interfaceId ||\\n ERC721Upgradeable.supportsInterface(interfaceId);\\n }\\n}\\n\",\"keccak256\":\"0x2d52681fe71b3802669c94ef39596f4cd032a7431c303a444a52ef20c12a1d4e\",\"license\":\"GPL-3.0\"},\"contracts/IExpandedNFT.sol\":{\"content\":\"// SPDX-License-Identifier: GPL-3.0\\npragma solidity ^0.8.15;\\n\\ninterface IExpandedNFT {\\n function mintEdition(address to) external payable returns (uint256);\\n function mintEditions(address[] memory to) external payable returns (uint256);\\n function numberCanMint() external view returns (uint256);\\n function owner() external view returns (address);\\n}\\n\",\"keccak256\":\"0xe1dc9fcdfab1ec02f10203a59e3fb819707f06515abe45674c5ecc814fc9c253\",\"license\":\"GPL-3.0\"},\"contracts/IPublicSharedMetadata.sol\":{\"content\":\"// SPDX-License-Identifier: GPL-3.0\\n\\npragma solidity ^0.8.15;\\n\\n/// Shared public library for on-chain NFT functions\\ninterface IPublicSharedMetadata {\\n /// @param unencoded bytes to base64-encode\\n function base64Encode(bytes memory unencoded)\\n external\\n pure\\n returns (string memory);\\n\\n /// Encodes the argument json bytes into base64-data uri format\\n /// @param json Raw json to base64 and turn into a data-uri\\n function encodeMetadataJSON(bytes memory json)\\n external\\n pure\\n returns (string memory);\\n\\n /// Proxy to openzeppelin's toString function\\n /// @param value number to return as a string\\n function numberToString(uint256 value)\\n external\\n pure\\n returns (string memory);\\n}\\n\",\"keccak256\":\"0x76caf66cf71c1067c845ff9de580b6e886c448915997ed1d6f37fe1a1ec489de\",\"license\":\"GPL-3.0\"},\"contracts/SharedNFTLogic.sol\":{\"content\":\"// SPDX-License-Identifier: GPL-3.0\\n\\npragma solidity ^0.8.15;\\n\\nimport {StringsUpgradeable} from \\\"@openzeppelin/contracts-upgradeable/utils/StringsUpgradeable.sol\\\";\\nimport {Base64} from \\\"base64-sol/base64.sol\\\";\\nimport {IPublicSharedMetadata} from \\\"./IPublicSharedMetadata.sol\\\";\\n\\n/// Shared NFT logic for rendering metadata associated with editions\\n/// @dev Can safely be used for generic base64Encode and numberToString functions\\ncontract SharedNFTLogic is IPublicSharedMetadata {\\n /// @param unencoded bytes to base64-encode\\n function base64Encode(bytes memory unencoded)\\n public\\n pure\\n override\\n returns (string memory)\\n {\\n return Base64.encode(unencoded);\\n }\\n\\n /// Proxy to openzeppelin's toString function\\n /// @param value number to return as a string\\n function numberToString(uint256 value)\\n public\\n pure\\n override\\n returns (string memory)\\n {\\n return StringsUpgradeable.toString(value);\\n }\\n\\n /// Generate edition metadata from storage information as base64-json blob\\n /// Combines the media data and metadata\\n /// @param name Name of NFT in metadata\\n /// @param description Description of NFT in metadata\\n /// @param imageUrl URL of image to render for edition\\n /// @param animationUrl URL of animation to render for edition\\n /// @param tokenOfEdition Token ID for specific token\\n /// @param editionSize Size of entire edition to show\\n function createMetadataEdition(\\n string memory name,\\n string memory description,\\n string memory imageUrl,\\n string memory animationUrl,\\n uint256 tokenOfEdition,\\n uint256 editionSize\\n ) external pure returns (string memory) {\\n string memory _tokenMediaData = tokenMediaData(\\n imageUrl,\\n animationUrl,\\n tokenOfEdition\\n );\\n bytes memory json = createMetadataJSON(\\n name,\\n description,\\n _tokenMediaData,\\n tokenOfEdition,\\n editionSize\\n );\\n return encodeMetadataJSON(json);\\n }\\n\\n /// Function to create the metadata json string for the nft edition\\n /// @param name Name of NFT in metadata\\n /// @param description Description of NFT in metadata\\n /// @param mediaData Data for media to include in json object\\n /// @param tokenOfEdition Token ID for specific token\\n /// @param editionSize Size of entire edition to show\\n function createMetadataJSON(\\n string memory name,\\n string memory description,\\n string memory mediaData,\\n uint256 tokenOfEdition,\\n uint256 editionSize\\n ) public pure returns (bytes memory) {\\n bytes memory editionSizeText;\\n if (editionSize > 0) {\\n editionSizeText = abi.encodePacked(\\n \\\"/\\\",\\n numberToString(editionSize)\\n );\\n }\\n return\\n // solhint-disable quotes\\n abi.encodePacked(\\n '{\\\"name\\\": \\\"',\\n name,\\n \\\" \\\",\\n numberToString(tokenOfEdition),\\n editionSizeText,\\n '\\\", \\\"',\\n 'description\\\": \\\"',\\n description,\\n '\\\", \\\"',\\n mediaData,\\n 'properties\\\": {\\\"number\\\": ',\\n numberToString(tokenOfEdition),\\n ',',\\n '\\\"}}'\\n );\\n // solhint-enable quotes\\n }\\n\\n /// Encodes the argument json bytes into base64-data uri format\\n /// @param json Raw json to base64 and turn into a data-uri\\n function encodeMetadataJSON(bytes memory json)\\n public\\n pure\\n override\\n returns (string memory)\\n {\\n return\\n string(\\n abi.encodePacked(\\n \\\"data:application/json;base64,\\\",\\n base64Encode(json)\\n )\\n );\\n }\\n\\n /// Generates edition metadata from storage information as base64-json blob\\n /// Combines the media data and metadata\\n /// @param imageUrl URL of image to render for edition\\n /// @param animationUrl URL of animation to render for edition\\n function tokenMediaData(\\n string memory imageUrl,\\n string memory animationUrl,\\n uint256 tokenOfEdition\\n ) public pure returns (string memory) {\\n bool hasImage = bytes(imageUrl).length > 0;\\n bool hasAnimation = bytes(animationUrl).length > 0;\\n if (hasImage && hasAnimation) {\\n return\\n // solhint-disable quotes\\n string(\\n abi.encodePacked(\\n 'image\\\": \\\"',\\n imageUrl,\\n \\\"?id=\\\",\\n numberToString(tokenOfEdition),\\n '\\\", \\\"animation_url\\\": \\\"',\\n animationUrl,\\n \\\"?id=\\\",\\n numberToString(tokenOfEdition),\\n '\\\", \\\"'\\n )\\n );\\n // solhint-enable quotes\\n }\\n if (hasImage) {\\n return\\n // solhint-disable quotes\\n string(\\n abi.encodePacked(\\n 'image\\\": \\\"',\\n imageUrl,\\n \\\"?id=\\\",\\n numberToString(tokenOfEdition),\\n '\\\", \\\"'\\n )\\n );\\n // solhint-enable quotes\\n }\\n if (hasAnimation) {\\n return\\n // solhint-disable quotes\\n string(\\n abi.encodePacked(\\n 'animation_url\\\": \\\"',\\n animationUrl,\\n \\\"?id=\\\",\\n numberToString(tokenOfEdition),\\n '\\\", \\\"'\\n )\\n );\\n // solhint-enable quotes\\n }\\n\\n return \\\"\\\";\\n }\\n}\\n\",\"keccak256\":\"0xb1c53433d1f74e358472e64cbb82db20af76022599c34147ce42d458fc4647b8\",\"license\":\"GPL-3.0\"}},\"version\":1}", - "bytecode": "0x608060405234801561001057600080fd5b5060405161062338038061062383398101604081905261002f91610054565b600180546001600160a01b0319166001600160a01b0392909216919091179055610084565b60006020828403121561006657600080fd5b81516001600160a01b038116811461007d57600080fd5b9392505050565b610590806100936000396000f3fe608060405234801561001057600080fd5b50600436106100415760003560e01c80635c60da1b146100465780636cdf1b3914610076578063c473a8f314610097575b600080fd5b600154610059906001600160a01b031681565b6040516001600160a01b0390911681526020015b60405180910390f35b6100896100843660046103f2565b6100aa565b60405190815260200161006d565b6100596100a536600461047c565b610216565b60008082116100f85760405162461bcd60e51b8152602060048201526015602482015274044726f702073697a65206d757374206265203e203605c1b60448201526064015b60405180910390fd5b600154600090610142906001600160a01b0316610113835490565b60405160200161012591815260200190565b60405160208183030381529060405261013d90610495565b6102b5565b6040516393d2392360e01b81529091506001600160a01b038216906393d23923906101799033908a908a908a908a90600401610509565b600060405180830381600087803b15801561019357600080fd5b505af11580156101a7573d6000803e3d6000fd5b5050505060006101b660005490565b604080518681526001600160a01b0385166020820152919250339183917f7260749523ea87a695e55bf7fb40009887d330ef68e44c5784725bc97fa4a6d7910160405180910390a361020c600080546001019055565b9695505050505050565b60015460408051602081018490526000926102af926001600160a01b03909116910160405160208183030381529060405261025090610495565b30604051733d602d80600a3d3981f3363d3d373d3d3d363d7360601b8152606093841b60148201526f5af43d82803e903d91602b57fd5bf3ff60801b6028820152921b6038830152604c8201526037808220606c830152605591012090565b92915050565b6000604051733d602d80600a3d3981f3363d3d373d3d3d363d7360601b81528360601b60148201526e5af43d82803e903d91602b57fd5bf360881b6028820152826037826000f59150506001600160a01b0381166102af5760405162461bcd60e51b8152602060048201526017602482015276115490cc4c4d8dce8818dc99585d194c8819985a5b1959604a1b60448201526064016100ef565b634e487b7160e01b600052604160045260246000fd5b600082601f83011261037657600080fd5b813567ffffffffffffffff808211156103915761039161034f565b604051601f8301601f19908116603f011681019082821181831017156103b9576103b961034f565b816040528381528660208588010111156103d257600080fd5b836020870160208301376000602085830101528094505050505092915050565b6000806000806080858703121561040857600080fd5b84356001600160a01b038116811461041f57600080fd5b9350602085013567ffffffffffffffff8082111561043c57600080fd5b61044888838901610365565b9450604087013591508082111561045e57600080fd5b5061046b87828801610365565b949793965093946060013593505050565b60006020828403121561048e57600080fd5b5035919050565b805160208083015191908110156104b6576000198160200360031b1b821691505b50919050565b6000815180845260005b818110156104e2576020818501810151868301820152016104c6565b818111156104f4576000602083870101525b50601f01601f19169290920160200192915050565b6001600160a01b0386811682528516602082015260a060408201819052600090610535908301866104bc565b828103606084015261054781866104bc565b915050826080830152969550505050505056fea26469706673582212205739155a0581f7740f3d374a258961524445c7b6e0ad7bbe8c0894a839deea3464736f6c634300080f0033", - "deployedBytecode": "0x608060405234801561001057600080fd5b50600436106100415760003560e01c80635c60da1b146100465780636cdf1b3914610076578063c473a8f314610097575b600080fd5b600154610059906001600160a01b031681565b6040516001600160a01b0390911681526020015b60405180910390f35b6100896100843660046103f2565b6100aa565b60405190815260200161006d565b6100596100a536600461047c565b610216565b60008082116100f85760405162461bcd60e51b8152602060048201526015602482015274044726f702073697a65206d757374206265203e203605c1b60448201526064015b60405180910390fd5b600154600090610142906001600160a01b0316610113835490565b60405160200161012591815260200190565b60405160208183030381529060405261013d90610495565b6102b5565b6040516393d2392360e01b81529091506001600160a01b038216906393d23923906101799033908a908a908a908a90600401610509565b600060405180830381600087803b15801561019357600080fd5b505af11580156101a7573d6000803e3d6000fd5b5050505060006101b660005490565b604080518681526001600160a01b0385166020820152919250339183917f7260749523ea87a695e55bf7fb40009887d330ef68e44c5784725bc97fa4a6d7910160405180910390a361020c600080546001019055565b9695505050505050565b60015460408051602081018490526000926102af926001600160a01b03909116910160405160208183030381529060405261025090610495565b30604051733d602d80600a3d3981f3363d3d373d3d3d363d7360601b8152606093841b60148201526f5af43d82803e903d91602b57fd5bf3ff60801b6028820152921b6038830152604c8201526037808220606c830152605591012090565b92915050565b6000604051733d602d80600a3d3981f3363d3d373d3d3d363d7360601b81528360601b60148201526e5af43d82803e903d91602b57fd5bf360881b6028820152826037826000f59150506001600160a01b0381166102af5760405162461bcd60e51b8152602060048201526017602482015276115490cc4c4d8dce8818dc99585d194c8819985a5b1959604a1b60448201526064016100ef565b634e487b7160e01b600052604160045260246000fd5b600082601f83011261037657600080fd5b813567ffffffffffffffff808211156103915761039161034f565b604051601f8301601f19908116603f011681019082821181831017156103b9576103b961034f565b816040528381528660208588010111156103d257600080fd5b836020870160208301376000602085830101528094505050505092915050565b6000806000806080858703121561040857600080fd5b84356001600160a01b038116811461041f57600080fd5b9350602085013567ffffffffffffffff8082111561043c57600080fd5b61044888838901610365565b9450604087013591508082111561045e57600080fd5b5061046b87828801610365565b949793965093946060013593505050565b60006020828403121561048e57600080fd5b5035919050565b805160208083015191908110156104b6576000198160200360031b1b821691505b50919050565b6000815180845260005b818110156104e2576020818501810151868301820152016104c6565b818111156104f4576000602083870101525b50601f01601f19169290920160200192915050565b6001600160a01b0386811682528516602082015260a060408201819052600090610535908301866104bc565b828103606084015261054781866104bc565b915050826080830152969550505050505056fea26469706673582212205739155a0581f7740f3d374a258961524445c7b6e0ad7bbe8c0894a839deea3464736f6c634300080f0033", + "solcInputHash": "e6a2ffa4bb274580347e73637376a7e0", + "metadata": "{\"compiler\":{\"version\":\"0.8.15+commit.e14f2714\"},\"language\":\"Solidity\",\"output\":{\"abi\":[{\"inputs\":[{\"internalType\":\"address\",\"name\":\"_implementation\",\"type\":\"address\"}],\"stateMutability\":\"nonpayable\",\"type\":\"constructor\"},{\"anonymous\":false,\"inputs\":[{\"indexed\":true,\"internalType\":\"uint256\",\"name\":\"dropId\",\"type\":\"uint256\"},{\"indexed\":true,\"internalType\":\"address\",\"name\":\"creator\",\"type\":\"address\"},{\"indexed\":false,\"internalType\":\"uint256\",\"name\":\"dropSize\",\"type\":\"uint256\"},{\"indexed\":false,\"internalType\":\"address\",\"name\":\"dropContractAddress\",\"type\":\"address\"}],\"name\":\"CreatedDrop\",\"type\":\"event\"},{\"inputs\":[{\"internalType\":\"address\",\"name\":\"_artistWallet\",\"type\":\"address\"},{\"internalType\":\"string\",\"name\":\"_name\",\"type\":\"string\"},{\"internalType\":\"string\",\"name\":\"_symbol\",\"type\":\"string\"},{\"internalType\":\"uint256\",\"name\":\"_dropSize\",\"type\":\"uint256\"}],\"name\":\"createDrop\",\"outputs\":[{\"internalType\":\"uint256\",\"name\":\"\",\"type\":\"uint256\"}],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"uint256\",\"name\":\"dropId\",\"type\":\"uint256\"}],\"name\":\"getDropAtId\",\"outputs\":[{\"internalType\":\"contract ExpandedNFT\",\"name\":\"\",\"type\":\"address\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[],\"name\":\"implementation\",\"outputs\":[{\"internalType\":\"address\",\"name\":\"\",\"type\":\"address\"}],\"stateMutability\":\"view\",\"type\":\"function\"}],\"devdoc\":{\"events\":{\"CreatedDrop(uint256,address,uint256,address)\":{\"params\":{\"dropId\":\"ID of newly created drop\"}}},\"kind\":\"dev\",\"methods\":{\"constructor\":{\"params\":{\"_implementation\":\"ExpandedNFT logic implementation contract to clone\"}},\"createDrop(address,string,string,uint256)\":{\"params\":{\"_artistWallet\":\"User that created the drop\",\"_name\":\"Name of the drop contract\",\"_symbol\":\"Symbol of the drop contract\"}},\"getDropAtId(uint256)\":{\"params\":{\"dropId\":\"id of drop to get contract for\"},\"returns\":{\"_0\":\"ExpandedNFT Drop NFT contract\"}}},\"version\":1},\"userdoc\":{\"events\":{\"CreatedDrop(uint256,address,uint256,address)\":{\"notice\":\"Emitted when a drop is created reserving the corresponding token IDs.\"}},\"kind\":\"user\",\"methods\":{\"constructor\":{\"notice\":\"Initializes factory with address of implementation logic\"},\"createDrop(address,string,string,uint256)\":{\"notice\":\"Creates a new drop contract as a factory with a deterministic address Important: None of these fields (except the Url fields with the same hash) can be changed after calling\"},\"getDropAtId(uint256)\":{\"notice\":\"Get drop given the created ID\"},\"implementation()\":{\"notice\":\"Address for implementation of ExpandedNFT to clone\"}},\"version\":1}},\"settings\":{\"compilationTarget\":{\"contracts/DropCreator.sol\":\"DropCreator\"},\"evmVersion\":\"london\",\"libraries\":{},\"metadata\":{\"bytecodeHash\":\"ipfs\",\"useLiteralContent\":true},\"optimizer\":{\"enabled\":true,\"runs\":100},\"remappings\":[]},\"sources\":{\"@openzeppelin/contracts-upgradeable/access/OwnableUpgradeable.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n// OpenZeppelin Contracts (last updated v4.7.0) (access/Ownable.sol)\\n\\npragma solidity ^0.8.0;\\n\\nimport \\\"../utils/ContextUpgradeable.sol\\\";\\nimport \\\"../proxy/utils/Initializable.sol\\\";\\n\\n/**\\n * @dev Contract module which provides a basic access control mechanism, where\\n * there is an account (an owner) that can be granted exclusive access to\\n * specific functions.\\n *\\n * By default, the owner account will be the one that deploys the contract. This\\n * can later be changed with {transferOwnership}.\\n *\\n * This module is used through inheritance. It will make available the modifier\\n * `onlyOwner`, which can be applied to your functions to restrict their use to\\n * the owner.\\n */\\nabstract contract OwnableUpgradeable is Initializable, ContextUpgradeable {\\n address private _owner;\\n\\n event OwnershipTransferred(address indexed previousOwner, address indexed newOwner);\\n\\n /**\\n * @dev Initializes the contract setting the deployer as the initial owner.\\n */\\n function __Ownable_init() internal onlyInitializing {\\n __Ownable_init_unchained();\\n }\\n\\n function __Ownable_init_unchained() internal onlyInitializing {\\n _transferOwnership(_msgSender());\\n }\\n\\n /**\\n * @dev Throws if called by any account other than the owner.\\n */\\n modifier onlyOwner() {\\n _checkOwner();\\n _;\\n }\\n\\n /**\\n * @dev Returns the address of the current owner.\\n */\\n function owner() public view virtual returns (address) {\\n return _owner;\\n }\\n\\n /**\\n * @dev Throws if the sender is not the owner.\\n */\\n function _checkOwner() internal view virtual {\\n require(owner() == _msgSender(), \\\"Ownable: caller is not the owner\\\");\\n }\\n\\n /**\\n * @dev Leaves the contract without owner. It will not be possible to call\\n * `onlyOwner` functions anymore. Can only be called by the current owner.\\n *\\n * NOTE: Renouncing ownership will leave the contract without an owner,\\n * thereby removing any functionality that is only available to the owner.\\n */\\n function renounceOwnership() public virtual onlyOwner {\\n _transferOwnership(address(0));\\n }\\n\\n /**\\n * @dev Transfers ownership of the contract to a new account (`newOwner`).\\n * Can only be called by the current owner.\\n */\\n function transferOwnership(address newOwner) public virtual onlyOwner {\\n require(newOwner != address(0), \\\"Ownable: new owner is the zero address\\\");\\n _transferOwnership(newOwner);\\n }\\n\\n /**\\n * @dev Transfers ownership of the contract to a new account (`newOwner`).\\n * Internal function without access restriction.\\n */\\n function _transferOwnership(address newOwner) internal virtual {\\n address oldOwner = _owner;\\n _owner = newOwner;\\n emit OwnershipTransferred(oldOwner, newOwner);\\n }\\n\\n /**\\n * @dev This empty reserved space is put in place to allow future versions to add new\\n * variables without shifting down storage in the inheritance chain.\\n * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps\\n */\\n uint256[49] private __gap;\\n}\\n\",\"keccak256\":\"0x247c62047745915c0af6b955470a72d1696ebad4352d7d3011aef1a2463cd888\",\"license\":\"MIT\"},\"@openzeppelin/contracts-upgradeable/interfaces/IERC2981Upgradeable.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n// OpenZeppelin Contracts (last updated v4.6.0) (interfaces/IERC2981.sol)\\n\\npragma solidity ^0.8.0;\\n\\nimport \\\"../utils/introspection/IERC165Upgradeable.sol\\\";\\n\\n/**\\n * @dev Interface for the NFT Royalty Standard.\\n *\\n * A standardized way to retrieve royalty payment information for non-fungible tokens (NFTs) to enable universal\\n * support for royalty payments across all NFT marketplaces and ecosystem participants.\\n *\\n * _Available since v4.5._\\n */\\ninterface IERC2981Upgradeable is IERC165Upgradeable {\\n /**\\n * @dev Returns how much royalty is owed and to whom, based on a sale price that may be denominated in any unit of\\n * exchange. The royalty amount is denominated and should be paid in that same unit of exchange.\\n */\\n function royaltyInfo(uint256 tokenId, uint256 salePrice)\\n external\\n view\\n returns (address receiver, uint256 royaltyAmount);\\n}\\n\",\"keccak256\":\"0xa8ff557539dcfed5706eddde2aa929e06bb1764e71aa8c1048a78970bf3ca37d\",\"license\":\"MIT\"},\"@openzeppelin/contracts-upgradeable/proxy/ClonesUpgradeable.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n// OpenZeppelin Contracts (last updated v4.7.0) (proxy/Clones.sol)\\n\\npragma solidity ^0.8.0;\\n\\n/**\\n * @dev https://eips.ethereum.org/EIPS/eip-1167[EIP 1167] is a standard for\\n * deploying minimal proxy contracts, also known as \\\"clones\\\".\\n *\\n * > To simply and cheaply clone contract functionality in an immutable way, this standard specifies\\n * > a minimal bytecode implementation that delegates all calls to a known, fixed address.\\n *\\n * The library includes functions to deploy a proxy using either `create` (traditional deployment) or `create2`\\n * (salted deterministic deployment). It also includes functions to predict the addresses of clones deployed using the\\n * deterministic method.\\n *\\n * _Available since v3.4._\\n */\\nlibrary ClonesUpgradeable {\\n /**\\n * @dev Deploys and returns the address of a clone that mimics the behaviour of `implementation`.\\n *\\n * This function uses the create opcode, which should never revert.\\n */\\n function clone(address implementation) internal returns (address instance) {\\n /// @solidity memory-safe-assembly\\n assembly {\\n let ptr := mload(0x40)\\n mstore(ptr, 0x3d602d80600a3d3981f3363d3d373d3d3d363d73000000000000000000000000)\\n mstore(add(ptr, 0x14), shl(0x60, implementation))\\n mstore(add(ptr, 0x28), 0x5af43d82803e903d91602b57fd5bf30000000000000000000000000000000000)\\n instance := create(0, ptr, 0x37)\\n }\\n require(instance != address(0), \\\"ERC1167: create failed\\\");\\n }\\n\\n /**\\n * @dev Deploys and returns the address of a clone that mimics the behaviour of `implementation`.\\n *\\n * This function uses the create2 opcode and a `salt` to deterministically deploy\\n * the clone. Using the same `implementation` and `salt` multiple time will revert, since\\n * the clones cannot be deployed twice at the same address.\\n */\\n function cloneDeterministic(address implementation, bytes32 salt) internal returns (address instance) {\\n /// @solidity memory-safe-assembly\\n assembly {\\n let ptr := mload(0x40)\\n mstore(ptr, 0x3d602d80600a3d3981f3363d3d373d3d3d363d73000000000000000000000000)\\n mstore(add(ptr, 0x14), shl(0x60, implementation))\\n mstore(add(ptr, 0x28), 0x5af43d82803e903d91602b57fd5bf30000000000000000000000000000000000)\\n instance := create2(0, ptr, 0x37, salt)\\n }\\n require(instance != address(0), \\\"ERC1167: create2 failed\\\");\\n }\\n\\n /**\\n * @dev Computes the address of a clone deployed using {Clones-cloneDeterministic}.\\n */\\n function predictDeterministicAddress(\\n address implementation,\\n bytes32 salt,\\n address deployer\\n ) internal pure returns (address predicted) {\\n /// @solidity memory-safe-assembly\\n assembly {\\n let ptr := mload(0x40)\\n mstore(ptr, 0x3d602d80600a3d3981f3363d3d373d3d3d363d73000000000000000000000000)\\n mstore(add(ptr, 0x14), shl(0x60, implementation))\\n mstore(add(ptr, 0x28), 0x5af43d82803e903d91602b57fd5bf3ff00000000000000000000000000000000)\\n mstore(add(ptr, 0x38), shl(0x60, deployer))\\n mstore(add(ptr, 0x4c), salt)\\n mstore(add(ptr, 0x6c), keccak256(ptr, 0x37))\\n predicted := keccak256(add(ptr, 0x37), 0x55)\\n }\\n }\\n\\n /**\\n * @dev Computes the address of a clone deployed using {Clones-cloneDeterministic}.\\n */\\n function predictDeterministicAddress(address implementation, bytes32 salt)\\n internal\\n view\\n returns (address predicted)\\n {\\n return predictDeterministicAddress(implementation, salt, address(this));\\n }\\n}\\n\",\"keccak256\":\"0x3734e36dc4de32780bfd344a94b85b6aab985fd6e42672983ed251ba7754a2e0\",\"license\":\"MIT\"},\"@openzeppelin/contracts-upgradeable/proxy/utils/Initializable.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n// OpenZeppelin Contracts (last updated v4.7.0) (proxy/utils/Initializable.sol)\\n\\npragma solidity ^0.8.2;\\n\\nimport \\\"../../utils/AddressUpgradeable.sol\\\";\\n\\n/**\\n * @dev This is a base contract to aid in writing upgradeable contracts, or any kind of contract that will be deployed\\n * behind a proxy. Since proxied contracts do not make use of a constructor, it's common to move constructor logic to an\\n * external initializer function, usually called `initialize`. It then becomes necessary to protect this initializer\\n * function so it can only be called once. The {initializer} modifier provided by this contract will have this effect.\\n *\\n * The initialization functions use a version number. Once a version number is used, it is consumed and cannot be\\n * reused. This mechanism prevents re-execution of each \\\"step\\\" but allows the creation of new initialization steps in\\n * case an upgrade adds a module that needs to be initialized.\\n *\\n * For example:\\n *\\n * [.hljs-theme-light.nopadding]\\n * ```\\n * contract MyToken is ERC20Upgradeable {\\n * function initialize() initializer public {\\n * __ERC20_init(\\\"MyToken\\\", \\\"MTK\\\");\\n * }\\n * }\\n * contract MyTokenV2 is MyToken, ERC20PermitUpgradeable {\\n * function initializeV2() reinitializer(2) public {\\n * __ERC20Permit_init(\\\"MyToken\\\");\\n * }\\n * }\\n * ```\\n *\\n * TIP: To avoid leaving the proxy in an uninitialized state, the initializer function should be called as early as\\n * possible by providing the encoded function call as the `_data` argument to {ERC1967Proxy-constructor}.\\n *\\n * CAUTION: When used with inheritance, manual care must be taken to not invoke a parent initializer twice, or to ensure\\n * that all initializers are idempotent. This is not verified automatically as constructors are by Solidity.\\n *\\n * [CAUTION]\\n * ====\\n * Avoid leaving a contract uninitialized.\\n *\\n * An uninitialized contract can be taken over by an attacker. This applies to both a proxy and its implementation\\n * contract, which may impact the proxy. To prevent the implementation contract from being used, you should invoke\\n * the {_disableInitializers} function in the constructor to automatically lock it when it is deployed:\\n *\\n * [.hljs-theme-light.nopadding]\\n * ```\\n * /// @custom:oz-upgrades-unsafe-allow constructor\\n * constructor() {\\n * _disableInitializers();\\n * }\\n * ```\\n * ====\\n */\\nabstract contract Initializable {\\n /**\\n * @dev Indicates that the contract has been initialized.\\n * @custom:oz-retyped-from bool\\n */\\n uint8 private _initialized;\\n\\n /**\\n * @dev Indicates that the contract is in the process of being initialized.\\n */\\n bool private _initializing;\\n\\n /**\\n * @dev Triggered when the contract has been initialized or reinitialized.\\n */\\n event Initialized(uint8 version);\\n\\n /**\\n * @dev A modifier that defines a protected initializer function that can be invoked at most once. In its scope,\\n * `onlyInitializing` functions can be used to initialize parent contracts. Equivalent to `reinitializer(1)`.\\n */\\n modifier initializer() {\\n bool isTopLevelCall = !_initializing;\\n require(\\n (isTopLevelCall && _initialized < 1) || (!AddressUpgradeable.isContract(address(this)) && _initialized == 1),\\n \\\"Initializable: contract is already initialized\\\"\\n );\\n _initialized = 1;\\n if (isTopLevelCall) {\\n _initializing = true;\\n }\\n _;\\n if (isTopLevelCall) {\\n _initializing = false;\\n emit Initialized(1);\\n }\\n }\\n\\n /**\\n * @dev A modifier that defines a protected reinitializer function that can be invoked at most once, and only if the\\n * contract hasn't been initialized to a greater version before. In its scope, `onlyInitializing` functions can be\\n * used to initialize parent contracts.\\n *\\n * `initializer` is equivalent to `reinitializer(1)`, so a reinitializer may be used after the original\\n * initialization step. This is essential to configure modules that are added through upgrades and that require\\n * initialization.\\n *\\n * Note that versions can jump in increments greater than 1; this implies that if multiple reinitializers coexist in\\n * a contract, executing them in the right order is up to the developer or operator.\\n */\\n modifier reinitializer(uint8 version) {\\n require(!_initializing && _initialized < version, \\\"Initializable: contract is already initialized\\\");\\n _initialized = version;\\n _initializing = true;\\n _;\\n _initializing = false;\\n emit Initialized(version);\\n }\\n\\n /**\\n * @dev Modifier to protect an initialization function so that it can only be invoked by functions with the\\n * {initializer} and {reinitializer} modifiers, directly or indirectly.\\n */\\n modifier onlyInitializing() {\\n require(_initializing, \\\"Initializable: contract is not initializing\\\");\\n _;\\n }\\n\\n /**\\n * @dev Locks the contract, preventing any future reinitialization. This cannot be part of an initializer call.\\n * Calling this in the constructor of a contract will prevent that contract from being initialized or reinitialized\\n * to any version. It is recommended to use this to lock implementation contracts that are designed to be called\\n * through proxies.\\n */\\n function _disableInitializers() internal virtual {\\n require(!_initializing, \\\"Initializable: contract is initializing\\\");\\n if (_initialized < type(uint8).max) {\\n _initialized = type(uint8).max;\\n emit Initialized(type(uint8).max);\\n }\\n }\\n}\\n\",\"keccak256\":\"0x0203dcadc5737d9ef2c211d6fa15d18ebc3b30dfa51903b64870b01a062b0b4e\",\"license\":\"MIT\"},\"@openzeppelin/contracts-upgradeable/token/ERC20/IERC20Upgradeable.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n// OpenZeppelin Contracts (last updated v4.6.0) (token/ERC20/IERC20.sol)\\n\\npragma solidity ^0.8.0;\\n\\n/**\\n * @dev Interface of the ERC20 standard as defined in the EIP.\\n */\\ninterface IERC20Upgradeable {\\n /**\\n * @dev Emitted when `value` tokens are moved from one account (`from`) to\\n * another (`to`).\\n *\\n * Note that `value` may be zero.\\n */\\n event Transfer(address indexed from, address indexed to, uint256 value);\\n\\n /**\\n * @dev Emitted when the allowance of a `spender` for an `owner` is set by\\n * a call to {approve}. `value` is the new allowance.\\n */\\n event Approval(address indexed owner, address indexed spender, uint256 value);\\n\\n /**\\n * @dev Returns the amount of tokens in existence.\\n */\\n function totalSupply() external view returns (uint256);\\n\\n /**\\n * @dev Returns the amount of tokens owned by `account`.\\n */\\n function balanceOf(address account) external view returns (uint256);\\n\\n /**\\n * @dev Moves `amount` tokens from the caller's account to `to`.\\n *\\n * Returns a boolean value indicating whether the operation succeeded.\\n *\\n * Emits a {Transfer} event.\\n */\\n function transfer(address to, uint256 amount) external returns (bool);\\n\\n /**\\n * @dev Returns the remaining number of tokens that `spender` will be\\n * allowed to spend on behalf of `owner` through {transferFrom}. This is\\n * zero by default.\\n *\\n * This value changes when {approve} or {transferFrom} are called.\\n */\\n function allowance(address owner, address spender) external view returns (uint256);\\n\\n /**\\n * @dev Sets `amount` as the allowance of `spender` over the caller's tokens.\\n *\\n * Returns a boolean value indicating whether the operation succeeded.\\n *\\n * IMPORTANT: Beware that changing an allowance with this method brings the risk\\n * that someone may use both the old and the new allowance by unfortunate\\n * transaction ordering. One possible solution to mitigate this race\\n * condition is to first reduce the spender's allowance to 0 and set the\\n * desired value afterwards:\\n * https://github.com/ethereum/EIPs/issues/20#issuecomment-263524729\\n *\\n * Emits an {Approval} event.\\n */\\n function approve(address spender, uint256 amount) external returns (bool);\\n\\n /**\\n * @dev Moves `amount` tokens from `from` to `to` using the\\n * allowance mechanism. `amount` is then deducted from the caller's\\n * allowance.\\n *\\n * Returns a boolean value indicating whether the operation succeeded.\\n *\\n * Emits a {Transfer} event.\\n */\\n function transferFrom(\\n address from,\\n address to,\\n uint256 amount\\n ) external returns (bool);\\n}\\n\",\"keccak256\":\"0x4e733d3164f73f461eaf9d8087a7ad1ea180bdc8ba0d3d61b0e1ae16d8e63dff\",\"license\":\"MIT\"},\"@openzeppelin/contracts-upgradeable/token/ERC721/ERC721Upgradeable.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n// OpenZeppelin Contracts (last updated v4.7.0) (token/ERC721/ERC721.sol)\\n\\npragma solidity ^0.8.0;\\n\\nimport \\\"./IERC721Upgradeable.sol\\\";\\nimport \\\"./IERC721ReceiverUpgradeable.sol\\\";\\nimport \\\"./extensions/IERC721MetadataUpgradeable.sol\\\";\\nimport \\\"../../utils/AddressUpgradeable.sol\\\";\\nimport \\\"../../utils/ContextUpgradeable.sol\\\";\\nimport \\\"../../utils/StringsUpgradeable.sol\\\";\\nimport \\\"../../utils/introspection/ERC165Upgradeable.sol\\\";\\nimport \\\"../../proxy/utils/Initializable.sol\\\";\\n\\n/**\\n * @dev Implementation of https://eips.ethereum.org/EIPS/eip-721[ERC721] Non-Fungible Token Standard, including\\n * the Metadata extension, but not including the Enumerable extension, which is available separately as\\n * {ERC721Enumerable}.\\n */\\ncontract ERC721Upgradeable is Initializable, ContextUpgradeable, ERC165Upgradeable, IERC721Upgradeable, IERC721MetadataUpgradeable {\\n using AddressUpgradeable for address;\\n using StringsUpgradeable for uint256;\\n\\n // Token name\\n string private _name;\\n\\n // Token symbol\\n string private _symbol;\\n\\n // Mapping from token ID to owner address\\n mapping(uint256 => address) private _owners;\\n\\n // Mapping owner address to token count\\n mapping(address => uint256) private _balances;\\n\\n // Mapping from token ID to approved address\\n mapping(uint256 => address) private _tokenApprovals;\\n\\n // Mapping from owner to operator approvals\\n mapping(address => mapping(address => bool)) private _operatorApprovals;\\n\\n /**\\n * @dev Initializes the contract by setting a `name` and a `symbol` to the token collection.\\n */\\n function __ERC721_init(string memory name_, string memory symbol_) internal onlyInitializing {\\n __ERC721_init_unchained(name_, symbol_);\\n }\\n\\n function __ERC721_init_unchained(string memory name_, string memory symbol_) internal onlyInitializing {\\n _name = name_;\\n _symbol = symbol_;\\n }\\n\\n /**\\n * @dev See {IERC165-supportsInterface}.\\n */\\n function supportsInterface(bytes4 interfaceId) public view virtual override(ERC165Upgradeable, IERC165Upgradeable) returns (bool) {\\n return\\n interfaceId == type(IERC721Upgradeable).interfaceId ||\\n interfaceId == type(IERC721MetadataUpgradeable).interfaceId ||\\n super.supportsInterface(interfaceId);\\n }\\n\\n /**\\n * @dev See {IERC721-balanceOf}.\\n */\\n function balanceOf(address owner) public view virtual override returns (uint256) {\\n require(owner != address(0), \\\"ERC721: address zero is not a valid owner\\\");\\n return _balances[owner];\\n }\\n\\n /**\\n * @dev See {IERC721-ownerOf}.\\n */\\n function ownerOf(uint256 tokenId) public view virtual override returns (address) {\\n address owner = _owners[tokenId];\\n require(owner != address(0), \\\"ERC721: invalid token ID\\\");\\n return owner;\\n }\\n\\n /**\\n * @dev See {IERC721Metadata-name}.\\n */\\n function name() public view virtual override returns (string memory) {\\n return _name;\\n }\\n\\n /**\\n * @dev See {IERC721Metadata-symbol}.\\n */\\n function symbol() public view virtual override returns (string memory) {\\n return _symbol;\\n }\\n\\n /**\\n * @dev See {IERC721Metadata-tokenURI}.\\n */\\n function tokenURI(uint256 tokenId) public view virtual override returns (string memory) {\\n _requireMinted(tokenId);\\n\\n string memory baseURI = _baseURI();\\n return bytes(baseURI).length > 0 ? string(abi.encodePacked(baseURI, tokenId.toString())) : \\\"\\\";\\n }\\n\\n /**\\n * @dev Base URI for computing {tokenURI}. If set, the resulting URI for each\\n * token will be the concatenation of the `baseURI` and the `tokenId`. Empty\\n * by default, can be overridden in child contracts.\\n */\\n function _baseURI() internal view virtual returns (string memory) {\\n return \\\"\\\";\\n }\\n\\n /**\\n * @dev See {IERC721-approve}.\\n */\\n function approve(address to, uint256 tokenId) public virtual override {\\n address owner = ERC721Upgradeable.ownerOf(tokenId);\\n require(to != owner, \\\"ERC721: approval to current owner\\\");\\n\\n require(\\n _msgSender() == owner || isApprovedForAll(owner, _msgSender()),\\n \\\"ERC721: approve caller is not token owner nor approved for all\\\"\\n );\\n\\n _approve(to, tokenId);\\n }\\n\\n /**\\n * @dev See {IERC721-getApproved}.\\n */\\n function getApproved(uint256 tokenId) public view virtual override returns (address) {\\n _requireMinted(tokenId);\\n\\n return _tokenApprovals[tokenId];\\n }\\n\\n /**\\n * @dev See {IERC721-setApprovalForAll}.\\n */\\n function setApprovalForAll(address operator, bool approved) public virtual override {\\n _setApprovalForAll(_msgSender(), operator, approved);\\n }\\n\\n /**\\n * @dev See {IERC721-isApprovedForAll}.\\n */\\n function isApprovedForAll(address owner, address operator) public view virtual override returns (bool) {\\n return _operatorApprovals[owner][operator];\\n }\\n\\n /**\\n * @dev See {IERC721-transferFrom}.\\n */\\n function transferFrom(\\n address from,\\n address to,\\n uint256 tokenId\\n ) public virtual override {\\n //solhint-disable-next-line max-line-length\\n require(_isApprovedOrOwner(_msgSender(), tokenId), \\\"ERC721: caller is not token owner nor approved\\\");\\n\\n _transfer(from, to, tokenId);\\n }\\n\\n /**\\n * @dev See {IERC721-safeTransferFrom}.\\n */\\n function safeTransferFrom(\\n address from,\\n address to,\\n uint256 tokenId\\n ) public virtual override {\\n safeTransferFrom(from, to, tokenId, \\\"\\\");\\n }\\n\\n /**\\n * @dev See {IERC721-safeTransferFrom}.\\n */\\n function safeTransferFrom(\\n address from,\\n address to,\\n uint256 tokenId,\\n bytes memory data\\n ) public virtual override {\\n require(_isApprovedOrOwner(_msgSender(), tokenId), \\\"ERC721: caller is not token owner nor approved\\\");\\n _safeTransfer(from, to, tokenId, data);\\n }\\n\\n /**\\n * @dev Safely transfers `tokenId` token from `from` to `to`, checking first that contract recipients\\n * are aware of the ERC721 protocol to prevent tokens from being forever locked.\\n *\\n * `data` is additional data, it has no specified format and it is sent in call to `to`.\\n *\\n * This internal function is equivalent to {safeTransferFrom}, and can be used to e.g.\\n * implement alternative mechanisms to perform token transfer, such as signature-based.\\n *\\n * Requirements:\\n *\\n * - `from` cannot be the zero address.\\n * - `to` cannot be the zero address.\\n * - `tokenId` token must exist and be owned by `from`.\\n * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer.\\n *\\n * Emits a {Transfer} event.\\n */\\n function _safeTransfer(\\n address from,\\n address to,\\n uint256 tokenId,\\n bytes memory data\\n ) internal virtual {\\n _transfer(from, to, tokenId);\\n require(_checkOnERC721Received(from, to, tokenId, data), \\\"ERC721: transfer to non ERC721Receiver implementer\\\");\\n }\\n\\n /**\\n * @dev Returns whether `tokenId` exists.\\n *\\n * Tokens can be managed by their owner or approved accounts via {approve} or {setApprovalForAll}.\\n *\\n * Tokens start existing when they are minted (`_mint`),\\n * and stop existing when they are burned (`_burn`).\\n */\\n function _exists(uint256 tokenId) internal view virtual returns (bool) {\\n return _owners[tokenId] != address(0);\\n }\\n\\n /**\\n * @dev Returns whether `spender` is allowed to manage `tokenId`.\\n *\\n * Requirements:\\n *\\n * - `tokenId` must exist.\\n */\\n function _isApprovedOrOwner(address spender, uint256 tokenId) internal view virtual returns (bool) {\\n address owner = ERC721Upgradeable.ownerOf(tokenId);\\n return (spender == owner || isApprovedForAll(owner, spender) || getApproved(tokenId) == spender);\\n }\\n\\n /**\\n * @dev Safely mints `tokenId` and transfers it to `to`.\\n *\\n * Requirements:\\n *\\n * - `tokenId` must not exist.\\n * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer.\\n *\\n * Emits a {Transfer} event.\\n */\\n function _safeMint(address to, uint256 tokenId) internal virtual {\\n _safeMint(to, tokenId, \\\"\\\");\\n }\\n\\n /**\\n * @dev Same as {xref-ERC721-_safeMint-address-uint256-}[`_safeMint`], with an additional `data` parameter which is\\n * forwarded in {IERC721Receiver-onERC721Received} to contract recipients.\\n */\\n function _safeMint(\\n address to,\\n uint256 tokenId,\\n bytes memory data\\n ) internal virtual {\\n _mint(to, tokenId);\\n require(\\n _checkOnERC721Received(address(0), to, tokenId, data),\\n \\\"ERC721: transfer to non ERC721Receiver implementer\\\"\\n );\\n }\\n\\n /**\\n * @dev Mints `tokenId` and transfers it to `to`.\\n *\\n * WARNING: Usage of this method is discouraged, use {_safeMint} whenever possible\\n *\\n * Requirements:\\n *\\n * - `tokenId` must not exist.\\n * - `to` cannot be the zero address.\\n *\\n * Emits a {Transfer} event.\\n */\\n function _mint(address to, uint256 tokenId) internal virtual {\\n require(to != address(0), \\\"ERC721: mint to the zero address\\\");\\n require(!_exists(tokenId), \\\"ERC721: token already minted\\\");\\n\\n _beforeTokenTransfer(address(0), to, tokenId);\\n\\n _balances[to] += 1;\\n _owners[tokenId] = to;\\n\\n emit Transfer(address(0), to, tokenId);\\n\\n _afterTokenTransfer(address(0), to, tokenId);\\n }\\n\\n /**\\n * @dev Destroys `tokenId`.\\n * The approval is cleared when the token is burned.\\n *\\n * Requirements:\\n *\\n * - `tokenId` must exist.\\n *\\n * Emits a {Transfer} event.\\n */\\n function _burn(uint256 tokenId) internal virtual {\\n address owner = ERC721Upgradeable.ownerOf(tokenId);\\n\\n _beforeTokenTransfer(owner, address(0), tokenId);\\n\\n // Clear approvals\\n _approve(address(0), tokenId);\\n\\n _balances[owner] -= 1;\\n delete _owners[tokenId];\\n\\n emit Transfer(owner, address(0), tokenId);\\n\\n _afterTokenTransfer(owner, address(0), tokenId);\\n }\\n\\n /**\\n * @dev Transfers `tokenId` from `from` to `to`.\\n * As opposed to {transferFrom}, this imposes no restrictions on msg.sender.\\n *\\n * Requirements:\\n *\\n * - `to` cannot be the zero address.\\n * - `tokenId` token must be owned by `from`.\\n *\\n * Emits a {Transfer} event.\\n */\\n function _transfer(\\n address from,\\n address to,\\n uint256 tokenId\\n ) internal virtual {\\n require(ERC721Upgradeable.ownerOf(tokenId) == from, \\\"ERC721: transfer from incorrect owner\\\");\\n require(to != address(0), \\\"ERC721: transfer to the zero address\\\");\\n\\n _beforeTokenTransfer(from, to, tokenId);\\n\\n // Clear approvals from the previous owner\\n _approve(address(0), tokenId);\\n\\n _balances[from] -= 1;\\n _balances[to] += 1;\\n _owners[tokenId] = to;\\n\\n emit Transfer(from, to, tokenId);\\n\\n _afterTokenTransfer(from, to, tokenId);\\n }\\n\\n /**\\n * @dev Approve `to` to operate on `tokenId`\\n *\\n * Emits an {Approval} event.\\n */\\n function _approve(address to, uint256 tokenId) internal virtual {\\n _tokenApprovals[tokenId] = to;\\n emit Approval(ERC721Upgradeable.ownerOf(tokenId), to, tokenId);\\n }\\n\\n /**\\n * @dev Approve `operator` to operate on all of `owner` tokens\\n *\\n * Emits an {ApprovalForAll} event.\\n */\\n function _setApprovalForAll(\\n address owner,\\n address operator,\\n bool approved\\n ) internal virtual {\\n require(owner != operator, \\\"ERC721: approve to caller\\\");\\n _operatorApprovals[owner][operator] = approved;\\n emit ApprovalForAll(owner, operator, approved);\\n }\\n\\n /**\\n * @dev Reverts if the `tokenId` has not been minted yet.\\n */\\n function _requireMinted(uint256 tokenId) internal view virtual {\\n require(_exists(tokenId), \\\"ERC721: invalid token ID\\\");\\n }\\n\\n /**\\n * @dev Internal function to invoke {IERC721Receiver-onERC721Received} on a target address.\\n * The call is not executed if the target address is not a contract.\\n *\\n * @param from address representing the previous owner of the given token ID\\n * @param to target address that will receive the tokens\\n * @param tokenId uint256 ID of the token to be transferred\\n * @param data bytes optional data to send along with the call\\n * @return bool whether the call correctly returned the expected magic value\\n */\\n function _checkOnERC721Received(\\n address from,\\n address to,\\n uint256 tokenId,\\n bytes memory data\\n ) private returns (bool) {\\n if (to.isContract()) {\\n try IERC721ReceiverUpgradeable(to).onERC721Received(_msgSender(), from, tokenId, data) returns (bytes4 retval) {\\n return retval == IERC721ReceiverUpgradeable.onERC721Received.selector;\\n } catch (bytes memory reason) {\\n if (reason.length == 0) {\\n revert(\\\"ERC721: transfer to non ERC721Receiver implementer\\\");\\n } else {\\n /// @solidity memory-safe-assembly\\n assembly {\\n revert(add(32, reason), mload(reason))\\n }\\n }\\n }\\n } else {\\n return true;\\n }\\n }\\n\\n /**\\n * @dev Hook that is called before any token transfer. This includes minting\\n * and burning.\\n *\\n * Calling conditions:\\n *\\n * - When `from` and `to` are both non-zero, ``from``'s `tokenId` will be\\n * transferred to `to`.\\n * - When `from` is zero, `tokenId` will be minted for `to`.\\n * - When `to` is zero, ``from``'s `tokenId` will be burned.\\n * - `from` and `to` are never both zero.\\n *\\n * To learn more about hooks, head to xref:ROOT:extending-contracts.adoc#using-hooks[Using Hooks].\\n */\\n function _beforeTokenTransfer(\\n address from,\\n address to,\\n uint256 tokenId\\n ) internal virtual {}\\n\\n /**\\n * @dev Hook that is called after any transfer of tokens. This includes\\n * minting and burning.\\n *\\n * Calling conditions:\\n *\\n * - when `from` and `to` are both non-zero.\\n * - `from` and `to` are never both zero.\\n *\\n * To learn more about hooks, head to xref:ROOT:extending-contracts.adoc#using-hooks[Using Hooks].\\n */\\n function _afterTokenTransfer(\\n address from,\\n address to,\\n uint256 tokenId\\n ) internal virtual {}\\n\\n /**\\n * @dev This empty reserved space is put in place to allow future versions to add new\\n * variables without shifting down storage in the inheritance chain.\\n * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps\\n */\\n uint256[44] private __gap;\\n}\\n\",\"keccak256\":\"0x5331c8909221d9f9f3851cfadd5959d0873413a2c27e30e0f2fa234158c1c6cf\",\"license\":\"MIT\"},\"@openzeppelin/contracts-upgradeable/token/ERC721/IERC721ReceiverUpgradeable.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n// OpenZeppelin Contracts (last updated v4.6.0) (token/ERC721/IERC721Receiver.sol)\\n\\npragma solidity ^0.8.0;\\n\\n/**\\n * @title ERC721 token receiver interface\\n * @dev Interface for any contract that wants to support safeTransfers\\n * from ERC721 asset contracts.\\n */\\ninterface IERC721ReceiverUpgradeable {\\n /**\\n * @dev Whenever an {IERC721} `tokenId` token is transferred to this contract via {IERC721-safeTransferFrom}\\n * by `operator` from `from`, this function is called.\\n *\\n * It must return its Solidity selector to confirm the token transfer.\\n * If any other value is returned or the interface is not implemented by the recipient, the transfer will be reverted.\\n *\\n * The selector can be obtained in Solidity with `IERC721Receiver.onERC721Received.selector`.\\n */\\n function onERC721Received(\\n address operator,\\n address from,\\n uint256 tokenId,\\n bytes calldata data\\n ) external returns (bytes4);\\n}\\n\",\"keccak256\":\"0xbb2ed8106d94aeae6858e2551a1e7174df73994b77b13ebd120ccaaef80155f5\",\"license\":\"MIT\"},\"@openzeppelin/contracts-upgradeable/token/ERC721/IERC721Upgradeable.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n// OpenZeppelin Contracts (last updated v4.7.0) (token/ERC721/IERC721.sol)\\n\\npragma solidity ^0.8.0;\\n\\nimport \\\"../../utils/introspection/IERC165Upgradeable.sol\\\";\\n\\n/**\\n * @dev Required interface of an ERC721 compliant contract.\\n */\\ninterface IERC721Upgradeable is IERC165Upgradeable {\\n /**\\n * @dev Emitted when `tokenId` token is transferred from `from` to `to`.\\n */\\n event Transfer(address indexed from, address indexed to, uint256 indexed tokenId);\\n\\n /**\\n * @dev Emitted when `owner` enables `approved` to manage the `tokenId` token.\\n */\\n event Approval(address indexed owner, address indexed approved, uint256 indexed tokenId);\\n\\n /**\\n * @dev Emitted when `owner` enables or disables (`approved`) `operator` to manage all of its assets.\\n */\\n event ApprovalForAll(address indexed owner, address indexed operator, bool approved);\\n\\n /**\\n * @dev Returns the number of tokens in ``owner``'s account.\\n */\\n function balanceOf(address owner) external view returns (uint256 balance);\\n\\n /**\\n * @dev Returns the owner of the `tokenId` token.\\n *\\n * Requirements:\\n *\\n * - `tokenId` must exist.\\n */\\n function ownerOf(uint256 tokenId) external view returns (address owner);\\n\\n /**\\n * @dev Safely transfers `tokenId` token from `from` to `to`.\\n *\\n * Requirements:\\n *\\n * - `from` cannot be the zero address.\\n * - `to` cannot be the zero address.\\n * - `tokenId` token must exist and be owned by `from`.\\n * - If the caller is not `from`, it must be approved to move this token by either {approve} or {setApprovalForAll}.\\n * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer.\\n *\\n * Emits a {Transfer} event.\\n */\\n function safeTransferFrom(\\n address from,\\n address to,\\n uint256 tokenId,\\n bytes calldata data\\n ) external;\\n\\n /**\\n * @dev Safely transfers `tokenId` token from `from` to `to`, checking first that contract recipients\\n * are aware of the ERC721 protocol to prevent tokens from being forever locked.\\n *\\n * Requirements:\\n *\\n * - `from` cannot be the zero address.\\n * - `to` cannot be the zero address.\\n * - `tokenId` token must exist and be owned by `from`.\\n * - If the caller is not `from`, it must have been allowed to move this token by either {approve} or {setApprovalForAll}.\\n * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer.\\n *\\n * Emits a {Transfer} event.\\n */\\n function safeTransferFrom(\\n address from,\\n address to,\\n uint256 tokenId\\n ) external;\\n\\n /**\\n * @dev Transfers `tokenId` token from `from` to `to`.\\n *\\n * WARNING: Usage of this method is discouraged, use {safeTransferFrom} whenever possible.\\n *\\n * Requirements:\\n *\\n * - `from` cannot be the zero address.\\n * - `to` cannot be the zero address.\\n * - `tokenId` token must be owned by `from`.\\n * - If the caller is not `from`, it must be approved to move this token by either {approve} or {setApprovalForAll}.\\n *\\n * Emits a {Transfer} event.\\n */\\n function transferFrom(\\n address from,\\n address to,\\n uint256 tokenId\\n ) external;\\n\\n /**\\n * @dev Gives permission to `to` to transfer `tokenId` token to another account.\\n * The approval is cleared when the token is transferred.\\n *\\n * Only a single account can be approved at a time, so approving the zero address clears previous approvals.\\n *\\n * Requirements:\\n *\\n * - The caller must own the token or be an approved operator.\\n * - `tokenId` must exist.\\n *\\n * Emits an {Approval} event.\\n */\\n function approve(address to, uint256 tokenId) external;\\n\\n /**\\n * @dev Approve or remove `operator` as an operator for the caller.\\n * Operators can call {transferFrom} or {safeTransferFrom} for any token owned by the caller.\\n *\\n * Requirements:\\n *\\n * - The `operator` cannot be the caller.\\n *\\n * Emits an {ApprovalForAll} event.\\n */\\n function setApprovalForAll(address operator, bool _approved) external;\\n\\n /**\\n * @dev Returns the account approved for `tokenId` token.\\n *\\n * Requirements:\\n *\\n * - `tokenId` must exist.\\n */\\n function getApproved(uint256 tokenId) external view returns (address operator);\\n\\n /**\\n * @dev Returns if the `operator` is allowed to manage all of the assets of `owner`.\\n *\\n * See {setApprovalForAll}\\n */\\n function isApprovedForAll(address owner, address operator) external view returns (bool);\\n}\\n\",\"keccak256\":\"0x016298e66a5810253c6c905e61966bb31c8775c3f3517bf946ff56ee31d6c005\",\"license\":\"MIT\"},\"@openzeppelin/contracts-upgradeable/token/ERC721/extensions/IERC721MetadataUpgradeable.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n// OpenZeppelin Contracts v4.4.1 (token/ERC721/extensions/IERC721Metadata.sol)\\n\\npragma solidity ^0.8.0;\\n\\nimport \\\"../IERC721Upgradeable.sol\\\";\\n\\n/**\\n * @title ERC-721 Non-Fungible Token Standard, optional metadata extension\\n * @dev See https://eips.ethereum.org/EIPS/eip-721\\n */\\ninterface IERC721MetadataUpgradeable is IERC721Upgradeable {\\n /**\\n * @dev Returns the token collection name.\\n */\\n function name() external view returns (string memory);\\n\\n /**\\n * @dev Returns the token collection symbol.\\n */\\n function symbol() external view returns (string memory);\\n\\n /**\\n * @dev Returns the Uniform Resource Identifier (URI) for `tokenId` token.\\n */\\n function tokenURI(uint256 tokenId) external view returns (string memory);\\n}\\n\",\"keccak256\":\"0x95a471796eb5f030fdc438660bebec121ad5d063763e64d92376ffb4b5ce8b70\",\"license\":\"MIT\"},\"@openzeppelin/contracts-upgradeable/utils/AddressUpgradeable.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n// OpenZeppelin Contracts (last updated v4.7.0) (utils/Address.sol)\\n\\npragma solidity ^0.8.1;\\n\\n/**\\n * @dev Collection of functions related to the address type\\n */\\nlibrary AddressUpgradeable {\\n /**\\n * @dev Returns true if `account` is a contract.\\n *\\n * [IMPORTANT]\\n * ====\\n * It is unsafe to assume that an address for which this function returns\\n * false is an externally-owned account (EOA) and not a contract.\\n *\\n * Among others, `isContract` will return false for the following\\n * types of addresses:\\n *\\n * - an externally-owned account\\n * - a contract in construction\\n * - an address where a contract will be created\\n * - an address where a contract lived, but was destroyed\\n * ====\\n *\\n * [IMPORTANT]\\n * ====\\n * You shouldn't rely on `isContract` to protect against flash loan attacks!\\n *\\n * Preventing calls from contracts is highly discouraged. It breaks composability, breaks support for smart wallets\\n * like Gnosis Safe, and does not provide security since it can be circumvented by calling from a contract\\n * constructor.\\n * ====\\n */\\n function isContract(address account) internal view returns (bool) {\\n // This method relies on extcodesize/address.code.length, which returns 0\\n // for contracts in construction, since the code is only stored at the end\\n // of the constructor execution.\\n\\n return account.code.length > 0;\\n }\\n\\n /**\\n * @dev Replacement for Solidity's `transfer`: sends `amount` wei to\\n * `recipient`, forwarding all available gas and reverting on errors.\\n *\\n * https://eips.ethereum.org/EIPS/eip-1884[EIP1884] increases the gas cost\\n * of certain opcodes, possibly making contracts go over the 2300 gas limit\\n * imposed by `transfer`, making them unable to receive funds via\\n * `transfer`. {sendValue} removes this limitation.\\n *\\n * https://diligence.consensys.net/posts/2019/09/stop-using-soliditys-transfer-now/[Learn more].\\n *\\n * IMPORTANT: because control is transferred to `recipient`, care must be\\n * taken to not create reentrancy vulnerabilities. Consider using\\n * {ReentrancyGuard} or the\\n * https://solidity.readthedocs.io/en/v0.5.11/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern].\\n */\\n function sendValue(address payable recipient, uint256 amount) internal {\\n require(address(this).balance >= amount, \\\"Address: insufficient balance\\\");\\n\\n (bool success, ) = recipient.call{value: amount}(\\\"\\\");\\n require(success, \\\"Address: unable to send value, recipient may have reverted\\\");\\n }\\n\\n /**\\n * @dev Performs a Solidity function call using a low level `call`. A\\n * plain `call` is an unsafe replacement for a function call: use this\\n * function instead.\\n *\\n * If `target` reverts with a revert reason, it is bubbled up by this\\n * function (like regular Solidity function calls).\\n *\\n * Returns the raw returned data. To convert to the expected return value,\\n * use https://solidity.readthedocs.io/en/latest/units-and-global-variables.html?highlight=abi.decode#abi-encoding-and-decoding-functions[`abi.decode`].\\n *\\n * Requirements:\\n *\\n * - `target` must be a contract.\\n * - calling `target` with `data` must not revert.\\n *\\n * _Available since v3.1._\\n */\\n function functionCall(address target, bytes memory data) internal returns (bytes memory) {\\n return functionCall(target, data, \\\"Address: low-level call failed\\\");\\n }\\n\\n /**\\n * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], but with\\n * `errorMessage` as a fallback revert reason when `target` reverts.\\n *\\n * _Available since v3.1._\\n */\\n function functionCall(\\n address target,\\n bytes memory data,\\n string memory errorMessage\\n ) internal returns (bytes memory) {\\n return functionCallWithValue(target, data, 0, errorMessage);\\n }\\n\\n /**\\n * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],\\n * but also transferring `value` wei to `target`.\\n *\\n * Requirements:\\n *\\n * - the calling contract must have an ETH balance of at least `value`.\\n * - the called Solidity function must be `payable`.\\n *\\n * _Available since v3.1._\\n */\\n function functionCallWithValue(\\n address target,\\n bytes memory data,\\n uint256 value\\n ) internal returns (bytes memory) {\\n return functionCallWithValue(target, data, value, \\\"Address: low-level call with value failed\\\");\\n }\\n\\n /**\\n * @dev Same as {xref-Address-functionCallWithValue-address-bytes-uint256-}[`functionCallWithValue`], but\\n * with `errorMessage` as a fallback revert reason when `target` reverts.\\n *\\n * _Available since v3.1._\\n */\\n function functionCallWithValue(\\n address target,\\n bytes memory data,\\n uint256 value,\\n string memory errorMessage\\n ) internal returns (bytes memory) {\\n require(address(this).balance >= value, \\\"Address: insufficient balance for call\\\");\\n require(isContract(target), \\\"Address: call to non-contract\\\");\\n\\n (bool success, bytes memory returndata) = target.call{value: value}(data);\\n return verifyCallResult(success, returndata, errorMessage);\\n }\\n\\n /**\\n * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],\\n * but performing a static call.\\n *\\n * _Available since v3.3._\\n */\\n function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) {\\n return functionStaticCall(target, data, \\\"Address: low-level static call failed\\\");\\n }\\n\\n /**\\n * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],\\n * but performing a static call.\\n *\\n * _Available since v3.3._\\n */\\n function functionStaticCall(\\n address target,\\n bytes memory data,\\n string memory errorMessage\\n ) internal view returns (bytes memory) {\\n require(isContract(target), \\\"Address: static call to non-contract\\\");\\n\\n (bool success, bytes memory returndata) = target.staticcall(data);\\n return verifyCallResult(success, returndata, errorMessage);\\n }\\n\\n /**\\n * @dev Tool to verifies that a low level call was successful, and revert if it wasn't, either by bubbling the\\n * revert reason using the provided one.\\n *\\n * _Available since v4.3._\\n */\\n function verifyCallResult(\\n bool success,\\n bytes memory returndata,\\n string memory errorMessage\\n ) internal pure returns (bytes memory) {\\n if (success) {\\n return returndata;\\n } else {\\n // Look for revert reason and bubble it up if present\\n if (returndata.length > 0) {\\n // The easiest way to bubble the revert reason is using memory via assembly\\n /// @solidity memory-safe-assembly\\n assembly {\\n let returndata_size := mload(returndata)\\n revert(add(32, returndata), returndata_size)\\n }\\n } else {\\n revert(errorMessage);\\n }\\n }\\n }\\n}\\n\",\"keccak256\":\"0x611aa3f23e59cfdd1863c536776407b3e33d695152a266fa7cfb34440a29a8a3\",\"license\":\"MIT\"},\"@openzeppelin/contracts-upgradeable/utils/ContextUpgradeable.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n// OpenZeppelin Contracts v4.4.1 (utils/Context.sol)\\n\\npragma solidity ^0.8.0;\\nimport \\\"../proxy/utils/Initializable.sol\\\";\\n\\n/**\\n * @dev Provides information about the current execution context, including the\\n * sender of the transaction and its data. While these are generally available\\n * via msg.sender and msg.data, they should not be accessed in such a direct\\n * manner, since when dealing with meta-transactions the account sending and\\n * paying for execution may not be the actual sender (as far as an application\\n * is concerned).\\n *\\n * This contract is only required for intermediate, library-like contracts.\\n */\\nabstract contract ContextUpgradeable is Initializable {\\n function __Context_init() internal onlyInitializing {\\n }\\n\\n function __Context_init_unchained() internal onlyInitializing {\\n }\\n function _msgSender() internal view virtual returns (address) {\\n return msg.sender;\\n }\\n\\n function _msgData() internal view virtual returns (bytes calldata) {\\n return msg.data;\\n }\\n\\n /**\\n * @dev This empty reserved space is put in place to allow future versions to add new\\n * variables without shifting down storage in the inheritance chain.\\n * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps\\n */\\n uint256[50] private __gap;\\n}\\n\",\"keccak256\":\"0x963ea7f0b48b032eef72fe3a7582edf78408d6f834115b9feadd673a4d5bd149\",\"license\":\"MIT\"},\"@openzeppelin/contracts-upgradeable/utils/CountersUpgradeable.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n// OpenZeppelin Contracts v4.4.1 (utils/Counters.sol)\\n\\npragma solidity ^0.8.0;\\n\\n/**\\n * @title Counters\\n * @author Matt Condon (@shrugs)\\n * @dev Provides counters that can only be incremented, decremented or reset. This can be used e.g. to track the number\\n * of elements in a mapping, issuing ERC721 ids, or counting request ids.\\n *\\n * Include with `using Counters for Counters.Counter;`\\n */\\nlibrary CountersUpgradeable {\\n struct Counter {\\n // This variable should never be directly accessed by users of the library: interactions must be restricted to\\n // the library's function. As of Solidity v0.5.2, this cannot be enforced, though there is a proposal to add\\n // this feature: see https://github.com/ethereum/solidity/issues/4637\\n uint256 _value; // default: 0\\n }\\n\\n function current(Counter storage counter) internal view returns (uint256) {\\n return counter._value;\\n }\\n\\n function increment(Counter storage counter) internal {\\n unchecked {\\n counter._value += 1;\\n }\\n }\\n\\n function decrement(Counter storage counter) internal {\\n uint256 value = counter._value;\\n require(value > 0, \\\"Counter: decrement overflow\\\");\\n unchecked {\\n counter._value = value - 1;\\n }\\n }\\n\\n function reset(Counter storage counter) internal {\\n counter._value = 0;\\n }\\n}\\n\",\"keccak256\":\"0x798741e231b22b81e2dd2eddaaf8832dee4baf5cd8e2dbaa5c1dd12a1c053c4d\",\"license\":\"MIT\"},\"@openzeppelin/contracts-upgradeable/utils/StringsUpgradeable.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n// OpenZeppelin Contracts (last updated v4.7.0) (utils/Strings.sol)\\n\\npragma solidity ^0.8.0;\\n\\n/**\\n * @dev String operations.\\n */\\nlibrary StringsUpgradeable {\\n bytes16 private constant _HEX_SYMBOLS = \\\"0123456789abcdef\\\";\\n uint8 private constant _ADDRESS_LENGTH = 20;\\n\\n /**\\n * @dev Converts a `uint256` to its ASCII `string` decimal representation.\\n */\\n function toString(uint256 value) internal pure returns (string memory) {\\n // Inspired by OraclizeAPI's implementation - MIT licence\\n // https://github.com/oraclize/ethereum-api/blob/b42146b063c7d6ee1358846c198246239e9360e8/oraclizeAPI_0.4.25.sol\\n\\n if (value == 0) {\\n return \\\"0\\\";\\n }\\n uint256 temp = value;\\n uint256 digits;\\n while (temp != 0) {\\n digits++;\\n temp /= 10;\\n }\\n bytes memory buffer = new bytes(digits);\\n while (value != 0) {\\n digits -= 1;\\n buffer[digits] = bytes1(uint8(48 + uint256(value % 10)));\\n value /= 10;\\n }\\n return string(buffer);\\n }\\n\\n /**\\n * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation.\\n */\\n function toHexString(uint256 value) internal pure returns (string memory) {\\n if (value == 0) {\\n return \\\"0x00\\\";\\n }\\n uint256 temp = value;\\n uint256 length = 0;\\n while (temp != 0) {\\n length++;\\n temp >>= 8;\\n }\\n return toHexString(value, length);\\n }\\n\\n /**\\n * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation with fixed length.\\n */\\n function toHexString(uint256 value, uint256 length) internal pure returns (string memory) {\\n bytes memory buffer = new bytes(2 * length + 2);\\n buffer[0] = \\\"0\\\";\\n buffer[1] = \\\"x\\\";\\n for (uint256 i = 2 * length + 1; i > 1; --i) {\\n buffer[i] = _HEX_SYMBOLS[value & 0xf];\\n value >>= 4;\\n }\\n require(value == 0, \\\"Strings: hex length insufficient\\\");\\n return string(buffer);\\n }\\n\\n /**\\n * @dev Converts an `address` with fixed length of 20 bytes to its not checksummed ASCII `string` hexadecimal representation.\\n */\\n function toHexString(address addr) internal pure returns (string memory) {\\n return toHexString(uint256(uint160(addr)), _ADDRESS_LENGTH);\\n }\\n}\\n\",\"keccak256\":\"0xea5339a7fff0ed42b45be56a88efdd0b2ddde9fa480dc99fef9a6a4c5b776863\",\"license\":\"MIT\"},\"@openzeppelin/contracts-upgradeable/utils/introspection/ERC165Upgradeable.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n// OpenZeppelin Contracts v4.4.1 (utils/introspection/ERC165.sol)\\n\\npragma solidity ^0.8.0;\\n\\nimport \\\"./IERC165Upgradeable.sol\\\";\\nimport \\\"../../proxy/utils/Initializable.sol\\\";\\n\\n/**\\n * @dev Implementation of the {IERC165} interface.\\n *\\n * Contracts that want to implement ERC165 should inherit from this contract and override {supportsInterface} to check\\n * for the additional interface id that will be supported. For example:\\n *\\n * ```solidity\\n * function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) {\\n * return interfaceId == type(MyInterface).interfaceId || super.supportsInterface(interfaceId);\\n * }\\n * ```\\n *\\n * Alternatively, {ERC165Storage} provides an easier to use but more expensive implementation.\\n */\\nabstract contract ERC165Upgradeable is Initializable, IERC165Upgradeable {\\n function __ERC165_init() internal onlyInitializing {\\n }\\n\\n function __ERC165_init_unchained() internal onlyInitializing {\\n }\\n /**\\n * @dev See {IERC165-supportsInterface}.\\n */\\n function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) {\\n return interfaceId == type(IERC165Upgradeable).interfaceId;\\n }\\n\\n /**\\n * @dev This empty reserved space is put in place to allow future versions to add new\\n * variables without shifting down storage in the inheritance chain.\\n * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps\\n */\\n uint256[50] private __gap;\\n}\\n\",\"keccak256\":\"0x9a3b990bd56d139df3e454a9edf1c64668530b5a77fc32eb063bc206f958274a\",\"license\":\"MIT\"},\"@openzeppelin/contracts-upgradeable/utils/introspection/IERC165Upgradeable.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n// OpenZeppelin Contracts v4.4.1 (utils/introspection/IERC165.sol)\\n\\npragma solidity ^0.8.0;\\n\\n/**\\n * @dev Interface of the ERC165 standard, as defined in the\\n * https://eips.ethereum.org/EIPS/eip-165[EIP].\\n *\\n * Implementers can declare support of contract interfaces, which can then be\\n * queried by others ({ERC165Checker}).\\n *\\n * For an implementation, see {ERC165}.\\n */\\ninterface IERC165Upgradeable {\\n /**\\n * @dev Returns true if this contract implements the interface defined by\\n * `interfaceId`. See the corresponding\\n * https://eips.ethereum.org/EIPS/eip-165#how-interfaces-are-identified[EIP section]\\n * to learn more about how these ids are created.\\n *\\n * This function call must use less than 30 000 gas.\\n */\\n function supportsInterface(bytes4 interfaceId) external view returns (bool);\\n}\\n\",\"keccak256\":\"0xc6cef87559d0aeffdf0a99803de655938a7779ec0a3cd5d4383483ad85565a09\",\"license\":\"MIT\"},\"base64-sol/base64.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n\\npragma solidity >=0.6.0;\\n\\n/// @title Base64\\n/// @author Brecht Devos - \\n/// @notice Provides functions for encoding/decoding base64\\nlibrary Base64 {\\n string internal constant TABLE_ENCODE = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/';\\n bytes internal constant TABLE_DECODE = hex\\\"0000000000000000000000000000000000000000000000000000000000000000\\\"\\n hex\\\"00000000000000000000003e0000003f3435363738393a3b3c3d000000000000\\\"\\n hex\\\"00000102030405060708090a0b0c0d0e0f101112131415161718190000000000\\\"\\n hex\\\"001a1b1c1d1e1f202122232425262728292a2b2c2d2e2f303132330000000000\\\";\\n\\n function encode(bytes memory data) internal pure returns (string memory) {\\n if (data.length == 0) return '';\\n\\n // load the table into memory\\n string memory table = TABLE_ENCODE;\\n\\n // multiply by 4/3 rounded up\\n uint256 encodedLen = 4 * ((data.length + 2) / 3);\\n\\n // add some extra buffer at the end required for the writing\\n string memory result = new string(encodedLen + 32);\\n\\n assembly {\\n // set the actual output length\\n mstore(result, encodedLen)\\n\\n // prepare the lookup table\\n let tablePtr := add(table, 1)\\n\\n // input ptr\\n let dataPtr := data\\n let endPtr := add(dataPtr, mload(data))\\n\\n // result ptr, jump over length\\n let resultPtr := add(result, 32)\\n\\n // run over the input, 3 bytes at a time\\n for {} lt(dataPtr, endPtr) {}\\n {\\n // read 3 bytes\\n dataPtr := add(dataPtr, 3)\\n let input := mload(dataPtr)\\n\\n // write 4 characters\\n mstore8(resultPtr, mload(add(tablePtr, and(shr(18, input), 0x3F))))\\n resultPtr := add(resultPtr, 1)\\n mstore8(resultPtr, mload(add(tablePtr, and(shr(12, input), 0x3F))))\\n resultPtr := add(resultPtr, 1)\\n mstore8(resultPtr, mload(add(tablePtr, and(shr( 6, input), 0x3F))))\\n resultPtr := add(resultPtr, 1)\\n mstore8(resultPtr, mload(add(tablePtr, and( input, 0x3F))))\\n resultPtr := add(resultPtr, 1)\\n }\\n\\n // padding with '='\\n switch mod(mload(data), 3)\\n case 1 { mstore(sub(resultPtr, 2), shl(240, 0x3d3d)) }\\n case 2 { mstore(sub(resultPtr, 1), shl(248, 0x3d)) }\\n }\\n\\n return result;\\n }\\n\\n function decode(string memory _data) internal pure returns (bytes memory) {\\n bytes memory data = bytes(_data);\\n\\n if (data.length == 0) return new bytes(0);\\n require(data.length % 4 == 0, \\\"invalid base64 decoder input\\\");\\n\\n // load the table into memory\\n bytes memory table = TABLE_DECODE;\\n\\n // every 4 characters represent 3 bytes\\n uint256 decodedLen = (data.length / 4) * 3;\\n\\n // add some extra buffer at the end required for the writing\\n bytes memory result = new bytes(decodedLen + 32);\\n\\n assembly {\\n // padding with '='\\n let lastBytes := mload(add(data, mload(data)))\\n if eq(and(lastBytes, 0xFF), 0x3d) {\\n decodedLen := sub(decodedLen, 1)\\n if eq(and(lastBytes, 0xFFFF), 0x3d3d) {\\n decodedLen := sub(decodedLen, 1)\\n }\\n }\\n\\n // set the actual output length\\n mstore(result, decodedLen)\\n\\n // prepare the lookup table\\n let tablePtr := add(table, 1)\\n\\n // input ptr\\n let dataPtr := data\\n let endPtr := add(dataPtr, mload(data))\\n\\n // result ptr, jump over length\\n let resultPtr := add(result, 32)\\n\\n // run over the input, 4 characters at a time\\n for {} lt(dataPtr, endPtr) {}\\n {\\n // read 4 characters\\n dataPtr := add(dataPtr, 4)\\n let input := mload(dataPtr)\\n\\n // write 3 bytes\\n let output := add(\\n add(\\n shl(18, and(mload(add(tablePtr, and(shr(24, input), 0xFF))), 0xFF)),\\n shl(12, and(mload(add(tablePtr, and(shr(16, input), 0xFF))), 0xFF))),\\n add(\\n shl( 6, and(mload(add(tablePtr, and(shr( 8, input), 0xFF))), 0xFF)),\\n and(mload(add(tablePtr, and( input , 0xFF))), 0xFF)\\n )\\n )\\n mstore(resultPtr, shl(232, output))\\n resultPtr := add(resultPtr, 3)\\n }\\n }\\n\\n return result;\\n }\\n}\\n\",\"keccak256\":\"0xa73959e6ef0b693e4423a562e612370160b934a75e618361ddd8c9c4b8ddbaaf\",\"license\":\"MIT\"},\"contracts/DropCreator.sol\":{\"content\":\"// SPDX-License-Identifier: GPL-3.0\\n\\n/**\\n\\n ExpandedNFTs\\n \\n */\\n\\npragma solidity ^0.8.15;\\n\\nimport {ClonesUpgradeable} from \\\"@openzeppelin/contracts-upgradeable/proxy/ClonesUpgradeable.sol\\\";\\nimport {CountersUpgradeable} from \\\"@openzeppelin/contracts-upgradeable/utils/CountersUpgradeable.sol\\\";\\n\\nimport \\\"./ExpandedNFT.sol\\\";\\n\\ncontract DropCreator {\\n using CountersUpgradeable for CountersUpgradeable.Counter;\\n\\n /// Counter for current contract id upgraded\\n CountersUpgradeable.Counter private _atContract;\\n\\n /// Address for implementation of ExpandedNFT to clone\\n address public implementation;\\n\\n /// Initializes factory with address of implementation logic\\n /// @param _implementation ExpandedNFT logic implementation contract to clone\\n constructor(address _implementation) {\\n implementation = _implementation;\\n }\\n\\n /// Creates a new drop contract as a factory with a deterministic address\\n /// Important: None of these fields (except the Url fields with the same hash) can be changed after calling\\n /// @param _artistWallet User that created the drop\\n /// @param _name Name of the drop contract\\n /// @param _symbol Symbol of the drop contract\\n function createDrop(\\n address _artistWallet,\\n string memory _name,\\n string memory _symbol,\\n uint256 _dropSize\\n ) external returns (uint256) {\\n require(_dropSize > 0, \\\"Drop size must be > 0\\\");\\n\\n address newContract = ClonesUpgradeable.cloneDeterministic(\\n implementation,\\n bytes32(abi.encodePacked(_atContract.current()))\\n );\\n\\n ExpandedNFT(newContract).initialize(\\n msg.sender,\\n _artistWallet,\\n _name,\\n _symbol,\\n _dropSize\\n );\\n\\n uint256 newId = _atContract.current(); \\n emit CreatedDrop(newId, msg.sender, _dropSize, newContract);\\n // Returns the ID of the recently created minting contract\\n // Also increments for the next contract creation call\\n _atContract.increment();\\n return newId;\\n }\\n\\n /// Get drop given the created ID\\n /// @param dropId id of drop to get contract for\\n /// @return ExpandedNFT Drop NFT contract\\n function getDropAtId(uint256 dropId)\\n external\\n view\\n returns (ExpandedNFT)\\n {\\n return\\n ExpandedNFT(\\n ClonesUpgradeable.predictDeterministicAddress(\\n implementation,\\n bytes32(abi.encodePacked(dropId)),\\n address(this)\\n )\\n );\\n }\\n\\n /// Emitted when a drop is created reserving the corresponding token IDs.\\n /// @param dropId ID of newly created drop\\n event CreatedDrop(\\n uint256 indexed dropId,\\n address indexed creator,\\n uint256 dropSize,\\n address dropContractAddress\\n );\\n}\\n\",\"keccak256\":\"0xd6a5ada49de66390c0a846b5221a6ddcca1941ec61eaa168dbabd030437dbe45\",\"license\":\"GPL-3.0\"},\"contracts/ExpandedNFT.sol\":{\"content\":\"// SPDX-License-Identifier: GPL-3.0\\n\\n/**\\n\\n ExpandedNFTs\\n\\n */\\n\\npragma solidity ^0.8.15;\\n\\nimport {ERC721Upgradeable} from \\\"@openzeppelin/contracts-upgradeable/token/ERC721/ERC721Upgradeable.sol\\\";\\nimport {IERC2981Upgradeable, IERC165Upgradeable} from \\\"@openzeppelin/contracts-upgradeable/interfaces/IERC2981Upgradeable.sol\\\";\\nimport {OwnableUpgradeable} from \\\"@openzeppelin/contracts-upgradeable/access/OwnableUpgradeable.sol\\\";\\nimport {AddressUpgradeable} from \\\"@openzeppelin/contracts-upgradeable/utils/AddressUpgradeable.sol\\\";\\nimport {IERC20Upgradeable} from \\\"@openzeppelin/contracts-upgradeable/token/ERC20/IERC20Upgradeable.sol\\\";\\n\\nimport {SharedNFTLogic} from \\\"./SharedNFTLogic.sol\\\";\\nimport {IExpandedNFT} from \\\"./IExpandedNFT.sol\\\";\\n/**\\n This is a smart contract for handling dynamic contract minting.\\n\\n @dev This allows creators to mint a unique serial drop of an expanded NFT within a custom contract\\n @author Zien\\n Repository: https://github.com/joinzien/expanded-nft\\n*/\\ncontract ExpandedNFT is\\n ERC721Upgradeable,\\n IExpandedNFT,\\n IERC2981Upgradeable,\\n OwnableUpgradeable\\n{\\n enum WhoCanMint{ ONLY_OWNER, VIPS, MEMBERS, ANYONE }\\n\\n enum ExpandedNFTStates{ UNMINTED, MINTED, REDEEM_STARTED, SET_OFFER_TERMS, ACCEPTED_OFFER, PRODUCTION_COMPLETE, REDEEMED }\\n \\n event PriceChanged(uint256 amount);\\n event EditionSold(uint256 price, address owner);\\n event WhoCanMintChanged(WhoCanMint minters);\\n\\n // State change events\\n event RedeemStarted(uint256 tokenId, address owner);\\n event RedeemAborted(uint256 tokenId, address owner); \\n event OfferTermsSet(uint256 tokenId);\\n event OfferAccepted(uint256 tokenId);\\n event OfferRejected(uint256 tokenId);\\n event ProductionComplete(uint256 tokenId);\\n event DeliveryAccepted(uint256 tokenId);\\n\\n struct PerToken { \\n // Hashmap of the Edition ID to the current \\n ExpandedNFTStates editionState;\\n\\n // Redemption price\\n uint256 editionFee; \\n\\n // Edition description\\n string description;\\n\\n // Minted\\n\\n // animation_url field in the metadata\\n string animationUrl;\\n // Hash for the associated animation\\n bytes32 animationHash;\\n // Image in the metadata\\n string imageUrl;\\n // Hash for the associated image\\n bytes32 imageHash;\\n\\n // Redeemed\\n\\n // animation_url field in the metadata\\n string redeemedAnimationUrl;\\n // Hash for the associated animation\\n bytes32 redeemedAnimationHash;\\n // Image in the metadata\\n string redeemedImageUrl;\\n // Hash for the associated image\\n bytes32 redeemedImageHash;\\n // Condition report in the metadata\\n string conditionReportUrl;\\n // Hash for the condition report\\n bytes32 conditionReportHash;\\n }\\n\\n struct Pricing { \\n // Royalty amount in bps\\n uint256 royaltyBPS;\\n\\n // Split amount to the platforms. the artist in bps\\n uint256 splitBPS;\\n\\n // Price for VIP sales\\n uint256 vipSalePrice;\\n\\n // Price for member sales\\n uint256 membersSalePrice; \\n\\n // Price for VIP sales\\n uint256 vipMintLimit;\\n\\n // Price for member sales\\n uint256 membersMintLimit;\\n\\n // Price for general sales\\n uint256 generalMintLimit; \\n\\n // Addresses allowed to mint edition\\n mapping(address => bool) allowedMinters;\\n // VIP Addresses allowed to mint edition\\n mapping(address => bool) vipAllowedMinters;\\n\\n // Who can currently mint\\n WhoCanMint whoCanMint;\\n\\n // Mint counts for each address\\n mapping(address => uint256) mintCounts; \\n }\\n\\n // metadata\\n string public description;\\n\\n // Artists wallet address\\n address private _artistWallet;\\n\\n // Per Token data\\n mapping(uint256 => PerToken) private _perTokenMetadata;\\n\\n // Total size of the drop that can be minted\\n uint256 public dropSize;\\n\\n uint256 private _loadedMetadata;\\n\\n // reservation list\\n uint256 private _reserveCount;\\n mapping(uint256 => address) private _reserveAddress;\\n mapping(uint256 => uint256) private _reserveTokenId;\\n\\n mapping(uint256 => bool) private _tokenClaimed; \\n uint256 private _claimCount; \\n uint256 private _currentIndex;\\n\\n Pricing private _pricing;\\n\\n // Price for general sales\\n uint256 public salePrice;\\n\\n // ERC20 interface for the payment token\\n IERC20Upgradeable private _paymentTokenERC20;\\n\\n // NFT rendering logic contract\\n SharedNFTLogic private immutable _sharedNFTLogic;\\n\\n // Global constructor for factory\\n constructor(SharedNFTLogic sharedNFTLogic) {\\n _sharedNFTLogic = sharedNFTLogic;\\n _pricing.whoCanMint = WhoCanMint.ONLY_OWNER;\\n }\\n\\n /**\\n @param _owner wallet addres for the user that owns and can mint the drop, gets royalty and sales payouts and can update the base url if needed.\\n @param artistWallet wallet address for thr User that created the drop\\n @param _name Name of drop, used in the title as \\\"$NAME NUMBER/TOTAL\\\"\\n @param _symbol Symbol of the new token contract\\n @param _dropSize Number of editions that can be minted in total. \\n @dev Function to create a new drop. Can only be called by the allowed creator\\n Sets the only allowed minter to the address that creates/owns the drop.\\n This can be re-assigned or updated later\\n */\\n function initialize(\\n address _owner,\\n address artistWallet,\\n string memory _name,\\n string memory _symbol,\\n uint256 _dropSize\\n ) public initializer {\\n require(_dropSize > 0, \\\"Drop size must be > 0\\\");\\n\\n __ERC721_init(_name, _symbol);\\n __Ownable_init();\\n\\n // Set ownership to original sender of contract call\\n transferOwnership(_owner);\\n\\n _artistWallet = artistWallet;\\n dropSize = _dropSize;\\n\\n // Set edition id start to be 1 not 0\\n _claimCount = 0; \\n _currentIndex = 1;\\n\\n // Set the metadata\\n description = _name;\\n _loadedMetadata = 0; \\n }\\n\\n /**\\n @param _description Description of the edition, used in the description field of the NFT\\n @param imageUrl Image URL of the the edition. Strongly encouraged to be used, if necessary, only animation URL can be used. One of animation and image url need to exist in a drop to render the NFT.\\n @param imageHash SHA256 of the given image in bytes32 format (0xHASH). If no image is included, the hash can be zero.\\n @param animationUrl Animation URL of the edition. Not required, but if omitted image URL needs to be included. This follows the opensea spec for NFTs\\n @param animationHash The associated hash of the animation in sha-256 bytes32 format. If animation is omitted the hash can be zero.\\n @dev Function to create a new drop. Can only be called by the allowed creator\\n Sets the only allowed minter to the address that creates/owns the drop.\\n This can be re-assigned or updated later\\n */\\n function loadMetadataChunk(\\n uint256 startOffset,\\n uint256 count,\\n string[] memory _description,\\n string[] memory animationUrl,\\n bytes32[] memory animationHash,\\n string[] memory imageUrl,\\n bytes32[] memory imageHash\\n\\n ) public {\\n require(_description.length == count, \\\"Data size mismatch\\\");\\n require(animationUrl.length == count, \\\"Data size mismatch\\\");\\n require(animationHash.length == count, \\\"Data size mismatch\\\");\\n require(imageUrl.length == count, \\\"Data size mismatch\\\");\\n require(imageHash.length == count, \\\"Data size mismatch\\\");\\n\\n for (uint i = 0; i < count; i++) {\\n uint index = startOffset + i + 1;\\n \\n _perTokenMetadata[index].description = _description[i];\\n _perTokenMetadata[index].imageUrl = imageUrl[i];\\n _perTokenMetadata[index].imageHash = imageHash[i];\\n _perTokenMetadata[index].animationUrl = animationUrl[i];\\n _perTokenMetadata[index].animationHash = animationHash[i];\\n\\n }\\n\\n _loadedMetadata += count;\\n }\\n\\n function metadataloaded() public view returns (bool){\\n return (_loadedMetadata >= dropSize);\\n }\\n\\n /// @dev returns the number of minted tokens within the drop\\n function totalSupply() public view returns (uint256) {\\n return _claimCount;\\n }\\n\\n /// @dev returns the royalty BPS\\n function getRoyaltyBPS() public view returns (uint256) {\\n return _pricing.royaltyBPS;\\n }\\n\\n /// @dev returns the split BPS\\n function getSplitBPS() public view returns (uint256) {\\n return _pricing.splitBPS;\\n }\\n\\n /// @dev returns the VIP sale price\\n function getVIPSalePrice() public view returns (uint256) {\\n return _pricing.vipSalePrice;\\n }\\n\\n /// @dev returns the member sale price\\n function getMembersSalePrice() public view returns (uint256) {\\n return _pricing.membersSalePrice;\\n }\\n\\n /// @dev returns the VIP mint limit\\n function getVIPMintLimit() public view returns (uint256) {\\n return _pricing.vipMintLimit;\\n }\\n\\n /// @dev returns the member mint limit\\n function getMembersMintLimit() public view returns (uint256) {\\n return _pricing.membersMintLimit;\\n }\\n\\n /// @dev returns the general mint limit\\n function getGeneralMintLimit() public view returns (uint256) {\\n return salePrice;\\n }\\n\\n /// @dev returns who can mint\\n function getWhoCanMint() public view returns (uint256) {\\n return uint256(_pricing.whoCanMint);\\n }\\n\\n /**\\n Simple eth-based sales function\\n More complex sales functions can be implemented through IExpandedNFT interface\\n */\\n\\n /**\\n @dev This allows the user to purchase an edition\\n at the given price in the contract.\\n */\\n\\n function purchase() external payable returns (uint256) {\\n uint256 currentPrice = _currentSalesPrice();\\n emit EditionSold(currentPrice, msg.sender);\\n\\n address[] memory toMint = new address[](1);\\n toMint[0] = msg.sender;\\n\\n return _mintEditionsBody(toMint); \\n }\\n\\n /**\\n @param to address to send the newly minted edition to\\n @dev This mints one edition to the given address by an allowed minter on the edition instance.\\n */\\n function mintEdition(address to) external payable override returns (uint256) {\\n address[] memory toMint = new address[](1);\\n toMint[0] = to;\\n\\n return _mintEditionsBody(toMint); \\n }\\n\\n /**\\n @param recipients list of addresses to send the newly minted editions to\\n @dev This mints multiple editions to the given list of addresses.\\n */\\n function mintEditions(address[] memory recipients)\\n external payable override returns (uint256)\\n {\\n return _mintEditionsBody(recipients);\\n } \\n\\n /**\\n @param recipients list of addresses to send the newly minted editions to\\n @dev This mints multiple editions to the given list of addresses.\\n */\\n function _mintEditionsBody(address[] memory recipients)\\n internal returns (uint256)\\n {\\n require(_loadedMetadata >= dropSize, \\\"Not all metadata loaded\\\");\\n\\n require(_isAllowedToMint(), \\\"Needs to be an allowed minter\\\");\\n\\n uint256 currentPrice = _currentSalesPrice();\\n require(currentPrice > 0, \\\"Not for sale\\\");\\n require(msg.value == (currentPrice * recipients.length), \\\"Wrong price\\\");\\n\\n require((_pricing.mintCounts[msg.sender] + recipients.length - 1) < _currentMintLimit(), \\\"Exceeded mint limit\\\");\\n\\n require(_claimCount + recipients.length <= dropSize, \\\"Over drop size\\\");\\n\\n if (_pricing.whoCanMint == WhoCanMint.VIPS) {\\n return _vipMintEditions(recipients);\\n }\\n\\n return _mintEditions(recipients);\\n } \\n\\n /**\\n @dev Private function to mint without any access checks.\\n Called by the public edition minting functions.\\n */\\n function _vipMintEditions(address[] memory recipients)\\n internal\\n returns (uint256)\\n {\\n address currentMinter = msg.sender;\\n\\n uint256 unclaimed = 0;\\n uint256 firstUnclaimed = _reserveCount;\\n\\n for (uint256 r = 0; r < _reserveCount; r++) {\\n if (_reserveAddress[r] == currentMinter) {\\n uint256 id = _reserveTokenId[r];\\n\\n if (_tokenClaimed[id] != true) {\\n if (r < firstUnclaimed) {\\n firstUnclaimed = r; \\n }\\n\\n unclaimed++;\\n }\\n }\\n }\\n\\n require(unclaimed >= recipients.length, \\\"Can not mint all editions\\\");\\n\\n uint256 idToMint = 1;\\n\\n uint256 reservationCounter = firstUnclaimed;\\n for (uint256 i = 0; i < recipients.length; i++) {\\n while (_reserveAddress[reservationCounter] != currentMinter) {\\n reservationCounter++;\\n } \\n\\n idToMint = _reserveTokenId[reservationCounter];\\n\\n _mint(recipients[i], idToMint);\\n\\n _perTokenMetadata[idToMint].editionState = ExpandedNFTStates.MINTED;\\n _tokenClaimed[idToMint] = true;\\n _pricing.mintCounts[currentMinter]++;\\n _claimCount++;\\n\\n reservationCounter++;\\n }\\n\\n return idToMint; \\n } \\n\\n /**\\n @dev Private function to mint without any access checks.\\n Called by the public edition minting functions.\\n */\\n function _mintEditions(address[] memory recipients)\\n internal\\n returns (uint256)\\n {\\n address currentMinter = msg.sender;\\n \\n for (uint256 i = 0; i < recipients.length; i++) {\\n while (_tokenClaimed[_currentIndex] == true) {\\n _currentIndex++;\\n } \\n\\n _mint(recipients[i], _currentIndex);\\n\\n _perTokenMetadata[_currentIndex].editionState = ExpandedNFTStates.MINTED;\\n _tokenClaimed[_currentIndex] = true;\\n _pricing.mintCounts[currentMinter]++;\\n _claimCount++;\\n }\\n\\n return _currentIndex; \\n } \\n\\n /**\\n @param _royaltyBPS BPS of the royalty set on the contract. Can be 0 for no royalty.\\n @param _splitBPS BPS of the royalty set on the contract. Can be 0 for no royalty. \\n @param _vipSalePrice Sale price for VIPs\\n @param _membersSalePrice SalePrice for Members \\n @param _generalSalePrice SalePrice for the general public \\n @param _vipMintLimit Mint limit for VIPs\\n @param _membersMintLimit Mint limit for Members \\n @param _generalMintLimit Mint limit for the general public \\n @dev Set various pricing related values\\n */\\n function setPricing (\\n uint256 _royaltyBPS,\\n uint256 _splitBPS,\\n uint256 _vipSalePrice,\\n uint256 _membersSalePrice, \\n uint256 _generalSalePrice,\\n uint256 _vipMintLimit,\\n uint256 _membersMintLimit,\\n uint256 _generalMintLimit \\n ) external onlyOwner { \\n _pricing.royaltyBPS = _royaltyBPS;\\n _pricing.splitBPS = _splitBPS;\\n\\n _pricing.vipSalePrice = _vipSalePrice;\\n _pricing.membersSalePrice = _membersSalePrice;\\n salePrice = _generalSalePrice;\\n\\n _pricing.vipMintLimit = _vipMintLimit;\\n _pricing.membersMintLimit = _membersMintLimit;\\n _pricing.generalMintLimit = _generalMintLimit;\\n\\n emit PriceChanged(salePrice);\\n }\\n\\n /**\\n @dev returns the current ETH sales price\\n based on who can currently mint.\\n */\\n function _currentSalesPrice() internal view returns (uint256){\\n if (_pricing.whoCanMint == WhoCanMint.VIPS) {\\n return _pricing.vipSalePrice;\\n } else if (_pricing.whoCanMint == WhoCanMint.MEMBERS) {\\n return _pricing.membersSalePrice;\\n } else if (_pricing.whoCanMint == WhoCanMint.ANYONE) {\\n return salePrice;\\n } \\n \\n return 0; \\n }\\n\\n /**\\n @param wallets A list of wallets\\n @param tokenIDs A list of tokenId to reserve \\n @dev Set various pricing related values\\n */\\n function reserve (address[] calldata wallets, uint256[] calldata tokenIDs) \\n external onlyOwner { \\n for (uint256 i = 0; i < wallets.length; i++) {\\n _reserveAddress[_reserveCount] = wallets[i]; \\n _reserveTokenId[_reserveCount] = tokenIDs[i]; \\n _reserveCount++;\\n }\\n }\\n\\n /**\\n @dev returns the current loimit on edition that \\n can be minted by one wallet\\n */\\n function _currentMintLimit() internal view returns (uint256){\\n if (_pricing.whoCanMint == WhoCanMint.VIPS) {\\n return _pricing.vipMintLimit;\\n } else if (_pricing.whoCanMint == WhoCanMint.MEMBERS) {\\n return _pricing.membersMintLimit;\\n } else if (_pricing.whoCanMint == WhoCanMint.ANYONE) {\\n return _pricing.generalMintLimit;\\n } \\n \\n return 0; \\n }\\n\\n /**\\n @param _salePrice if sale price is 0 sale is stopped, otherwise that amount \\n of ETH is needed to start the sale.\\n @dev This sets a simple ETH sales price\\n Setting a sales price allows users to mint the drop until it sells out.\\n For more granular sales, use an external sales contract.\\n */\\n function setSalePrice(uint256 _salePrice) external onlyOwner {\\n salePrice = _salePrice;\\n\\n _pricing.whoCanMint = WhoCanMint.ANYONE;\\n\\n emit WhoCanMintChanged(_pricing.whoCanMint);\\n emit PriceChanged(salePrice);\\n }\\n\\n /**\\n @param _salePrice if sale price is 0 sale is stopped, otherwise that amount \\n of ETH is needed to start the sale.\\n @dev This sets the VIP ETH sales price\\n Setting a sales price allows users to mint the drop until it sells out.\\n For more granular sales, use an external sales contract.\\n */\\n function setVIPSalePrice(uint256 _salePrice) external onlyOwner {\\n _pricing.vipSalePrice = _salePrice;\\n\\n _pricing.whoCanMint = WhoCanMint.VIPS;\\n\\n emit WhoCanMintChanged(_pricing.whoCanMint);\\n emit PriceChanged(salePrice);\\n }\\n\\n /**\\n @param _salePrice if sale price is 0 sale is stopped, otherwise that amount \\n of ETH is needed to start the sale.\\n @dev This sets the members ETH sales price\\n Setting a sales price allows users to mint the drop until it sells out.\\n For more granular sales, use an external sales contract.\\n */\\n function setMembersSalePrice(uint256 _salePrice) external onlyOwner {\\n _pricing.membersSalePrice = _salePrice;\\n\\n _pricing.whoCanMint = WhoCanMint.MEMBERS;\\n\\n emit WhoCanMintChanged(_pricing.whoCanMint);\\n emit PriceChanged(salePrice);\\n } \\n\\n\\n /**\\n @param vipSalePrice if sale price is 0 sale is stopped, otherwise that amount \\n of ETH is needed to start the sale.\\n @param membersSalePrice if sale price is 0 sale is stopped, otherwise that amount \\n of ETH is needed to start the sale.\\n @param generalSalePrice if sale price is 0 sale is stopped, otherwise that amount \\n of ETH is needed to start the sale. \\n @dev This sets the members ETH sales price\\n Setting a sales price allows users to mint the drop until it sells out.\\n For more granular sales, use an external sales contract.\\n */\\n function setSalePrices(uint256 vipSalePrice, uint256 membersSalePrice, uint256 generalSalePrice) external onlyOwner {\\n _pricing.vipSalePrice = vipSalePrice;\\n _pricing.membersSalePrice = membersSalePrice;\\n salePrice = generalSalePrice; \\n\\n emit PriceChanged(salePrice);\\n } \\n\\n /**\\n @dev This withdraws ETH from the contract to the contract owner.\\n */\\n function withdraw() external onlyOwner {\\n uint256 currentBalance = address(this).balance;\\n if (currentBalance > 0) {\\n uint256 platformFee = (currentBalance * _pricing.splitBPS) / 10000;\\n uint256 artistFee = currentBalance - platformFee;\\n\\n AddressUpgradeable.sendValue(payable(owner()), platformFee);\\n AddressUpgradeable.sendValue(payable(_artistWallet), artistFee);\\n }\\n\\n if (address(_paymentTokenERC20) != address(0x0)) {\\n uint256 currentBalanceERC20 = _paymentTokenERC20.balanceOf(address(this));\\n if (currentBalanceERC20 > 0) {\\n uint256 platformFee = (currentBalanceERC20 * _pricing.splitBPS) / 10000;\\n uint256 artistFee = currentBalanceERC20 - platformFee;\\n\\n _paymentTokenERC20.transfer(owner(), platformFee);\\n _paymentTokenERC20.transfer(_artistWallet, artistFee);\\n }\\n }\\n }\\n\\n /**\\n @dev This helper function checks if the msg.sender is allowed to mint the\\n given edition id.\\n */\\n function _isAllowedToMint() internal view returns (bool) {\\n if (_pricing.whoCanMint == WhoCanMint.ANYONE) {\\n return true;\\n }\\n\\n if (_pricing.whoCanMint == WhoCanMint.MEMBERS) {\\n if (_pricing.vipAllowedMinters[msg.sender]) {\\n return true;\\n } \\n\\n if (_pricing.allowedMinters[msg.sender]) {\\n return true;\\n } \\n }\\n\\n if (_pricing.whoCanMint == WhoCanMint.VIPS) {\\n if (_pricing.vipAllowedMinters[msg.sender]) {\\n return true;\\n } \\n }\\n\\n if (owner() == msg.sender) {\\n return true;\\n }\\n\\n return false;\\n }\\n\\n /**\\n Simple override for owner interface.\\n */\\n function owner()\\n public\\n view\\n override(OwnableUpgradeable, IExpandedNFT)\\n returns (address)\\n {\\n return super.owner();\\n }\\n\\n /**\\n return the artists wallet address\\n */\\n function getArtistWallet()\\n public\\n view\\n returns (address)\\n {\\n return _artistWallet;\\n }\\n\\n /**\\n set the artists wallet address\\n */\\n function setArtistWallet(address wallet)\\n public\\n onlyOwner\\n {\\n _artistWallet = wallet;\\n } \\n\\n /**\\n return the payment tokens address\\n */\\n function getPaymentToken()\\n public\\n view\\n returns (address)\\n {\\n return address(_paymentTokenERC20);\\n }\\n\\n /**\\n set a new payment token address\\n */\\n function setPaymentToken(address paymentToken)\\n public\\n onlyOwner\\n {\\n if (address(_paymentTokenERC20) != address(0x0)) {\\n require(_paymentTokenERC20.balanceOf(address(this)) == 0, \\\"token must have 0 balance\\\");\\n }\\n\\n _paymentTokenERC20 = IERC20Upgradeable(paymentToken);\\n } \\n\\n /**\\n @dev Sets the types of users who is allowed to mint.\\n */\\n function getAllowedMinter() public view returns (WhoCanMint){\\n return _pricing.whoCanMint;\\n }\\n\\n /**\\n @param minters WhoCanMint enum of minter types\\n @dev Sets the types of users who is allowed to mint.\\n */\\n function setAllowedMinter(WhoCanMint minters) public onlyOwner {\\n require(((minters >= WhoCanMint.ONLY_OWNER) && (minters <= WhoCanMint.ANYONE)), \\\"Needs to be a valid minter type\\\");\\n\\n _pricing.whoCanMint = minters;\\n emit WhoCanMintChanged(minters);\\n }\\n\\n /**\\n @param minter address to set approved minting status for\\n @param allowed boolean if that address is allowed to mint\\n @dev Sets the approved minting status of the given address.\\n This requires that msg.sender is the owner of the given edition id.\\n If the ZeroAddress (address(0x0)) is set as a minter,\\n anyone will be allowed to mint.\\n This setup is similar to setApprovalForAll in the ERC721 spec.\\n */\\n function setApprovedMinters(uint256 count, address[] calldata minter, bool[] calldata allowed) public onlyOwner {\\n for (uint256 i = 0; i < count; i++) {\\n _pricing.allowedMinters[minter[i]] = allowed[i];\\n }\\n }\\n\\n /**\\n @param minter address to set approved minting status for\\n @param allowed boolean if that address is allowed to mint\\n @dev Sets the approved minting status of the given address.\\n This requires that msg.sender is the owner of the given edition id.\\n If the ZeroAddress (address(0x0)) is set as a minter,\\n anyone will be allowed to mint.\\n This setup is similar to setApprovalForAll in the ERC721 spec.\\n */\\n function setApprovedVIPMinters(uint256 count, address[] calldata minter, bool[] calldata allowed) public onlyOwner {\\n for (uint256 i = 0; i < count; i++) {\\n _pricing.vipAllowedMinters[minter[i]] = allowed[i];\\n }\\n }\\n\\n /**\\n @dev Allows for updates of edition urls by the owner of the edition.\\n Only URLs can be updated (data-uris are supported), hashes cannot be updated.\\n */\\n function updateEditionURLs(\\n uint256 tokenId,\\n string memory imageUrl,\\n string memory animationUrl\\n ) public onlyOwner {\\n _perTokenMetadata[tokenId].imageUrl = imageUrl;\\n _perTokenMetadata[tokenId].animationUrl = animationUrl;\\n }\\n\\n /// Returns the number of editions allowed to mint\\n function numberCanMint() public view override returns (uint256) {\\n return dropSize - _claimCount;\\n }\\n\\n /**\\n @param tokenId Token ID to burn\\n User burn function for token id \\n */\\n function burn(uint256 tokenId) public {\\n require(_isApprovedOrOwner(_msgSender(), tokenId), \\\"Not approved\\\");\\n _burn(tokenId);\\n }\\n\\n function redeem(uint256 tokenId) public {\\n require(_exists(tokenId), \\\"No token\\\");\\n require(_isApprovedOrOwner(_msgSender(), tokenId), \\\"Not approved\\\");\\n\\n require((_perTokenMetadata[tokenId].editionState == ExpandedNFTStates.MINTED), \\\"You currently can not redeem\\\");\\n\\n _perTokenMetadata[tokenId].editionState = ExpandedNFTStates.REDEEM_STARTED;\\n emit RedeemStarted(tokenId, _msgSender());\\n }\\n\\n function abortRedemption(uint256 tokenId) public {\\n require(_exists(tokenId), \\\"No token\\\");\\n require(_isApprovedOrOwner(_msgSender(), tokenId), \\\"Not approved\\\");\\n\\n require((_perTokenMetadata[tokenId].editionState == ExpandedNFTStates.REDEEM_STARTED), \\\"You currently can not redeem\\\");\\n\\n _perTokenMetadata[tokenId].editionState = ExpandedNFTStates.MINTED;\\n emit RedeemAborted(tokenId, _msgSender());\\n }\\n\\n function setOfferTerms(uint256 tokenId, uint256 fee) public onlyOwner {\\n require(_exists(tokenId), \\\"No token\\\"); \\n require((_perTokenMetadata[tokenId].editionState == ExpandedNFTStates.REDEEM_STARTED), \\\"Wrong state\\\");\\n\\n _perTokenMetadata[tokenId].editionState = ExpandedNFTStates.SET_OFFER_TERMS;\\n _perTokenMetadata[tokenId].editionFee = fee;\\n\\n emit OfferTermsSet(tokenId);\\n }\\n\\n function rejectOfferTerms(uint256 tokenId) public {\\n require(_exists(tokenId), \\\"No token\\\"); \\n require(_isApprovedOrOwner(_msgSender(), tokenId), \\\"Not approved\\\");\\n\\n require((_perTokenMetadata[tokenId].editionState == ExpandedNFTStates.SET_OFFER_TERMS), \\\"You currently can not redeem\\\");\\n\\n _perTokenMetadata[tokenId].editionState = ExpandedNFTStates.MINTED;\\n\\n emit OfferRejected(tokenId);\\n }\\n\\n function acceptOfferTerms(uint256 tokenId, uint256 paymentAmount) external {\\n require(_exists(tokenId), \\\"No token\\\"); \\n require(_isApprovedOrOwner(_msgSender(), tokenId), \\\"Not approved\\\");\\n\\n require((_perTokenMetadata[tokenId].editionState == ExpandedNFTStates.SET_OFFER_TERMS), \\\"You currently can not redeem\\\");\\n\\n require(paymentAmount >= _perTokenMetadata[tokenId].editionFee, \\\"Wrong price\\\");\\n require(_paymentTokenERC20.allowance(_msgSender(), address(this)) >= _perTokenMetadata[tokenId].editionFee, \\\"Insufficient allowance\\\");\\n\\n bool success = _paymentTokenERC20.transferFrom(_msgSender(), address(this), _perTokenMetadata[tokenId].editionFee);\\n require(success, \\\"Could not transfer token\\\");\\n\\n _perTokenMetadata[tokenId].editionState = ExpandedNFTStates.ACCEPTED_OFFER; \\n\\n emit OfferAccepted(tokenId);\\n }\\n\\n function productionComplete(\\n uint256 tokenId,\\n string memory _description,\\n string memory animationUrl,\\n bytes32 animationHash,\\n string memory imageUrl,\\n bytes32 imageHash, \\n string memory conditionReportUrl,\\n bytes32 conditionReportHash \\n ) public onlyOwner {\\n require(_exists(tokenId), \\\"No token\\\"); \\n require((_perTokenMetadata[tokenId].editionState == ExpandedNFTStates.ACCEPTED_OFFER), \\\"You currently can not redeem\\\");\\n\\n // Set the NFT to display as redeemed\\n _perTokenMetadata[tokenId].description = _description;\\n _perTokenMetadata[tokenId].redeemedAnimationUrl = animationUrl;\\n _perTokenMetadata[tokenId].redeemedAnimationHash = animationHash;\\n _perTokenMetadata[tokenId].redeemedImageUrl = imageUrl;\\n _perTokenMetadata[tokenId].redeemedImageHash = imageHash;\\n _perTokenMetadata[tokenId].conditionReportUrl = conditionReportUrl;\\n _perTokenMetadata[tokenId].conditionReportHash = conditionReportHash;\\n\\n _perTokenMetadata[tokenId].editionState = ExpandedNFTStates.PRODUCTION_COMPLETE;\\n\\n emit ProductionComplete(tokenId);\\n }\\n\\n function acceptDelivery(uint256 tokenId) public {\\n require(_exists(tokenId), \\\"No token\\\"); \\n require(_isApprovedOrOwner(_msgSender(), tokenId), \\\"Not approved\\\");\\n\\n require((_perTokenMetadata[tokenId].editionState == ExpandedNFTStates.PRODUCTION_COMPLETE), \\\"You currently can not redeem\\\");\\n\\n _perTokenMetadata[tokenId].editionState = ExpandedNFTStates.REDEEMED;\\n\\n emit OfferRejected(tokenId);\\n }\\n\\n /**\\n @dev Get URIs for the condition report\\n @return conditionReportUrl, conditionReportHash\\n */\\n function getConditionReport(uint256 tokenId)\\n public\\n view\\n returns (\\n string memory,\\n bytes32\\n )\\n {\\n return (_perTokenMetadata[tokenId].conditionReportUrl, _perTokenMetadata[tokenId].conditionReportHash);\\n }\\n\\n /**\\n @dev Get royalty information for token\\n @param _salePrice Sale price for the token\\n */\\n function royaltyInfo(uint256, uint256 _salePrice)\\n external\\n view\\n override\\n returns (address receiver, uint256 royaltyAmount)\\n {\\n if (owner() == address(0x0)) {\\n return (owner(), 0);\\n }\\n return (owner(), (_salePrice * _pricing.royaltyBPS) / 10_000);\\n }\\n\\n /**\\n @dev Get URIs for edition NFT\\n @return _imageUrl, _imageHash, _animationUrl, _animationHash\\n */\\n function getURIs(uint256 tokenId)\\n public\\n view\\n returns (\\n string memory,\\n bytes32,\\n string memory,\\n bytes32\\n )\\n {\\n if (_perTokenMetadata[tokenId].editionState == ExpandedNFTStates.REDEEMED) { \\n return (_perTokenMetadata[tokenId].redeemedImageUrl, _perTokenMetadata[tokenId].redeemedImageHash,\\n _perTokenMetadata[tokenId].redeemedAnimationUrl, _perTokenMetadata[tokenId].redeemedAnimationHash);\\n }\\n\\n return (_perTokenMetadata[tokenId].imageUrl, _perTokenMetadata[tokenId].imageHash,\\n _perTokenMetadata[tokenId].animationUrl, _perTokenMetadata[tokenId].animationHash);\\n }\\n\\n /**\\n @dev Get URI for given token id\\n @param tokenId token id to get uri for\\n @return base64-encoded json metadata object\\n */\\n function tokenURI(uint256 tokenId)\\n public\\n view\\n override\\n returns (string memory)\\n {\\n require(_exists(tokenId), \\\"No token\\\");\\n\\n if (_perTokenMetadata[tokenId].editionState == ExpandedNFTStates.REDEEMED) {\\n return\\n _sharedNFTLogic.createMetadataEdition(\\n name(),\\n _perTokenMetadata[tokenId].description,\\n _perTokenMetadata[tokenId].redeemedImageUrl,\\n _perTokenMetadata[tokenId].redeemedAnimationUrl,\\n tokenId,\\n dropSize\\n );\\n }\\n\\n return\\n _sharedNFTLogic.createMetadataEdition(\\n name(),\\n _perTokenMetadata[tokenId].description,\\n _perTokenMetadata[tokenId].imageUrl,\\n _perTokenMetadata[tokenId].animationUrl,\\n tokenId,\\n dropSize\\n );\\n }\\n\\n function supportsInterface(bytes4 interfaceId)\\n public\\n view\\n override(ERC721Upgradeable, IERC165Upgradeable)\\n returns (bool)\\n {\\n return\\n type(IERC2981Upgradeable).interfaceId == interfaceId ||\\n ERC721Upgradeable.supportsInterface(interfaceId);\\n }\\n}\\n\",\"keccak256\":\"0x1fb27bcf9324d5b43b44cc4a97220da3b3ffddf5cce218be885e5e731e7f1a1c\",\"license\":\"GPL-3.0\"},\"contracts/IExpandedNFT.sol\":{\"content\":\"// SPDX-License-Identifier: GPL-3.0\\npragma solidity ^0.8.15;\\n\\ninterface IExpandedNFT {\\n function mintEdition(address to) external payable returns (uint256);\\n function mintEditions(address[] memory to) external payable returns (uint256);\\n function numberCanMint() external view returns (uint256);\\n function owner() external view returns (address);\\n}\\n\",\"keccak256\":\"0xe1dc9fcdfab1ec02f10203a59e3fb819707f06515abe45674c5ecc814fc9c253\",\"license\":\"GPL-3.0\"},\"contracts/IPublicSharedMetadata.sol\":{\"content\":\"// SPDX-License-Identifier: GPL-3.0\\n\\npragma solidity ^0.8.15;\\n\\n/// Shared public library for on-chain NFT functions\\ninterface IPublicSharedMetadata {\\n /// @param unencoded bytes to base64-encode\\n function base64Encode(bytes memory unencoded)\\n external\\n pure\\n returns (string memory);\\n\\n /// Encodes the argument json bytes into base64-data uri format\\n /// @param json Raw json to base64 and turn into a data-uri\\n function encodeMetadataJSON(bytes memory json)\\n external\\n pure\\n returns (string memory);\\n\\n /// Proxy to openzeppelin's toString function\\n /// @param value number to return as a string\\n function numberToString(uint256 value)\\n external\\n pure\\n returns (string memory);\\n}\\n\",\"keccak256\":\"0x76caf66cf71c1067c845ff9de580b6e886c448915997ed1d6f37fe1a1ec489de\",\"license\":\"GPL-3.0\"},\"contracts/SharedNFTLogic.sol\":{\"content\":\"// SPDX-License-Identifier: GPL-3.0\\n\\npragma solidity ^0.8.15;\\n\\nimport {StringsUpgradeable} from \\\"@openzeppelin/contracts-upgradeable/utils/StringsUpgradeable.sol\\\";\\nimport {Base64} from \\\"base64-sol/base64.sol\\\";\\nimport {IPublicSharedMetadata} from \\\"./IPublicSharedMetadata.sol\\\";\\n\\n/// Shared NFT logic for rendering metadata associated with editions\\n/// @dev Can safely be used for generic base64Encode and numberToString functions\\ncontract SharedNFTLogic is IPublicSharedMetadata {\\n /// @param unencoded bytes to base64-encode\\n function base64Encode(bytes memory unencoded)\\n public\\n pure\\n override\\n returns (string memory)\\n {\\n return Base64.encode(unencoded);\\n }\\n\\n /// Proxy to openzeppelin's toString function\\n /// @param value number to return as a string\\n function numberToString(uint256 value)\\n public\\n pure\\n override\\n returns (string memory)\\n {\\n return StringsUpgradeable.toString(value);\\n }\\n\\n /// Generate edition metadata from storage information as base64-json blob\\n /// Combines the media data and metadata\\n /// @param name Name of NFT in metadata\\n /// @param description Description of NFT in metadata\\n /// @param imageUrl URL of image to render for edition\\n /// @param animationUrl URL of animation to render for edition\\n /// @param tokenOfEdition Token ID for specific token\\n /// @param editionSize Size of entire edition to show\\n function createMetadataEdition(\\n string memory name,\\n string memory description,\\n string memory imageUrl,\\n string memory animationUrl,\\n uint256 tokenOfEdition,\\n uint256 editionSize\\n ) external pure returns (string memory) {\\n string memory _tokenMediaData = tokenMediaData(\\n imageUrl,\\n animationUrl,\\n tokenOfEdition\\n );\\n bytes memory json = createMetadataJSON(\\n name,\\n description,\\n _tokenMediaData,\\n tokenOfEdition,\\n editionSize\\n );\\n return encodeMetadataJSON(json);\\n }\\n\\n /// Function to create the metadata json string for the nft edition\\n /// @param name Name of NFT in metadata\\n /// @param description Description of NFT in metadata\\n /// @param mediaData Data for media to include in json object\\n /// @param tokenOfEdition Token ID for specific token\\n /// @param editionSize Size of entire edition to show\\n function createMetadataJSON(\\n string memory name,\\n string memory description,\\n string memory mediaData,\\n uint256 tokenOfEdition,\\n uint256 editionSize\\n ) public pure returns (bytes memory) {\\n bytes memory editionSizeText;\\n if (editionSize > 0) {\\n editionSizeText = abi.encodePacked(\\n \\\"/\\\",\\n numberToString(editionSize)\\n );\\n }\\n return\\n // solhint-disable quotes\\n abi.encodePacked(\\n '{\\\"name\\\": \\\"',\\n name,\\n \\\" \\\",\\n numberToString(tokenOfEdition),\\n editionSizeText,\\n '\\\", \\\"',\\n 'description\\\": \\\"',\\n description,\\n '\\\", \\\"',\\n mediaData,\\n 'properties\\\": {\\\"number\\\": ',\\n numberToString(tokenOfEdition),\\n '}}'\\n );\\n // solhint-enable quotes\\n }\\n\\n /// Encodes the argument json bytes into base64-data uri format\\n /// @param json Raw json to base64 and turn into a data-uri\\n function encodeMetadataJSON(bytes memory json)\\n public\\n pure\\n override\\n returns (string memory)\\n {\\n return\\n string(\\n abi.encodePacked(\\n \\\"data:application/json;base64,\\\",\\n base64Encode(json)\\n )\\n );\\n }\\n\\n /// Generates edition metadata from storage information as base64-json blob\\n /// Combines the media data and metadata\\n /// @param imageUrl URL of image to render for edition\\n /// @param animationUrl URL of animation to render for edition\\n function tokenMediaData(\\n string memory imageUrl,\\n string memory animationUrl,\\n uint256 tokenOfEdition\\n ) public pure returns (string memory) {\\n bool hasImage = bytes(imageUrl).length > 0;\\n bool hasAnimation = bytes(animationUrl).length > 0;\\n if (hasImage && hasAnimation) {\\n return\\n // solhint-disable quotes\\n string(\\n abi.encodePacked(\\n 'image\\\": \\\"',\\n imageUrl,\\n \\\"?id=\\\",\\n numberToString(tokenOfEdition),\\n '\\\", \\\"animation_url\\\": \\\"',\\n animationUrl,\\n \\\"?id=\\\",\\n numberToString(tokenOfEdition),\\n '\\\", \\\"'\\n )\\n );\\n // solhint-enable quotes\\n }\\n if (hasImage) {\\n return\\n // solhint-disable quotes\\n string(\\n abi.encodePacked(\\n 'image\\\": \\\"',\\n imageUrl,\\n \\\"?id=\\\",\\n numberToString(tokenOfEdition),\\n '\\\", \\\"'\\n )\\n );\\n // solhint-enable quotes\\n }\\n if (hasAnimation) {\\n return\\n // solhint-disable quotes\\n string(\\n abi.encodePacked(\\n 'animation_url\\\": \\\"',\\n animationUrl,\\n \\\"?id=\\\",\\n numberToString(tokenOfEdition),\\n '\\\", \\\"'\\n )\\n );\\n // solhint-enable quotes\\n }\\n\\n return \\\"\\\";\\n }\\n}\\n\",\"keccak256\":\"0x869febd1ab7ac26132dc40cc132cc527f480131b1d2b92a4a65af60ca16660a5\",\"license\":\"GPL-3.0\"}},\"version\":1}", + "bytecode": "0x608060405234801561001057600080fd5b5060405161062338038061062383398101604081905261002f91610054565b600180546001600160a01b0319166001600160a01b0392909216919091179055610084565b60006020828403121561006657600080fd5b81516001600160a01b038116811461007d57600080fd5b9392505050565b610590806100936000396000f3fe608060405234801561001057600080fd5b50600436106100415760003560e01c80635c60da1b146100465780636cdf1b3914610076578063c473a8f314610097575b600080fd5b600154610059906001600160a01b031681565b6040516001600160a01b0390911681526020015b60405180910390f35b6100896100843660046103f2565b6100aa565b60405190815260200161006d565b6100596100a536600461047c565b610216565b60008082116100f85760405162461bcd60e51b8152602060048201526015602482015274044726f702073697a65206d757374206265203e203605c1b60448201526064015b60405180910390fd5b600154600090610142906001600160a01b0316610113835490565b60405160200161012591815260200190565b60405160208183030381529060405261013d90610495565b6102b5565b6040516393d2392360e01b81529091506001600160a01b038216906393d23923906101799033908a908a908a908a90600401610509565b600060405180830381600087803b15801561019357600080fd5b505af11580156101a7573d6000803e3d6000fd5b5050505060006101b660005490565b604080518681526001600160a01b0385166020820152919250339183917f7260749523ea87a695e55bf7fb40009887d330ef68e44c5784725bc97fa4a6d7910160405180910390a361020c600080546001019055565b9695505050505050565b60015460408051602081018490526000926102af926001600160a01b03909116910160405160208183030381529060405261025090610495565b30604051733d602d80600a3d3981f3363d3d373d3d3d363d7360601b8152606093841b60148201526f5af43d82803e903d91602b57fd5bf3ff60801b6028820152921b6038830152604c8201526037808220606c830152605591012090565b92915050565b6000604051733d602d80600a3d3981f3363d3d373d3d3d363d7360601b81528360601b60148201526e5af43d82803e903d91602b57fd5bf360881b6028820152826037826000f59150506001600160a01b0381166102af5760405162461bcd60e51b8152602060048201526017602482015276115490cc4c4d8dce8818dc99585d194c8819985a5b1959604a1b60448201526064016100ef565b634e487b7160e01b600052604160045260246000fd5b600082601f83011261037657600080fd5b813567ffffffffffffffff808211156103915761039161034f565b604051601f8301601f19908116603f011681019082821181831017156103b9576103b961034f565b816040528381528660208588010111156103d257600080fd5b836020870160208301376000602085830101528094505050505092915050565b6000806000806080858703121561040857600080fd5b84356001600160a01b038116811461041f57600080fd5b9350602085013567ffffffffffffffff8082111561043c57600080fd5b61044888838901610365565b9450604087013591508082111561045e57600080fd5b5061046b87828801610365565b949793965093946060013593505050565b60006020828403121561048e57600080fd5b5035919050565b805160208083015191908110156104b6576000198160200360031b1b821691505b50919050565b6000815180845260005b818110156104e2576020818501810151868301820152016104c6565b818111156104f4576000602083870101525b50601f01601f19169290920160200192915050565b6001600160a01b0386811682528516602082015260a060408201819052600090610535908301866104bc565b828103606084015261054781866104bc565b915050826080830152969550505050505056fea264697066735822122035b6b094818288413fd275bb0fd5b6c5cc55d83c2d21b0fe4a039adf91a3f82e64736f6c634300080f0033", + "deployedBytecode": "0x608060405234801561001057600080fd5b50600436106100415760003560e01c80635c60da1b146100465780636cdf1b3914610076578063c473a8f314610097575b600080fd5b600154610059906001600160a01b031681565b6040516001600160a01b0390911681526020015b60405180910390f35b6100896100843660046103f2565b6100aa565b60405190815260200161006d565b6100596100a536600461047c565b610216565b60008082116100f85760405162461bcd60e51b8152602060048201526015602482015274044726f702073697a65206d757374206265203e203605c1b60448201526064015b60405180910390fd5b600154600090610142906001600160a01b0316610113835490565b60405160200161012591815260200190565b60405160208183030381529060405261013d90610495565b6102b5565b6040516393d2392360e01b81529091506001600160a01b038216906393d23923906101799033908a908a908a908a90600401610509565b600060405180830381600087803b15801561019357600080fd5b505af11580156101a7573d6000803e3d6000fd5b5050505060006101b660005490565b604080518681526001600160a01b0385166020820152919250339183917f7260749523ea87a695e55bf7fb40009887d330ef68e44c5784725bc97fa4a6d7910160405180910390a361020c600080546001019055565b9695505050505050565b60015460408051602081018490526000926102af926001600160a01b03909116910160405160208183030381529060405261025090610495565b30604051733d602d80600a3d3981f3363d3d373d3d3d363d7360601b8152606093841b60148201526f5af43d82803e903d91602b57fd5bf3ff60801b6028820152921b6038830152604c8201526037808220606c830152605591012090565b92915050565b6000604051733d602d80600a3d3981f3363d3d373d3d3d363d7360601b81528360601b60148201526e5af43d82803e903d91602b57fd5bf360881b6028820152826037826000f59150506001600160a01b0381166102af5760405162461bcd60e51b8152602060048201526017602482015276115490cc4c4d8dce8818dc99585d194c8819985a5b1959604a1b60448201526064016100ef565b634e487b7160e01b600052604160045260246000fd5b600082601f83011261037657600080fd5b813567ffffffffffffffff808211156103915761039161034f565b604051601f8301601f19908116603f011681019082821181831017156103b9576103b961034f565b816040528381528660208588010111156103d257600080fd5b836020870160208301376000602085830101528094505050505092915050565b6000806000806080858703121561040857600080fd5b84356001600160a01b038116811461041f57600080fd5b9350602085013567ffffffffffffffff8082111561043c57600080fd5b61044888838901610365565b9450604087013591508082111561045e57600080fd5b5061046b87828801610365565b949793965093946060013593505050565b60006020828403121561048e57600080fd5b5035919050565b805160208083015191908110156104b6576000198160200360031b1b821691505b50919050565b6000815180845260005b818110156104e2576020818501810151868301820152016104c6565b818111156104f4576000602083870101525b50601f01601f19169290920160200192915050565b6001600160a01b0386811682528516602082015260a060408201819052600090610535908301866104bc565b828103606084015261054781866104bc565b915050826080830152969550505050505056fea264697066735822122035b6b094818288413fd275bb0fd5b6c5cc55d83c2d21b0fe4a039adf91a3f82e64736f6c634300080f0033", "devdoc": { "events": { "CreatedDrop(uint256,address,uint256,address)": { diff --git a/deployments/rinkeby/ExpandedNFT.json b/deployments/rinkeby/ExpandedNFT.json index c9c200f6..ebaacab7 100644 --- a/deployments/rinkeby/ExpandedNFT.json +++ b/deployments/rinkeby/ExpandedNFT.json @@ -1,5 +1,5 @@ { - "address": "0x856e6c62c5564e9013053b3988E7E4256F3887dC", + "address": "0xfddbEb53227C2CB1729C3C35c6480c79603EC373", "abi": [ { "inputs": [ @@ -683,22 +683,22 @@ }, { "internalType": "string[]", - "name": "imageUrl", + "name": "animationUrl", "type": "string[]" }, { "internalType": "bytes32[]", - "name": "imageHash", + "name": "animationHash", "type": "bytes32[]" }, { "internalType": "string[]", - "name": "animationUrl", + "name": "imageUrl", "type": "string[]" }, { "internalType": "bytes32[]", - "name": "animationHash", + "name": "imageHash", "type": "bytes32[]" } ], @@ -1383,29 +1383,29 @@ "type": "function" } ], - "transactionHash": "0xf67c0fdf48f35e82faf5be105a489357351263a87a62b33f6a38b70728290e97", + "transactionHash": "0xfc4efa3fd6925546adb2c1afb3502a80ceeb18f8f2ba66bcd9fe983ec6a0b904", "receipt": { "to": null, "from": "0xaD1fcD83DE77518d3D1b769F22B0A169eD55A919", - "contractAddress": "0x856e6c62c5564e9013053b3988E7E4256F3887dC", - "transactionIndex": 12, + "contractAddress": "0xfddbEb53227C2CB1729C3C35c6480c79603EC373", + "transactionIndex": 16, "gasUsed": "4143710", "logsBloom": "0x00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000", - "blockHash": "0x500c85a1ccda0693351b80987473ccbfd401a4cc6f013521ae20d6a6c0a7e9c8", - "transactionHash": "0xf67c0fdf48f35e82faf5be105a489357351263a87a62b33f6a38b70728290e97", + "blockHash": "0xb12122dc267940ea23e9a2f4fa86e68136dedaf33373ece371c2905101c8741a", + "transactionHash": "0xfc4efa3fd6925546adb2c1afb3502a80ceeb18f8f2ba66bcd9fe983ec6a0b904", "logs": [], - "blockNumber": 11091889, - "cumulativeGasUsed": "17535949", + "blockNumber": 11092052, + "cumulativeGasUsed": "10671373", "status": 1, "byzantium": true }, "args": [ - "0x952dc1F837Bae76512cBBC744B4e341E8b22a048" + "0x691fB71440A5d5A4A48d8F65020E67F011e413b7" ], - "solcInputHash": "f22aeb84b70c3f3bdbb59fc583a836f8", - "metadata": "{\"compiler\":{\"version\":\"0.8.15+commit.e14f2714\"},\"language\":\"Solidity\",\"output\":{\"abi\":[{\"inputs\":[{\"internalType\":\"contract SharedNFTLogic\",\"name\":\"sharedNFTLogic\",\"type\":\"address\"}],\"stateMutability\":\"nonpayable\",\"type\":\"constructor\"},{\"anonymous\":false,\"inputs\":[{\"indexed\":true,\"internalType\":\"address\",\"name\":\"owner\",\"type\":\"address\"},{\"indexed\":true,\"internalType\":\"address\",\"name\":\"approved\",\"type\":\"address\"},{\"indexed\":true,\"internalType\":\"uint256\",\"name\":\"tokenId\",\"type\":\"uint256\"}],\"name\":\"Approval\",\"type\":\"event\"},{\"anonymous\":false,\"inputs\":[{\"indexed\":true,\"internalType\":\"address\",\"name\":\"owner\",\"type\":\"address\"},{\"indexed\":true,\"internalType\":\"address\",\"name\":\"operator\",\"type\":\"address\"},{\"indexed\":false,\"internalType\":\"bool\",\"name\":\"approved\",\"type\":\"bool\"}],\"name\":\"ApprovalForAll\",\"type\":\"event\"},{\"anonymous\":false,\"inputs\":[{\"indexed\":false,\"internalType\":\"uint256\",\"name\":\"tokenId\",\"type\":\"uint256\"}],\"name\":\"DeliveryAccepted\",\"type\":\"event\"},{\"anonymous\":false,\"inputs\":[{\"indexed\":false,\"internalType\":\"uint256\",\"name\":\"price\",\"type\":\"uint256\"},{\"indexed\":false,\"internalType\":\"address\",\"name\":\"owner\",\"type\":\"address\"}],\"name\":\"EditionSold\",\"type\":\"event\"},{\"anonymous\":false,\"inputs\":[{\"indexed\":false,\"internalType\":\"uint8\",\"name\":\"version\",\"type\":\"uint8\"}],\"name\":\"Initialized\",\"type\":\"event\"},{\"anonymous\":false,\"inputs\":[{\"indexed\":false,\"internalType\":\"uint256\",\"name\":\"tokenId\",\"type\":\"uint256\"}],\"name\":\"OfferAccepted\",\"type\":\"event\"},{\"anonymous\":false,\"inputs\":[{\"indexed\":false,\"internalType\":\"uint256\",\"name\":\"tokenId\",\"type\":\"uint256\"}],\"name\":\"OfferRejected\",\"type\":\"event\"},{\"anonymous\":false,\"inputs\":[{\"indexed\":false,\"internalType\":\"uint256\",\"name\":\"tokenId\",\"type\":\"uint256\"}],\"name\":\"OfferTermsSet\",\"type\":\"event\"},{\"anonymous\":false,\"inputs\":[{\"indexed\":true,\"internalType\":\"address\",\"name\":\"previousOwner\",\"type\":\"address\"},{\"indexed\":true,\"internalType\":\"address\",\"name\":\"newOwner\",\"type\":\"address\"}],\"name\":\"OwnershipTransferred\",\"type\":\"event\"},{\"anonymous\":false,\"inputs\":[{\"indexed\":false,\"internalType\":\"uint256\",\"name\":\"amount\",\"type\":\"uint256\"}],\"name\":\"PriceChanged\",\"type\":\"event\"},{\"anonymous\":false,\"inputs\":[{\"indexed\":false,\"internalType\":\"uint256\",\"name\":\"tokenId\",\"type\":\"uint256\"}],\"name\":\"ProductionComplete\",\"type\":\"event\"},{\"anonymous\":false,\"inputs\":[{\"indexed\":false,\"internalType\":\"uint256\",\"name\":\"tokenId\",\"type\":\"uint256\"},{\"indexed\":false,\"internalType\":\"address\",\"name\":\"owner\",\"type\":\"address\"}],\"name\":\"RedeemAborted\",\"type\":\"event\"},{\"anonymous\":false,\"inputs\":[{\"indexed\":false,\"internalType\":\"uint256\",\"name\":\"tokenId\",\"type\":\"uint256\"},{\"indexed\":false,\"internalType\":\"address\",\"name\":\"owner\",\"type\":\"address\"}],\"name\":\"RedeemStarted\",\"type\":\"event\"},{\"anonymous\":false,\"inputs\":[{\"indexed\":true,\"internalType\":\"address\",\"name\":\"from\",\"type\":\"address\"},{\"indexed\":true,\"internalType\":\"address\",\"name\":\"to\",\"type\":\"address\"},{\"indexed\":true,\"internalType\":\"uint256\",\"name\":\"tokenId\",\"type\":\"uint256\"}],\"name\":\"Transfer\",\"type\":\"event\"},{\"anonymous\":false,\"inputs\":[{\"indexed\":false,\"internalType\":\"enum ExpandedNFT.WhoCanMint\",\"name\":\"minters\",\"type\":\"uint8\"}],\"name\":\"WhoCanMintChanged\",\"type\":\"event\"},{\"inputs\":[{\"internalType\":\"uint256\",\"name\":\"tokenId\",\"type\":\"uint256\"}],\"name\":\"abortRedemption\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"uint256\",\"name\":\"tokenId\",\"type\":\"uint256\"}],\"name\":\"acceptDelivery\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"uint256\",\"name\":\"tokenId\",\"type\":\"uint256\"},{\"internalType\":\"uint256\",\"name\":\"paymentAmount\",\"type\":\"uint256\"}],\"name\":\"acceptOfferTerms\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"address\",\"name\":\"to\",\"type\":\"address\"},{\"internalType\":\"uint256\",\"name\":\"tokenId\",\"type\":\"uint256\"}],\"name\":\"approve\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"address\",\"name\":\"owner\",\"type\":\"address\"}],\"name\":\"balanceOf\",\"outputs\":[{\"internalType\":\"uint256\",\"name\":\"\",\"type\":\"uint256\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"uint256\",\"name\":\"tokenId\",\"type\":\"uint256\"}],\"name\":\"burn\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[],\"name\":\"description\",\"outputs\":[{\"internalType\":\"string\",\"name\":\"\",\"type\":\"string\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[],\"name\":\"dropSize\",\"outputs\":[{\"internalType\":\"uint256\",\"name\":\"\",\"type\":\"uint256\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[],\"name\":\"getAllowedMinter\",\"outputs\":[{\"internalType\":\"enum ExpandedNFT.WhoCanMint\",\"name\":\"\",\"type\":\"uint8\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"uint256\",\"name\":\"tokenId\",\"type\":\"uint256\"}],\"name\":\"getApproved\",\"outputs\":[{\"internalType\":\"address\",\"name\":\"\",\"type\":\"address\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[],\"name\":\"getArtistWallet\",\"outputs\":[{\"internalType\":\"address\",\"name\":\"\",\"type\":\"address\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"uint256\",\"name\":\"tokenId\",\"type\":\"uint256\"}],\"name\":\"getConditionReport\",\"outputs\":[{\"internalType\":\"string\",\"name\":\"\",\"type\":\"string\"},{\"internalType\":\"bytes32\",\"name\":\"\",\"type\":\"bytes32\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[],\"name\":\"getGeneralMintLimit\",\"outputs\":[{\"internalType\":\"uint256\",\"name\":\"\",\"type\":\"uint256\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[],\"name\":\"getMembersMintLimit\",\"outputs\":[{\"internalType\":\"uint256\",\"name\":\"\",\"type\":\"uint256\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[],\"name\":\"getMembersSalePrice\",\"outputs\":[{\"internalType\":\"uint256\",\"name\":\"\",\"type\":\"uint256\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[],\"name\":\"getPaymentToken\",\"outputs\":[{\"internalType\":\"address\",\"name\":\"\",\"type\":\"address\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[],\"name\":\"getRoyaltyBPS\",\"outputs\":[{\"internalType\":\"uint256\",\"name\":\"\",\"type\":\"uint256\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[],\"name\":\"getSplitBPS\",\"outputs\":[{\"internalType\":\"uint256\",\"name\":\"\",\"type\":\"uint256\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"uint256\",\"name\":\"tokenId\",\"type\":\"uint256\"}],\"name\":\"getURIs\",\"outputs\":[{\"internalType\":\"string\",\"name\":\"\",\"type\":\"string\"},{\"internalType\":\"bytes32\",\"name\":\"\",\"type\":\"bytes32\"},{\"internalType\":\"string\",\"name\":\"\",\"type\":\"string\"},{\"internalType\":\"bytes32\",\"name\":\"\",\"type\":\"bytes32\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[],\"name\":\"getVIPMintLimit\",\"outputs\":[{\"internalType\":\"uint256\",\"name\":\"\",\"type\":\"uint256\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[],\"name\":\"getVIPSalePrice\",\"outputs\":[{\"internalType\":\"uint256\",\"name\":\"\",\"type\":\"uint256\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[],\"name\":\"getWhoCanMint\",\"outputs\":[{\"internalType\":\"uint256\",\"name\":\"\",\"type\":\"uint256\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"address\",\"name\":\"_owner\",\"type\":\"address\"},{\"internalType\":\"address\",\"name\":\"artistWallet\",\"type\":\"address\"},{\"internalType\":\"string\",\"name\":\"_name\",\"type\":\"string\"},{\"internalType\":\"string\",\"name\":\"_symbol\",\"type\":\"string\"},{\"internalType\":\"uint256\",\"name\":\"_dropSize\",\"type\":\"uint256\"}],\"name\":\"initialize\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"address\",\"name\":\"owner\",\"type\":\"address\"},{\"internalType\":\"address\",\"name\":\"operator\",\"type\":\"address\"}],\"name\":\"isApprovedForAll\",\"outputs\":[{\"internalType\":\"bool\",\"name\":\"\",\"type\":\"bool\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"uint256\",\"name\":\"startOffset\",\"type\":\"uint256\"},{\"internalType\":\"uint256\",\"name\":\"count\",\"type\":\"uint256\"},{\"internalType\":\"string[]\",\"name\":\"_description\",\"type\":\"string[]\"},{\"internalType\":\"string[]\",\"name\":\"imageUrl\",\"type\":\"string[]\"},{\"internalType\":\"bytes32[]\",\"name\":\"imageHash\",\"type\":\"bytes32[]\"},{\"internalType\":\"string[]\",\"name\":\"animationUrl\",\"type\":\"string[]\"},{\"internalType\":\"bytes32[]\",\"name\":\"animationHash\",\"type\":\"bytes32[]\"}],\"name\":\"loadMetadataChunk\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[],\"name\":\"metadataloaded\",\"outputs\":[{\"internalType\":\"bool\",\"name\":\"\",\"type\":\"bool\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"address\",\"name\":\"to\",\"type\":\"address\"}],\"name\":\"mintEdition\",\"outputs\":[{\"internalType\":\"uint256\",\"name\":\"\",\"type\":\"uint256\"}],\"stateMutability\":\"payable\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"address[]\",\"name\":\"recipients\",\"type\":\"address[]\"}],\"name\":\"mintEditions\",\"outputs\":[{\"internalType\":\"uint256\",\"name\":\"\",\"type\":\"uint256\"}],\"stateMutability\":\"payable\",\"type\":\"function\"},{\"inputs\":[],\"name\":\"name\",\"outputs\":[{\"internalType\":\"string\",\"name\":\"\",\"type\":\"string\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[],\"name\":\"numberCanMint\",\"outputs\":[{\"internalType\":\"uint256\",\"name\":\"\",\"type\":\"uint256\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[],\"name\":\"owner\",\"outputs\":[{\"internalType\":\"address\",\"name\":\"\",\"type\":\"address\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"uint256\",\"name\":\"tokenId\",\"type\":\"uint256\"}],\"name\":\"ownerOf\",\"outputs\":[{\"internalType\":\"address\",\"name\":\"\",\"type\":\"address\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"uint256\",\"name\":\"tokenId\",\"type\":\"uint256\"},{\"internalType\":\"string\",\"name\":\"_description\",\"type\":\"string\"},{\"internalType\":\"string\",\"name\":\"animationUrl\",\"type\":\"string\"},{\"internalType\":\"bytes32\",\"name\":\"animationHash\",\"type\":\"bytes32\"},{\"internalType\":\"string\",\"name\":\"imageUrl\",\"type\":\"string\"},{\"internalType\":\"bytes32\",\"name\":\"imageHash\",\"type\":\"bytes32\"},{\"internalType\":\"string\",\"name\":\"conditionReportUrl\",\"type\":\"string\"},{\"internalType\":\"bytes32\",\"name\":\"conditionReportHash\",\"type\":\"bytes32\"}],\"name\":\"productionComplete\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[],\"name\":\"purchase\",\"outputs\":[{\"internalType\":\"uint256\",\"name\":\"\",\"type\":\"uint256\"}],\"stateMutability\":\"payable\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"uint256\",\"name\":\"tokenId\",\"type\":\"uint256\"}],\"name\":\"redeem\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"uint256\",\"name\":\"tokenId\",\"type\":\"uint256\"}],\"name\":\"rejectOfferTerms\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[],\"name\":\"renounceOwnership\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"address[]\",\"name\":\"wallets\",\"type\":\"address[]\"},{\"internalType\":\"uint256[]\",\"name\":\"tokenIDs\",\"type\":\"uint256[]\"}],\"name\":\"reserve\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"uint256\",\"name\":\"\",\"type\":\"uint256\"},{\"internalType\":\"uint256\",\"name\":\"_salePrice\",\"type\":\"uint256\"}],\"name\":\"royaltyInfo\",\"outputs\":[{\"internalType\":\"address\",\"name\":\"receiver\",\"type\":\"address\"},{\"internalType\":\"uint256\",\"name\":\"royaltyAmount\",\"type\":\"uint256\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"address\",\"name\":\"from\",\"type\":\"address\"},{\"internalType\":\"address\",\"name\":\"to\",\"type\":\"address\"},{\"internalType\":\"uint256\",\"name\":\"tokenId\",\"type\":\"uint256\"}],\"name\":\"safeTransferFrom\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"address\",\"name\":\"from\",\"type\":\"address\"},{\"internalType\":\"address\",\"name\":\"to\",\"type\":\"address\"},{\"internalType\":\"uint256\",\"name\":\"tokenId\",\"type\":\"uint256\"},{\"internalType\":\"bytes\",\"name\":\"data\",\"type\":\"bytes\"}],\"name\":\"safeTransferFrom\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[],\"name\":\"salePrice\",\"outputs\":[{\"internalType\":\"uint256\",\"name\":\"\",\"type\":\"uint256\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"enum ExpandedNFT.WhoCanMint\",\"name\":\"minters\",\"type\":\"uint8\"}],\"name\":\"setAllowedMinter\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"address\",\"name\":\"operator\",\"type\":\"address\"},{\"internalType\":\"bool\",\"name\":\"approved\",\"type\":\"bool\"}],\"name\":\"setApprovalForAll\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"uint256\",\"name\":\"count\",\"type\":\"uint256\"},{\"internalType\":\"address[]\",\"name\":\"minter\",\"type\":\"address[]\"},{\"internalType\":\"bool[]\",\"name\":\"allowed\",\"type\":\"bool[]\"}],\"name\":\"setApprovedMinters\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"uint256\",\"name\":\"count\",\"type\":\"uint256\"},{\"internalType\":\"address[]\",\"name\":\"minter\",\"type\":\"address[]\"},{\"internalType\":\"bool[]\",\"name\":\"allowed\",\"type\":\"bool[]\"}],\"name\":\"setApprovedVIPMinters\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"address\",\"name\":\"wallet\",\"type\":\"address\"}],\"name\":\"setArtistWallet\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"uint256\",\"name\":\"_salePrice\",\"type\":\"uint256\"}],\"name\":\"setMembersSalePrice\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"uint256\",\"name\":\"tokenId\",\"type\":\"uint256\"},{\"internalType\":\"uint256\",\"name\":\"fee\",\"type\":\"uint256\"}],\"name\":\"setOfferTerms\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"address\",\"name\":\"paymentToken\",\"type\":\"address\"}],\"name\":\"setPaymentToken\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"uint256\",\"name\":\"_royaltyBPS\",\"type\":\"uint256\"},{\"internalType\":\"uint256\",\"name\":\"_splitBPS\",\"type\":\"uint256\"},{\"internalType\":\"uint256\",\"name\":\"_vipSalePrice\",\"type\":\"uint256\"},{\"internalType\":\"uint256\",\"name\":\"_membersSalePrice\",\"type\":\"uint256\"},{\"internalType\":\"uint256\",\"name\":\"_generalSalePrice\",\"type\":\"uint256\"},{\"internalType\":\"uint256\",\"name\":\"_vipMintLimit\",\"type\":\"uint256\"},{\"internalType\":\"uint256\",\"name\":\"_membersMintLimit\",\"type\":\"uint256\"},{\"internalType\":\"uint256\",\"name\":\"_generalMintLimit\",\"type\":\"uint256\"}],\"name\":\"setPricing\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"uint256\",\"name\":\"_salePrice\",\"type\":\"uint256\"}],\"name\":\"setSalePrice\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"uint256\",\"name\":\"vipSalePrice\",\"type\":\"uint256\"},{\"internalType\":\"uint256\",\"name\":\"membersSalePrice\",\"type\":\"uint256\"},{\"internalType\":\"uint256\",\"name\":\"generalSalePrice\",\"type\":\"uint256\"}],\"name\":\"setSalePrices\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"uint256\",\"name\":\"_salePrice\",\"type\":\"uint256\"}],\"name\":\"setVIPSalePrice\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"bytes4\",\"name\":\"interfaceId\",\"type\":\"bytes4\"}],\"name\":\"supportsInterface\",\"outputs\":[{\"internalType\":\"bool\",\"name\":\"\",\"type\":\"bool\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[],\"name\":\"symbol\",\"outputs\":[{\"internalType\":\"string\",\"name\":\"\",\"type\":\"string\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"uint256\",\"name\":\"tokenId\",\"type\":\"uint256\"}],\"name\":\"tokenURI\",\"outputs\":[{\"internalType\":\"string\",\"name\":\"\",\"type\":\"string\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[],\"name\":\"totalSupply\",\"outputs\":[{\"internalType\":\"uint256\",\"name\":\"\",\"type\":\"uint256\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"address\",\"name\":\"from\",\"type\":\"address\"},{\"internalType\":\"address\",\"name\":\"to\",\"type\":\"address\"},{\"internalType\":\"uint256\",\"name\":\"tokenId\",\"type\":\"uint256\"}],\"name\":\"transferFrom\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"address\",\"name\":\"newOwner\",\"type\":\"address\"}],\"name\":\"transferOwnership\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"uint256\",\"name\":\"tokenId\",\"type\":\"uint256\"},{\"internalType\":\"string\",\"name\":\"imageUrl\",\"type\":\"string\"},{\"internalType\":\"string\",\"name\":\"animationUrl\",\"type\":\"string\"}],\"name\":\"updateEditionURLs\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[],\"name\":\"withdraw\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"}],\"devdoc\":{\"author\":\"Zien Repository: https://github.com/joinzien/expanded-nft\",\"details\":\"This allows creators to mint a unique serial drop of an expanded NFT within a custom contract\",\"kind\":\"dev\",\"methods\":{\"approve(address,uint256)\":{\"details\":\"See {IERC721-approve}.\"},\"balanceOf(address)\":{\"details\":\"See {IERC721-balanceOf}.\"},\"burn(uint256)\":{\"params\":{\"tokenId\":\"Token ID to burn User burn function for token id \"}},\"getAllowedMinter()\":{\"details\":\"Sets the types of users who is allowed to mint.\"},\"getApproved(uint256)\":{\"details\":\"See {IERC721-getApproved}.\"},\"getConditionReport(uint256)\":{\"details\":\"Get URIs for the condition report\",\"returns\":{\"_0\":\"_imageUrl, _imageHash\"}},\"getGeneralMintLimit()\":{\"details\":\"returns the general mint limit\"},\"getMembersMintLimit()\":{\"details\":\"returns the member mint limit\"},\"getMembersSalePrice()\":{\"details\":\"returns the member sale price\"},\"getRoyaltyBPS()\":{\"details\":\"returns the royalty BPS\"},\"getSplitBPS()\":{\"details\":\"returns the split BPS\"},\"getURIs(uint256)\":{\"details\":\"Get URIs for edition NFT\",\"returns\":{\"_0\":\"_imageUrl, _imageHash, _animationUrl, _animationHash\"}},\"getVIPMintLimit()\":{\"details\":\"returns the VIP mint limit\"},\"getVIPSalePrice()\":{\"details\":\"returns the VIP sale price\"},\"getWhoCanMint()\":{\"details\":\"returns who can mint\"},\"initialize(address,address,string,string,uint256)\":{\"details\":\"Function to create a new drop. Can only be called by the allowed creator Sets the only allowed minter to the address that creates/owns the drop. This can be re-assigned or updated later\",\"params\":{\"_dropSize\":\"Number of editions that can be minted in total. \",\"_name\":\"Name of drop, used in the title as \\\"$NAME NUMBER/TOTAL\\\"\",\"_owner\":\"wallet addres for the user that owns and can mint the drop, gets royalty and sales payouts and can update the base url if needed.\",\"_symbol\":\"Symbol of the new token contract\",\"artistWallet\":\"wallet address for thr User that created the drop\"}},\"isApprovedForAll(address,address)\":{\"details\":\"See {IERC721-isApprovedForAll}.\"},\"loadMetadataChunk(uint256,uint256,string[],string[],bytes32[],string[],bytes32[])\":{\"details\":\"Function to create a new drop. Can only be called by the allowed creator Sets the only allowed minter to the address that creates/owns the drop. This can be re-assigned or updated later\",\"params\":{\"_description\":\"Description of the edition, used in the description field of the NFT\",\"animationHash\":\"The associated hash of the animation in sha-256 bytes32 format. If animation is omitted the hash can be zero.\",\"animationUrl\":\"Animation URL of the edition. Not required, but if omitted image URL needs to be included. This follows the opensea spec for NFTs\",\"imageHash\":\"SHA256 of the given image in bytes32 format (0xHASH). If no image is included, the hash can be zero.\",\"imageUrl\":\"Image URL of the the edition. Strongly encouraged to be used, if necessary, only animation URL can be used. One of animation and image url need to exist in a drop to render the NFT.\"}},\"mintEdition(address)\":{\"details\":\"This mints one edition to the given address by an allowed minter on the edition instance.\",\"params\":{\"to\":\"address to send the newly minted edition to\"}},\"mintEditions(address[])\":{\"details\":\"This mints multiple editions to the given list of addresses.\",\"params\":{\"recipients\":\"list of addresses to send the newly minted editions to\"}},\"name()\":{\"details\":\"See {IERC721Metadata-name}.\"},\"ownerOf(uint256)\":{\"details\":\"See {IERC721-ownerOf}.\"},\"purchase()\":{\"details\":\"This allows the user to purchase an edition at the given price in the contract.\"},\"renounceOwnership()\":{\"details\":\"Leaves the contract without owner. It will not be possible to call `onlyOwner` functions anymore. Can only be called by the current owner. NOTE: Renouncing ownership will leave the contract without an owner, thereby removing any functionality that is only available to the owner.\"},\"reserve(address[],uint256[])\":{\"details\":\"Set various pricing related values\",\"params\":{\"tokenIDs\":\"A list of tokenId to reserve \",\"wallets\":\"A list of wallets\"}},\"royaltyInfo(uint256,uint256)\":{\"details\":\"Get royalty information for token\",\"params\":{\"_salePrice\":\"Sale price for the token\"}},\"safeTransferFrom(address,address,uint256)\":{\"details\":\"See {IERC721-safeTransferFrom}.\"},\"safeTransferFrom(address,address,uint256,bytes)\":{\"details\":\"See {IERC721-safeTransferFrom}.\"},\"setAllowedMinter(uint8)\":{\"details\":\"Sets the types of users who is allowed to mint.\",\"params\":{\"minters\":\"WhoCanMint enum of minter types\"}},\"setApprovalForAll(address,bool)\":{\"details\":\"See {IERC721-setApprovalForAll}.\"},\"setApprovedMinters(uint256,address[],bool[])\":{\"details\":\"Sets the approved minting status of the given address. This requires that msg.sender is the owner of the given edition id. If the ZeroAddress (address(0x0)) is set as a minter, anyone will be allowed to mint. This setup is similar to setApprovalForAll in the ERC721 spec.\",\"params\":{\"allowed\":\"boolean if that address is allowed to mint\",\"minter\":\"address to set approved minting status for\"}},\"setApprovedVIPMinters(uint256,address[],bool[])\":{\"details\":\"Sets the approved minting status of the given address. This requires that msg.sender is the owner of the given edition id. If the ZeroAddress (address(0x0)) is set as a minter, anyone will be allowed to mint. This setup is similar to setApprovalForAll in the ERC721 spec.\",\"params\":{\"allowed\":\"boolean if that address is allowed to mint\",\"minter\":\"address to set approved minting status for\"}},\"setMembersSalePrice(uint256)\":{\"details\":\"This sets the members ETH sales price Setting a sales price allows users to mint the drop until it sells out. For more granular sales, use an external sales contract.\",\"params\":{\"_salePrice\":\"if sale price is 0 sale is stopped, otherwise that amount of ETH is needed to start the sale.\"}},\"setPricing(uint256,uint256,uint256,uint256,uint256,uint256,uint256,uint256)\":{\"details\":\"Set various pricing related values\",\"params\":{\"_generalMintLimit\":\"Mint limit for the general public \",\"_generalSalePrice\":\"SalePrice for the general public \",\"_membersMintLimit\":\"Mint limit for Members \",\"_membersSalePrice\":\"SalePrice for Members \",\"_royaltyBPS\":\"BPS of the royalty set on the contract. Can be 0 for no royalty.\",\"_splitBPS\":\"BPS of the royalty set on the contract. Can be 0 for no royalty. \",\"_vipMintLimit\":\"Mint limit for VIPs\",\"_vipSalePrice\":\"Sale price for VIPs\"}},\"setSalePrice(uint256)\":{\"details\":\"This sets a simple ETH sales price Setting a sales price allows users to mint the drop until it sells out. For more granular sales, use an external sales contract.\",\"params\":{\"_salePrice\":\"if sale price is 0 sale is stopped, otherwise that amount of ETH is needed to start the sale.\"}},\"setSalePrices(uint256,uint256,uint256)\":{\"details\":\"This sets the members ETH sales price Setting a sales price allows users to mint the drop until it sells out. For more granular sales, use an external sales contract.\",\"params\":{\"generalSalePrice\":\"if sale price is 0 sale is stopped, otherwise that amount of ETH is needed to start the sale. \",\"membersSalePrice\":\"if sale price is 0 sale is stopped, otherwise that amount of ETH is needed to start the sale.\",\"vipSalePrice\":\"if sale price is 0 sale is stopped, otherwise that amount of ETH is needed to start the sale.\"}},\"setVIPSalePrice(uint256)\":{\"details\":\"This sets the VIP ETH sales price Setting a sales price allows users to mint the drop until it sells out. For more granular sales, use an external sales contract.\",\"params\":{\"_salePrice\":\"if sale price is 0 sale is stopped, otherwise that amount of ETH is needed to start the sale.\"}},\"symbol()\":{\"details\":\"See {IERC721Metadata-symbol}.\"},\"tokenURI(uint256)\":{\"details\":\"Get URI for given token id\",\"params\":{\"tokenId\":\"token id to get uri for\"},\"returns\":{\"_0\":\"base64-encoded json metadata object\"}},\"totalSupply()\":{\"details\":\"returns the number of minted tokens within the drop\"},\"transferFrom(address,address,uint256)\":{\"details\":\"See {IERC721-transferFrom}.\"},\"transferOwnership(address)\":{\"details\":\"Transfers ownership of the contract to a new account (`newOwner`). Can only be called by the current owner.\"},\"updateEditionURLs(uint256,string,string)\":{\"details\":\"Allows for updates of edition urls by the owner of the edition. Only URLs can be updated (data-uris are supported), hashes cannot be updated.\"},\"withdraw()\":{\"details\":\"This withdraws ETH from the contract to the contract owner.\"}},\"version\":1},\"userdoc\":{\"kind\":\"user\",\"methods\":{\"getArtistWallet()\":{\"notice\":\"return the artists wallet address\"},\"getPaymentToken()\":{\"notice\":\"return the payment tokens address\"},\"numberCanMint()\":{\"notice\":\"Returns the number of editions allowed to mint\"},\"owner()\":{\"notice\":\"Simple override for owner interface.\"},\"setArtistWallet(address)\":{\"notice\":\"set the artists wallet address\"},\"setPaymentToken(address)\":{\"notice\":\"set a new payment token address\"}},\"notice\":\"This is a smart contract for handling dynamic contract minting.\",\"version\":1}},\"settings\":{\"compilationTarget\":{\"contracts/ExpandedNFT.sol\":\"ExpandedNFT\"},\"evmVersion\":\"london\",\"libraries\":{},\"metadata\":{\"bytecodeHash\":\"ipfs\",\"useLiteralContent\":true},\"optimizer\":{\"enabled\":true,\"runs\":100},\"remappings\":[]},\"sources\":{\"@openzeppelin/contracts-upgradeable/access/OwnableUpgradeable.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n// OpenZeppelin Contracts (last updated v4.7.0) (access/Ownable.sol)\\n\\npragma solidity ^0.8.0;\\n\\nimport \\\"../utils/ContextUpgradeable.sol\\\";\\nimport \\\"../proxy/utils/Initializable.sol\\\";\\n\\n/**\\n * @dev Contract module which provides a basic access control mechanism, where\\n * there is an account (an owner) that can be granted exclusive access to\\n * specific functions.\\n *\\n * By default, the owner account will be the one that deploys the contract. This\\n * can later be changed with {transferOwnership}.\\n *\\n * This module is used through inheritance. It will make available the modifier\\n * `onlyOwner`, which can be applied to your functions to restrict their use to\\n * the owner.\\n */\\nabstract contract OwnableUpgradeable is Initializable, ContextUpgradeable {\\n address private _owner;\\n\\n event OwnershipTransferred(address indexed previousOwner, address indexed newOwner);\\n\\n /**\\n * @dev Initializes the contract setting the deployer as the initial owner.\\n */\\n function __Ownable_init() internal onlyInitializing {\\n __Ownable_init_unchained();\\n }\\n\\n function __Ownable_init_unchained() internal onlyInitializing {\\n _transferOwnership(_msgSender());\\n }\\n\\n /**\\n * @dev Throws if called by any account other than the owner.\\n */\\n modifier onlyOwner() {\\n _checkOwner();\\n _;\\n }\\n\\n /**\\n * @dev Returns the address of the current owner.\\n */\\n function owner() public view virtual returns (address) {\\n return _owner;\\n }\\n\\n /**\\n * @dev Throws if the sender is not the owner.\\n */\\n function _checkOwner() internal view virtual {\\n require(owner() == _msgSender(), \\\"Ownable: caller is not the owner\\\");\\n }\\n\\n /**\\n * @dev Leaves the contract without owner. It will not be possible to call\\n * `onlyOwner` functions anymore. Can only be called by the current owner.\\n *\\n * NOTE: Renouncing ownership will leave the contract without an owner,\\n * thereby removing any functionality that is only available to the owner.\\n */\\n function renounceOwnership() public virtual onlyOwner {\\n _transferOwnership(address(0));\\n }\\n\\n /**\\n * @dev Transfers ownership of the contract to a new account (`newOwner`).\\n * Can only be called by the current owner.\\n */\\n function transferOwnership(address newOwner) public virtual onlyOwner {\\n require(newOwner != address(0), \\\"Ownable: new owner is the zero address\\\");\\n _transferOwnership(newOwner);\\n }\\n\\n /**\\n * @dev Transfers ownership of the contract to a new account (`newOwner`).\\n * Internal function without access restriction.\\n */\\n function _transferOwnership(address newOwner) internal virtual {\\n address oldOwner = _owner;\\n _owner = newOwner;\\n emit OwnershipTransferred(oldOwner, newOwner);\\n }\\n\\n /**\\n * @dev This empty reserved space is put in place to allow future versions to add new\\n * variables without shifting down storage in the inheritance chain.\\n * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps\\n */\\n uint256[49] private __gap;\\n}\\n\",\"keccak256\":\"0x247c62047745915c0af6b955470a72d1696ebad4352d7d3011aef1a2463cd888\",\"license\":\"MIT\"},\"@openzeppelin/contracts-upgradeable/interfaces/IERC2981Upgradeable.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n// OpenZeppelin Contracts (last updated v4.6.0) (interfaces/IERC2981.sol)\\n\\npragma solidity ^0.8.0;\\n\\nimport \\\"../utils/introspection/IERC165Upgradeable.sol\\\";\\n\\n/**\\n * @dev Interface for the NFT Royalty Standard.\\n *\\n * A standardized way to retrieve royalty payment information for non-fungible tokens (NFTs) to enable universal\\n * support for royalty payments across all NFT marketplaces and ecosystem participants.\\n *\\n * _Available since v4.5._\\n */\\ninterface IERC2981Upgradeable is IERC165Upgradeable {\\n /**\\n * @dev Returns how much royalty is owed and to whom, based on a sale price that may be denominated in any unit of\\n * exchange. The royalty amount is denominated and should be paid in that same unit of exchange.\\n */\\n function royaltyInfo(uint256 tokenId, uint256 salePrice)\\n external\\n view\\n returns (address receiver, uint256 royaltyAmount);\\n}\\n\",\"keccak256\":\"0xa8ff557539dcfed5706eddde2aa929e06bb1764e71aa8c1048a78970bf3ca37d\",\"license\":\"MIT\"},\"@openzeppelin/contracts-upgradeable/proxy/utils/Initializable.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n// OpenZeppelin Contracts (last updated v4.7.0) (proxy/utils/Initializable.sol)\\n\\npragma solidity ^0.8.2;\\n\\nimport \\\"../../utils/AddressUpgradeable.sol\\\";\\n\\n/**\\n * @dev This is a base contract to aid in writing upgradeable contracts, or any kind of contract that will be deployed\\n * behind a proxy. Since proxied contracts do not make use of a constructor, it's common to move constructor logic to an\\n * external initializer function, usually called `initialize`. It then becomes necessary to protect this initializer\\n * function so it can only be called once. The {initializer} modifier provided by this contract will have this effect.\\n *\\n * The initialization functions use a version number. Once a version number is used, it is consumed and cannot be\\n * reused. This mechanism prevents re-execution of each \\\"step\\\" but allows the creation of new initialization steps in\\n * case an upgrade adds a module that needs to be initialized.\\n *\\n * For example:\\n *\\n * [.hljs-theme-light.nopadding]\\n * ```\\n * contract MyToken is ERC20Upgradeable {\\n * function initialize() initializer public {\\n * __ERC20_init(\\\"MyToken\\\", \\\"MTK\\\");\\n * }\\n * }\\n * contract MyTokenV2 is MyToken, ERC20PermitUpgradeable {\\n * function initializeV2() reinitializer(2) public {\\n * __ERC20Permit_init(\\\"MyToken\\\");\\n * }\\n * }\\n * ```\\n *\\n * TIP: To avoid leaving the proxy in an uninitialized state, the initializer function should be called as early as\\n * possible by providing the encoded function call as the `_data` argument to {ERC1967Proxy-constructor}.\\n *\\n * CAUTION: When used with inheritance, manual care must be taken to not invoke a parent initializer twice, or to ensure\\n * that all initializers are idempotent. This is not verified automatically as constructors are by Solidity.\\n *\\n * [CAUTION]\\n * ====\\n * Avoid leaving a contract uninitialized.\\n *\\n * An uninitialized contract can be taken over by an attacker. This applies to both a proxy and its implementation\\n * contract, which may impact the proxy. To prevent the implementation contract from being used, you should invoke\\n * the {_disableInitializers} function in the constructor to automatically lock it when it is deployed:\\n *\\n * [.hljs-theme-light.nopadding]\\n * ```\\n * /// @custom:oz-upgrades-unsafe-allow constructor\\n * constructor() {\\n * _disableInitializers();\\n * }\\n * ```\\n * ====\\n */\\nabstract contract Initializable {\\n /**\\n * @dev Indicates that the contract has been initialized.\\n * @custom:oz-retyped-from bool\\n */\\n uint8 private _initialized;\\n\\n /**\\n * @dev Indicates that the contract is in the process of being initialized.\\n */\\n bool private _initializing;\\n\\n /**\\n * @dev Triggered when the contract has been initialized or reinitialized.\\n */\\n event Initialized(uint8 version);\\n\\n /**\\n * @dev A modifier that defines a protected initializer function that can be invoked at most once. In its scope,\\n * `onlyInitializing` functions can be used to initialize parent contracts. Equivalent to `reinitializer(1)`.\\n */\\n modifier initializer() {\\n bool isTopLevelCall = !_initializing;\\n require(\\n (isTopLevelCall && _initialized < 1) || (!AddressUpgradeable.isContract(address(this)) && _initialized == 1),\\n \\\"Initializable: contract is already initialized\\\"\\n );\\n _initialized = 1;\\n if (isTopLevelCall) {\\n _initializing = true;\\n }\\n _;\\n if (isTopLevelCall) {\\n _initializing = false;\\n emit Initialized(1);\\n }\\n }\\n\\n /**\\n * @dev A modifier that defines a protected reinitializer function that can be invoked at most once, and only if the\\n * contract hasn't been initialized to a greater version before. In its scope, `onlyInitializing` functions can be\\n * used to initialize parent contracts.\\n *\\n * `initializer` is equivalent to `reinitializer(1)`, so a reinitializer may be used after the original\\n * initialization step. This is essential to configure modules that are added through upgrades and that require\\n * initialization.\\n *\\n * Note that versions can jump in increments greater than 1; this implies that if multiple reinitializers coexist in\\n * a contract, executing them in the right order is up to the developer or operator.\\n */\\n modifier reinitializer(uint8 version) {\\n require(!_initializing && _initialized < version, \\\"Initializable: contract is already initialized\\\");\\n _initialized = version;\\n _initializing = true;\\n _;\\n _initializing = false;\\n emit Initialized(version);\\n }\\n\\n /**\\n * @dev Modifier to protect an initialization function so that it can only be invoked by functions with the\\n * {initializer} and {reinitializer} modifiers, directly or indirectly.\\n */\\n modifier onlyInitializing() {\\n require(_initializing, \\\"Initializable: contract is not initializing\\\");\\n _;\\n }\\n\\n /**\\n * @dev Locks the contract, preventing any future reinitialization. This cannot be part of an initializer call.\\n * Calling this in the constructor of a contract will prevent that contract from being initialized or reinitialized\\n * to any version. It is recommended to use this to lock implementation contracts that are designed to be called\\n * through proxies.\\n */\\n function _disableInitializers() internal virtual {\\n require(!_initializing, \\\"Initializable: contract is initializing\\\");\\n if (_initialized < type(uint8).max) {\\n _initialized = type(uint8).max;\\n emit Initialized(type(uint8).max);\\n }\\n }\\n}\\n\",\"keccak256\":\"0x0203dcadc5737d9ef2c211d6fa15d18ebc3b30dfa51903b64870b01a062b0b4e\",\"license\":\"MIT\"},\"@openzeppelin/contracts-upgradeable/token/ERC20/IERC20Upgradeable.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n// OpenZeppelin Contracts (last updated v4.6.0) (token/ERC20/IERC20.sol)\\n\\npragma solidity ^0.8.0;\\n\\n/**\\n * @dev Interface of the ERC20 standard as defined in the EIP.\\n */\\ninterface IERC20Upgradeable {\\n /**\\n * @dev Emitted when `value` tokens are moved from one account (`from`) to\\n * another (`to`).\\n *\\n * Note that `value` may be zero.\\n */\\n event Transfer(address indexed from, address indexed to, uint256 value);\\n\\n /**\\n * @dev Emitted when the allowance of a `spender` for an `owner` is set by\\n * a call to {approve}. `value` is the new allowance.\\n */\\n event Approval(address indexed owner, address indexed spender, uint256 value);\\n\\n /**\\n * @dev Returns the amount of tokens in existence.\\n */\\n function totalSupply() external view returns (uint256);\\n\\n /**\\n * @dev Returns the amount of tokens owned by `account`.\\n */\\n function balanceOf(address account) external view returns (uint256);\\n\\n /**\\n * @dev Moves `amount` tokens from the caller's account to `to`.\\n *\\n * Returns a boolean value indicating whether the operation succeeded.\\n *\\n * Emits a {Transfer} event.\\n */\\n function transfer(address to, uint256 amount) external returns (bool);\\n\\n /**\\n * @dev Returns the remaining number of tokens that `spender` will be\\n * allowed to spend on behalf of `owner` through {transferFrom}. This is\\n * zero by default.\\n *\\n * This value changes when {approve} or {transferFrom} are called.\\n */\\n function allowance(address owner, address spender) external view returns (uint256);\\n\\n /**\\n * @dev Sets `amount` as the allowance of `spender` over the caller's tokens.\\n *\\n * Returns a boolean value indicating whether the operation succeeded.\\n *\\n * IMPORTANT: Beware that changing an allowance with this method brings the risk\\n * that someone may use both the old and the new allowance by unfortunate\\n * transaction ordering. One possible solution to mitigate this race\\n * condition is to first reduce the spender's allowance to 0 and set the\\n * desired value afterwards:\\n * https://github.com/ethereum/EIPs/issues/20#issuecomment-263524729\\n *\\n * Emits an {Approval} event.\\n */\\n function approve(address spender, uint256 amount) external returns (bool);\\n\\n /**\\n * @dev Moves `amount` tokens from `from` to `to` using the\\n * allowance mechanism. `amount` is then deducted from the caller's\\n * allowance.\\n *\\n * Returns a boolean value indicating whether the operation succeeded.\\n *\\n * Emits a {Transfer} event.\\n */\\n function transferFrom(\\n address from,\\n address to,\\n uint256 amount\\n ) external returns (bool);\\n}\\n\",\"keccak256\":\"0x4e733d3164f73f461eaf9d8087a7ad1ea180bdc8ba0d3d61b0e1ae16d8e63dff\",\"license\":\"MIT\"},\"@openzeppelin/contracts-upgradeable/token/ERC721/ERC721Upgradeable.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n// OpenZeppelin Contracts (last updated v4.7.0) (token/ERC721/ERC721.sol)\\n\\npragma solidity ^0.8.0;\\n\\nimport \\\"./IERC721Upgradeable.sol\\\";\\nimport \\\"./IERC721ReceiverUpgradeable.sol\\\";\\nimport \\\"./extensions/IERC721MetadataUpgradeable.sol\\\";\\nimport \\\"../../utils/AddressUpgradeable.sol\\\";\\nimport \\\"../../utils/ContextUpgradeable.sol\\\";\\nimport \\\"../../utils/StringsUpgradeable.sol\\\";\\nimport \\\"../../utils/introspection/ERC165Upgradeable.sol\\\";\\nimport \\\"../../proxy/utils/Initializable.sol\\\";\\n\\n/**\\n * @dev Implementation of https://eips.ethereum.org/EIPS/eip-721[ERC721] Non-Fungible Token Standard, including\\n * the Metadata extension, but not including the Enumerable extension, which is available separately as\\n * {ERC721Enumerable}.\\n */\\ncontract ERC721Upgradeable is Initializable, ContextUpgradeable, ERC165Upgradeable, IERC721Upgradeable, IERC721MetadataUpgradeable {\\n using AddressUpgradeable for address;\\n using StringsUpgradeable for uint256;\\n\\n // Token name\\n string private _name;\\n\\n // Token symbol\\n string private _symbol;\\n\\n // Mapping from token ID to owner address\\n mapping(uint256 => address) private _owners;\\n\\n // Mapping owner address to token count\\n mapping(address => uint256) private _balances;\\n\\n // Mapping from token ID to approved address\\n mapping(uint256 => address) private _tokenApprovals;\\n\\n // Mapping from owner to operator approvals\\n mapping(address => mapping(address => bool)) private _operatorApprovals;\\n\\n /**\\n * @dev Initializes the contract by setting a `name` and a `symbol` to the token collection.\\n */\\n function __ERC721_init(string memory name_, string memory symbol_) internal onlyInitializing {\\n __ERC721_init_unchained(name_, symbol_);\\n }\\n\\n function __ERC721_init_unchained(string memory name_, string memory symbol_) internal onlyInitializing {\\n _name = name_;\\n _symbol = symbol_;\\n }\\n\\n /**\\n * @dev See {IERC165-supportsInterface}.\\n */\\n function supportsInterface(bytes4 interfaceId) public view virtual override(ERC165Upgradeable, IERC165Upgradeable) returns (bool) {\\n return\\n interfaceId == type(IERC721Upgradeable).interfaceId ||\\n interfaceId == type(IERC721MetadataUpgradeable).interfaceId ||\\n super.supportsInterface(interfaceId);\\n }\\n\\n /**\\n * @dev See {IERC721-balanceOf}.\\n */\\n function balanceOf(address owner) public view virtual override returns (uint256) {\\n require(owner != address(0), \\\"ERC721: address zero is not a valid owner\\\");\\n return _balances[owner];\\n }\\n\\n /**\\n * @dev See {IERC721-ownerOf}.\\n */\\n function ownerOf(uint256 tokenId) public view virtual override returns (address) {\\n address owner = _owners[tokenId];\\n require(owner != address(0), \\\"ERC721: invalid token ID\\\");\\n return owner;\\n }\\n\\n /**\\n * @dev See {IERC721Metadata-name}.\\n */\\n function name() public view virtual override returns (string memory) {\\n return _name;\\n }\\n\\n /**\\n * @dev See {IERC721Metadata-symbol}.\\n */\\n function symbol() public view virtual override returns (string memory) {\\n return _symbol;\\n }\\n\\n /**\\n * @dev See {IERC721Metadata-tokenURI}.\\n */\\n function tokenURI(uint256 tokenId) public view virtual override returns (string memory) {\\n _requireMinted(tokenId);\\n\\n string memory baseURI = _baseURI();\\n return bytes(baseURI).length > 0 ? string(abi.encodePacked(baseURI, tokenId.toString())) : \\\"\\\";\\n }\\n\\n /**\\n * @dev Base URI for computing {tokenURI}. If set, the resulting URI for each\\n * token will be the concatenation of the `baseURI` and the `tokenId`. Empty\\n * by default, can be overridden in child contracts.\\n */\\n function _baseURI() internal view virtual returns (string memory) {\\n return \\\"\\\";\\n }\\n\\n /**\\n * @dev See {IERC721-approve}.\\n */\\n function approve(address to, uint256 tokenId) public virtual override {\\n address owner = ERC721Upgradeable.ownerOf(tokenId);\\n require(to != owner, \\\"ERC721: approval to current owner\\\");\\n\\n require(\\n _msgSender() == owner || isApprovedForAll(owner, _msgSender()),\\n \\\"ERC721: approve caller is not token owner nor approved for all\\\"\\n );\\n\\n _approve(to, tokenId);\\n }\\n\\n /**\\n * @dev See {IERC721-getApproved}.\\n */\\n function getApproved(uint256 tokenId) public view virtual override returns (address) {\\n _requireMinted(tokenId);\\n\\n return _tokenApprovals[tokenId];\\n }\\n\\n /**\\n * @dev See {IERC721-setApprovalForAll}.\\n */\\n function setApprovalForAll(address operator, bool approved) public virtual override {\\n _setApprovalForAll(_msgSender(), operator, approved);\\n }\\n\\n /**\\n * @dev See {IERC721-isApprovedForAll}.\\n */\\n function isApprovedForAll(address owner, address operator) public view virtual override returns (bool) {\\n return _operatorApprovals[owner][operator];\\n }\\n\\n /**\\n * @dev See {IERC721-transferFrom}.\\n */\\n function transferFrom(\\n address from,\\n address to,\\n uint256 tokenId\\n ) public virtual override {\\n //solhint-disable-next-line max-line-length\\n require(_isApprovedOrOwner(_msgSender(), tokenId), \\\"ERC721: caller is not token owner nor approved\\\");\\n\\n _transfer(from, to, tokenId);\\n }\\n\\n /**\\n * @dev See {IERC721-safeTransferFrom}.\\n */\\n function safeTransferFrom(\\n address from,\\n address to,\\n uint256 tokenId\\n ) public virtual override {\\n safeTransferFrom(from, to, tokenId, \\\"\\\");\\n }\\n\\n /**\\n * @dev See {IERC721-safeTransferFrom}.\\n */\\n function safeTransferFrom(\\n address from,\\n address to,\\n uint256 tokenId,\\n bytes memory data\\n ) public virtual override {\\n require(_isApprovedOrOwner(_msgSender(), tokenId), \\\"ERC721: caller is not token owner nor approved\\\");\\n _safeTransfer(from, to, tokenId, data);\\n }\\n\\n /**\\n * @dev Safely transfers `tokenId` token from `from` to `to`, checking first that contract recipients\\n * are aware of the ERC721 protocol to prevent tokens from being forever locked.\\n *\\n * `data` is additional data, it has no specified format and it is sent in call to `to`.\\n *\\n * This internal function is equivalent to {safeTransferFrom}, and can be used to e.g.\\n * implement alternative mechanisms to perform token transfer, such as signature-based.\\n *\\n * Requirements:\\n *\\n * - `from` cannot be the zero address.\\n * - `to` cannot be the zero address.\\n * - `tokenId` token must exist and be owned by `from`.\\n * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer.\\n *\\n * Emits a {Transfer} event.\\n */\\n function _safeTransfer(\\n address from,\\n address to,\\n uint256 tokenId,\\n bytes memory data\\n ) internal virtual {\\n _transfer(from, to, tokenId);\\n require(_checkOnERC721Received(from, to, tokenId, data), \\\"ERC721: transfer to non ERC721Receiver implementer\\\");\\n }\\n\\n /**\\n * @dev Returns whether `tokenId` exists.\\n *\\n * Tokens can be managed by their owner or approved accounts via {approve} or {setApprovalForAll}.\\n *\\n * Tokens start existing when they are minted (`_mint`),\\n * and stop existing when they are burned (`_burn`).\\n */\\n function _exists(uint256 tokenId) internal view virtual returns (bool) {\\n return _owners[tokenId] != address(0);\\n }\\n\\n /**\\n * @dev Returns whether `spender` is allowed to manage `tokenId`.\\n *\\n * Requirements:\\n *\\n * - `tokenId` must exist.\\n */\\n function _isApprovedOrOwner(address spender, uint256 tokenId) internal view virtual returns (bool) {\\n address owner = ERC721Upgradeable.ownerOf(tokenId);\\n return (spender == owner || isApprovedForAll(owner, spender) || getApproved(tokenId) == spender);\\n }\\n\\n /**\\n * @dev Safely mints `tokenId` and transfers it to `to`.\\n *\\n * Requirements:\\n *\\n * - `tokenId` must not exist.\\n * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer.\\n *\\n * Emits a {Transfer} event.\\n */\\n function _safeMint(address to, uint256 tokenId) internal virtual {\\n _safeMint(to, tokenId, \\\"\\\");\\n }\\n\\n /**\\n * @dev Same as {xref-ERC721-_safeMint-address-uint256-}[`_safeMint`], with an additional `data` parameter which is\\n * forwarded in {IERC721Receiver-onERC721Received} to contract recipients.\\n */\\n function _safeMint(\\n address to,\\n uint256 tokenId,\\n bytes memory data\\n ) internal virtual {\\n _mint(to, tokenId);\\n require(\\n _checkOnERC721Received(address(0), to, tokenId, data),\\n \\\"ERC721: transfer to non ERC721Receiver implementer\\\"\\n );\\n }\\n\\n /**\\n * @dev Mints `tokenId` and transfers it to `to`.\\n *\\n * WARNING: Usage of this method is discouraged, use {_safeMint} whenever possible\\n *\\n * Requirements:\\n *\\n * - `tokenId` must not exist.\\n * - `to` cannot be the zero address.\\n *\\n * Emits a {Transfer} event.\\n */\\n function _mint(address to, uint256 tokenId) internal virtual {\\n require(to != address(0), \\\"ERC721: mint to the zero address\\\");\\n require(!_exists(tokenId), \\\"ERC721: token already minted\\\");\\n\\n _beforeTokenTransfer(address(0), to, tokenId);\\n\\n _balances[to] += 1;\\n _owners[tokenId] = to;\\n\\n emit Transfer(address(0), to, tokenId);\\n\\n _afterTokenTransfer(address(0), to, tokenId);\\n }\\n\\n /**\\n * @dev Destroys `tokenId`.\\n * The approval is cleared when the token is burned.\\n *\\n * Requirements:\\n *\\n * - `tokenId` must exist.\\n *\\n * Emits a {Transfer} event.\\n */\\n function _burn(uint256 tokenId) internal virtual {\\n address owner = ERC721Upgradeable.ownerOf(tokenId);\\n\\n _beforeTokenTransfer(owner, address(0), tokenId);\\n\\n // Clear approvals\\n _approve(address(0), tokenId);\\n\\n _balances[owner] -= 1;\\n delete _owners[tokenId];\\n\\n emit Transfer(owner, address(0), tokenId);\\n\\n _afterTokenTransfer(owner, address(0), tokenId);\\n }\\n\\n /**\\n * @dev Transfers `tokenId` from `from` to `to`.\\n * As opposed to {transferFrom}, this imposes no restrictions on msg.sender.\\n *\\n * Requirements:\\n *\\n * - `to` cannot be the zero address.\\n * - `tokenId` token must be owned by `from`.\\n *\\n * Emits a {Transfer} event.\\n */\\n function _transfer(\\n address from,\\n address to,\\n uint256 tokenId\\n ) internal virtual {\\n require(ERC721Upgradeable.ownerOf(tokenId) == from, \\\"ERC721: transfer from incorrect owner\\\");\\n require(to != address(0), \\\"ERC721: transfer to the zero address\\\");\\n\\n _beforeTokenTransfer(from, to, tokenId);\\n\\n // Clear approvals from the previous owner\\n _approve(address(0), tokenId);\\n\\n _balances[from] -= 1;\\n _balances[to] += 1;\\n _owners[tokenId] = to;\\n\\n emit Transfer(from, to, tokenId);\\n\\n _afterTokenTransfer(from, to, tokenId);\\n }\\n\\n /**\\n * @dev Approve `to` to operate on `tokenId`\\n *\\n * Emits an {Approval} event.\\n */\\n function _approve(address to, uint256 tokenId) internal virtual {\\n _tokenApprovals[tokenId] = to;\\n emit Approval(ERC721Upgradeable.ownerOf(tokenId), to, tokenId);\\n }\\n\\n /**\\n * @dev Approve `operator` to operate on all of `owner` tokens\\n *\\n * Emits an {ApprovalForAll} event.\\n */\\n function _setApprovalForAll(\\n address owner,\\n address operator,\\n bool approved\\n ) internal virtual {\\n require(owner != operator, \\\"ERC721: approve to caller\\\");\\n _operatorApprovals[owner][operator] = approved;\\n emit ApprovalForAll(owner, operator, approved);\\n }\\n\\n /**\\n * @dev Reverts if the `tokenId` has not been minted yet.\\n */\\n function _requireMinted(uint256 tokenId) internal view virtual {\\n require(_exists(tokenId), \\\"ERC721: invalid token ID\\\");\\n }\\n\\n /**\\n * @dev Internal function to invoke {IERC721Receiver-onERC721Received} on a target address.\\n * The call is not executed if the target address is not a contract.\\n *\\n * @param from address representing the previous owner of the given token ID\\n * @param to target address that will receive the tokens\\n * @param tokenId uint256 ID of the token to be transferred\\n * @param data bytes optional data to send along with the call\\n * @return bool whether the call correctly returned the expected magic value\\n */\\n function _checkOnERC721Received(\\n address from,\\n address to,\\n uint256 tokenId,\\n bytes memory data\\n ) private returns (bool) {\\n if (to.isContract()) {\\n try IERC721ReceiverUpgradeable(to).onERC721Received(_msgSender(), from, tokenId, data) returns (bytes4 retval) {\\n return retval == IERC721ReceiverUpgradeable.onERC721Received.selector;\\n } catch (bytes memory reason) {\\n if (reason.length == 0) {\\n revert(\\\"ERC721: transfer to non ERC721Receiver implementer\\\");\\n } else {\\n /// @solidity memory-safe-assembly\\n assembly {\\n revert(add(32, reason), mload(reason))\\n }\\n }\\n }\\n } else {\\n return true;\\n }\\n }\\n\\n /**\\n * @dev Hook that is called before any token transfer. This includes minting\\n * and burning.\\n *\\n * Calling conditions:\\n *\\n * - When `from` and `to` are both non-zero, ``from``'s `tokenId` will be\\n * transferred to `to`.\\n * - When `from` is zero, `tokenId` will be minted for `to`.\\n * - When `to` is zero, ``from``'s `tokenId` will be burned.\\n * - `from` and `to` are never both zero.\\n *\\n * To learn more about hooks, head to xref:ROOT:extending-contracts.adoc#using-hooks[Using Hooks].\\n */\\n function _beforeTokenTransfer(\\n address from,\\n address to,\\n uint256 tokenId\\n ) internal virtual {}\\n\\n /**\\n * @dev Hook that is called after any transfer of tokens. This includes\\n * minting and burning.\\n *\\n * Calling conditions:\\n *\\n * - when `from` and `to` are both non-zero.\\n * - `from` and `to` are never both zero.\\n *\\n * To learn more about hooks, head to xref:ROOT:extending-contracts.adoc#using-hooks[Using Hooks].\\n */\\n function _afterTokenTransfer(\\n address from,\\n address to,\\n uint256 tokenId\\n ) internal virtual {}\\n\\n /**\\n * @dev This empty reserved space is put in place to allow future versions to add new\\n * variables without shifting down storage in the inheritance chain.\\n * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps\\n */\\n uint256[44] private __gap;\\n}\\n\",\"keccak256\":\"0x5331c8909221d9f9f3851cfadd5959d0873413a2c27e30e0f2fa234158c1c6cf\",\"license\":\"MIT\"},\"@openzeppelin/contracts-upgradeable/token/ERC721/IERC721ReceiverUpgradeable.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n// OpenZeppelin Contracts (last updated v4.6.0) (token/ERC721/IERC721Receiver.sol)\\n\\npragma solidity ^0.8.0;\\n\\n/**\\n * @title ERC721 token receiver interface\\n * @dev Interface for any contract that wants to support safeTransfers\\n * from ERC721 asset contracts.\\n */\\ninterface IERC721ReceiverUpgradeable {\\n /**\\n * @dev Whenever an {IERC721} `tokenId` token is transferred to this contract via {IERC721-safeTransferFrom}\\n * by `operator` from `from`, this function is called.\\n *\\n * It must return its Solidity selector to confirm the token transfer.\\n * If any other value is returned or the interface is not implemented by the recipient, the transfer will be reverted.\\n *\\n * The selector can be obtained in Solidity with `IERC721Receiver.onERC721Received.selector`.\\n */\\n function onERC721Received(\\n address operator,\\n address from,\\n uint256 tokenId,\\n bytes calldata data\\n ) external returns (bytes4);\\n}\\n\",\"keccak256\":\"0xbb2ed8106d94aeae6858e2551a1e7174df73994b77b13ebd120ccaaef80155f5\",\"license\":\"MIT\"},\"@openzeppelin/contracts-upgradeable/token/ERC721/IERC721Upgradeable.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n// OpenZeppelin Contracts (last updated v4.7.0) (token/ERC721/IERC721.sol)\\n\\npragma solidity ^0.8.0;\\n\\nimport \\\"../../utils/introspection/IERC165Upgradeable.sol\\\";\\n\\n/**\\n * @dev Required interface of an ERC721 compliant contract.\\n */\\ninterface IERC721Upgradeable is IERC165Upgradeable {\\n /**\\n * @dev Emitted when `tokenId` token is transferred from `from` to `to`.\\n */\\n event Transfer(address indexed from, address indexed to, uint256 indexed tokenId);\\n\\n /**\\n * @dev Emitted when `owner` enables `approved` to manage the `tokenId` token.\\n */\\n event Approval(address indexed owner, address indexed approved, uint256 indexed tokenId);\\n\\n /**\\n * @dev Emitted when `owner` enables or disables (`approved`) `operator` to manage all of its assets.\\n */\\n event ApprovalForAll(address indexed owner, address indexed operator, bool approved);\\n\\n /**\\n * @dev Returns the number of tokens in ``owner``'s account.\\n */\\n function balanceOf(address owner) external view returns (uint256 balance);\\n\\n /**\\n * @dev Returns the owner of the `tokenId` token.\\n *\\n * Requirements:\\n *\\n * - `tokenId` must exist.\\n */\\n function ownerOf(uint256 tokenId) external view returns (address owner);\\n\\n /**\\n * @dev Safely transfers `tokenId` token from `from` to `to`.\\n *\\n * Requirements:\\n *\\n * - `from` cannot be the zero address.\\n * - `to` cannot be the zero address.\\n * - `tokenId` token must exist and be owned by `from`.\\n * - If the caller is not `from`, it must be approved to move this token by either {approve} or {setApprovalForAll}.\\n * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer.\\n *\\n * Emits a {Transfer} event.\\n */\\n function safeTransferFrom(\\n address from,\\n address to,\\n uint256 tokenId,\\n bytes calldata data\\n ) external;\\n\\n /**\\n * @dev Safely transfers `tokenId` token from `from` to `to`, checking first that contract recipients\\n * are aware of the ERC721 protocol to prevent tokens from being forever locked.\\n *\\n * Requirements:\\n *\\n * - `from` cannot be the zero address.\\n * - `to` cannot be the zero address.\\n * - `tokenId` token must exist and be owned by `from`.\\n * - If the caller is not `from`, it must have been allowed to move this token by either {approve} or {setApprovalForAll}.\\n * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer.\\n *\\n * Emits a {Transfer} event.\\n */\\n function safeTransferFrom(\\n address from,\\n address to,\\n uint256 tokenId\\n ) external;\\n\\n /**\\n * @dev Transfers `tokenId` token from `from` to `to`.\\n *\\n * WARNING: Usage of this method is discouraged, use {safeTransferFrom} whenever possible.\\n *\\n * Requirements:\\n *\\n * - `from` cannot be the zero address.\\n * - `to` cannot be the zero address.\\n * - `tokenId` token must be owned by `from`.\\n * - If the caller is not `from`, it must be approved to move this token by either {approve} or {setApprovalForAll}.\\n *\\n * Emits a {Transfer} event.\\n */\\n function transferFrom(\\n address from,\\n address to,\\n uint256 tokenId\\n ) external;\\n\\n /**\\n * @dev Gives permission to `to` to transfer `tokenId` token to another account.\\n * The approval is cleared when the token is transferred.\\n *\\n * Only a single account can be approved at a time, so approving the zero address clears previous approvals.\\n *\\n * Requirements:\\n *\\n * - The caller must own the token or be an approved operator.\\n * - `tokenId` must exist.\\n *\\n * Emits an {Approval} event.\\n */\\n function approve(address to, uint256 tokenId) external;\\n\\n /**\\n * @dev Approve or remove `operator` as an operator for the caller.\\n * Operators can call {transferFrom} or {safeTransferFrom} for any token owned by the caller.\\n *\\n * Requirements:\\n *\\n * - The `operator` cannot be the caller.\\n *\\n * Emits an {ApprovalForAll} event.\\n */\\n function setApprovalForAll(address operator, bool _approved) external;\\n\\n /**\\n * @dev Returns the account approved for `tokenId` token.\\n *\\n * Requirements:\\n *\\n * - `tokenId` must exist.\\n */\\n function getApproved(uint256 tokenId) external view returns (address operator);\\n\\n /**\\n * @dev Returns if the `operator` is allowed to manage all of the assets of `owner`.\\n *\\n * See {setApprovalForAll}\\n */\\n function isApprovedForAll(address owner, address operator) external view returns (bool);\\n}\\n\",\"keccak256\":\"0x016298e66a5810253c6c905e61966bb31c8775c3f3517bf946ff56ee31d6c005\",\"license\":\"MIT\"},\"@openzeppelin/contracts-upgradeable/token/ERC721/extensions/IERC721MetadataUpgradeable.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n// OpenZeppelin Contracts v4.4.1 (token/ERC721/extensions/IERC721Metadata.sol)\\n\\npragma solidity ^0.8.0;\\n\\nimport \\\"../IERC721Upgradeable.sol\\\";\\n\\n/**\\n * @title ERC-721 Non-Fungible Token Standard, optional metadata extension\\n * @dev See https://eips.ethereum.org/EIPS/eip-721\\n */\\ninterface IERC721MetadataUpgradeable is IERC721Upgradeable {\\n /**\\n * @dev Returns the token collection name.\\n */\\n function name() external view returns (string memory);\\n\\n /**\\n * @dev Returns the token collection symbol.\\n */\\n function symbol() external view returns (string memory);\\n\\n /**\\n * @dev Returns the Uniform Resource Identifier (URI) for `tokenId` token.\\n */\\n function tokenURI(uint256 tokenId) external view returns (string memory);\\n}\\n\",\"keccak256\":\"0x95a471796eb5f030fdc438660bebec121ad5d063763e64d92376ffb4b5ce8b70\",\"license\":\"MIT\"},\"@openzeppelin/contracts-upgradeable/utils/AddressUpgradeable.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n// OpenZeppelin Contracts (last updated v4.7.0) (utils/Address.sol)\\n\\npragma solidity ^0.8.1;\\n\\n/**\\n * @dev Collection of functions related to the address type\\n */\\nlibrary AddressUpgradeable {\\n /**\\n * @dev Returns true if `account` is a contract.\\n *\\n * [IMPORTANT]\\n * ====\\n * It is unsafe to assume that an address for which this function returns\\n * false is an externally-owned account (EOA) and not a contract.\\n *\\n * Among others, `isContract` will return false for the following\\n * types of addresses:\\n *\\n * - an externally-owned account\\n * - a contract in construction\\n * - an address where a contract will be created\\n * - an address where a contract lived, but was destroyed\\n * ====\\n *\\n * [IMPORTANT]\\n * ====\\n * You shouldn't rely on `isContract` to protect against flash loan attacks!\\n *\\n * Preventing calls from contracts is highly discouraged. It breaks composability, breaks support for smart wallets\\n * like Gnosis Safe, and does not provide security since it can be circumvented by calling from a contract\\n * constructor.\\n * ====\\n */\\n function isContract(address account) internal view returns (bool) {\\n // This method relies on extcodesize/address.code.length, which returns 0\\n // for contracts in construction, since the code is only stored at the end\\n // of the constructor execution.\\n\\n return account.code.length > 0;\\n }\\n\\n /**\\n * @dev Replacement for Solidity's `transfer`: sends `amount` wei to\\n * `recipient`, forwarding all available gas and reverting on errors.\\n *\\n * https://eips.ethereum.org/EIPS/eip-1884[EIP1884] increases the gas cost\\n * of certain opcodes, possibly making contracts go over the 2300 gas limit\\n * imposed by `transfer`, making them unable to receive funds via\\n * `transfer`. {sendValue} removes this limitation.\\n *\\n * https://diligence.consensys.net/posts/2019/09/stop-using-soliditys-transfer-now/[Learn more].\\n *\\n * IMPORTANT: because control is transferred to `recipient`, care must be\\n * taken to not create reentrancy vulnerabilities. Consider using\\n * {ReentrancyGuard} or the\\n * https://solidity.readthedocs.io/en/v0.5.11/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern].\\n */\\n function sendValue(address payable recipient, uint256 amount) internal {\\n require(address(this).balance >= amount, \\\"Address: insufficient balance\\\");\\n\\n (bool success, ) = recipient.call{value: amount}(\\\"\\\");\\n require(success, \\\"Address: unable to send value, recipient may have reverted\\\");\\n }\\n\\n /**\\n * @dev Performs a Solidity function call using a low level `call`. A\\n * plain `call` is an unsafe replacement for a function call: use this\\n * function instead.\\n *\\n * If `target` reverts with a revert reason, it is bubbled up by this\\n * function (like regular Solidity function calls).\\n *\\n * Returns the raw returned data. To convert to the expected return value,\\n * use https://solidity.readthedocs.io/en/latest/units-and-global-variables.html?highlight=abi.decode#abi-encoding-and-decoding-functions[`abi.decode`].\\n *\\n * Requirements:\\n *\\n * - `target` must be a contract.\\n * - calling `target` with `data` must not revert.\\n *\\n * _Available since v3.1._\\n */\\n function functionCall(address target, bytes memory data) internal returns (bytes memory) {\\n return functionCall(target, data, \\\"Address: low-level call failed\\\");\\n }\\n\\n /**\\n * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], but with\\n * `errorMessage` as a fallback revert reason when `target` reverts.\\n *\\n * _Available since v3.1._\\n */\\n function functionCall(\\n address target,\\n bytes memory data,\\n string memory errorMessage\\n ) internal returns (bytes memory) {\\n return functionCallWithValue(target, data, 0, errorMessage);\\n }\\n\\n /**\\n * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],\\n * but also transferring `value` wei to `target`.\\n *\\n * Requirements:\\n *\\n * - the calling contract must have an ETH balance of at least `value`.\\n * - the called Solidity function must be `payable`.\\n *\\n * _Available since v3.1._\\n */\\n function functionCallWithValue(\\n address target,\\n bytes memory data,\\n uint256 value\\n ) internal returns (bytes memory) {\\n return functionCallWithValue(target, data, value, \\\"Address: low-level call with value failed\\\");\\n }\\n\\n /**\\n * @dev Same as {xref-Address-functionCallWithValue-address-bytes-uint256-}[`functionCallWithValue`], but\\n * with `errorMessage` as a fallback revert reason when `target` reverts.\\n *\\n * _Available since v3.1._\\n */\\n function functionCallWithValue(\\n address target,\\n bytes memory data,\\n uint256 value,\\n string memory errorMessage\\n ) internal returns (bytes memory) {\\n require(address(this).balance >= value, \\\"Address: insufficient balance for call\\\");\\n require(isContract(target), \\\"Address: call to non-contract\\\");\\n\\n (bool success, bytes memory returndata) = target.call{value: value}(data);\\n return verifyCallResult(success, returndata, errorMessage);\\n }\\n\\n /**\\n * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],\\n * but performing a static call.\\n *\\n * _Available since v3.3._\\n */\\n function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) {\\n return functionStaticCall(target, data, \\\"Address: low-level static call failed\\\");\\n }\\n\\n /**\\n * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],\\n * but performing a static call.\\n *\\n * _Available since v3.3._\\n */\\n function functionStaticCall(\\n address target,\\n bytes memory data,\\n string memory errorMessage\\n ) internal view returns (bytes memory) {\\n require(isContract(target), \\\"Address: static call to non-contract\\\");\\n\\n (bool success, bytes memory returndata) = target.staticcall(data);\\n return verifyCallResult(success, returndata, errorMessage);\\n }\\n\\n /**\\n * @dev Tool to verifies that a low level call was successful, and revert if it wasn't, either by bubbling the\\n * revert reason using the provided one.\\n *\\n * _Available since v4.3._\\n */\\n function verifyCallResult(\\n bool success,\\n bytes memory returndata,\\n string memory errorMessage\\n ) internal pure returns (bytes memory) {\\n if (success) {\\n return returndata;\\n } else {\\n // Look for revert reason and bubble it up if present\\n if (returndata.length > 0) {\\n // The easiest way to bubble the revert reason is using memory via assembly\\n /// @solidity memory-safe-assembly\\n assembly {\\n let returndata_size := mload(returndata)\\n revert(add(32, returndata), returndata_size)\\n }\\n } else {\\n revert(errorMessage);\\n }\\n }\\n }\\n}\\n\",\"keccak256\":\"0x611aa3f23e59cfdd1863c536776407b3e33d695152a266fa7cfb34440a29a8a3\",\"license\":\"MIT\"},\"@openzeppelin/contracts-upgradeable/utils/ContextUpgradeable.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n// OpenZeppelin Contracts v4.4.1 (utils/Context.sol)\\n\\npragma solidity ^0.8.0;\\nimport \\\"../proxy/utils/Initializable.sol\\\";\\n\\n/**\\n * @dev Provides information about the current execution context, including the\\n * sender of the transaction and its data. While these are generally available\\n * via msg.sender and msg.data, they should not be accessed in such a direct\\n * manner, since when dealing with meta-transactions the account sending and\\n * paying for execution may not be the actual sender (as far as an application\\n * is concerned).\\n *\\n * This contract is only required for intermediate, library-like contracts.\\n */\\nabstract contract ContextUpgradeable is Initializable {\\n function __Context_init() internal onlyInitializing {\\n }\\n\\n function __Context_init_unchained() internal onlyInitializing {\\n }\\n function _msgSender() internal view virtual returns (address) {\\n return msg.sender;\\n }\\n\\n function _msgData() internal view virtual returns (bytes calldata) {\\n return msg.data;\\n }\\n\\n /**\\n * @dev This empty reserved space is put in place to allow future versions to add new\\n * variables without shifting down storage in the inheritance chain.\\n * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps\\n */\\n uint256[50] private __gap;\\n}\\n\",\"keccak256\":\"0x963ea7f0b48b032eef72fe3a7582edf78408d6f834115b9feadd673a4d5bd149\",\"license\":\"MIT\"},\"@openzeppelin/contracts-upgradeable/utils/StringsUpgradeable.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n// OpenZeppelin Contracts (last updated v4.7.0) (utils/Strings.sol)\\n\\npragma solidity ^0.8.0;\\n\\n/**\\n * @dev String operations.\\n */\\nlibrary StringsUpgradeable {\\n bytes16 private constant _HEX_SYMBOLS = \\\"0123456789abcdef\\\";\\n uint8 private constant _ADDRESS_LENGTH = 20;\\n\\n /**\\n * @dev Converts a `uint256` to its ASCII `string` decimal representation.\\n */\\n function toString(uint256 value) internal pure returns (string memory) {\\n // Inspired by OraclizeAPI's implementation - MIT licence\\n // https://github.com/oraclize/ethereum-api/blob/b42146b063c7d6ee1358846c198246239e9360e8/oraclizeAPI_0.4.25.sol\\n\\n if (value == 0) {\\n return \\\"0\\\";\\n }\\n uint256 temp = value;\\n uint256 digits;\\n while (temp != 0) {\\n digits++;\\n temp /= 10;\\n }\\n bytes memory buffer = new bytes(digits);\\n while (value != 0) {\\n digits -= 1;\\n buffer[digits] = bytes1(uint8(48 + uint256(value % 10)));\\n value /= 10;\\n }\\n return string(buffer);\\n }\\n\\n /**\\n * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation.\\n */\\n function toHexString(uint256 value) internal pure returns (string memory) {\\n if (value == 0) {\\n return \\\"0x00\\\";\\n }\\n uint256 temp = value;\\n uint256 length = 0;\\n while (temp != 0) {\\n length++;\\n temp >>= 8;\\n }\\n return toHexString(value, length);\\n }\\n\\n /**\\n * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation with fixed length.\\n */\\n function toHexString(uint256 value, uint256 length) internal pure returns (string memory) {\\n bytes memory buffer = new bytes(2 * length + 2);\\n buffer[0] = \\\"0\\\";\\n buffer[1] = \\\"x\\\";\\n for (uint256 i = 2 * length + 1; i > 1; --i) {\\n buffer[i] = _HEX_SYMBOLS[value & 0xf];\\n value >>= 4;\\n }\\n require(value == 0, \\\"Strings: hex length insufficient\\\");\\n return string(buffer);\\n }\\n\\n /**\\n * @dev Converts an `address` with fixed length of 20 bytes to its not checksummed ASCII `string` hexadecimal representation.\\n */\\n function toHexString(address addr) internal pure returns (string memory) {\\n return toHexString(uint256(uint160(addr)), _ADDRESS_LENGTH);\\n }\\n}\\n\",\"keccak256\":\"0xea5339a7fff0ed42b45be56a88efdd0b2ddde9fa480dc99fef9a6a4c5b776863\",\"license\":\"MIT\"},\"@openzeppelin/contracts-upgradeable/utils/introspection/ERC165Upgradeable.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n// OpenZeppelin Contracts v4.4.1 (utils/introspection/ERC165.sol)\\n\\npragma solidity ^0.8.0;\\n\\nimport \\\"./IERC165Upgradeable.sol\\\";\\nimport \\\"../../proxy/utils/Initializable.sol\\\";\\n\\n/**\\n * @dev Implementation of the {IERC165} interface.\\n *\\n * Contracts that want to implement ERC165 should inherit from this contract and override {supportsInterface} to check\\n * for the additional interface id that will be supported. For example:\\n *\\n * ```solidity\\n * function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) {\\n * return interfaceId == type(MyInterface).interfaceId || super.supportsInterface(interfaceId);\\n * }\\n * ```\\n *\\n * Alternatively, {ERC165Storage} provides an easier to use but more expensive implementation.\\n */\\nabstract contract ERC165Upgradeable is Initializable, IERC165Upgradeable {\\n function __ERC165_init() internal onlyInitializing {\\n }\\n\\n function __ERC165_init_unchained() internal onlyInitializing {\\n }\\n /**\\n * @dev See {IERC165-supportsInterface}.\\n */\\n function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) {\\n return interfaceId == type(IERC165Upgradeable).interfaceId;\\n }\\n\\n /**\\n * @dev This empty reserved space is put in place to allow future versions to add new\\n * variables without shifting down storage in the inheritance chain.\\n * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps\\n */\\n uint256[50] private __gap;\\n}\\n\",\"keccak256\":\"0x9a3b990bd56d139df3e454a9edf1c64668530b5a77fc32eb063bc206f958274a\",\"license\":\"MIT\"},\"@openzeppelin/contracts-upgradeable/utils/introspection/IERC165Upgradeable.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n// OpenZeppelin Contracts v4.4.1 (utils/introspection/IERC165.sol)\\n\\npragma solidity ^0.8.0;\\n\\n/**\\n * @dev Interface of the ERC165 standard, as defined in the\\n * https://eips.ethereum.org/EIPS/eip-165[EIP].\\n *\\n * Implementers can declare support of contract interfaces, which can then be\\n * queried by others ({ERC165Checker}).\\n *\\n * For an implementation, see {ERC165}.\\n */\\ninterface IERC165Upgradeable {\\n /**\\n * @dev Returns true if this contract implements the interface defined by\\n * `interfaceId`. See the corresponding\\n * https://eips.ethereum.org/EIPS/eip-165#how-interfaces-are-identified[EIP section]\\n * to learn more about how these ids are created.\\n *\\n * This function call must use less than 30 000 gas.\\n */\\n function supportsInterface(bytes4 interfaceId) external view returns (bool);\\n}\\n\",\"keccak256\":\"0xc6cef87559d0aeffdf0a99803de655938a7779ec0a3cd5d4383483ad85565a09\",\"license\":\"MIT\"},\"base64-sol/base64.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n\\npragma solidity >=0.6.0;\\n\\n/// @title Base64\\n/// @author Brecht Devos - \\n/// @notice Provides functions for encoding/decoding base64\\nlibrary Base64 {\\n string internal constant TABLE_ENCODE = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/';\\n bytes internal constant TABLE_DECODE = hex\\\"0000000000000000000000000000000000000000000000000000000000000000\\\"\\n hex\\\"00000000000000000000003e0000003f3435363738393a3b3c3d000000000000\\\"\\n hex\\\"00000102030405060708090a0b0c0d0e0f101112131415161718190000000000\\\"\\n hex\\\"001a1b1c1d1e1f202122232425262728292a2b2c2d2e2f303132330000000000\\\";\\n\\n function encode(bytes memory data) internal pure returns (string memory) {\\n if (data.length == 0) return '';\\n\\n // load the table into memory\\n string memory table = TABLE_ENCODE;\\n\\n // multiply by 4/3 rounded up\\n uint256 encodedLen = 4 * ((data.length + 2) / 3);\\n\\n // add some extra buffer at the end required for the writing\\n string memory result = new string(encodedLen + 32);\\n\\n assembly {\\n // set the actual output length\\n mstore(result, encodedLen)\\n\\n // prepare the lookup table\\n let tablePtr := add(table, 1)\\n\\n // input ptr\\n let dataPtr := data\\n let endPtr := add(dataPtr, mload(data))\\n\\n // result ptr, jump over length\\n let resultPtr := add(result, 32)\\n\\n // run over the input, 3 bytes at a time\\n for {} lt(dataPtr, endPtr) {}\\n {\\n // read 3 bytes\\n dataPtr := add(dataPtr, 3)\\n let input := mload(dataPtr)\\n\\n // write 4 characters\\n mstore8(resultPtr, mload(add(tablePtr, and(shr(18, input), 0x3F))))\\n resultPtr := add(resultPtr, 1)\\n mstore8(resultPtr, mload(add(tablePtr, and(shr(12, input), 0x3F))))\\n resultPtr := add(resultPtr, 1)\\n mstore8(resultPtr, mload(add(tablePtr, and(shr( 6, input), 0x3F))))\\n resultPtr := add(resultPtr, 1)\\n mstore8(resultPtr, mload(add(tablePtr, and( input, 0x3F))))\\n resultPtr := add(resultPtr, 1)\\n }\\n\\n // padding with '='\\n switch mod(mload(data), 3)\\n case 1 { mstore(sub(resultPtr, 2), shl(240, 0x3d3d)) }\\n case 2 { mstore(sub(resultPtr, 1), shl(248, 0x3d)) }\\n }\\n\\n return result;\\n }\\n\\n function decode(string memory _data) internal pure returns (bytes memory) {\\n bytes memory data = bytes(_data);\\n\\n if (data.length == 0) return new bytes(0);\\n require(data.length % 4 == 0, \\\"invalid base64 decoder input\\\");\\n\\n // load the table into memory\\n bytes memory table = TABLE_DECODE;\\n\\n // every 4 characters represent 3 bytes\\n uint256 decodedLen = (data.length / 4) * 3;\\n\\n // add some extra buffer at the end required for the writing\\n bytes memory result = new bytes(decodedLen + 32);\\n\\n assembly {\\n // padding with '='\\n let lastBytes := mload(add(data, mload(data)))\\n if eq(and(lastBytes, 0xFF), 0x3d) {\\n decodedLen := sub(decodedLen, 1)\\n if eq(and(lastBytes, 0xFFFF), 0x3d3d) {\\n decodedLen := sub(decodedLen, 1)\\n }\\n }\\n\\n // set the actual output length\\n mstore(result, decodedLen)\\n\\n // prepare the lookup table\\n let tablePtr := add(table, 1)\\n\\n // input ptr\\n let dataPtr := data\\n let endPtr := add(dataPtr, mload(data))\\n\\n // result ptr, jump over length\\n let resultPtr := add(result, 32)\\n\\n // run over the input, 4 characters at a time\\n for {} lt(dataPtr, endPtr) {}\\n {\\n // read 4 characters\\n dataPtr := add(dataPtr, 4)\\n let input := mload(dataPtr)\\n\\n // write 3 bytes\\n let output := add(\\n add(\\n shl(18, and(mload(add(tablePtr, and(shr(24, input), 0xFF))), 0xFF)),\\n shl(12, and(mload(add(tablePtr, and(shr(16, input), 0xFF))), 0xFF))),\\n add(\\n shl( 6, and(mload(add(tablePtr, and(shr( 8, input), 0xFF))), 0xFF)),\\n and(mload(add(tablePtr, and( input , 0xFF))), 0xFF)\\n )\\n )\\n mstore(resultPtr, shl(232, output))\\n resultPtr := add(resultPtr, 3)\\n }\\n }\\n\\n return result;\\n }\\n}\\n\",\"keccak256\":\"0xa73959e6ef0b693e4423a562e612370160b934a75e618361ddd8c9c4b8ddbaaf\",\"license\":\"MIT\"},\"contracts/ExpandedNFT.sol\":{\"content\":\"// SPDX-License-Identifier: GPL-3.0\\n\\n/**\\n\\n ExpandedNFTs\\n\\n */\\n\\npragma solidity ^0.8.15;\\n\\nimport {ERC721Upgradeable} from \\\"@openzeppelin/contracts-upgradeable/token/ERC721/ERC721Upgradeable.sol\\\";\\nimport {IERC2981Upgradeable, IERC165Upgradeable} from \\\"@openzeppelin/contracts-upgradeable/interfaces/IERC2981Upgradeable.sol\\\";\\nimport {OwnableUpgradeable} from \\\"@openzeppelin/contracts-upgradeable/access/OwnableUpgradeable.sol\\\";\\nimport {AddressUpgradeable} from \\\"@openzeppelin/contracts-upgradeable/utils/AddressUpgradeable.sol\\\";\\nimport {IERC20Upgradeable} from \\\"@openzeppelin/contracts-upgradeable/token/ERC20/IERC20Upgradeable.sol\\\";\\n\\nimport {SharedNFTLogic} from \\\"./SharedNFTLogic.sol\\\";\\nimport {IExpandedNFT} from \\\"./IExpandedNFT.sol\\\";\\n/**\\n This is a smart contract for handling dynamic contract minting.\\n\\n @dev This allows creators to mint a unique serial drop of an expanded NFT within a custom contract\\n @author Zien\\n Repository: https://github.com/joinzien/expanded-nft\\n*/\\ncontract ExpandedNFT is\\n ERC721Upgradeable,\\n IExpandedNFT,\\n IERC2981Upgradeable,\\n OwnableUpgradeable\\n{\\n enum WhoCanMint{ ONLY_OWNER, VIPS, MEMBERS, ANYONE }\\n\\n enum ExpandedNFTStates{ UNMINTED, MINTED, REDEEM_STARTED, SET_OFFER_TERMS, ACCEPTED_OFFER, PRODUCTION_COMPLETE, REDEEMED }\\n \\n event PriceChanged(uint256 amount);\\n event EditionSold(uint256 price, address owner);\\n event WhoCanMintChanged(WhoCanMint minters);\\n\\n // State change events\\n event RedeemStarted(uint256 tokenId, address owner);\\n event RedeemAborted(uint256 tokenId, address owner); \\n event OfferTermsSet(uint256 tokenId);\\n event OfferAccepted(uint256 tokenId);\\n event OfferRejected(uint256 tokenId);\\n event ProductionComplete(uint256 tokenId);\\n event DeliveryAccepted(uint256 tokenId);\\n\\n struct PerToken { \\n // Hashmap of the Edition ID to the current \\n ExpandedNFTStates editionState;\\n\\n // Redemption price\\n uint256 editionFee; \\n\\n // Edition description\\n string description;\\n\\n // Minted\\n\\n // animation_url field in the metadata\\n string animationUrl;\\n // Hash for the associated animation\\n bytes32 animationHash;\\n // Image in the metadata\\n string imageUrl;\\n // Hash for the associated image\\n bytes32 imageHash;\\n\\n // Redeemed\\n\\n // animation_url field in the metadata\\n string redeemedAnimationUrl;\\n // Hash for the associated animation\\n bytes32 redeemedAnimationHash;\\n // Image in the metadata\\n string redeemedImageUrl;\\n // Hash for the associated image\\n bytes32 redeemedImageHash;\\n // Condition report in the metadata\\n string conditionReportUrl;\\n // Hash for the condition report\\n bytes32 conditionReportHash;\\n }\\n\\n struct Pricing { \\n // Royalty amount in bps\\n uint256 royaltyBPS;\\n\\n // Split amount to the platforms. the artist in bps\\n uint256 splitBPS;\\n\\n // Price for VIP sales\\n uint256 vipSalePrice;\\n\\n // Price for member sales\\n uint256 membersSalePrice; \\n\\n // Price for VIP sales\\n uint256 vipMintLimit;\\n\\n // Price for member sales\\n uint256 membersMintLimit;\\n\\n // Price for general sales\\n uint256 generalMintLimit; \\n\\n // Addresses allowed to mint edition\\n mapping(address => bool) allowedMinters;\\n // VIP Addresses allowed to mint edition\\n mapping(address => bool) vipAllowedMinters;\\n\\n // Who can currently mint\\n WhoCanMint whoCanMint;\\n\\n // Mint counts for each address\\n mapping(address => uint256) mintCounts; \\n }\\n\\n // metadata\\n string public description;\\n\\n // Artists wallet address\\n address private _artistWallet;\\n\\n // Per Token data\\n mapping(uint256 => PerToken) private _perTokenMetadata;\\n\\n // Total size of the drop that can be minted\\n uint256 public dropSize;\\n\\n uint256 private _loadedMetadata;\\n\\n // reservation list\\n uint256 private _reserveCount;\\n mapping(uint256 => address) private _reserveAddress;\\n mapping(uint256 => uint256) private _reserveTokenId;\\n\\n mapping(uint256 => bool) private _tokenClaimed; \\n uint256 private _claimCount; \\n uint256 private _currentIndex;\\n\\n Pricing private _pricing;\\n\\n // Price for general sales\\n uint256 public salePrice;\\n\\n // ERC20 interface for the payment token\\n IERC20Upgradeable private _paymentTokenERC20;\\n\\n // NFT rendering logic contract\\n SharedNFTLogic private immutable _sharedNFTLogic;\\n\\n // Global constructor for factory\\n constructor(SharedNFTLogic sharedNFTLogic) {\\n _sharedNFTLogic = sharedNFTLogic;\\n _pricing.whoCanMint = WhoCanMint.ONLY_OWNER;\\n }\\n\\n /**\\n @param _owner wallet addres for the user that owns and can mint the drop, gets royalty and sales payouts and can update the base url if needed.\\n @param artistWallet wallet address for thr User that created the drop\\n @param _name Name of drop, used in the title as \\\"$NAME NUMBER/TOTAL\\\"\\n @param _symbol Symbol of the new token contract\\n @param _dropSize Number of editions that can be minted in total. \\n @dev Function to create a new drop. Can only be called by the allowed creator\\n Sets the only allowed minter to the address that creates/owns the drop.\\n This can be re-assigned or updated later\\n */\\n function initialize(\\n address _owner,\\n address artistWallet,\\n string memory _name,\\n string memory _symbol,\\n uint256 _dropSize\\n ) public initializer {\\n require(_dropSize > 0, \\\"Drop size must be > 0\\\");\\n\\n __ERC721_init(_name, _symbol);\\n __Ownable_init();\\n\\n // Set ownership to original sender of contract call\\n transferOwnership(_owner);\\n\\n _artistWallet = artistWallet;\\n dropSize = _dropSize;\\n\\n // Set edition id start to be 1 not 0\\n _claimCount = 0; \\n _currentIndex = 1;\\n\\n // Set the metadata\\n description = _name;\\n _loadedMetadata = 0; \\n }\\n\\n /**\\n @param _description Description of the edition, used in the description field of the NFT\\n @param imageUrl Image URL of the the edition. Strongly encouraged to be used, if necessary, only animation URL can be used. One of animation and image url need to exist in a drop to render the NFT.\\n @param imageHash SHA256 of the given image in bytes32 format (0xHASH). If no image is included, the hash can be zero.\\n @param animationUrl Animation URL of the edition. Not required, but if omitted image URL needs to be included. This follows the opensea spec for NFTs\\n @param animationHash The associated hash of the animation in sha-256 bytes32 format. If animation is omitted the hash can be zero.\\n @dev Function to create a new drop. Can only be called by the allowed creator\\n Sets the only allowed minter to the address that creates/owns the drop.\\n This can be re-assigned or updated later\\n */\\n function loadMetadataChunk(\\n uint256 startOffset,\\n uint256 count,\\n string[] memory _description,\\n string[] memory imageUrl,\\n bytes32[] memory imageHash,\\n string[] memory animationUrl,\\n bytes32[] memory animationHash\\n ) public {\\n require(_description.length == count, \\\"Data size mismatch\\\");\\n require(animationUrl.length == count, \\\"Data size mismatch\\\");\\n require(animationHash.length == count, \\\"Data size mismatch\\\");\\n require(imageUrl.length == count, \\\"Data size mismatch\\\");\\n require(imageHash.length == count, \\\"Data size mismatch\\\");\\n\\n for (uint i = 0; i < count; i++) {\\n uint index = startOffset + i + 1;\\n \\n _perTokenMetadata[index].description = _description[i];\\n\\n _perTokenMetadata[index].animationUrl = animationUrl[i];\\n _perTokenMetadata[index].animationHash = animationHash[i];\\n _perTokenMetadata[index].imageUrl = imageUrl[i];\\n _perTokenMetadata[index].imageHash = imageHash[i];\\n }\\n\\n _loadedMetadata += count;\\n }\\n\\n function metadataloaded() public view returns (bool){\\n return (_loadedMetadata >= dropSize);\\n }\\n\\n /// @dev returns the number of minted tokens within the drop\\n function totalSupply() public view returns (uint256) {\\n return _claimCount;\\n }\\n\\n /// @dev returns the royalty BPS\\n function getRoyaltyBPS() public view returns (uint256) {\\n return _pricing.royaltyBPS;\\n }\\n\\n /// @dev returns the split BPS\\n function getSplitBPS() public view returns (uint256) {\\n return _pricing.splitBPS;\\n }\\n\\n /// @dev returns the VIP sale price\\n function getVIPSalePrice() public view returns (uint256) {\\n return _pricing.vipSalePrice;\\n }\\n\\n /// @dev returns the member sale price\\n function getMembersSalePrice() public view returns (uint256) {\\n return _pricing.membersSalePrice;\\n }\\n\\n /// @dev returns the VIP mint limit\\n function getVIPMintLimit() public view returns (uint256) {\\n return _pricing.vipMintLimit;\\n }\\n\\n /// @dev returns the member mint limit\\n function getMembersMintLimit() public view returns (uint256) {\\n return _pricing.membersMintLimit;\\n }\\n\\n /// @dev returns the general mint limit\\n function getGeneralMintLimit() public view returns (uint256) {\\n return salePrice;\\n }\\n\\n /// @dev returns who can mint\\n function getWhoCanMint() public view returns (uint256) {\\n return uint256(_pricing.whoCanMint);\\n }\\n\\n /**\\n Simple eth-based sales function\\n More complex sales functions can be implemented through IExpandedNFT interface\\n */\\n\\n /**\\n @dev This allows the user to purchase an edition\\n at the given price in the contract.\\n */\\n\\n function purchase() external payable returns (uint256) {\\n uint256 currentPrice = _currentSalesPrice();\\n emit EditionSold(currentPrice, msg.sender);\\n\\n address[] memory toMint = new address[](1);\\n toMint[0] = msg.sender;\\n\\n return _mintEditionsBody(toMint); \\n }\\n\\n /**\\n @param to address to send the newly minted edition to\\n @dev This mints one edition to the given address by an allowed minter on the edition instance.\\n */\\n function mintEdition(address to) external payable override returns (uint256) {\\n address[] memory toMint = new address[](1);\\n toMint[0] = to;\\n\\n return _mintEditionsBody(toMint); \\n }\\n\\n /**\\n @param recipients list of addresses to send the newly minted editions to\\n @dev This mints multiple editions to the given list of addresses.\\n */\\n function mintEditions(address[] memory recipients)\\n external payable override returns (uint256)\\n {\\n return _mintEditionsBody(recipients);\\n } \\n\\n /**\\n @param recipients list of addresses to send the newly minted editions to\\n @dev This mints multiple editions to the given list of addresses.\\n */\\n function _mintEditionsBody(address[] memory recipients)\\n internal returns (uint256)\\n {\\n require(_loadedMetadata >= dropSize, \\\"Not all metadata loaded\\\");\\n\\n require(_isAllowedToMint(), \\\"Needs to be an allowed minter\\\");\\n\\n uint256 currentPrice = _currentSalesPrice();\\n require(currentPrice > 0, \\\"Not for sale\\\");\\n require(msg.value == (currentPrice * recipients.length), \\\"Wrong price\\\");\\n\\n require((_pricing.mintCounts[msg.sender] + recipients.length - 1) < _currentMintLimit(), \\\"Exceeded mint limit\\\");\\n\\n require(_claimCount + recipients.length <= dropSize, \\\"Over drop size\\\");\\n\\n if (_pricing.whoCanMint == WhoCanMint.VIPS) {\\n return _vipMintEditions(recipients);\\n }\\n\\n return _mintEditions(recipients);\\n } \\n\\n /**\\n @dev Private function to mint without any access checks.\\n Called by the public edition minting functions.\\n */\\n function _vipMintEditions(address[] memory recipients)\\n internal\\n returns (uint256)\\n {\\n address currentMinter = msg.sender;\\n\\n uint256 unclaimed = 0;\\n uint256 firstUnclaimed = _reserveCount;\\n\\n for (uint256 r = 0; r < _reserveCount; r++) {\\n if (_reserveAddress[r] == currentMinter) {\\n uint256 id = _reserveTokenId[r];\\n\\n if (_tokenClaimed[id] != true) {\\n if (r < firstUnclaimed) {\\n firstUnclaimed = r; \\n }\\n\\n unclaimed++;\\n }\\n }\\n }\\n\\n require(unclaimed >= recipients.length, \\\"Can not mint all editions\\\");\\n\\n uint256 idToMint = 1;\\n\\n uint256 reservationCounter = firstUnclaimed;\\n for (uint256 i = 0; i < recipients.length; i++) {\\n while (_reserveAddress[reservationCounter] != currentMinter) {\\n reservationCounter++;\\n } \\n\\n idToMint = _reserveTokenId[reservationCounter];\\n\\n _mint(recipients[i], idToMint);\\n\\n _perTokenMetadata[idToMint].editionState = ExpandedNFTStates.MINTED;\\n _tokenClaimed[idToMint] = true;\\n _pricing.mintCounts[currentMinter]++;\\n _claimCount++;\\n\\n reservationCounter++;\\n }\\n\\n return idToMint; \\n } \\n\\n /**\\n @dev Private function to mint without any access checks.\\n Called by the public edition minting functions.\\n */\\n function _mintEditions(address[] memory recipients)\\n internal\\n returns (uint256)\\n {\\n address currentMinter = msg.sender;\\n \\n for (uint256 i = 0; i < recipients.length; i++) {\\n while (_tokenClaimed[_currentIndex] == true) {\\n _currentIndex++;\\n } \\n\\n _mint(recipients[i], _currentIndex);\\n\\n _perTokenMetadata[_currentIndex].editionState = ExpandedNFTStates.MINTED;\\n _tokenClaimed[_currentIndex] = true;\\n _pricing.mintCounts[currentMinter]++;\\n _claimCount++;\\n }\\n\\n return _currentIndex; \\n } \\n\\n /**\\n @param _royaltyBPS BPS of the royalty set on the contract. Can be 0 for no royalty.\\n @param _splitBPS BPS of the royalty set on the contract. Can be 0 for no royalty. \\n @param _vipSalePrice Sale price for VIPs\\n @param _membersSalePrice SalePrice for Members \\n @param _generalSalePrice SalePrice for the general public \\n @param _vipMintLimit Mint limit for VIPs\\n @param _membersMintLimit Mint limit for Members \\n @param _generalMintLimit Mint limit for the general public \\n @dev Set various pricing related values\\n */\\n function setPricing (\\n uint256 _royaltyBPS,\\n uint256 _splitBPS,\\n uint256 _vipSalePrice,\\n uint256 _membersSalePrice, \\n uint256 _generalSalePrice,\\n uint256 _vipMintLimit,\\n uint256 _membersMintLimit,\\n uint256 _generalMintLimit \\n ) external onlyOwner { \\n _pricing.royaltyBPS = _royaltyBPS;\\n _pricing.splitBPS = _splitBPS;\\n\\n _pricing.vipSalePrice = _vipSalePrice;\\n _pricing.membersSalePrice = _membersSalePrice;\\n salePrice = _generalSalePrice;\\n\\n _pricing.vipMintLimit = _vipMintLimit;\\n _pricing.membersMintLimit = _membersMintLimit;\\n _pricing.generalMintLimit = _generalMintLimit;\\n\\n emit PriceChanged(salePrice);\\n }\\n\\n /**\\n @dev returns the current ETH sales price\\n based on who can currently mint.\\n */\\n function _currentSalesPrice() internal view returns (uint256){\\n if (_pricing.whoCanMint == WhoCanMint.VIPS) {\\n return _pricing.vipSalePrice;\\n } else if (_pricing.whoCanMint == WhoCanMint.MEMBERS) {\\n return _pricing.membersSalePrice;\\n } else if (_pricing.whoCanMint == WhoCanMint.ANYONE) {\\n return salePrice;\\n } \\n \\n return 0; \\n }\\n\\n /**\\n @param wallets A list of wallets\\n @param tokenIDs A list of tokenId to reserve \\n @dev Set various pricing related values\\n */\\n function reserve (address[] calldata wallets, uint256[] calldata tokenIDs) \\n external onlyOwner { \\n for (uint256 i = 0; i < wallets.length; i++) {\\n _reserveAddress[_reserveCount] = wallets[i]; \\n _reserveTokenId[_reserveCount] = tokenIDs[i]; \\n _reserveCount++;\\n }\\n }\\n\\n /**\\n @dev returns the current loimit on edition that \\n can be minted by one wallet\\n */\\n function _currentMintLimit() internal view returns (uint256){\\n if (_pricing.whoCanMint == WhoCanMint.VIPS) {\\n return _pricing.vipMintLimit;\\n } else if (_pricing.whoCanMint == WhoCanMint.MEMBERS) {\\n return _pricing.membersMintLimit;\\n } else if (_pricing.whoCanMint == WhoCanMint.ANYONE) {\\n return _pricing.generalMintLimit;\\n } \\n \\n return 0; \\n }\\n\\n /**\\n @param _salePrice if sale price is 0 sale is stopped, otherwise that amount \\n of ETH is needed to start the sale.\\n @dev This sets a simple ETH sales price\\n Setting a sales price allows users to mint the drop until it sells out.\\n For more granular sales, use an external sales contract.\\n */\\n function setSalePrice(uint256 _salePrice) external onlyOwner {\\n salePrice = _salePrice;\\n\\n _pricing.whoCanMint = WhoCanMint.ANYONE;\\n\\n emit WhoCanMintChanged(_pricing.whoCanMint);\\n emit PriceChanged(salePrice);\\n }\\n\\n /**\\n @param _salePrice if sale price is 0 sale is stopped, otherwise that amount \\n of ETH is needed to start the sale.\\n @dev This sets the VIP ETH sales price\\n Setting a sales price allows users to mint the drop until it sells out.\\n For more granular sales, use an external sales contract.\\n */\\n function setVIPSalePrice(uint256 _salePrice) external onlyOwner {\\n _pricing.vipSalePrice = _salePrice;\\n\\n _pricing.whoCanMint = WhoCanMint.VIPS;\\n\\n emit WhoCanMintChanged(_pricing.whoCanMint);\\n emit PriceChanged(salePrice);\\n }\\n\\n /**\\n @param _salePrice if sale price is 0 sale is stopped, otherwise that amount \\n of ETH is needed to start the sale.\\n @dev This sets the members ETH sales price\\n Setting a sales price allows users to mint the drop until it sells out.\\n For more granular sales, use an external sales contract.\\n */\\n function setMembersSalePrice(uint256 _salePrice) external onlyOwner {\\n _pricing.membersSalePrice = _salePrice;\\n\\n _pricing.whoCanMint = WhoCanMint.MEMBERS;\\n\\n emit WhoCanMintChanged(_pricing.whoCanMint);\\n emit PriceChanged(salePrice);\\n } \\n\\n\\n /**\\n @param vipSalePrice if sale price is 0 sale is stopped, otherwise that amount \\n of ETH is needed to start the sale.\\n @param membersSalePrice if sale price is 0 sale is stopped, otherwise that amount \\n of ETH is needed to start the sale.\\n @param generalSalePrice if sale price is 0 sale is stopped, otherwise that amount \\n of ETH is needed to start the sale. \\n @dev This sets the members ETH sales price\\n Setting a sales price allows users to mint the drop until it sells out.\\n For more granular sales, use an external sales contract.\\n */\\n function setSalePrices(uint256 vipSalePrice, uint256 membersSalePrice, uint256 generalSalePrice) external onlyOwner {\\n _pricing.vipSalePrice = vipSalePrice;\\n _pricing.membersSalePrice = membersSalePrice;\\n salePrice = generalSalePrice; \\n\\n emit PriceChanged(salePrice);\\n } \\n\\n /**\\n @dev This withdraws ETH from the contract to the contract owner.\\n */\\n function withdraw() external onlyOwner {\\n uint256 currentBalance = address(this).balance;\\n if (currentBalance > 0) {\\n uint256 platformFee = (currentBalance * _pricing.splitBPS) / 10000;\\n uint256 artistFee = currentBalance - platformFee;\\n\\n AddressUpgradeable.sendValue(payable(owner()), platformFee);\\n AddressUpgradeable.sendValue(payable(_artistWallet), artistFee);\\n }\\n\\n if (address(_paymentTokenERC20) != address(0x0)) {\\n uint256 currentBalanceERC20 = _paymentTokenERC20.balanceOf(address(this));\\n if (currentBalanceERC20 > 0) {\\n uint256 platformFee = (currentBalanceERC20 * _pricing.splitBPS) / 10000;\\n uint256 artistFee = currentBalanceERC20 - platformFee;\\n\\n _paymentTokenERC20.transfer(owner(), platformFee);\\n _paymentTokenERC20.transfer(_artistWallet, artistFee);\\n }\\n }\\n }\\n\\n /**\\n @dev This helper function checks if the msg.sender is allowed to mint the\\n given edition id.\\n */\\n function _isAllowedToMint() internal view returns (bool) {\\n if (_pricing.whoCanMint == WhoCanMint.ANYONE) {\\n return true;\\n }\\n\\n if (_pricing.whoCanMint == WhoCanMint.MEMBERS) {\\n if (_pricing.vipAllowedMinters[msg.sender]) {\\n return true;\\n } \\n\\n if (_pricing.allowedMinters[msg.sender]) {\\n return true;\\n } \\n }\\n\\n if (_pricing.whoCanMint == WhoCanMint.VIPS) {\\n if (_pricing.vipAllowedMinters[msg.sender]) {\\n return true;\\n } \\n }\\n\\n if (owner() == msg.sender) {\\n return true;\\n }\\n\\n return false;\\n }\\n\\n /**\\n Simple override for owner interface.\\n */\\n function owner()\\n public\\n view\\n override(OwnableUpgradeable, IExpandedNFT)\\n returns (address)\\n {\\n return super.owner();\\n }\\n\\n /**\\n return the artists wallet address\\n */\\n function getArtistWallet()\\n public\\n view\\n returns (address)\\n {\\n return _artistWallet;\\n }\\n\\n /**\\n set the artists wallet address\\n */\\n function setArtistWallet(address wallet)\\n public\\n onlyOwner\\n {\\n _artistWallet = wallet;\\n } \\n\\n /**\\n return the payment tokens address\\n */\\n function getPaymentToken()\\n public\\n view\\n returns (address)\\n {\\n return address(_paymentTokenERC20);\\n }\\n\\n /**\\n set a new payment token address\\n */\\n function setPaymentToken(address paymentToken)\\n public\\n onlyOwner\\n {\\n if (address(_paymentTokenERC20) != address(0x0)) {\\n require(_paymentTokenERC20.balanceOf(address(this)) == 0, \\\"token must have 0 balance\\\");\\n }\\n\\n _paymentTokenERC20 = IERC20Upgradeable(paymentToken);\\n } \\n\\n /**\\n @dev Sets the types of users who is allowed to mint.\\n */\\n function getAllowedMinter() public view returns (WhoCanMint){\\n return _pricing.whoCanMint;\\n }\\n\\n /**\\n @param minters WhoCanMint enum of minter types\\n @dev Sets the types of users who is allowed to mint.\\n */\\n function setAllowedMinter(WhoCanMint minters) public onlyOwner {\\n require(((minters >= WhoCanMint.ONLY_OWNER) && (minters <= WhoCanMint.ANYONE)), \\\"Needs to be a valid minter type\\\");\\n\\n _pricing.whoCanMint = minters;\\n emit WhoCanMintChanged(minters);\\n }\\n\\n /**\\n @param minter address to set approved minting status for\\n @param allowed boolean if that address is allowed to mint\\n @dev Sets the approved minting status of the given address.\\n This requires that msg.sender is the owner of the given edition id.\\n If the ZeroAddress (address(0x0)) is set as a minter,\\n anyone will be allowed to mint.\\n This setup is similar to setApprovalForAll in the ERC721 spec.\\n */\\n function setApprovedMinters(uint256 count, address[] calldata minter, bool[] calldata allowed) public onlyOwner {\\n for (uint256 i = 0; i < count; i++) {\\n _pricing.allowedMinters[minter[i]] = allowed[i];\\n }\\n }\\n\\n /**\\n @param minter address to set approved minting status for\\n @param allowed boolean if that address is allowed to mint\\n @dev Sets the approved minting status of the given address.\\n This requires that msg.sender is the owner of the given edition id.\\n If the ZeroAddress (address(0x0)) is set as a minter,\\n anyone will be allowed to mint.\\n This setup is similar to setApprovalForAll in the ERC721 spec.\\n */\\n function setApprovedVIPMinters(uint256 count, address[] calldata minter, bool[] calldata allowed) public onlyOwner {\\n for (uint256 i = 0; i < count; i++) {\\n _pricing.vipAllowedMinters[minter[i]] = allowed[i];\\n }\\n }\\n\\n /**\\n @dev Allows for updates of edition urls by the owner of the edition.\\n Only URLs can be updated (data-uris are supported), hashes cannot be updated.\\n */\\n function updateEditionURLs(\\n uint256 tokenId,\\n string memory imageUrl,\\n string memory animationUrl\\n ) public onlyOwner {\\n _perTokenMetadata[tokenId].imageUrl = imageUrl;\\n _perTokenMetadata[tokenId].animationUrl = animationUrl;\\n }\\n\\n /// Returns the number of editions allowed to mint\\n function numberCanMint() public view override returns (uint256) {\\n return dropSize - _claimCount;\\n }\\n\\n /**\\n @param tokenId Token ID to burn\\n User burn function for token id \\n */\\n function burn(uint256 tokenId) public {\\n require(_isApprovedOrOwner(_msgSender(), tokenId), \\\"Not approved\\\");\\n _burn(tokenId);\\n }\\n\\n function redeem(uint256 tokenId) public {\\n require(_exists(tokenId), \\\"No token\\\");\\n require(_isApprovedOrOwner(_msgSender(), tokenId), \\\"Not approved\\\");\\n\\n require((_perTokenMetadata[tokenId].editionState == ExpandedNFTStates.MINTED), \\\"You currently can not redeem\\\");\\n\\n _perTokenMetadata[tokenId].editionState = ExpandedNFTStates.REDEEM_STARTED;\\n emit RedeemStarted(tokenId, _msgSender());\\n }\\n\\n function abortRedemption(uint256 tokenId) public {\\n require(_exists(tokenId), \\\"No token\\\");\\n require(_isApprovedOrOwner(_msgSender(), tokenId), \\\"Not approved\\\");\\n\\n require((_perTokenMetadata[tokenId].editionState == ExpandedNFTStates.REDEEM_STARTED), \\\"You currently can not redeem\\\");\\n\\n _perTokenMetadata[tokenId].editionState = ExpandedNFTStates.MINTED;\\n emit RedeemAborted(tokenId, _msgSender());\\n }\\n\\n function setOfferTerms(uint256 tokenId, uint256 fee) public onlyOwner {\\n require(_exists(tokenId), \\\"No token\\\"); \\n require((_perTokenMetadata[tokenId].editionState == ExpandedNFTStates.REDEEM_STARTED), \\\"Wrong state\\\");\\n\\n _perTokenMetadata[tokenId].editionState = ExpandedNFTStates.SET_OFFER_TERMS;\\n _perTokenMetadata[tokenId].editionFee = fee;\\n\\n emit OfferTermsSet(tokenId);\\n }\\n\\n function rejectOfferTerms(uint256 tokenId) public {\\n require(_exists(tokenId), \\\"No token\\\"); \\n require(_isApprovedOrOwner(_msgSender(), tokenId), \\\"Not approved\\\");\\n\\n require((_perTokenMetadata[tokenId].editionState == ExpandedNFTStates.SET_OFFER_TERMS), \\\"You currently can not redeem\\\");\\n\\n _perTokenMetadata[tokenId].editionState = ExpandedNFTStates.MINTED;\\n\\n emit OfferRejected(tokenId);\\n }\\n\\n function acceptOfferTerms(uint256 tokenId, uint256 paymentAmount) external {\\n require(_exists(tokenId), \\\"No token\\\"); \\n require(_isApprovedOrOwner(_msgSender(), tokenId), \\\"Not approved\\\");\\n\\n require((_perTokenMetadata[tokenId].editionState == ExpandedNFTStates.SET_OFFER_TERMS), \\\"You currently can not redeem\\\");\\n\\n require(paymentAmount >= _perTokenMetadata[tokenId].editionFee, \\\"Wrong price\\\");\\n require(_paymentTokenERC20.allowance(_msgSender(), address(this)) >= _perTokenMetadata[tokenId].editionFee, \\\"Insufficient allowance\\\");\\n\\n bool success = _paymentTokenERC20.transferFrom(_msgSender(), address(this), _perTokenMetadata[tokenId].editionFee);\\n require(success, \\\"Could not transfer token\\\");\\n\\n _perTokenMetadata[tokenId].editionState = ExpandedNFTStates.ACCEPTED_OFFER; \\n\\n emit OfferAccepted(tokenId);\\n }\\n\\n function productionComplete(\\n uint256 tokenId,\\n string memory _description,\\n string memory animationUrl,\\n bytes32 animationHash,\\n string memory imageUrl,\\n bytes32 imageHash, \\n string memory conditionReportUrl,\\n bytes32 conditionReportHash \\n ) public onlyOwner {\\n require(_exists(tokenId), \\\"No token\\\"); \\n require((_perTokenMetadata[tokenId].editionState == ExpandedNFTStates.ACCEPTED_OFFER), \\\"You currently can not redeem\\\");\\n\\n // Set the NFT to display as redeemed\\n _perTokenMetadata[tokenId].description = _description;\\n _perTokenMetadata[tokenId].redeemedAnimationUrl = animationUrl;\\n _perTokenMetadata[tokenId].redeemedAnimationHash = animationHash;\\n _perTokenMetadata[tokenId].redeemedImageUrl = imageUrl;\\n _perTokenMetadata[tokenId].redeemedImageHash = imageHash;\\n _perTokenMetadata[tokenId].conditionReportUrl = conditionReportUrl;\\n _perTokenMetadata[tokenId].conditionReportHash = conditionReportHash;\\n\\n _perTokenMetadata[tokenId].editionState = ExpandedNFTStates.PRODUCTION_COMPLETE;\\n\\n emit ProductionComplete(tokenId);\\n }\\n\\n function acceptDelivery(uint256 tokenId) public {\\n require(_exists(tokenId), \\\"No token\\\"); \\n require(_isApprovedOrOwner(_msgSender(), tokenId), \\\"Not approved\\\");\\n\\n require((_perTokenMetadata[tokenId].editionState == ExpandedNFTStates.PRODUCTION_COMPLETE), \\\"You currently can not redeem\\\");\\n\\n _perTokenMetadata[tokenId].editionState = ExpandedNFTStates.REDEEMED;\\n\\n emit OfferRejected(tokenId);\\n }\\n\\n /**\\n @dev Get URIs for the condition report\\n @return _imageUrl, _imageHash\\n */\\n function getConditionReport(uint256 tokenId)\\n public\\n view\\n returns (\\n string memory,\\n bytes32\\n )\\n {\\n return (_perTokenMetadata[tokenId].conditionReportUrl, _perTokenMetadata[tokenId].conditionReportHash);\\n }\\n\\n /**\\n @dev Get royalty information for token\\n @param _salePrice Sale price for the token\\n */\\n function royaltyInfo(uint256, uint256 _salePrice)\\n external\\n view\\n override\\n returns (address receiver, uint256 royaltyAmount)\\n {\\n if (owner() == address(0x0)) {\\n return (owner(), 0);\\n }\\n return (owner(), (_salePrice * _pricing.royaltyBPS) / 10_000);\\n }\\n\\n /**\\n @dev Get URIs for edition NFT\\n @return _imageUrl, _imageHash, _animationUrl, _animationHash\\n */\\n function getURIs(uint256 tokenId)\\n public\\n view\\n returns (\\n string memory,\\n bytes32,\\n string memory,\\n bytes32\\n )\\n {\\n if (_perTokenMetadata[tokenId].editionState == ExpandedNFTStates.REDEEMED) { \\n return (_perTokenMetadata[tokenId].redeemedImageUrl, _perTokenMetadata[tokenId].redeemedImageHash,\\n _perTokenMetadata[tokenId].redeemedAnimationUrl, _perTokenMetadata[tokenId].redeemedAnimationHash);\\n }\\n\\n return (_perTokenMetadata[tokenId].imageUrl, _perTokenMetadata[tokenId].imageHash,\\n _perTokenMetadata[tokenId].animationUrl, _perTokenMetadata[tokenId].animationHash);\\n }\\n\\n /**\\n @dev Get URI for given token id\\n @param tokenId token id to get uri for\\n @return base64-encoded json metadata object\\n */\\n function tokenURI(uint256 tokenId)\\n public\\n view\\n override\\n returns (string memory)\\n {\\n require(_exists(tokenId), \\\"No token\\\");\\n\\n if (_perTokenMetadata[tokenId].editionState == ExpandedNFTStates.REDEEMED) {\\n return\\n _sharedNFTLogic.createMetadataEdition(\\n name(),\\n _perTokenMetadata[tokenId].description,\\n _perTokenMetadata[tokenId].redeemedImageUrl,\\n _perTokenMetadata[tokenId].redeemedAnimationUrl,\\n tokenId,\\n dropSize\\n );\\n }\\n\\n return\\n _sharedNFTLogic.createMetadataEdition(\\n name(),\\n _perTokenMetadata[tokenId].description,\\n _perTokenMetadata[tokenId].imageUrl,\\n _perTokenMetadata[tokenId].animationUrl,\\n tokenId,\\n dropSize\\n );\\n }\\n\\n function supportsInterface(bytes4 interfaceId)\\n public\\n view\\n override(ERC721Upgradeable, IERC165Upgradeable)\\n returns (bool)\\n {\\n return\\n type(IERC2981Upgradeable).interfaceId == interfaceId ||\\n ERC721Upgradeable.supportsInterface(interfaceId);\\n }\\n}\\n\",\"keccak256\":\"0x2d52681fe71b3802669c94ef39596f4cd032a7431c303a444a52ef20c12a1d4e\",\"license\":\"GPL-3.0\"},\"contracts/IExpandedNFT.sol\":{\"content\":\"// SPDX-License-Identifier: GPL-3.0\\npragma solidity ^0.8.15;\\n\\ninterface IExpandedNFT {\\n function mintEdition(address to) external payable returns (uint256);\\n function mintEditions(address[] memory to) external payable returns (uint256);\\n function numberCanMint() external view returns (uint256);\\n function owner() external view returns (address);\\n}\\n\",\"keccak256\":\"0xe1dc9fcdfab1ec02f10203a59e3fb819707f06515abe45674c5ecc814fc9c253\",\"license\":\"GPL-3.0\"},\"contracts/IPublicSharedMetadata.sol\":{\"content\":\"// SPDX-License-Identifier: GPL-3.0\\n\\npragma solidity ^0.8.15;\\n\\n/// Shared public library for on-chain NFT functions\\ninterface IPublicSharedMetadata {\\n /// @param unencoded bytes to base64-encode\\n function base64Encode(bytes memory unencoded)\\n external\\n pure\\n returns (string memory);\\n\\n /// Encodes the argument json bytes into base64-data uri format\\n /// @param json Raw json to base64 and turn into a data-uri\\n function encodeMetadataJSON(bytes memory json)\\n external\\n pure\\n returns (string memory);\\n\\n /// Proxy to openzeppelin's toString function\\n /// @param value number to return as a string\\n function numberToString(uint256 value)\\n external\\n pure\\n returns (string memory);\\n}\\n\",\"keccak256\":\"0x76caf66cf71c1067c845ff9de580b6e886c448915997ed1d6f37fe1a1ec489de\",\"license\":\"GPL-3.0\"},\"contracts/SharedNFTLogic.sol\":{\"content\":\"// SPDX-License-Identifier: GPL-3.0\\n\\npragma solidity ^0.8.15;\\n\\nimport {StringsUpgradeable} from \\\"@openzeppelin/contracts-upgradeable/utils/StringsUpgradeable.sol\\\";\\nimport {Base64} from \\\"base64-sol/base64.sol\\\";\\nimport {IPublicSharedMetadata} from \\\"./IPublicSharedMetadata.sol\\\";\\n\\n/// Shared NFT logic for rendering metadata associated with editions\\n/// @dev Can safely be used for generic base64Encode and numberToString functions\\ncontract SharedNFTLogic is IPublicSharedMetadata {\\n /// @param unencoded bytes to base64-encode\\n function base64Encode(bytes memory unencoded)\\n public\\n pure\\n override\\n returns (string memory)\\n {\\n return Base64.encode(unencoded);\\n }\\n\\n /// Proxy to openzeppelin's toString function\\n /// @param value number to return as a string\\n function numberToString(uint256 value)\\n public\\n pure\\n override\\n returns (string memory)\\n {\\n return StringsUpgradeable.toString(value);\\n }\\n\\n /// Generate edition metadata from storage information as base64-json blob\\n /// Combines the media data and metadata\\n /// @param name Name of NFT in metadata\\n /// @param description Description of NFT in metadata\\n /// @param imageUrl URL of image to render for edition\\n /// @param animationUrl URL of animation to render for edition\\n /// @param tokenOfEdition Token ID for specific token\\n /// @param editionSize Size of entire edition to show\\n function createMetadataEdition(\\n string memory name,\\n string memory description,\\n string memory imageUrl,\\n string memory animationUrl,\\n uint256 tokenOfEdition,\\n uint256 editionSize\\n ) external pure returns (string memory) {\\n string memory _tokenMediaData = tokenMediaData(\\n imageUrl,\\n animationUrl,\\n tokenOfEdition\\n );\\n bytes memory json = createMetadataJSON(\\n name,\\n description,\\n _tokenMediaData,\\n tokenOfEdition,\\n editionSize\\n );\\n return encodeMetadataJSON(json);\\n }\\n\\n /// Function to create the metadata json string for the nft edition\\n /// @param name Name of NFT in metadata\\n /// @param description Description of NFT in metadata\\n /// @param mediaData Data for media to include in json object\\n /// @param tokenOfEdition Token ID for specific token\\n /// @param editionSize Size of entire edition to show\\n function createMetadataJSON(\\n string memory name,\\n string memory description,\\n string memory mediaData,\\n uint256 tokenOfEdition,\\n uint256 editionSize\\n ) public pure returns (bytes memory) {\\n bytes memory editionSizeText;\\n if (editionSize > 0) {\\n editionSizeText = abi.encodePacked(\\n \\\"/\\\",\\n numberToString(editionSize)\\n );\\n }\\n return\\n // solhint-disable quotes\\n abi.encodePacked(\\n '{\\\"name\\\": \\\"',\\n name,\\n \\\" \\\",\\n numberToString(tokenOfEdition),\\n editionSizeText,\\n '\\\", \\\"',\\n 'description\\\": \\\"',\\n description,\\n '\\\", \\\"',\\n mediaData,\\n 'properties\\\": {\\\"number\\\": ',\\n numberToString(tokenOfEdition),\\n ',',\\n '\\\"}}'\\n );\\n // solhint-enable quotes\\n }\\n\\n /// Encodes the argument json bytes into base64-data uri format\\n /// @param json Raw json to base64 and turn into a data-uri\\n function encodeMetadataJSON(bytes memory json)\\n public\\n pure\\n override\\n returns (string memory)\\n {\\n return\\n string(\\n abi.encodePacked(\\n \\\"data:application/json;base64,\\\",\\n base64Encode(json)\\n )\\n );\\n }\\n\\n /// Generates edition metadata from storage information as base64-json blob\\n /// Combines the media data and metadata\\n /// @param imageUrl URL of image to render for edition\\n /// @param animationUrl URL of animation to render for edition\\n function tokenMediaData(\\n string memory imageUrl,\\n string memory animationUrl,\\n uint256 tokenOfEdition\\n ) public pure returns (string memory) {\\n bool hasImage = bytes(imageUrl).length > 0;\\n bool hasAnimation = bytes(animationUrl).length > 0;\\n if (hasImage && hasAnimation) {\\n return\\n // solhint-disable quotes\\n string(\\n abi.encodePacked(\\n 'image\\\": \\\"',\\n imageUrl,\\n \\\"?id=\\\",\\n numberToString(tokenOfEdition),\\n '\\\", \\\"animation_url\\\": \\\"',\\n animationUrl,\\n \\\"?id=\\\",\\n numberToString(tokenOfEdition),\\n '\\\", \\\"'\\n )\\n );\\n // solhint-enable quotes\\n }\\n if (hasImage) {\\n return\\n // solhint-disable quotes\\n string(\\n abi.encodePacked(\\n 'image\\\": \\\"',\\n imageUrl,\\n \\\"?id=\\\",\\n numberToString(tokenOfEdition),\\n '\\\", \\\"'\\n )\\n );\\n // solhint-enable quotes\\n }\\n if (hasAnimation) {\\n return\\n // solhint-disable quotes\\n string(\\n abi.encodePacked(\\n 'animation_url\\\": \\\"',\\n animationUrl,\\n \\\"?id=\\\",\\n numberToString(tokenOfEdition),\\n '\\\", \\\"'\\n )\\n );\\n // solhint-enable quotes\\n }\\n\\n return \\\"\\\";\\n }\\n}\\n\",\"keccak256\":\"0xb1c53433d1f74e358472e64cbb82db20af76022599c34147ce42d458fc4647b8\",\"license\":\"GPL-3.0\"}},\"version\":1}", - "bytecode": "0x60a06040523480156200001157600080fd5b5060405162004a9038038062004a90833981016040819052620000349162000050565b6001600160a01b031660805260dd805460ff1916905562000082565b6000602082840312156200006357600080fd5b81516001600160a01b03811681146200007b57600080fd5b9392505050565b6080516149eb620000a5600039600081816124ed01526125ae01526149eb6000f3fe60806040526004361061032d5760003560e01c806369e1cac1116101a7578063b88d4fde116100ed578063db006a7511610090578063db006a7514610955578063de5d62a014610975578063e985e9c514610995578063ee5df550146109b5578063f2fde38b146109d5578063f51f96dd146109f5578063fc51e0c314610a0b578063fce57fd914610a3957600080fd5b8063b88d4fde14610873578063bcda01af14610893578063bf82ae49146108b3578063c87b56dd146108d3578063c9d359d6146108f3578063d375b6fc1461090d578063d41c3a6514610922578063d5344e361461094057600080fd5b80637b422c4c116101555780637b422c4c146107a15780638c503ca4146107c15780638da5cb5b146107e157806393d23923146107f657806395d89b4114610816578063a1279cfc1461082b578063a22cb46514610840578063a66ff0af1461086057600080fd5b806369e1cac1146106d75780636a326ab1146106f757806370a0823114610717578063715018a6146107375780637284e4161461074c578063787e5efc14610761578063799b7c5f1461078157600080fd5b8063392774151161027757806342966c681161021a57806342966c68146105f557806346d9c4bf14610615578063499e16721461062a57806352c9143e1461063f5780636352211e1461065f57806364e9293e1461067f57806364edfbf0146106af578063660243d0146106b757600080fd5b8063392774151461052257806339dc13bc146105405780633ccfd60b146105605780633da165d4146105755780633e1dc1071461058b578063402635cf146105a0578063410a1095146105b557806342842e0e146105d557600080fd5b80630f6a9349116102df5780630f6a93491461042c57806318160ddd1461043f5780631919fed71461045457806323b872dd146104745780632a55205a14610494578063367de39e146104c257806336f3ec76146104e25780633892aa521461050257600080fd5b806301ffc9a714610332578063036c8ea11461036757806306fdde0314610389578063081812fc146103ab578063095ea7b3146103d857806309949abe146103f85780630b65b6e714610417575b600080fd5b34801561033e57600080fd5b5061035261034d366004613991565b610a59565b60405190151581526020015b60405180910390f35b34801561037357600080fd5b506103876103823660046139ae565b610a84565b005b34801561039557600080fd5b5061039e610ae0565b60405161035e9190613a5b565b3480156103b757600080fd5b506103cb6103c6366004613a6e565b610b72565b60405161035e9190613a87565b3480156103e457600080fd5b506103876103f3366004613ab7565b610b99565b34801561040457600080fd5b5060d4545b60405190815260200161035e565b34801561042357600080fd5b50610409610cb3565b61040961043a366004613b4a565b610cca565b34801561044b57600080fd5b5060d254610409565b34801561046057600080fd5b5061038761046f366004613a6e565b610cd5565b34801561048057600080fd5b5061038761048f366004613be6565b610d62565b3480156104a057600080fd5b506104b46104af366004613c22565b610d94565b60405161035e929190613c44565b3480156104ce57600080fd5b506103876104dd366004613c5d565b610df0565b3480156104ee57600080fd5b506103876104fd366004613c7e565b610ecb565b34801561050e57600080fd5b5061038761051d366004613cee565b610f09565b34801561052e57600080fd5b5060ca546001600160a01b03166103cb565b34801561054c57600080fd5b5061038761055b366004613a6e565b610fb0565b34801561056c57600080fd5b5061038761108c565b34801561058157600080fd5b5061040960cc5481565b34801561059757600080fd5b5060d954610409565b3480156105ac57600080fd5b5060d554610409565b3480156105c157600080fd5b5060dd5460ff1660405161035e9190613d7d565b3480156105e157600080fd5b506103876105f0366004613be6565b6112aa565b34801561060157600080fd5b50610387610610366004613a6e565b6112c5565b34801561062157600080fd5b5060df54610409565b34801561063657600080fd5b5060d754610409565b34801561064b57600080fd5b5061038761065a366004613e25565b6112f3565b34801561066b57600080fd5b506103cb61067a366004613a6e565b611431565b34801561068b57600080fd5b5061069f61069a366004613a6e565b611466565b60405161035e9493929190613ef8565b61040961172b565b3480156106c357600080fd5b506103876106d2366004613c22565b6117d7565b3480156106e357600080fd5b506103876106f2366004613f35565b611aa0565b34801561070357600080fd5b50610387610712366004613fa0565b611b60565b34801561072357600080fd5b50610409610732366004613fa0565b611c55565b34801561074357600080fd5b50610387611cdb565b34801561075857600080fd5b5061039e611cef565b34801561076d57600080fd5b5061038761077c366004613a6e565b611d7d565b34801561078d57600080fd5b5061038761079c366004613cee565b611d9d565b3480156107ad57600080fd5b506103876107bc366004613a6e565b611e3c565b3480156107cd57600080fd5b506103876107dc366004613fbb565b611ee8565b3480156107ed57600080fd5b506103cb611f2d565b34801561080257600080fd5b50610387610811366004614027565b611f41565b34801561082257600080fd5b5061039e6120ed565b34801561083757600080fd5b5060d654610409565b34801561084c57600080fd5b5061038761085b3660046140c1565b6120fc565b61040961086e366004613fa0565b612107565b34801561087f57600080fd5b5061038761088e3660046140f8565b612170565b34801561089f57600080fd5b506103876108ae366004614258565b6121a1565b3480156108bf57600080fd5b506103876108ce366004613a6e565b6123b4565b3480156108df57600080fd5b5061039e6108ee366004613a6e565b61249a565b3480156108ff57600080fd5b5060cc5460cd541015610352565b34801561091957600080fd5b50610409612628565b34801561092e57600080fd5b5060e0546001600160a01b03166103cb565b34801561094c57600080fd5b5060d854610409565b34801561096157600080fd5b50610387610970366004613a6e565b612642565b34801561098157600080fd5b50610387610990366004613a6e565b61270f565b3480156109a157600080fd5b506103526109b036600461433c565b612730565b3480156109c157600080fd5b506103876109d0366004613c22565b61275e565b3480156109e157600080fd5b506103876109f0366004613fa0565b612842565b348015610a0157600080fd5b5061040960df5481565b348015610a1757600080fd5b50610a2b610a26366004613a6e565b6128b8565b60405161035e92919061436f565b348015610a4557600080fd5b50610387610a54366004613fa0565b61296b565b600063152a902d60e11b6001600160e01b031983161480610a7e5750610a7e82612995565b92915050565b610a8c6129e5565b60d488905560d587905560d686905560d785905560df84905560d883905560d982905560da819055604051848152600080516020614996833981519152906020015b60405180910390a15050505050505050565b606060658054610aef90614391565b80601f0160208091040260200160405190810160405280929190818152602001828054610b1b90614391565b8015610b685780601f10610b3d57610100808354040283529160200191610b68565b820191906000526020600020905b815481529060010190602001808311610b4b57829003601f168201915b5050505050905090565b6000610b7d82612a44565b506000908152606960205260409020546001600160a01b031690565b6000610ba482611431565b9050806001600160a01b0316836001600160a01b031603610c165760405162461bcd60e51b815260206004820152602160248201527f4552433732313a20617070726f76616c20746f2063757272656e74206f776e656044820152603960f91b60648201526084015b60405180910390fd5b336001600160a01b0382161480610c325750610c328133612730565b610ca45760405162461bcd60e51b815260206004820152603e60248201527f4552433732313a20617070726f76652063616c6c6572206973206e6f7420746f60448201527f6b656e206f776e6572206e6f7220617070726f76656420666f7220616c6c00006064820152608401610c0d565b610cae8383612a69565b505050565b600060d25460cc54610cc591906143e1565b905090565b6000610a7e82612ad7565b610cdd6129e5565b60df81905560dd80546003919060ff19166001835b021790555060dd546040517f40cb75607c032a2aac1c3565feb3b70e354cbd56e8f8fa36773cba5750464c5191610d2e9160ff90911690613d7d565b60405180910390a160008051602061499683398151915260df54604051610d5791815260200190565b60405180910390a150565b610d6d335b82612ce4565b610d895760405162461bcd60e51b8152600401610c0d906143f8565b610cae838383612d43565b60008080610da0611f2d565b6001600160a01b031603610dc157610db6611f2d565b600091509150610de9565b610dc9611f2d565b60d45461271090610dda9086614446565b610de49190614465565b915091505b9250929050565b610df86129e5565b6000816003811115610e0c57610e0c613d67565b10158015610e2c57506003816003811115610e2957610e29613d67565b11155b610e785760405162461bcd60e51b815260206004820152601f60248201527f4e6565647320746f20626520612076616c6964206d696e7465722074797065006044820152606401610c0d565b60dd805482919060ff19166001836003811115610e9757610e97613d67565b02179055507f40cb75607c032a2aac1c3565feb3b70e354cbd56e8f8fa36773cba5750464c5181604051610d579190613d7d565b610ed36129e5565b60d683905560d782905560df819055604051818152600080516020614996833981519152906020015b60405180910390a1505050565b610f116129e5565b60005b85811015610fa857828282818110610f2e57610f2e614487565b9050602002016020810190610f43919061449d565b60dc6000878785818110610f5957610f59614487565b9050602002016020810190610f6e9190613fa0565b6001600160a01b031681526020810191909152604001600020805460ff191691151591909117905580610fa0816144ba565b915050610f14565b505050505050565b610fb981612ecd565b610fd55760405162461bcd60e51b8152600401610c0d906144d3565b610fde33610d67565b610ffa5760405162461bcd60e51b8152600401610c0d906144f5565b6003600082815260cb602052604090205460ff16600681111561101f5761101f613d67565b1461103c5760405162461bcd60e51b8152600401610c0d9061451b565b600081815260cb6020526040902080546001919060ff191682805b02179055506040518181527facb9cd98d41dc8854281172ab2a764d949b62b51776edd911f85376ed2eca64b90602001610d57565b6110946129e5565b4780156110f35760d554600090612710906110af9084614446565b6110b99190614465565b905060006110c782846143e1565b90506110da6110d4611f2d565b83612eea565b60ca546110f0906001600160a01b031682612eea565b50505b60e0546001600160a01b0316156112a75760e0546040516370a0823160e01b81526000916001600160a01b0316906370a0823190611135903090600401613a87565b602060405180830381865afa158015611152573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906111769190614552565b905080156112a55760d554600090612710906111929084614446565b61119c9190614465565b905060006111aa82846143e1565b60e0549091506001600160a01b031663a9059cbb6111c6611f2d565b846040518363ffffffff1660e01b81526004016111e4929190613c44565b6020604051808303816000875af1158015611203573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611227919061456b565b5060e05460ca5460405163a9059cbb60e01b81526001600160a01b039283169263a9059cbb9261125e929116908590600401613c44565b6020604051808303816000875af115801561127d573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906112a1919061456b565b5050505b505b50565b610cae83838360405180602001604052806000815250612170565b6112ce33610d67565b6112ea5760405162461bcd60e51b8152600401610c0d906144f5565b6112a781613003565b6112fb6129e5565b61130488612ecd565b6113205760405162461bcd60e51b8152600401610c0d906144d3565b6004600089815260cb602052604090205460ff16600681111561134557611345613d67565b146113625760405162461bcd60e51b8152600401610c0d9061451b565b600088815260cb6020526040902060020161137d88826145ce565b50600088815260cb6020526040902060070161139987826145ce565b50600088815260cb60205260409020600881018690556009016113bc85826145ce565b50600088815260cb60205260409020600a8101849055600b016113df83826145ce565b50600088815260cb6020908152604091829020600c8101849055805460ff1916600517905590518981527fadb6d4f651f6fbf8b37026019ee48fdb86abf9ec94eb4a5f0acf72ad8c7872a39101610ace565b6000818152606760205260408120546001600160a01b031680610a7e5760405162461bcd60e51b8152600401610c0d9061468d565b6060600081816006600086815260cb602052604090205460ff16600681111561149157611491613d67565b036115df57600085815260cb60205260409020600a81015460088201546009830180549093600701919084906114c690614391565b80601f01602080910402602001604051908101604052809291908181526020018280546114f290614391565b801561153f5780601f106115145761010080835404028352916020019161153f565b820191906000526020600020905b81548152906001019060200180831161152257829003601f168201915b5050505050935081805461155290614391565b80601f016020809104026020016040519081016040528092919081815260200182805461157e90614391565b80156115cb5780601f106115a0576101008083540402835291602001916115cb565b820191906000526020600020905b8154815290600101906020018083116115ae57829003601f168201915b505050505091509350935093509350611724565b600085815260cb602052604090206006810154600482015460058301805490936003019190849061160f90614391565b80601f016020809104026020016040519081016040528092919081815260200182805461163b90614391565b80156116885780601f1061165d57610100808354040283529160200191611688565b820191906000526020600020905b81548152906001019060200180831161166b57829003601f168201915b5050505050935081805461169b90614391565b80601f01602080910402602001604051908101604052809291908181526020018280546116c790614391565b80156117145780601f106116e957610100808354040283529160200191611714565b820191906000526020600020905b8154815290600101906020018083116116f757829003601f168201915b5050505050915093509350935093505b9193509193565b60008061173661308d565b604080518281523360208201529192507f60a6c75698fadb72223808131f9f9bb9db3afa32122db6d94fb8fc985a504baa910160405180910390a16040805160018082528183019092526000916020808301908036833701905050905033816000815181106117a7576117a7614487565b60200260200101906001600160a01b031690816001600160a01b0316815250506117d081612ad7565b9250505090565b6117e082612ecd565b6117fc5760405162461bcd60e51b8152600401610c0d906144d3565b611807335b83612ce4565b6118235760405162461bcd60e51b8152600401610c0d906144f5565b6003600083815260cb602052604090205460ff16600681111561184857611848613d67565b146118655760405162461bcd60e51b8152600401610c0d9061451b565b600082815260cb60205260409020600101548110156118965760405162461bcd60e51b8152600401610c0d906146bf565b600082815260cb602052604090206001015460e0546001600160a01b031663dd62ed3e336040516001600160e01b031960e084901b1681526001600160a01b039091166004820152306024820152604401602060405180830381865afa158015611904573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906119289190614552565b101561196f5760405162461bcd60e51b8152602060048201526016602482015275496e73756666696369656e7420616c6c6f77616e636560501b6044820152606401610c0d565b60e0546000906001600160a01b03166323b872dd33600086815260cb60205260409081902060010154905160e084901b6001600160e01b03191681526001600160a01b03909216600483015230602483015260448201526064016020604051808303816000875af11580156119e8573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611a0c919061456b565b905080611a565760405162461bcd60e51b815260206004820152601860248201527721b7bab632103737ba103a3930b739b332b9103a37b5b2b760411b6044820152606401610c0d565b600083815260cb6020908152604091829020805460ff1916600417905590518481527f9bd6b4fd288008520fd788a93304e5688a401aea817ea8140ecf1fb8648f31919101610efc565b611aa86129e5565b60005b838110156112a157848482818110611ac557611ac5614487565b9050602002016020810190611ada9190613fa0565b60ce54600090815260cf6020526040902080546001600160a01b0319166001600160a01b0392909216919091179055828282818110611b1b57611b1b614487565b60ce8054600090815260d060209081526040822093029490940135909155805492909150611b48836144ba565b91905055508080611b58906144ba565b915050611aab565b611b686129e5565b60e0546001600160a01b031615611c335760e0546040516370a0823160e01b81526001600160a01b03909116906370a0823190611ba9903090600401613a87565b602060405180830381865afa158015611bc6573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611bea9190614552565b15611c335760405162461bcd60e51b8152602060048201526019602482015278746f6b656e206d757374206861766520302062616c616e636560381b6044820152606401610c0d565b60e080546001600160a01b0319166001600160a01b0392909216919091179055565b60006001600160a01b038216611cbf5760405162461bcd60e51b815260206004820152602960248201527f4552433732313a2061646472657373207a65726f206973206e6f7420612076616044820152683634b21037bbb732b960b91b6064820152608401610c0d565b506001600160a01b031660009081526068602052604090205490565b611ce36129e5565b611ced6000613104565b565b60c98054611cfc90614391565b80601f0160208091040260200160405190810160405280929190818152602001828054611d2890614391565b8015611d755780601f10611d4a57610100808354040283529160200191611d75565b820191906000526020600020905b815481529060010190602001808311611d5857829003601f168201915b505050505081565b611d856129e5565b60d681905560dd80546001919060ff19168280610cf2565b611da56129e5565b60005b85811015610fa857828282818110611dc257611dc2614487565b9050602002016020810190611dd7919061449d565b60db6000878785818110611ded57611ded614487565b9050602002016020810190611e029190613fa0565b6001600160a01b031681526020810191909152604001600020805460ff191691151591909117905580611e34816144ba565b915050611da8565b611e4581612ecd565b611e615760405162461bcd60e51b8152600401610c0d906144d3565b611e6a33610d67565b611e865760405162461bcd60e51b8152600401610c0d906144f5565b6005600082815260cb602052604090205460ff166006811115611eab57611eab613d67565b14611ec85760405162461bcd60e51b8152600401610c0d9061451b565b600081815260cb6020526040902080546006919060ff1916600183611057565b611ef06129e5565b600083815260cb60205260409020600501611f0b83826145ce565b50600083815260cb60205260409020600301611f2782826145ce565b50505050565b6000610cc56097546001600160a01b031690565b600054610100900460ff1615808015611f615750600054600160ff909116105b80611f7b5750303b158015611f7b575060005460ff166001145b611fde5760405162461bcd60e51b815260206004820152602e60248201527f496e697469616c697a61626c653a20636f6e747261637420697320616c72656160448201526d191e481a5b9a5d1a585b1a5e995960921b6064820152608401610c0d565b6000805460ff191660011790558015612001576000805461ff0019166101001790555b600082116120495760405162461bcd60e51b8152602060048201526015602482015274044726f702073697a65206d757374206265203e203605c1b6044820152606401610c0d565b6120538484613156565b61205b613187565b61206486612842565b60ca80546001600160a01b0319166001600160a01b03871617905560cc829055600060d255600160d35560c961209a85826145ce565b50600060cd558015610fa8576000805461ff0019169055604051600181527f7f26b83ff96e1f2b6a682f133852f6798a09c465da95921460cefb38474024989060200160405180910390a1505050505050565b606060668054610aef90614391565b6112a53383836131b6565b60408051600180825281830190925260009182919060208083019080368337019050509050828160008151811061214057612140614487565b60200260200101906001600160a01b031690816001600160a01b03168152505061216981612ad7565b9392505050565b61217933611801565b6121955760405162461bcd60e51b8152600401610c0d906143f8565b611f2784848484613280565b858551146121c15760405162461bcd60e51b8152600401610c0d906146e4565b858251146121e15760405162461bcd60e51b8152600401610c0d906146e4565b858151146122015760405162461bcd60e51b8152600401610c0d906146e4565b858451146122215760405162461bcd60e51b8152600401610c0d906146e4565b858351146122415760405162461bcd60e51b8152600401610c0d906146e4565b60005b86811015612393576000612258828a614710565b612263906001614710565b905086828151811061227757612277614487565b602002602001015160cb600083815260200190815260200160002060020190816122a191906145ce565b508382815181106122b4576122b4614487565b602002602001015160cb600083815260200190815260200160002060030190816122de91906145ce565b508282815181106122f1576122f1614487565b602002602001015160cb60008381526020019081526020016000206004018190555085828151811061232557612325614487565b602002602001015160cb6000838152602001908152602001600020600501908161234f91906145ce565b5084828151811061236257612362614487565b602090810291909101810151600092835260cb9091526040909120600601558061238b816144ba565b915050612244565b508560cd60008282546123a69190614710565b909155505050505050505050565b6123bd81612ecd565b6123d95760405162461bcd60e51b8152600401610c0d906144d3565b6123e233610d67565b6123fe5760405162461bcd60e51b8152600401610c0d906144f5565b6002600082815260cb602052604090205460ff16600681111561242357612423613d67565b146124405760405162461bcd60e51b8152600401610c0d9061451b565b600081815260cb60205260409020805460ff191660011790557f6c7c9f699aacfa5bacd2758023ab209b2d1e1c237bfa752bdc94cb87878cf1b781335b604080519283526001600160a01b03909116602083015201610d57565b60606124a582612ecd565b6124c15760405162461bcd60e51b8152600401610c0d906144d3565b6006600083815260cb602052604090205460ff1660068111156124e6576124e6613d67565b036125ac577f00000000000000000000000000000000000000000000000000000000000000006001600160a01b031663df30dba0612522610ae0565b600085815260cb60205260409081902060cc5491516001600160e01b031960e086901b1681526125679392600283019260098101926007909101918a916004016147a5565b600060405180830381865afa158015612584573d6000803e3d6000fd5b505050506040513d6000823e601f3d908101601f19168201604052610a7e9190810190614808565b7f00000000000000000000000000000000000000000000000000000000000000006001600160a01b031663df30dba06125e3610ae0565b600085815260cb60205260409081902060cc5491516001600160e01b031960e086901b1681526125679392600283019260058101926003909101918a916004016147a5565b60dd5460009060ff166003811115610cc557610cc5613d67565b61264b81612ecd565b6126675760405162461bcd60e51b8152600401610c0d906144d3565b61267033610d67565b61268c5760405162461bcd60e51b8152600401610c0d906144f5565b6001600082815260cb602052604090205460ff1660068111156126b1576126b1613d67565b146126ce5760405162461bcd60e51b8152600401610c0d9061451b565b600081815260cb60205260409020805460ff191660021790557fc290e40808ef06721539ad0d926f8d5f53d2dd6d5e28334308a012676f867416813361247d565b6127176129e5565b60d781905560dd80546002919060ff1916600183610cf2565b6001600160a01b039182166000908152606a6020908152604080832093909416825291909152205460ff1690565b6127666129e5565b61276f82612ecd565b61278b5760405162461bcd60e51b8152600401610c0d906144d3565b6002600083815260cb602052604090205460ff1660068111156127b0576127b0613d67565b146127eb5760405162461bcd60e51b815260206004820152600b60248201526a57726f6e6720737461746560a81b6044820152606401610c0d565b600082815260cb6020908152604091829020805460ff1916600317815560010183905590518381527f316b408b8cf323e8f20ef7a8c267eedc89bef19ca5db7fc30be938e55bf068f9910160405180910390a15050565b61284a6129e5565b6001600160a01b0381166128af5760405162461bcd60e51b815260206004820152602660248201527f4f776e61626c653a206e6577206f776e657220697320746865207a65726f206160448201526564647265737360d01b6064820152608401610c0d565b6112a781613104565b600081815260cb60205260408120600c810154600b9091018054606093929082906128e290614391565b80601f016020809104026020016040519081016040528092919081815260200182805461290e90614391565b801561295b5780601f106129305761010080835404028352916020019161295b565b820191906000526020600020905b81548152906001019060200180831161293e57829003601f168201915b5050505050915091509150915091565b6129736129e5565b60ca80546001600160a01b0319166001600160a01b0392909216919091179055565b60006001600160e01b031982166380ac58cd60e01b14806129c657506001600160e01b03198216635b5e139f60e01b145b80610a7e57506301ffc9a760e01b6001600160e01b0319831614610a7e565b336129ee611f2d565b6001600160a01b031614611ced5760405162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e65726044820152606401610c0d565b612a4d81612ecd565b6112a75760405162461bcd60e51b8152600401610c0d9061468d565b600081815260696020526040902080546001600160a01b0319166001600160a01b0384169081179091558190612a9e82611431565b6001600160a01b03167f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b92560405160405180910390a45050565b600060cc5460cd541015612b275760405162461bcd60e51b8152602060048201526017602482015276139bdd08185b1b081b595d1859185d18481b1bd8591959604a1b6044820152606401610c0d565b612b2f6132b3565b612b7b5760405162461bcd60e51b815260206004820152601d60248201527f4e6565647320746f20626520616e20616c6c6f776564206d696e7465720000006044820152606401610c0d565b6000612b8561308d565b905060008111612bc65760405162461bcd60e51b815260206004820152600c60248201526b4e6f7420666f722073616c6560a01b6044820152606401610c0d565b8251612bd29082614446565b3414612bf05760405162461bcd60e51b8152600401610c0d906146bf565b612bf861338c565b835133600090815260de6020526040902054600191612c1691614710565b612c2091906143e1565b10612c635760405162461bcd60e51b8152602060048201526013602482015272115e18d959591959081b5a5b9d081b1a5b5a5d606a1b6044820152606401610c0d565b60cc54835160d254612c759190614710565b1115612cb45760405162461bcd60e51b815260206004820152600e60248201526d4f7665722064726f702073697a6560901b6044820152606401610c0d565b600160dd5460ff166003811115612ccd57612ccd613d67565b03612cdb57612169836133fd565b612169836135ed565b600080612cf083611431565b9050806001600160a01b0316846001600160a01b03161480612d175750612d178185612730565b80612d3b5750836001600160a01b0316612d3084610b72565b6001600160a01b0316145b949350505050565b826001600160a01b0316612d5682611431565b6001600160a01b031614612dba5760405162461bcd60e51b815260206004820152602560248201527f4552433732313a207472616e736665722066726f6d20696e636f72726563742060448201526437bbb732b960d91b6064820152608401610c0d565b6001600160a01b038216612e1c5760405162461bcd60e51b8152602060048201526024808201527f4552433732313a207472616e7366657220746f20746865207a65726f206164646044820152637265737360e01b6064820152608401610c0d565b612e27600082612a69565b6001600160a01b0383166000908152606860205260408120805460019290612e509084906143e1565b90915550506001600160a01b0382166000908152606860205260408120805460019290612e7e908490614710565b909155505060008181526067602052604080822080546001600160a01b0319166001600160a01b03868116918217909255915184939187169160008051602061497683398151915291a4505050565b6000908152606760205260409020546001600160a01b0316151590565b80471015612f3a5760405162461bcd60e51b815260206004820152601d60248201527f416464726573733a20696e73756666696369656e742062616c616e63650000006044820152606401610c0d565b6000826001600160a01b03168260405160006040518083038185875af1925050503d8060008114612f87576040519150601f19603f3d011682016040523d82523d6000602084013e612f8c565b606091505b5050905080610cae5760405162461bcd60e51b815260206004820152603a60248201527f416464726573733a20756e61626c6520746f2073656e642076616c75652c207260448201527f6563697069656e74206d617920686176652072657665727465640000000000006064820152608401610c0d565b600061300e82611431565b905061301b600083612a69565b6001600160a01b03811660009081526068602052604081208054600192906130449084906143e1565b909155505060008281526067602052604080822080546001600160a01b0319169055518391906001600160a01b03841690600080516020614976833981519152908390a46112a5565b6000600160dd5460ff1660038111156130a8576130a8613d67565b036130b4575060d65490565b600260dd5460ff1660038111156130cd576130cd613d67565b036130d9575060d75490565b600360dd5460ff1660038111156130f2576130f2613d67565b036130fe575060df5490565b50600090565b609780546001600160a01b038381166001600160a01b0319831681179093556040519116919082907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e090600090a35050565b600054610100900460ff1661317d5760405162461bcd60e51b8152600401610c0d9061487e565b6112a582826136e8565b600054610100900460ff166131ae5760405162461bcd60e51b8152600401610c0d9061487e565b611ced613728565b816001600160a01b0316836001600160a01b0316036132135760405162461bcd60e51b815260206004820152601960248201527822a9219b99189d1030b8383937bb32903a379031b0b63632b960391b6044820152606401610c0d565b6001600160a01b038381166000818152606a6020908152604080832094871680845294825291829020805460ff191686151590811790915591519182527f17307eab39ab6107e8899845ad3d59bd9653f200f220920489ca2b5937696c31910160405180910390a3505050565b61328b848484612d43565b61329784848484613758565b611f275760405162461bcd60e51b8152600401610c0d906148c9565b6000600360dd5460ff1660038111156132ce576132ce613d67565b036132d95750600190565b600260dd5460ff1660038111156132f2576132f2613d67565b036133335733600090815260dc602052604090205460ff16156133155750600190565b33600090815260db602052604090205460ff16156133335750600190565b600160dd5460ff16600381111561334c5761334c613d67565b0361336f5733600090815260dc602052604090205460ff161561336f5750600190565b33613378611f2d565b6001600160a01b0316036130fe5750600190565b6000600160dd5460ff1660038111156133a7576133a7613d67565b036133b3575060d85490565b600260dd5460ff1660038111156133cc576133cc613d67565b036133d8575060d95490565b600360dd5460ff1660038111156133f1576133f1613d67565b036130fe575060da5490565b60ce5460009033908290815b60ce5481101561348c57600081815260cf60205260409020546001600160a01b0380861691160361347a57600081815260d0602090815260408083205480845260d19092529091205460ff161515600114613478578282101561346a578192505b83613474816144ba565b9450505b505b80613484816144ba565b915050613409565b5084518210156134da5760405162461bcd60e51b815260206004820152601960248201527843616e206e6f74206d696e7420616c6c2065646974696f6e7360381b6044820152606401610c0d565b60018160005b87518110156135e1575b600082815260cf60205260409020546001600160a01b0387811691161461351d5781613515816144ba565b9250506134ea565b60d0600083815260200190815260200160002054925061355688828151811061354857613548614487565b602002602001015184613859565b600083815260cb602090815260408083208054600160ff19918216811790925560d18452828520805490911690911790556001600160a01b038916835260de90915281208054916135a6836144ba565b909155505060d280549060006135bb836144ba565b919050555081806135cb906144ba565b92505080806135d9906144ba565b9150506134e0565b50909695505050505050565b600033815b83518110156136dd575b60d354600090815260d1602052604090205460ff1615156001036136345760d3805490600061362a836144ba565b91905055506135fc565b61365984828151811061364957613649614487565b602002602001015160d354613859565b60d38054600090815260cb602090815260408083208054600160ff1991821681179092559454845260d183528184208054909516179093556001600160a01b038516825260de90529081208054916136b0836144ba565b909155505060d280549060006136c5836144ba565b919050555080806136d5906144ba565b9150506135f2565b505060d35492915050565b600054610100900460ff1661370f5760405162461bcd60e51b8152600401610c0d9061487e565b606561371b83826145ce565b506066610cae82826145ce565b600054610100900460ff1661374f5760405162461bcd60e51b8152600401610c0d9061487e565b611ced33613104565b60006001600160a01b0384163b1561384e57604051630a85bd0160e11b81526001600160a01b0385169063150b7a029061379c90339089908890889060040161491b565b6020604051808303816000875af19250505080156137d7575060408051601f3d908101601f191682019092526137d491810190614958565b60015b613834573d808015613805576040519150601f19603f3d011682016040523d82523d6000602084013e61380a565b606091505b50805160000361382c5760405162461bcd60e51b8152600401610c0d906148c9565b805181602001fd5b6001600160e01b031916630a85bd0160e11b149050612d3b565b506001949350505050565b6001600160a01b0382166138af5760405162461bcd60e51b815260206004820181905260248201527f4552433732313a206d696e7420746f20746865207a65726f20616464726573736044820152606401610c0d565b6138b881612ecd565b156139055760405162461bcd60e51b815260206004820152601c60248201527f4552433732313a20746f6b656e20616c7265616479206d696e746564000000006044820152606401610c0d565b6001600160a01b038216600090815260686020526040812080546001929061392e908490614710565b909155505060008181526067602052604080822080546001600160a01b0319166001600160a01b0386169081179091559051839290600080516020614976833981519152908290a46112a5565b6001600160e01b0319811681146112a757600080fd5b6000602082840312156139a357600080fd5b81356121698161397b565b600080600080600080600080610100898b0312156139cb57600080fd5b505086359860208801359850604088013597606081013597506080810135965060a0810135955060c0810135945060e0013592509050565b60005b83811015613a1e578181015183820152602001613a06565b83811115611f275750506000910152565b60008151808452613a47816020860160208601613a03565b601f01601f19169290920160200192915050565b6020815260006121696020830184613a2f565b600060208284031215613a8057600080fd5b5035919050565b6001600160a01b0391909116815260200190565b80356001600160a01b0381168114613ab257600080fd5b919050565b60008060408385031215613aca57600080fd5b613ad383613a9b565b946020939093013593505050565b634e487b7160e01b600052604160045260246000fd5b604051601f8201601f191681016001600160401b0381118282101715613b1f57613b1f613ae1565b604052919050565b60006001600160401b03821115613b4057613b40613ae1565b5060051b60200190565b60006020808385031215613b5d57600080fd5b82356001600160401b03811115613b7357600080fd5b8301601f81018513613b8457600080fd5b8035613b97613b9282613b27565b613af7565b81815260059190911b82018301908381019087831115613bb657600080fd5b928401925b82841015613bdb57613bcc84613a9b565b82529284019290840190613bbb565b979650505050505050565b600080600060608486031215613bfb57600080fd5b613c0484613a9b565b9250613c1260208501613a9b565b9150604084013590509250925092565b60008060408385031215613c3557600080fd5b50508035926020909101359150565b6001600160a01b03929092168252602082015260400190565b600060208284031215613c6f57600080fd5b81356004811061216957600080fd5b600080600060608486031215613c9357600080fd5b505081359360208301359350604090920135919050565b60008083601f840112613cbc57600080fd5b5081356001600160401b03811115613cd357600080fd5b6020830191508360208260051b8501011115610de957600080fd5b600080600080600060608688031215613d0657600080fd5b8535945060208601356001600160401b0380821115613d2457600080fd5b613d3089838a01613caa565b90965094506040880135915080821115613d4957600080fd5b50613d5688828901613caa565b969995985093965092949392505050565b634e487b7160e01b600052602160045260246000fd5b6020810160048310613d9f57634e487b7160e01b600052602160045260246000fd5b91905290565b60006001600160401b03821115613dbe57613dbe613ae1565b50601f01601f191660200190565b6000613dda613b9284613da5565b9050828152838383011115613dee57600080fd5b828260208301376000602084830101529392505050565b600082601f830112613e1657600080fd5b61216983833560208501613dcc565b600080600080600080600080610100898b031215613e4257600080fd5b8835975060208901356001600160401b0380821115613e6057600080fd5b613e6c8c838d01613e05565b985060408b0135915080821115613e8257600080fd5b613e8e8c838d01613e05565b975060608b0135965060808b0135915080821115613eab57600080fd5b613eb78c838d01613e05565b955060a08b0135945060c08b0135915080821115613ed457600080fd5b50613ee18b828c01613e05565b92505060e089013590509295985092959890939650565b608081526000613f0b6080830187613a2f565b8560208401528281036040840152613f238186613a2f565b91505082606083015295945050505050565b60008060008060408587031215613f4b57600080fd5b84356001600160401b0380821115613f6257600080fd5b613f6e88838901613caa565b90965094506020870135915080821115613f8757600080fd5b50613f9487828801613caa565b95989497509550505050565b600060208284031215613fb257600080fd5b61216982613a9b565b600080600060608486031215613fd057600080fd5b8335925060208401356001600160401b0380821115613fee57600080fd5b613ffa87838801613e05565b9350604086013591508082111561401057600080fd5b5061401d86828701613e05565b9150509250925092565b600080600080600060a0868803121561403f57600080fd5b61404886613a9b565b945061405660208701613a9b565b935060408601356001600160401b038082111561407257600080fd5b61407e89838a01613e05565b9450606088013591508082111561409457600080fd5b506140a188828901613e05565b95989497509295608001359392505050565b80151581146112a757600080fd5b600080604083850312156140d457600080fd5b6140dd83613a9b565b915060208301356140ed816140b3565b809150509250929050565b6000806000806080858703121561410e57600080fd5b61411785613a9b565b935061412560208601613a9b565b92506040850135915060608501356001600160401b0381111561414757600080fd5b8501601f8101871361415857600080fd5b61416787823560208401613dcc565b91505092959194509250565b600082601f83011261418457600080fd5b81356020614194613b9283613b27565b82815260059290921b840181019181810190868411156141b357600080fd5b8286015b848110156141f25780356001600160401b038111156141d65760008081fd5b6141e48986838b0101613e05565b8452509183019183016141b7565b509695505050505050565b600082601f83011261420e57600080fd5b8135602061421e613b9283613b27565b82815260059290921b8401810191818101908684111561423d57600080fd5b8286015b848110156141f25780358352918301918301614241565b600080600080600080600060e0888a03121561427357600080fd5b873596506020880135955060408801356001600160401b038082111561429857600080fd5b6142a48b838c01614173565b965060608a01359150808211156142ba57600080fd5b6142c68b838c01614173565b955060808a01359150808211156142dc57600080fd5b6142e88b838c016141fd565b945060a08a01359150808211156142fe57600080fd5b61430a8b838c01614173565b935060c08a013591508082111561432057600080fd5b5061432d8a828b016141fd565b91505092959891949750929550565b6000806040838503121561434f57600080fd5b61435883613a9b565b915061436660208401613a9b565b90509250929050565b6040815260006143826040830185613a2f565b90508260208301529392505050565b600181811c908216806143a557607f821691505b6020821081036143c557634e487b7160e01b600052602260045260246000fd5b50919050565b634e487b7160e01b600052601160045260246000fd5b6000828210156143f3576143f36143cb565b500390565b6020808252602e908201527f4552433732313a2063616c6c6572206973206e6f7420746f6b656e206f776e6560408201526d1c881b9bdc88185c1c1c9bdd995960921b606082015260800190565b6000816000190483118215151615614460576144606143cb565b500290565b60008261448257634e487b7160e01b600052601260045260246000fd5b500490565b634e487b7160e01b600052603260045260246000fd5b6000602082840312156144af57600080fd5b8135612169816140b3565b6000600182016144cc576144cc6143cb565b5060010190565b6020808252600890820152672737903a37b5b2b760c11b604082015260600190565b6020808252600c908201526b139bdd08185c1c1c9bdd995960a21b604082015260600190565b6020808252601c908201527f596f752063757272656e746c792063616e206e6f742072656465656d00000000604082015260600190565b60006020828403121561456457600080fd5b5051919050565b60006020828403121561457d57600080fd5b8151612169816140b3565b601f821115610cae57600081815260208120601f850160051c810160208610156145af5750805b601f850160051c820191505b81811015610fa8578281556001016145bb565b81516001600160401b038111156145e7576145e7613ae1565b6145fb816145f58454614391565b84614588565b602080601f83116001811461463057600084156146185750858301515b600019600386901b1c1916600185901b178555610fa8565b600085815260208120601f198616915b8281101561465f57888601518255948401946001909101908401614640565b508582101561467d5787850151600019600388901b60f8161c191681555b5050505050600190811b01905550565b602080825260189082015277115490cdcc8c4e881a5b9d985b1a59081d1bdad95b88125160421b604082015260600190565b6020808252600b908201526a57726f6e6720707269636560a81b604082015260600190565b602080825260129082015271088c2e8c240e6d2f4ca40dad2e6dac2e8c6d60731b604082015260600190565b60008219821115614723576147236143cb565b500190565b6000815461473581614391565b808552602060018381168015614752576001811461476c5761479a565b60ff1985168884015283151560051b88018301955061479a565b866000528260002060005b858110156147925781548a8201860152908301908401614777565b890184019650505b505050505092915050565b60c0815260006147b860c0830189613a2f565b82810360208401526147ca8189614728565b905082810360408401526147de8188614728565b905082810360608401526147f28187614728565b6080840195909552505060a00152949350505050565b60006020828403121561481a57600080fd5b81516001600160401b0381111561483057600080fd5b8201601f8101841361484157600080fd5b805161484f613b9282613da5565b81815285602083850101111561486457600080fd5b614875826020830160208601613a03565b95945050505050565b6020808252602b908201527f496e697469616c697a61626c653a20636f6e7472616374206973206e6f74206960408201526a6e697469616c697a696e6760a81b606082015260800190565b60208082526032908201527f4552433732313a207472616e7366657220746f206e6f6e20455243373231526560408201527131b2b4bb32b91034b6b83632b6b2b73a32b960711b606082015260800190565b6001600160a01b038581168252841660208201526040810183905260806060820181905260009061494e90830184613a2f565b9695505050505050565b60006020828403121561496a57600080fd5b81516121698161397b56feddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3efa6dc15bdb68da224c66db4b3838d9a2b205138e8cff6774e57d0af91e196d622a2646970667358221220dfce55129c649678a19b6e90e8b7492d350514a682c63fa65494fd6f2383136a64736f6c634300080f0033", - "deployedBytecode": "0x60806040526004361061032d5760003560e01c806369e1cac1116101a7578063b88d4fde116100ed578063db006a7511610090578063db006a7514610955578063de5d62a014610975578063e985e9c514610995578063ee5df550146109b5578063f2fde38b146109d5578063f51f96dd146109f5578063fc51e0c314610a0b578063fce57fd914610a3957600080fd5b8063b88d4fde14610873578063bcda01af14610893578063bf82ae49146108b3578063c87b56dd146108d3578063c9d359d6146108f3578063d375b6fc1461090d578063d41c3a6514610922578063d5344e361461094057600080fd5b80637b422c4c116101555780637b422c4c146107a15780638c503ca4146107c15780638da5cb5b146107e157806393d23923146107f657806395d89b4114610816578063a1279cfc1461082b578063a22cb46514610840578063a66ff0af1461086057600080fd5b806369e1cac1146106d75780636a326ab1146106f757806370a0823114610717578063715018a6146107375780637284e4161461074c578063787e5efc14610761578063799b7c5f1461078157600080fd5b8063392774151161027757806342966c681161021a57806342966c68146105f557806346d9c4bf14610615578063499e16721461062a57806352c9143e1461063f5780636352211e1461065f57806364e9293e1461067f57806364edfbf0146106af578063660243d0146106b757600080fd5b8063392774151461052257806339dc13bc146105405780633ccfd60b146105605780633da165d4146105755780633e1dc1071461058b578063402635cf146105a0578063410a1095146105b557806342842e0e146105d557600080fd5b80630f6a9349116102df5780630f6a93491461042c57806318160ddd1461043f5780631919fed71461045457806323b872dd146104745780632a55205a14610494578063367de39e146104c257806336f3ec76146104e25780633892aa521461050257600080fd5b806301ffc9a714610332578063036c8ea11461036757806306fdde0314610389578063081812fc146103ab578063095ea7b3146103d857806309949abe146103f85780630b65b6e714610417575b600080fd5b34801561033e57600080fd5b5061035261034d366004613991565b610a59565b60405190151581526020015b60405180910390f35b34801561037357600080fd5b506103876103823660046139ae565b610a84565b005b34801561039557600080fd5b5061039e610ae0565b60405161035e9190613a5b565b3480156103b757600080fd5b506103cb6103c6366004613a6e565b610b72565b60405161035e9190613a87565b3480156103e457600080fd5b506103876103f3366004613ab7565b610b99565b34801561040457600080fd5b5060d4545b60405190815260200161035e565b34801561042357600080fd5b50610409610cb3565b61040961043a366004613b4a565b610cca565b34801561044b57600080fd5b5060d254610409565b34801561046057600080fd5b5061038761046f366004613a6e565b610cd5565b34801561048057600080fd5b5061038761048f366004613be6565b610d62565b3480156104a057600080fd5b506104b46104af366004613c22565b610d94565b60405161035e929190613c44565b3480156104ce57600080fd5b506103876104dd366004613c5d565b610df0565b3480156104ee57600080fd5b506103876104fd366004613c7e565b610ecb565b34801561050e57600080fd5b5061038761051d366004613cee565b610f09565b34801561052e57600080fd5b5060ca546001600160a01b03166103cb565b34801561054c57600080fd5b5061038761055b366004613a6e565b610fb0565b34801561056c57600080fd5b5061038761108c565b34801561058157600080fd5b5061040960cc5481565b34801561059757600080fd5b5060d954610409565b3480156105ac57600080fd5b5060d554610409565b3480156105c157600080fd5b5060dd5460ff1660405161035e9190613d7d565b3480156105e157600080fd5b506103876105f0366004613be6565b6112aa565b34801561060157600080fd5b50610387610610366004613a6e565b6112c5565b34801561062157600080fd5b5060df54610409565b34801561063657600080fd5b5060d754610409565b34801561064b57600080fd5b5061038761065a366004613e25565b6112f3565b34801561066b57600080fd5b506103cb61067a366004613a6e565b611431565b34801561068b57600080fd5b5061069f61069a366004613a6e565b611466565b60405161035e9493929190613ef8565b61040961172b565b3480156106c357600080fd5b506103876106d2366004613c22565b6117d7565b3480156106e357600080fd5b506103876106f2366004613f35565b611aa0565b34801561070357600080fd5b50610387610712366004613fa0565b611b60565b34801561072357600080fd5b50610409610732366004613fa0565b611c55565b34801561074357600080fd5b50610387611cdb565b34801561075857600080fd5b5061039e611cef565b34801561076d57600080fd5b5061038761077c366004613a6e565b611d7d565b34801561078d57600080fd5b5061038761079c366004613cee565b611d9d565b3480156107ad57600080fd5b506103876107bc366004613a6e565b611e3c565b3480156107cd57600080fd5b506103876107dc366004613fbb565b611ee8565b3480156107ed57600080fd5b506103cb611f2d565b34801561080257600080fd5b50610387610811366004614027565b611f41565b34801561082257600080fd5b5061039e6120ed565b34801561083757600080fd5b5060d654610409565b34801561084c57600080fd5b5061038761085b3660046140c1565b6120fc565b61040961086e366004613fa0565b612107565b34801561087f57600080fd5b5061038761088e3660046140f8565b612170565b34801561089f57600080fd5b506103876108ae366004614258565b6121a1565b3480156108bf57600080fd5b506103876108ce366004613a6e565b6123b4565b3480156108df57600080fd5b5061039e6108ee366004613a6e565b61249a565b3480156108ff57600080fd5b5060cc5460cd541015610352565b34801561091957600080fd5b50610409612628565b34801561092e57600080fd5b5060e0546001600160a01b03166103cb565b34801561094c57600080fd5b5060d854610409565b34801561096157600080fd5b50610387610970366004613a6e565b612642565b34801561098157600080fd5b50610387610990366004613a6e565b61270f565b3480156109a157600080fd5b506103526109b036600461433c565b612730565b3480156109c157600080fd5b506103876109d0366004613c22565b61275e565b3480156109e157600080fd5b506103876109f0366004613fa0565b612842565b348015610a0157600080fd5b5061040960df5481565b348015610a1757600080fd5b50610a2b610a26366004613a6e565b6128b8565b60405161035e92919061436f565b348015610a4557600080fd5b50610387610a54366004613fa0565b61296b565b600063152a902d60e11b6001600160e01b031983161480610a7e5750610a7e82612995565b92915050565b610a8c6129e5565b60d488905560d587905560d686905560d785905560df84905560d883905560d982905560da819055604051848152600080516020614996833981519152906020015b60405180910390a15050505050505050565b606060658054610aef90614391565b80601f0160208091040260200160405190810160405280929190818152602001828054610b1b90614391565b8015610b685780601f10610b3d57610100808354040283529160200191610b68565b820191906000526020600020905b815481529060010190602001808311610b4b57829003601f168201915b5050505050905090565b6000610b7d82612a44565b506000908152606960205260409020546001600160a01b031690565b6000610ba482611431565b9050806001600160a01b0316836001600160a01b031603610c165760405162461bcd60e51b815260206004820152602160248201527f4552433732313a20617070726f76616c20746f2063757272656e74206f776e656044820152603960f91b60648201526084015b60405180910390fd5b336001600160a01b0382161480610c325750610c328133612730565b610ca45760405162461bcd60e51b815260206004820152603e60248201527f4552433732313a20617070726f76652063616c6c6572206973206e6f7420746f60448201527f6b656e206f776e6572206e6f7220617070726f76656420666f7220616c6c00006064820152608401610c0d565b610cae8383612a69565b505050565b600060d25460cc54610cc591906143e1565b905090565b6000610a7e82612ad7565b610cdd6129e5565b60df81905560dd80546003919060ff19166001835b021790555060dd546040517f40cb75607c032a2aac1c3565feb3b70e354cbd56e8f8fa36773cba5750464c5191610d2e9160ff90911690613d7d565b60405180910390a160008051602061499683398151915260df54604051610d5791815260200190565b60405180910390a150565b610d6d335b82612ce4565b610d895760405162461bcd60e51b8152600401610c0d906143f8565b610cae838383612d43565b60008080610da0611f2d565b6001600160a01b031603610dc157610db6611f2d565b600091509150610de9565b610dc9611f2d565b60d45461271090610dda9086614446565b610de49190614465565b915091505b9250929050565b610df86129e5565b6000816003811115610e0c57610e0c613d67565b10158015610e2c57506003816003811115610e2957610e29613d67565b11155b610e785760405162461bcd60e51b815260206004820152601f60248201527f4e6565647320746f20626520612076616c6964206d696e7465722074797065006044820152606401610c0d565b60dd805482919060ff19166001836003811115610e9757610e97613d67565b02179055507f40cb75607c032a2aac1c3565feb3b70e354cbd56e8f8fa36773cba5750464c5181604051610d579190613d7d565b610ed36129e5565b60d683905560d782905560df819055604051818152600080516020614996833981519152906020015b60405180910390a1505050565b610f116129e5565b60005b85811015610fa857828282818110610f2e57610f2e614487565b9050602002016020810190610f43919061449d565b60dc6000878785818110610f5957610f59614487565b9050602002016020810190610f6e9190613fa0565b6001600160a01b031681526020810191909152604001600020805460ff191691151591909117905580610fa0816144ba565b915050610f14565b505050505050565b610fb981612ecd565b610fd55760405162461bcd60e51b8152600401610c0d906144d3565b610fde33610d67565b610ffa5760405162461bcd60e51b8152600401610c0d906144f5565b6003600082815260cb602052604090205460ff16600681111561101f5761101f613d67565b1461103c5760405162461bcd60e51b8152600401610c0d9061451b565b600081815260cb6020526040902080546001919060ff191682805b02179055506040518181527facb9cd98d41dc8854281172ab2a764d949b62b51776edd911f85376ed2eca64b90602001610d57565b6110946129e5565b4780156110f35760d554600090612710906110af9084614446565b6110b99190614465565b905060006110c782846143e1565b90506110da6110d4611f2d565b83612eea565b60ca546110f0906001600160a01b031682612eea565b50505b60e0546001600160a01b0316156112a75760e0546040516370a0823160e01b81526000916001600160a01b0316906370a0823190611135903090600401613a87565b602060405180830381865afa158015611152573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906111769190614552565b905080156112a55760d554600090612710906111929084614446565b61119c9190614465565b905060006111aa82846143e1565b60e0549091506001600160a01b031663a9059cbb6111c6611f2d565b846040518363ffffffff1660e01b81526004016111e4929190613c44565b6020604051808303816000875af1158015611203573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611227919061456b565b5060e05460ca5460405163a9059cbb60e01b81526001600160a01b039283169263a9059cbb9261125e929116908590600401613c44565b6020604051808303816000875af115801561127d573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906112a1919061456b565b5050505b505b50565b610cae83838360405180602001604052806000815250612170565b6112ce33610d67565b6112ea5760405162461bcd60e51b8152600401610c0d906144f5565b6112a781613003565b6112fb6129e5565b61130488612ecd565b6113205760405162461bcd60e51b8152600401610c0d906144d3565b6004600089815260cb602052604090205460ff16600681111561134557611345613d67565b146113625760405162461bcd60e51b8152600401610c0d9061451b565b600088815260cb6020526040902060020161137d88826145ce565b50600088815260cb6020526040902060070161139987826145ce565b50600088815260cb60205260409020600881018690556009016113bc85826145ce565b50600088815260cb60205260409020600a8101849055600b016113df83826145ce565b50600088815260cb6020908152604091829020600c8101849055805460ff1916600517905590518981527fadb6d4f651f6fbf8b37026019ee48fdb86abf9ec94eb4a5f0acf72ad8c7872a39101610ace565b6000818152606760205260408120546001600160a01b031680610a7e5760405162461bcd60e51b8152600401610c0d9061468d565b6060600081816006600086815260cb602052604090205460ff16600681111561149157611491613d67565b036115df57600085815260cb60205260409020600a81015460088201546009830180549093600701919084906114c690614391565b80601f01602080910402602001604051908101604052809291908181526020018280546114f290614391565b801561153f5780601f106115145761010080835404028352916020019161153f565b820191906000526020600020905b81548152906001019060200180831161152257829003601f168201915b5050505050935081805461155290614391565b80601f016020809104026020016040519081016040528092919081815260200182805461157e90614391565b80156115cb5780601f106115a0576101008083540402835291602001916115cb565b820191906000526020600020905b8154815290600101906020018083116115ae57829003601f168201915b505050505091509350935093509350611724565b600085815260cb602052604090206006810154600482015460058301805490936003019190849061160f90614391565b80601f016020809104026020016040519081016040528092919081815260200182805461163b90614391565b80156116885780601f1061165d57610100808354040283529160200191611688565b820191906000526020600020905b81548152906001019060200180831161166b57829003601f168201915b5050505050935081805461169b90614391565b80601f01602080910402602001604051908101604052809291908181526020018280546116c790614391565b80156117145780601f106116e957610100808354040283529160200191611714565b820191906000526020600020905b8154815290600101906020018083116116f757829003601f168201915b5050505050915093509350935093505b9193509193565b60008061173661308d565b604080518281523360208201529192507f60a6c75698fadb72223808131f9f9bb9db3afa32122db6d94fb8fc985a504baa910160405180910390a16040805160018082528183019092526000916020808301908036833701905050905033816000815181106117a7576117a7614487565b60200260200101906001600160a01b031690816001600160a01b0316815250506117d081612ad7565b9250505090565b6117e082612ecd565b6117fc5760405162461bcd60e51b8152600401610c0d906144d3565b611807335b83612ce4565b6118235760405162461bcd60e51b8152600401610c0d906144f5565b6003600083815260cb602052604090205460ff16600681111561184857611848613d67565b146118655760405162461bcd60e51b8152600401610c0d9061451b565b600082815260cb60205260409020600101548110156118965760405162461bcd60e51b8152600401610c0d906146bf565b600082815260cb602052604090206001015460e0546001600160a01b031663dd62ed3e336040516001600160e01b031960e084901b1681526001600160a01b039091166004820152306024820152604401602060405180830381865afa158015611904573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906119289190614552565b101561196f5760405162461bcd60e51b8152602060048201526016602482015275496e73756666696369656e7420616c6c6f77616e636560501b6044820152606401610c0d565b60e0546000906001600160a01b03166323b872dd33600086815260cb60205260409081902060010154905160e084901b6001600160e01b03191681526001600160a01b03909216600483015230602483015260448201526064016020604051808303816000875af11580156119e8573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611a0c919061456b565b905080611a565760405162461bcd60e51b815260206004820152601860248201527721b7bab632103737ba103a3930b739b332b9103a37b5b2b760411b6044820152606401610c0d565b600083815260cb6020908152604091829020805460ff1916600417905590518481527f9bd6b4fd288008520fd788a93304e5688a401aea817ea8140ecf1fb8648f31919101610efc565b611aa86129e5565b60005b838110156112a157848482818110611ac557611ac5614487565b9050602002016020810190611ada9190613fa0565b60ce54600090815260cf6020526040902080546001600160a01b0319166001600160a01b0392909216919091179055828282818110611b1b57611b1b614487565b60ce8054600090815260d060209081526040822093029490940135909155805492909150611b48836144ba565b91905055508080611b58906144ba565b915050611aab565b611b686129e5565b60e0546001600160a01b031615611c335760e0546040516370a0823160e01b81526001600160a01b03909116906370a0823190611ba9903090600401613a87565b602060405180830381865afa158015611bc6573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611bea9190614552565b15611c335760405162461bcd60e51b8152602060048201526019602482015278746f6b656e206d757374206861766520302062616c616e636560381b6044820152606401610c0d565b60e080546001600160a01b0319166001600160a01b0392909216919091179055565b60006001600160a01b038216611cbf5760405162461bcd60e51b815260206004820152602960248201527f4552433732313a2061646472657373207a65726f206973206e6f7420612076616044820152683634b21037bbb732b960b91b6064820152608401610c0d565b506001600160a01b031660009081526068602052604090205490565b611ce36129e5565b611ced6000613104565b565b60c98054611cfc90614391565b80601f0160208091040260200160405190810160405280929190818152602001828054611d2890614391565b8015611d755780601f10611d4a57610100808354040283529160200191611d75565b820191906000526020600020905b815481529060010190602001808311611d5857829003601f168201915b505050505081565b611d856129e5565b60d681905560dd80546001919060ff19168280610cf2565b611da56129e5565b60005b85811015610fa857828282818110611dc257611dc2614487565b9050602002016020810190611dd7919061449d565b60db6000878785818110611ded57611ded614487565b9050602002016020810190611e029190613fa0565b6001600160a01b031681526020810191909152604001600020805460ff191691151591909117905580611e34816144ba565b915050611da8565b611e4581612ecd565b611e615760405162461bcd60e51b8152600401610c0d906144d3565b611e6a33610d67565b611e865760405162461bcd60e51b8152600401610c0d906144f5565b6005600082815260cb602052604090205460ff166006811115611eab57611eab613d67565b14611ec85760405162461bcd60e51b8152600401610c0d9061451b565b600081815260cb6020526040902080546006919060ff1916600183611057565b611ef06129e5565b600083815260cb60205260409020600501611f0b83826145ce565b50600083815260cb60205260409020600301611f2782826145ce565b50505050565b6000610cc56097546001600160a01b031690565b600054610100900460ff1615808015611f615750600054600160ff909116105b80611f7b5750303b158015611f7b575060005460ff166001145b611fde5760405162461bcd60e51b815260206004820152602e60248201527f496e697469616c697a61626c653a20636f6e747261637420697320616c72656160448201526d191e481a5b9a5d1a585b1a5e995960921b6064820152608401610c0d565b6000805460ff191660011790558015612001576000805461ff0019166101001790555b600082116120495760405162461bcd60e51b8152602060048201526015602482015274044726f702073697a65206d757374206265203e203605c1b6044820152606401610c0d565b6120538484613156565b61205b613187565b61206486612842565b60ca80546001600160a01b0319166001600160a01b03871617905560cc829055600060d255600160d35560c961209a85826145ce565b50600060cd558015610fa8576000805461ff0019169055604051600181527f7f26b83ff96e1f2b6a682f133852f6798a09c465da95921460cefb38474024989060200160405180910390a1505050505050565b606060668054610aef90614391565b6112a53383836131b6565b60408051600180825281830190925260009182919060208083019080368337019050509050828160008151811061214057612140614487565b60200260200101906001600160a01b031690816001600160a01b03168152505061216981612ad7565b9392505050565b61217933611801565b6121955760405162461bcd60e51b8152600401610c0d906143f8565b611f2784848484613280565b858551146121c15760405162461bcd60e51b8152600401610c0d906146e4565b858251146121e15760405162461bcd60e51b8152600401610c0d906146e4565b858151146122015760405162461bcd60e51b8152600401610c0d906146e4565b858451146122215760405162461bcd60e51b8152600401610c0d906146e4565b858351146122415760405162461bcd60e51b8152600401610c0d906146e4565b60005b86811015612393576000612258828a614710565b612263906001614710565b905086828151811061227757612277614487565b602002602001015160cb600083815260200190815260200160002060020190816122a191906145ce565b508382815181106122b4576122b4614487565b602002602001015160cb600083815260200190815260200160002060030190816122de91906145ce565b508282815181106122f1576122f1614487565b602002602001015160cb60008381526020019081526020016000206004018190555085828151811061232557612325614487565b602002602001015160cb6000838152602001908152602001600020600501908161234f91906145ce565b5084828151811061236257612362614487565b602090810291909101810151600092835260cb9091526040909120600601558061238b816144ba565b915050612244565b508560cd60008282546123a69190614710565b909155505050505050505050565b6123bd81612ecd565b6123d95760405162461bcd60e51b8152600401610c0d906144d3565b6123e233610d67565b6123fe5760405162461bcd60e51b8152600401610c0d906144f5565b6002600082815260cb602052604090205460ff16600681111561242357612423613d67565b146124405760405162461bcd60e51b8152600401610c0d9061451b565b600081815260cb60205260409020805460ff191660011790557f6c7c9f699aacfa5bacd2758023ab209b2d1e1c237bfa752bdc94cb87878cf1b781335b604080519283526001600160a01b03909116602083015201610d57565b60606124a582612ecd565b6124c15760405162461bcd60e51b8152600401610c0d906144d3565b6006600083815260cb602052604090205460ff1660068111156124e6576124e6613d67565b036125ac577f00000000000000000000000000000000000000000000000000000000000000006001600160a01b031663df30dba0612522610ae0565b600085815260cb60205260409081902060cc5491516001600160e01b031960e086901b1681526125679392600283019260098101926007909101918a916004016147a5565b600060405180830381865afa158015612584573d6000803e3d6000fd5b505050506040513d6000823e601f3d908101601f19168201604052610a7e9190810190614808565b7f00000000000000000000000000000000000000000000000000000000000000006001600160a01b031663df30dba06125e3610ae0565b600085815260cb60205260409081902060cc5491516001600160e01b031960e086901b1681526125679392600283019260058101926003909101918a916004016147a5565b60dd5460009060ff166003811115610cc557610cc5613d67565b61264b81612ecd565b6126675760405162461bcd60e51b8152600401610c0d906144d3565b61267033610d67565b61268c5760405162461bcd60e51b8152600401610c0d906144f5565b6001600082815260cb602052604090205460ff1660068111156126b1576126b1613d67565b146126ce5760405162461bcd60e51b8152600401610c0d9061451b565b600081815260cb60205260409020805460ff191660021790557fc290e40808ef06721539ad0d926f8d5f53d2dd6d5e28334308a012676f867416813361247d565b6127176129e5565b60d781905560dd80546002919060ff1916600183610cf2565b6001600160a01b039182166000908152606a6020908152604080832093909416825291909152205460ff1690565b6127666129e5565b61276f82612ecd565b61278b5760405162461bcd60e51b8152600401610c0d906144d3565b6002600083815260cb602052604090205460ff1660068111156127b0576127b0613d67565b146127eb5760405162461bcd60e51b815260206004820152600b60248201526a57726f6e6720737461746560a81b6044820152606401610c0d565b600082815260cb6020908152604091829020805460ff1916600317815560010183905590518381527f316b408b8cf323e8f20ef7a8c267eedc89bef19ca5db7fc30be938e55bf068f9910160405180910390a15050565b61284a6129e5565b6001600160a01b0381166128af5760405162461bcd60e51b815260206004820152602660248201527f4f776e61626c653a206e6577206f776e657220697320746865207a65726f206160448201526564647265737360d01b6064820152608401610c0d565b6112a781613104565b600081815260cb60205260408120600c810154600b9091018054606093929082906128e290614391565b80601f016020809104026020016040519081016040528092919081815260200182805461290e90614391565b801561295b5780601f106129305761010080835404028352916020019161295b565b820191906000526020600020905b81548152906001019060200180831161293e57829003601f168201915b5050505050915091509150915091565b6129736129e5565b60ca80546001600160a01b0319166001600160a01b0392909216919091179055565b60006001600160e01b031982166380ac58cd60e01b14806129c657506001600160e01b03198216635b5e139f60e01b145b80610a7e57506301ffc9a760e01b6001600160e01b0319831614610a7e565b336129ee611f2d565b6001600160a01b031614611ced5760405162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e65726044820152606401610c0d565b612a4d81612ecd565b6112a75760405162461bcd60e51b8152600401610c0d9061468d565b600081815260696020526040902080546001600160a01b0319166001600160a01b0384169081179091558190612a9e82611431565b6001600160a01b03167f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b92560405160405180910390a45050565b600060cc5460cd541015612b275760405162461bcd60e51b8152602060048201526017602482015276139bdd08185b1b081b595d1859185d18481b1bd8591959604a1b6044820152606401610c0d565b612b2f6132b3565b612b7b5760405162461bcd60e51b815260206004820152601d60248201527f4e6565647320746f20626520616e20616c6c6f776564206d696e7465720000006044820152606401610c0d565b6000612b8561308d565b905060008111612bc65760405162461bcd60e51b815260206004820152600c60248201526b4e6f7420666f722073616c6560a01b6044820152606401610c0d565b8251612bd29082614446565b3414612bf05760405162461bcd60e51b8152600401610c0d906146bf565b612bf861338c565b835133600090815260de6020526040902054600191612c1691614710565b612c2091906143e1565b10612c635760405162461bcd60e51b8152602060048201526013602482015272115e18d959591959081b5a5b9d081b1a5b5a5d606a1b6044820152606401610c0d565b60cc54835160d254612c759190614710565b1115612cb45760405162461bcd60e51b815260206004820152600e60248201526d4f7665722064726f702073697a6560901b6044820152606401610c0d565b600160dd5460ff166003811115612ccd57612ccd613d67565b03612cdb57612169836133fd565b612169836135ed565b600080612cf083611431565b9050806001600160a01b0316846001600160a01b03161480612d175750612d178185612730565b80612d3b5750836001600160a01b0316612d3084610b72565b6001600160a01b0316145b949350505050565b826001600160a01b0316612d5682611431565b6001600160a01b031614612dba5760405162461bcd60e51b815260206004820152602560248201527f4552433732313a207472616e736665722066726f6d20696e636f72726563742060448201526437bbb732b960d91b6064820152608401610c0d565b6001600160a01b038216612e1c5760405162461bcd60e51b8152602060048201526024808201527f4552433732313a207472616e7366657220746f20746865207a65726f206164646044820152637265737360e01b6064820152608401610c0d565b612e27600082612a69565b6001600160a01b0383166000908152606860205260408120805460019290612e509084906143e1565b90915550506001600160a01b0382166000908152606860205260408120805460019290612e7e908490614710565b909155505060008181526067602052604080822080546001600160a01b0319166001600160a01b03868116918217909255915184939187169160008051602061497683398151915291a4505050565b6000908152606760205260409020546001600160a01b0316151590565b80471015612f3a5760405162461bcd60e51b815260206004820152601d60248201527f416464726573733a20696e73756666696369656e742062616c616e63650000006044820152606401610c0d565b6000826001600160a01b03168260405160006040518083038185875af1925050503d8060008114612f87576040519150601f19603f3d011682016040523d82523d6000602084013e612f8c565b606091505b5050905080610cae5760405162461bcd60e51b815260206004820152603a60248201527f416464726573733a20756e61626c6520746f2073656e642076616c75652c207260448201527f6563697069656e74206d617920686176652072657665727465640000000000006064820152608401610c0d565b600061300e82611431565b905061301b600083612a69565b6001600160a01b03811660009081526068602052604081208054600192906130449084906143e1565b909155505060008281526067602052604080822080546001600160a01b0319169055518391906001600160a01b03841690600080516020614976833981519152908390a46112a5565b6000600160dd5460ff1660038111156130a8576130a8613d67565b036130b4575060d65490565b600260dd5460ff1660038111156130cd576130cd613d67565b036130d9575060d75490565b600360dd5460ff1660038111156130f2576130f2613d67565b036130fe575060df5490565b50600090565b609780546001600160a01b038381166001600160a01b0319831681179093556040519116919082907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e090600090a35050565b600054610100900460ff1661317d5760405162461bcd60e51b8152600401610c0d9061487e565b6112a582826136e8565b600054610100900460ff166131ae5760405162461bcd60e51b8152600401610c0d9061487e565b611ced613728565b816001600160a01b0316836001600160a01b0316036132135760405162461bcd60e51b815260206004820152601960248201527822a9219b99189d1030b8383937bb32903a379031b0b63632b960391b6044820152606401610c0d565b6001600160a01b038381166000818152606a6020908152604080832094871680845294825291829020805460ff191686151590811790915591519182527f17307eab39ab6107e8899845ad3d59bd9653f200f220920489ca2b5937696c31910160405180910390a3505050565b61328b848484612d43565b61329784848484613758565b611f275760405162461bcd60e51b8152600401610c0d906148c9565b6000600360dd5460ff1660038111156132ce576132ce613d67565b036132d95750600190565b600260dd5460ff1660038111156132f2576132f2613d67565b036133335733600090815260dc602052604090205460ff16156133155750600190565b33600090815260db602052604090205460ff16156133335750600190565b600160dd5460ff16600381111561334c5761334c613d67565b0361336f5733600090815260dc602052604090205460ff161561336f5750600190565b33613378611f2d565b6001600160a01b0316036130fe5750600190565b6000600160dd5460ff1660038111156133a7576133a7613d67565b036133b3575060d85490565b600260dd5460ff1660038111156133cc576133cc613d67565b036133d8575060d95490565b600360dd5460ff1660038111156133f1576133f1613d67565b036130fe575060da5490565b60ce5460009033908290815b60ce5481101561348c57600081815260cf60205260409020546001600160a01b0380861691160361347a57600081815260d0602090815260408083205480845260d19092529091205460ff161515600114613478578282101561346a578192505b83613474816144ba565b9450505b505b80613484816144ba565b915050613409565b5084518210156134da5760405162461bcd60e51b815260206004820152601960248201527843616e206e6f74206d696e7420616c6c2065646974696f6e7360381b6044820152606401610c0d565b60018160005b87518110156135e1575b600082815260cf60205260409020546001600160a01b0387811691161461351d5781613515816144ba565b9250506134ea565b60d0600083815260200190815260200160002054925061355688828151811061354857613548614487565b602002602001015184613859565b600083815260cb602090815260408083208054600160ff19918216811790925560d18452828520805490911690911790556001600160a01b038916835260de90915281208054916135a6836144ba565b909155505060d280549060006135bb836144ba565b919050555081806135cb906144ba565b92505080806135d9906144ba565b9150506134e0565b50909695505050505050565b600033815b83518110156136dd575b60d354600090815260d1602052604090205460ff1615156001036136345760d3805490600061362a836144ba565b91905055506135fc565b61365984828151811061364957613649614487565b602002602001015160d354613859565b60d38054600090815260cb602090815260408083208054600160ff1991821681179092559454845260d183528184208054909516179093556001600160a01b038516825260de90529081208054916136b0836144ba565b909155505060d280549060006136c5836144ba565b919050555080806136d5906144ba565b9150506135f2565b505060d35492915050565b600054610100900460ff1661370f5760405162461bcd60e51b8152600401610c0d9061487e565b606561371b83826145ce565b506066610cae82826145ce565b600054610100900460ff1661374f5760405162461bcd60e51b8152600401610c0d9061487e565b611ced33613104565b60006001600160a01b0384163b1561384e57604051630a85bd0160e11b81526001600160a01b0385169063150b7a029061379c90339089908890889060040161491b565b6020604051808303816000875af19250505080156137d7575060408051601f3d908101601f191682019092526137d491810190614958565b60015b613834573d808015613805576040519150601f19603f3d011682016040523d82523d6000602084013e61380a565b606091505b50805160000361382c5760405162461bcd60e51b8152600401610c0d906148c9565b805181602001fd5b6001600160e01b031916630a85bd0160e11b149050612d3b565b506001949350505050565b6001600160a01b0382166138af5760405162461bcd60e51b815260206004820181905260248201527f4552433732313a206d696e7420746f20746865207a65726f20616464726573736044820152606401610c0d565b6138b881612ecd565b156139055760405162461bcd60e51b815260206004820152601c60248201527f4552433732313a20746f6b656e20616c7265616479206d696e746564000000006044820152606401610c0d565b6001600160a01b038216600090815260686020526040812080546001929061392e908490614710565b909155505060008181526067602052604080822080546001600160a01b0319166001600160a01b0386169081179091559051839290600080516020614976833981519152908290a46112a5565b6001600160e01b0319811681146112a757600080fd5b6000602082840312156139a357600080fd5b81356121698161397b565b600080600080600080600080610100898b0312156139cb57600080fd5b505086359860208801359850604088013597606081013597506080810135965060a0810135955060c0810135945060e0013592509050565b60005b83811015613a1e578181015183820152602001613a06565b83811115611f275750506000910152565b60008151808452613a47816020860160208601613a03565b601f01601f19169290920160200192915050565b6020815260006121696020830184613a2f565b600060208284031215613a8057600080fd5b5035919050565b6001600160a01b0391909116815260200190565b80356001600160a01b0381168114613ab257600080fd5b919050565b60008060408385031215613aca57600080fd5b613ad383613a9b565b946020939093013593505050565b634e487b7160e01b600052604160045260246000fd5b604051601f8201601f191681016001600160401b0381118282101715613b1f57613b1f613ae1565b604052919050565b60006001600160401b03821115613b4057613b40613ae1565b5060051b60200190565b60006020808385031215613b5d57600080fd5b82356001600160401b03811115613b7357600080fd5b8301601f81018513613b8457600080fd5b8035613b97613b9282613b27565b613af7565b81815260059190911b82018301908381019087831115613bb657600080fd5b928401925b82841015613bdb57613bcc84613a9b565b82529284019290840190613bbb565b979650505050505050565b600080600060608486031215613bfb57600080fd5b613c0484613a9b565b9250613c1260208501613a9b565b9150604084013590509250925092565b60008060408385031215613c3557600080fd5b50508035926020909101359150565b6001600160a01b03929092168252602082015260400190565b600060208284031215613c6f57600080fd5b81356004811061216957600080fd5b600080600060608486031215613c9357600080fd5b505081359360208301359350604090920135919050565b60008083601f840112613cbc57600080fd5b5081356001600160401b03811115613cd357600080fd5b6020830191508360208260051b8501011115610de957600080fd5b600080600080600060608688031215613d0657600080fd5b8535945060208601356001600160401b0380821115613d2457600080fd5b613d3089838a01613caa565b90965094506040880135915080821115613d4957600080fd5b50613d5688828901613caa565b969995985093965092949392505050565b634e487b7160e01b600052602160045260246000fd5b6020810160048310613d9f57634e487b7160e01b600052602160045260246000fd5b91905290565b60006001600160401b03821115613dbe57613dbe613ae1565b50601f01601f191660200190565b6000613dda613b9284613da5565b9050828152838383011115613dee57600080fd5b828260208301376000602084830101529392505050565b600082601f830112613e1657600080fd5b61216983833560208501613dcc565b600080600080600080600080610100898b031215613e4257600080fd5b8835975060208901356001600160401b0380821115613e6057600080fd5b613e6c8c838d01613e05565b985060408b0135915080821115613e8257600080fd5b613e8e8c838d01613e05565b975060608b0135965060808b0135915080821115613eab57600080fd5b613eb78c838d01613e05565b955060a08b0135945060c08b0135915080821115613ed457600080fd5b50613ee18b828c01613e05565b92505060e089013590509295985092959890939650565b608081526000613f0b6080830187613a2f565b8560208401528281036040840152613f238186613a2f565b91505082606083015295945050505050565b60008060008060408587031215613f4b57600080fd5b84356001600160401b0380821115613f6257600080fd5b613f6e88838901613caa565b90965094506020870135915080821115613f8757600080fd5b50613f9487828801613caa565b95989497509550505050565b600060208284031215613fb257600080fd5b61216982613a9b565b600080600060608486031215613fd057600080fd5b8335925060208401356001600160401b0380821115613fee57600080fd5b613ffa87838801613e05565b9350604086013591508082111561401057600080fd5b5061401d86828701613e05565b9150509250925092565b600080600080600060a0868803121561403f57600080fd5b61404886613a9b565b945061405660208701613a9b565b935060408601356001600160401b038082111561407257600080fd5b61407e89838a01613e05565b9450606088013591508082111561409457600080fd5b506140a188828901613e05565b95989497509295608001359392505050565b80151581146112a757600080fd5b600080604083850312156140d457600080fd5b6140dd83613a9b565b915060208301356140ed816140b3565b809150509250929050565b6000806000806080858703121561410e57600080fd5b61411785613a9b565b935061412560208601613a9b565b92506040850135915060608501356001600160401b0381111561414757600080fd5b8501601f8101871361415857600080fd5b61416787823560208401613dcc565b91505092959194509250565b600082601f83011261418457600080fd5b81356020614194613b9283613b27565b82815260059290921b840181019181810190868411156141b357600080fd5b8286015b848110156141f25780356001600160401b038111156141d65760008081fd5b6141e48986838b0101613e05565b8452509183019183016141b7565b509695505050505050565b600082601f83011261420e57600080fd5b8135602061421e613b9283613b27565b82815260059290921b8401810191818101908684111561423d57600080fd5b8286015b848110156141f25780358352918301918301614241565b600080600080600080600060e0888a03121561427357600080fd5b873596506020880135955060408801356001600160401b038082111561429857600080fd5b6142a48b838c01614173565b965060608a01359150808211156142ba57600080fd5b6142c68b838c01614173565b955060808a01359150808211156142dc57600080fd5b6142e88b838c016141fd565b945060a08a01359150808211156142fe57600080fd5b61430a8b838c01614173565b935060c08a013591508082111561432057600080fd5b5061432d8a828b016141fd565b91505092959891949750929550565b6000806040838503121561434f57600080fd5b61435883613a9b565b915061436660208401613a9b565b90509250929050565b6040815260006143826040830185613a2f565b90508260208301529392505050565b600181811c908216806143a557607f821691505b6020821081036143c557634e487b7160e01b600052602260045260246000fd5b50919050565b634e487b7160e01b600052601160045260246000fd5b6000828210156143f3576143f36143cb565b500390565b6020808252602e908201527f4552433732313a2063616c6c6572206973206e6f7420746f6b656e206f776e6560408201526d1c881b9bdc88185c1c1c9bdd995960921b606082015260800190565b6000816000190483118215151615614460576144606143cb565b500290565b60008261448257634e487b7160e01b600052601260045260246000fd5b500490565b634e487b7160e01b600052603260045260246000fd5b6000602082840312156144af57600080fd5b8135612169816140b3565b6000600182016144cc576144cc6143cb565b5060010190565b6020808252600890820152672737903a37b5b2b760c11b604082015260600190565b6020808252600c908201526b139bdd08185c1c1c9bdd995960a21b604082015260600190565b6020808252601c908201527f596f752063757272656e746c792063616e206e6f742072656465656d00000000604082015260600190565b60006020828403121561456457600080fd5b5051919050565b60006020828403121561457d57600080fd5b8151612169816140b3565b601f821115610cae57600081815260208120601f850160051c810160208610156145af5750805b601f850160051c820191505b81811015610fa8578281556001016145bb565b81516001600160401b038111156145e7576145e7613ae1565b6145fb816145f58454614391565b84614588565b602080601f83116001811461463057600084156146185750858301515b600019600386901b1c1916600185901b178555610fa8565b600085815260208120601f198616915b8281101561465f57888601518255948401946001909101908401614640565b508582101561467d5787850151600019600388901b60f8161c191681555b5050505050600190811b01905550565b602080825260189082015277115490cdcc8c4e881a5b9d985b1a59081d1bdad95b88125160421b604082015260600190565b6020808252600b908201526a57726f6e6720707269636560a81b604082015260600190565b602080825260129082015271088c2e8c240e6d2f4ca40dad2e6dac2e8c6d60731b604082015260600190565b60008219821115614723576147236143cb565b500190565b6000815461473581614391565b808552602060018381168015614752576001811461476c5761479a565b60ff1985168884015283151560051b88018301955061479a565b866000528260002060005b858110156147925781548a8201860152908301908401614777565b890184019650505b505050505092915050565b60c0815260006147b860c0830189613a2f565b82810360208401526147ca8189614728565b905082810360408401526147de8188614728565b905082810360608401526147f28187614728565b6080840195909552505060a00152949350505050565b60006020828403121561481a57600080fd5b81516001600160401b0381111561483057600080fd5b8201601f8101841361484157600080fd5b805161484f613b9282613da5565b81815285602083850101111561486457600080fd5b614875826020830160208601613a03565b95945050505050565b6020808252602b908201527f496e697469616c697a61626c653a20636f6e7472616374206973206e6f74206960408201526a6e697469616c697a696e6760a81b606082015260800190565b60208082526032908201527f4552433732313a207472616e7366657220746f206e6f6e20455243373231526560408201527131b2b4bb32b91034b6b83632b6b2b73a32b960711b606082015260800190565b6001600160a01b038581168252841660208201526040810183905260806060820181905260009061494e90830184613a2f565b9695505050505050565b60006020828403121561496a57600080fd5b81516121698161397b56feddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3efa6dc15bdb68da224c66db4b3838d9a2b205138e8cff6774e57d0af91e196d622a2646970667358221220dfce55129c649678a19b6e90e8b7492d350514a682c63fa65494fd6f2383136a64736f6c634300080f0033", + "solcInputHash": "e6a2ffa4bb274580347e73637376a7e0", + "metadata": "{\"compiler\":{\"version\":\"0.8.15+commit.e14f2714\"},\"language\":\"Solidity\",\"output\":{\"abi\":[{\"inputs\":[{\"internalType\":\"contract SharedNFTLogic\",\"name\":\"sharedNFTLogic\",\"type\":\"address\"}],\"stateMutability\":\"nonpayable\",\"type\":\"constructor\"},{\"anonymous\":false,\"inputs\":[{\"indexed\":true,\"internalType\":\"address\",\"name\":\"owner\",\"type\":\"address\"},{\"indexed\":true,\"internalType\":\"address\",\"name\":\"approved\",\"type\":\"address\"},{\"indexed\":true,\"internalType\":\"uint256\",\"name\":\"tokenId\",\"type\":\"uint256\"}],\"name\":\"Approval\",\"type\":\"event\"},{\"anonymous\":false,\"inputs\":[{\"indexed\":true,\"internalType\":\"address\",\"name\":\"owner\",\"type\":\"address\"},{\"indexed\":true,\"internalType\":\"address\",\"name\":\"operator\",\"type\":\"address\"},{\"indexed\":false,\"internalType\":\"bool\",\"name\":\"approved\",\"type\":\"bool\"}],\"name\":\"ApprovalForAll\",\"type\":\"event\"},{\"anonymous\":false,\"inputs\":[{\"indexed\":false,\"internalType\":\"uint256\",\"name\":\"tokenId\",\"type\":\"uint256\"}],\"name\":\"DeliveryAccepted\",\"type\":\"event\"},{\"anonymous\":false,\"inputs\":[{\"indexed\":false,\"internalType\":\"uint256\",\"name\":\"price\",\"type\":\"uint256\"},{\"indexed\":false,\"internalType\":\"address\",\"name\":\"owner\",\"type\":\"address\"}],\"name\":\"EditionSold\",\"type\":\"event\"},{\"anonymous\":false,\"inputs\":[{\"indexed\":false,\"internalType\":\"uint8\",\"name\":\"version\",\"type\":\"uint8\"}],\"name\":\"Initialized\",\"type\":\"event\"},{\"anonymous\":false,\"inputs\":[{\"indexed\":false,\"internalType\":\"uint256\",\"name\":\"tokenId\",\"type\":\"uint256\"}],\"name\":\"OfferAccepted\",\"type\":\"event\"},{\"anonymous\":false,\"inputs\":[{\"indexed\":false,\"internalType\":\"uint256\",\"name\":\"tokenId\",\"type\":\"uint256\"}],\"name\":\"OfferRejected\",\"type\":\"event\"},{\"anonymous\":false,\"inputs\":[{\"indexed\":false,\"internalType\":\"uint256\",\"name\":\"tokenId\",\"type\":\"uint256\"}],\"name\":\"OfferTermsSet\",\"type\":\"event\"},{\"anonymous\":false,\"inputs\":[{\"indexed\":true,\"internalType\":\"address\",\"name\":\"previousOwner\",\"type\":\"address\"},{\"indexed\":true,\"internalType\":\"address\",\"name\":\"newOwner\",\"type\":\"address\"}],\"name\":\"OwnershipTransferred\",\"type\":\"event\"},{\"anonymous\":false,\"inputs\":[{\"indexed\":false,\"internalType\":\"uint256\",\"name\":\"amount\",\"type\":\"uint256\"}],\"name\":\"PriceChanged\",\"type\":\"event\"},{\"anonymous\":false,\"inputs\":[{\"indexed\":false,\"internalType\":\"uint256\",\"name\":\"tokenId\",\"type\":\"uint256\"}],\"name\":\"ProductionComplete\",\"type\":\"event\"},{\"anonymous\":false,\"inputs\":[{\"indexed\":false,\"internalType\":\"uint256\",\"name\":\"tokenId\",\"type\":\"uint256\"},{\"indexed\":false,\"internalType\":\"address\",\"name\":\"owner\",\"type\":\"address\"}],\"name\":\"RedeemAborted\",\"type\":\"event\"},{\"anonymous\":false,\"inputs\":[{\"indexed\":false,\"internalType\":\"uint256\",\"name\":\"tokenId\",\"type\":\"uint256\"},{\"indexed\":false,\"internalType\":\"address\",\"name\":\"owner\",\"type\":\"address\"}],\"name\":\"RedeemStarted\",\"type\":\"event\"},{\"anonymous\":false,\"inputs\":[{\"indexed\":true,\"internalType\":\"address\",\"name\":\"from\",\"type\":\"address\"},{\"indexed\":true,\"internalType\":\"address\",\"name\":\"to\",\"type\":\"address\"},{\"indexed\":true,\"internalType\":\"uint256\",\"name\":\"tokenId\",\"type\":\"uint256\"}],\"name\":\"Transfer\",\"type\":\"event\"},{\"anonymous\":false,\"inputs\":[{\"indexed\":false,\"internalType\":\"enum ExpandedNFT.WhoCanMint\",\"name\":\"minters\",\"type\":\"uint8\"}],\"name\":\"WhoCanMintChanged\",\"type\":\"event\"},{\"inputs\":[{\"internalType\":\"uint256\",\"name\":\"tokenId\",\"type\":\"uint256\"}],\"name\":\"abortRedemption\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"uint256\",\"name\":\"tokenId\",\"type\":\"uint256\"}],\"name\":\"acceptDelivery\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"uint256\",\"name\":\"tokenId\",\"type\":\"uint256\"},{\"internalType\":\"uint256\",\"name\":\"paymentAmount\",\"type\":\"uint256\"}],\"name\":\"acceptOfferTerms\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"address\",\"name\":\"to\",\"type\":\"address\"},{\"internalType\":\"uint256\",\"name\":\"tokenId\",\"type\":\"uint256\"}],\"name\":\"approve\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"address\",\"name\":\"owner\",\"type\":\"address\"}],\"name\":\"balanceOf\",\"outputs\":[{\"internalType\":\"uint256\",\"name\":\"\",\"type\":\"uint256\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"uint256\",\"name\":\"tokenId\",\"type\":\"uint256\"}],\"name\":\"burn\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[],\"name\":\"description\",\"outputs\":[{\"internalType\":\"string\",\"name\":\"\",\"type\":\"string\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[],\"name\":\"dropSize\",\"outputs\":[{\"internalType\":\"uint256\",\"name\":\"\",\"type\":\"uint256\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[],\"name\":\"getAllowedMinter\",\"outputs\":[{\"internalType\":\"enum ExpandedNFT.WhoCanMint\",\"name\":\"\",\"type\":\"uint8\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"uint256\",\"name\":\"tokenId\",\"type\":\"uint256\"}],\"name\":\"getApproved\",\"outputs\":[{\"internalType\":\"address\",\"name\":\"\",\"type\":\"address\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[],\"name\":\"getArtistWallet\",\"outputs\":[{\"internalType\":\"address\",\"name\":\"\",\"type\":\"address\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"uint256\",\"name\":\"tokenId\",\"type\":\"uint256\"}],\"name\":\"getConditionReport\",\"outputs\":[{\"internalType\":\"string\",\"name\":\"\",\"type\":\"string\"},{\"internalType\":\"bytes32\",\"name\":\"\",\"type\":\"bytes32\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[],\"name\":\"getGeneralMintLimit\",\"outputs\":[{\"internalType\":\"uint256\",\"name\":\"\",\"type\":\"uint256\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[],\"name\":\"getMembersMintLimit\",\"outputs\":[{\"internalType\":\"uint256\",\"name\":\"\",\"type\":\"uint256\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[],\"name\":\"getMembersSalePrice\",\"outputs\":[{\"internalType\":\"uint256\",\"name\":\"\",\"type\":\"uint256\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[],\"name\":\"getPaymentToken\",\"outputs\":[{\"internalType\":\"address\",\"name\":\"\",\"type\":\"address\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[],\"name\":\"getRoyaltyBPS\",\"outputs\":[{\"internalType\":\"uint256\",\"name\":\"\",\"type\":\"uint256\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[],\"name\":\"getSplitBPS\",\"outputs\":[{\"internalType\":\"uint256\",\"name\":\"\",\"type\":\"uint256\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"uint256\",\"name\":\"tokenId\",\"type\":\"uint256\"}],\"name\":\"getURIs\",\"outputs\":[{\"internalType\":\"string\",\"name\":\"\",\"type\":\"string\"},{\"internalType\":\"bytes32\",\"name\":\"\",\"type\":\"bytes32\"},{\"internalType\":\"string\",\"name\":\"\",\"type\":\"string\"},{\"internalType\":\"bytes32\",\"name\":\"\",\"type\":\"bytes32\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[],\"name\":\"getVIPMintLimit\",\"outputs\":[{\"internalType\":\"uint256\",\"name\":\"\",\"type\":\"uint256\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[],\"name\":\"getVIPSalePrice\",\"outputs\":[{\"internalType\":\"uint256\",\"name\":\"\",\"type\":\"uint256\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[],\"name\":\"getWhoCanMint\",\"outputs\":[{\"internalType\":\"uint256\",\"name\":\"\",\"type\":\"uint256\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"address\",\"name\":\"_owner\",\"type\":\"address\"},{\"internalType\":\"address\",\"name\":\"artistWallet\",\"type\":\"address\"},{\"internalType\":\"string\",\"name\":\"_name\",\"type\":\"string\"},{\"internalType\":\"string\",\"name\":\"_symbol\",\"type\":\"string\"},{\"internalType\":\"uint256\",\"name\":\"_dropSize\",\"type\":\"uint256\"}],\"name\":\"initialize\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"address\",\"name\":\"owner\",\"type\":\"address\"},{\"internalType\":\"address\",\"name\":\"operator\",\"type\":\"address\"}],\"name\":\"isApprovedForAll\",\"outputs\":[{\"internalType\":\"bool\",\"name\":\"\",\"type\":\"bool\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"uint256\",\"name\":\"startOffset\",\"type\":\"uint256\"},{\"internalType\":\"uint256\",\"name\":\"count\",\"type\":\"uint256\"},{\"internalType\":\"string[]\",\"name\":\"_description\",\"type\":\"string[]\"},{\"internalType\":\"string[]\",\"name\":\"animationUrl\",\"type\":\"string[]\"},{\"internalType\":\"bytes32[]\",\"name\":\"animationHash\",\"type\":\"bytes32[]\"},{\"internalType\":\"string[]\",\"name\":\"imageUrl\",\"type\":\"string[]\"},{\"internalType\":\"bytes32[]\",\"name\":\"imageHash\",\"type\":\"bytes32[]\"}],\"name\":\"loadMetadataChunk\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[],\"name\":\"metadataloaded\",\"outputs\":[{\"internalType\":\"bool\",\"name\":\"\",\"type\":\"bool\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"address\",\"name\":\"to\",\"type\":\"address\"}],\"name\":\"mintEdition\",\"outputs\":[{\"internalType\":\"uint256\",\"name\":\"\",\"type\":\"uint256\"}],\"stateMutability\":\"payable\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"address[]\",\"name\":\"recipients\",\"type\":\"address[]\"}],\"name\":\"mintEditions\",\"outputs\":[{\"internalType\":\"uint256\",\"name\":\"\",\"type\":\"uint256\"}],\"stateMutability\":\"payable\",\"type\":\"function\"},{\"inputs\":[],\"name\":\"name\",\"outputs\":[{\"internalType\":\"string\",\"name\":\"\",\"type\":\"string\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[],\"name\":\"numberCanMint\",\"outputs\":[{\"internalType\":\"uint256\",\"name\":\"\",\"type\":\"uint256\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[],\"name\":\"owner\",\"outputs\":[{\"internalType\":\"address\",\"name\":\"\",\"type\":\"address\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"uint256\",\"name\":\"tokenId\",\"type\":\"uint256\"}],\"name\":\"ownerOf\",\"outputs\":[{\"internalType\":\"address\",\"name\":\"\",\"type\":\"address\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"uint256\",\"name\":\"tokenId\",\"type\":\"uint256\"},{\"internalType\":\"string\",\"name\":\"_description\",\"type\":\"string\"},{\"internalType\":\"string\",\"name\":\"animationUrl\",\"type\":\"string\"},{\"internalType\":\"bytes32\",\"name\":\"animationHash\",\"type\":\"bytes32\"},{\"internalType\":\"string\",\"name\":\"imageUrl\",\"type\":\"string\"},{\"internalType\":\"bytes32\",\"name\":\"imageHash\",\"type\":\"bytes32\"},{\"internalType\":\"string\",\"name\":\"conditionReportUrl\",\"type\":\"string\"},{\"internalType\":\"bytes32\",\"name\":\"conditionReportHash\",\"type\":\"bytes32\"}],\"name\":\"productionComplete\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[],\"name\":\"purchase\",\"outputs\":[{\"internalType\":\"uint256\",\"name\":\"\",\"type\":\"uint256\"}],\"stateMutability\":\"payable\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"uint256\",\"name\":\"tokenId\",\"type\":\"uint256\"}],\"name\":\"redeem\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"uint256\",\"name\":\"tokenId\",\"type\":\"uint256\"}],\"name\":\"rejectOfferTerms\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[],\"name\":\"renounceOwnership\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"address[]\",\"name\":\"wallets\",\"type\":\"address[]\"},{\"internalType\":\"uint256[]\",\"name\":\"tokenIDs\",\"type\":\"uint256[]\"}],\"name\":\"reserve\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"uint256\",\"name\":\"\",\"type\":\"uint256\"},{\"internalType\":\"uint256\",\"name\":\"_salePrice\",\"type\":\"uint256\"}],\"name\":\"royaltyInfo\",\"outputs\":[{\"internalType\":\"address\",\"name\":\"receiver\",\"type\":\"address\"},{\"internalType\":\"uint256\",\"name\":\"royaltyAmount\",\"type\":\"uint256\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"address\",\"name\":\"from\",\"type\":\"address\"},{\"internalType\":\"address\",\"name\":\"to\",\"type\":\"address\"},{\"internalType\":\"uint256\",\"name\":\"tokenId\",\"type\":\"uint256\"}],\"name\":\"safeTransferFrom\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"address\",\"name\":\"from\",\"type\":\"address\"},{\"internalType\":\"address\",\"name\":\"to\",\"type\":\"address\"},{\"internalType\":\"uint256\",\"name\":\"tokenId\",\"type\":\"uint256\"},{\"internalType\":\"bytes\",\"name\":\"data\",\"type\":\"bytes\"}],\"name\":\"safeTransferFrom\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[],\"name\":\"salePrice\",\"outputs\":[{\"internalType\":\"uint256\",\"name\":\"\",\"type\":\"uint256\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"enum ExpandedNFT.WhoCanMint\",\"name\":\"minters\",\"type\":\"uint8\"}],\"name\":\"setAllowedMinter\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"address\",\"name\":\"operator\",\"type\":\"address\"},{\"internalType\":\"bool\",\"name\":\"approved\",\"type\":\"bool\"}],\"name\":\"setApprovalForAll\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"uint256\",\"name\":\"count\",\"type\":\"uint256\"},{\"internalType\":\"address[]\",\"name\":\"minter\",\"type\":\"address[]\"},{\"internalType\":\"bool[]\",\"name\":\"allowed\",\"type\":\"bool[]\"}],\"name\":\"setApprovedMinters\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"uint256\",\"name\":\"count\",\"type\":\"uint256\"},{\"internalType\":\"address[]\",\"name\":\"minter\",\"type\":\"address[]\"},{\"internalType\":\"bool[]\",\"name\":\"allowed\",\"type\":\"bool[]\"}],\"name\":\"setApprovedVIPMinters\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"address\",\"name\":\"wallet\",\"type\":\"address\"}],\"name\":\"setArtistWallet\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"uint256\",\"name\":\"_salePrice\",\"type\":\"uint256\"}],\"name\":\"setMembersSalePrice\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"uint256\",\"name\":\"tokenId\",\"type\":\"uint256\"},{\"internalType\":\"uint256\",\"name\":\"fee\",\"type\":\"uint256\"}],\"name\":\"setOfferTerms\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"address\",\"name\":\"paymentToken\",\"type\":\"address\"}],\"name\":\"setPaymentToken\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"uint256\",\"name\":\"_royaltyBPS\",\"type\":\"uint256\"},{\"internalType\":\"uint256\",\"name\":\"_splitBPS\",\"type\":\"uint256\"},{\"internalType\":\"uint256\",\"name\":\"_vipSalePrice\",\"type\":\"uint256\"},{\"internalType\":\"uint256\",\"name\":\"_membersSalePrice\",\"type\":\"uint256\"},{\"internalType\":\"uint256\",\"name\":\"_generalSalePrice\",\"type\":\"uint256\"},{\"internalType\":\"uint256\",\"name\":\"_vipMintLimit\",\"type\":\"uint256\"},{\"internalType\":\"uint256\",\"name\":\"_membersMintLimit\",\"type\":\"uint256\"},{\"internalType\":\"uint256\",\"name\":\"_generalMintLimit\",\"type\":\"uint256\"}],\"name\":\"setPricing\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"uint256\",\"name\":\"_salePrice\",\"type\":\"uint256\"}],\"name\":\"setSalePrice\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"uint256\",\"name\":\"vipSalePrice\",\"type\":\"uint256\"},{\"internalType\":\"uint256\",\"name\":\"membersSalePrice\",\"type\":\"uint256\"},{\"internalType\":\"uint256\",\"name\":\"generalSalePrice\",\"type\":\"uint256\"}],\"name\":\"setSalePrices\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"uint256\",\"name\":\"_salePrice\",\"type\":\"uint256\"}],\"name\":\"setVIPSalePrice\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"bytes4\",\"name\":\"interfaceId\",\"type\":\"bytes4\"}],\"name\":\"supportsInterface\",\"outputs\":[{\"internalType\":\"bool\",\"name\":\"\",\"type\":\"bool\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[],\"name\":\"symbol\",\"outputs\":[{\"internalType\":\"string\",\"name\":\"\",\"type\":\"string\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"uint256\",\"name\":\"tokenId\",\"type\":\"uint256\"}],\"name\":\"tokenURI\",\"outputs\":[{\"internalType\":\"string\",\"name\":\"\",\"type\":\"string\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[],\"name\":\"totalSupply\",\"outputs\":[{\"internalType\":\"uint256\",\"name\":\"\",\"type\":\"uint256\"}],\"stateMutability\":\"view\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"address\",\"name\":\"from\",\"type\":\"address\"},{\"internalType\":\"address\",\"name\":\"to\",\"type\":\"address\"},{\"internalType\":\"uint256\",\"name\":\"tokenId\",\"type\":\"uint256\"}],\"name\":\"transferFrom\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"address\",\"name\":\"newOwner\",\"type\":\"address\"}],\"name\":\"transferOwnership\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"uint256\",\"name\":\"tokenId\",\"type\":\"uint256\"},{\"internalType\":\"string\",\"name\":\"imageUrl\",\"type\":\"string\"},{\"internalType\":\"string\",\"name\":\"animationUrl\",\"type\":\"string\"}],\"name\":\"updateEditionURLs\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"},{\"inputs\":[],\"name\":\"withdraw\",\"outputs\":[],\"stateMutability\":\"nonpayable\",\"type\":\"function\"}],\"devdoc\":{\"author\":\"Zien Repository: https://github.com/joinzien/expanded-nft\",\"details\":\"This allows creators to mint a unique serial drop of an expanded NFT within a custom contract\",\"kind\":\"dev\",\"methods\":{\"approve(address,uint256)\":{\"details\":\"See {IERC721-approve}.\"},\"balanceOf(address)\":{\"details\":\"See {IERC721-balanceOf}.\"},\"burn(uint256)\":{\"params\":{\"tokenId\":\"Token ID to burn User burn function for token id \"}},\"getAllowedMinter()\":{\"details\":\"Sets the types of users who is allowed to mint.\"},\"getApproved(uint256)\":{\"details\":\"See {IERC721-getApproved}.\"},\"getConditionReport(uint256)\":{\"details\":\"Get URIs for the condition report\",\"returns\":{\"_0\":\"conditionReportUrl, conditionReportHash\"}},\"getGeneralMintLimit()\":{\"details\":\"returns the general mint limit\"},\"getMembersMintLimit()\":{\"details\":\"returns the member mint limit\"},\"getMembersSalePrice()\":{\"details\":\"returns the member sale price\"},\"getRoyaltyBPS()\":{\"details\":\"returns the royalty BPS\"},\"getSplitBPS()\":{\"details\":\"returns the split BPS\"},\"getURIs(uint256)\":{\"details\":\"Get URIs for edition NFT\",\"returns\":{\"_0\":\"_imageUrl, _imageHash, _animationUrl, _animationHash\"}},\"getVIPMintLimit()\":{\"details\":\"returns the VIP mint limit\"},\"getVIPSalePrice()\":{\"details\":\"returns the VIP sale price\"},\"getWhoCanMint()\":{\"details\":\"returns who can mint\"},\"initialize(address,address,string,string,uint256)\":{\"details\":\"Function to create a new drop. Can only be called by the allowed creator Sets the only allowed minter to the address that creates/owns the drop. This can be re-assigned or updated later\",\"params\":{\"_dropSize\":\"Number of editions that can be minted in total. \",\"_name\":\"Name of drop, used in the title as \\\"$NAME NUMBER/TOTAL\\\"\",\"_owner\":\"wallet addres for the user that owns and can mint the drop, gets royalty and sales payouts and can update the base url if needed.\",\"_symbol\":\"Symbol of the new token contract\",\"artistWallet\":\"wallet address for thr User that created the drop\"}},\"isApprovedForAll(address,address)\":{\"details\":\"See {IERC721-isApprovedForAll}.\"},\"loadMetadataChunk(uint256,uint256,string[],string[],bytes32[],string[],bytes32[])\":{\"details\":\"Function to create a new drop. Can only be called by the allowed creator Sets the only allowed minter to the address that creates/owns the drop. This can be re-assigned or updated later\",\"params\":{\"_description\":\"Description of the edition, used in the description field of the NFT\",\"animationHash\":\"The associated hash of the animation in sha-256 bytes32 format. If animation is omitted the hash can be zero.\",\"animationUrl\":\"Animation URL of the edition. Not required, but if omitted image URL needs to be included. This follows the opensea spec for NFTs\",\"imageHash\":\"SHA256 of the given image in bytes32 format (0xHASH). If no image is included, the hash can be zero.\",\"imageUrl\":\"Image URL of the the edition. Strongly encouraged to be used, if necessary, only animation URL can be used. One of animation and image url need to exist in a drop to render the NFT.\"}},\"mintEdition(address)\":{\"details\":\"This mints one edition to the given address by an allowed minter on the edition instance.\",\"params\":{\"to\":\"address to send the newly minted edition to\"}},\"mintEditions(address[])\":{\"details\":\"This mints multiple editions to the given list of addresses.\",\"params\":{\"recipients\":\"list of addresses to send the newly minted editions to\"}},\"name()\":{\"details\":\"See {IERC721Metadata-name}.\"},\"ownerOf(uint256)\":{\"details\":\"See {IERC721-ownerOf}.\"},\"purchase()\":{\"details\":\"This allows the user to purchase an edition at the given price in the contract.\"},\"renounceOwnership()\":{\"details\":\"Leaves the contract without owner. It will not be possible to call `onlyOwner` functions anymore. Can only be called by the current owner. NOTE: Renouncing ownership will leave the contract without an owner, thereby removing any functionality that is only available to the owner.\"},\"reserve(address[],uint256[])\":{\"details\":\"Set various pricing related values\",\"params\":{\"tokenIDs\":\"A list of tokenId to reserve \",\"wallets\":\"A list of wallets\"}},\"royaltyInfo(uint256,uint256)\":{\"details\":\"Get royalty information for token\",\"params\":{\"_salePrice\":\"Sale price for the token\"}},\"safeTransferFrom(address,address,uint256)\":{\"details\":\"See {IERC721-safeTransferFrom}.\"},\"safeTransferFrom(address,address,uint256,bytes)\":{\"details\":\"See {IERC721-safeTransferFrom}.\"},\"setAllowedMinter(uint8)\":{\"details\":\"Sets the types of users who is allowed to mint.\",\"params\":{\"minters\":\"WhoCanMint enum of minter types\"}},\"setApprovalForAll(address,bool)\":{\"details\":\"See {IERC721-setApprovalForAll}.\"},\"setApprovedMinters(uint256,address[],bool[])\":{\"details\":\"Sets the approved minting status of the given address. This requires that msg.sender is the owner of the given edition id. If the ZeroAddress (address(0x0)) is set as a minter, anyone will be allowed to mint. This setup is similar to setApprovalForAll in the ERC721 spec.\",\"params\":{\"allowed\":\"boolean if that address is allowed to mint\",\"minter\":\"address to set approved minting status for\"}},\"setApprovedVIPMinters(uint256,address[],bool[])\":{\"details\":\"Sets the approved minting status of the given address. This requires that msg.sender is the owner of the given edition id. If the ZeroAddress (address(0x0)) is set as a minter, anyone will be allowed to mint. This setup is similar to setApprovalForAll in the ERC721 spec.\",\"params\":{\"allowed\":\"boolean if that address is allowed to mint\",\"minter\":\"address to set approved minting status for\"}},\"setMembersSalePrice(uint256)\":{\"details\":\"This sets the members ETH sales price Setting a sales price allows users to mint the drop until it sells out. For more granular sales, use an external sales contract.\",\"params\":{\"_salePrice\":\"if sale price is 0 sale is stopped, otherwise that amount of ETH is needed to start the sale.\"}},\"setPricing(uint256,uint256,uint256,uint256,uint256,uint256,uint256,uint256)\":{\"details\":\"Set various pricing related values\",\"params\":{\"_generalMintLimit\":\"Mint limit for the general public \",\"_generalSalePrice\":\"SalePrice for the general public \",\"_membersMintLimit\":\"Mint limit for Members \",\"_membersSalePrice\":\"SalePrice for Members \",\"_royaltyBPS\":\"BPS of the royalty set on the contract. Can be 0 for no royalty.\",\"_splitBPS\":\"BPS of the royalty set on the contract. Can be 0 for no royalty. \",\"_vipMintLimit\":\"Mint limit for VIPs\",\"_vipSalePrice\":\"Sale price for VIPs\"}},\"setSalePrice(uint256)\":{\"details\":\"This sets a simple ETH sales price Setting a sales price allows users to mint the drop until it sells out. For more granular sales, use an external sales contract.\",\"params\":{\"_salePrice\":\"if sale price is 0 sale is stopped, otherwise that amount of ETH is needed to start the sale.\"}},\"setSalePrices(uint256,uint256,uint256)\":{\"details\":\"This sets the members ETH sales price Setting a sales price allows users to mint the drop until it sells out. For more granular sales, use an external sales contract.\",\"params\":{\"generalSalePrice\":\"if sale price is 0 sale is stopped, otherwise that amount of ETH is needed to start the sale. \",\"membersSalePrice\":\"if sale price is 0 sale is stopped, otherwise that amount of ETH is needed to start the sale.\",\"vipSalePrice\":\"if sale price is 0 sale is stopped, otherwise that amount of ETH is needed to start the sale.\"}},\"setVIPSalePrice(uint256)\":{\"details\":\"This sets the VIP ETH sales price Setting a sales price allows users to mint the drop until it sells out. For more granular sales, use an external sales contract.\",\"params\":{\"_salePrice\":\"if sale price is 0 sale is stopped, otherwise that amount of ETH is needed to start the sale.\"}},\"symbol()\":{\"details\":\"See {IERC721Metadata-symbol}.\"},\"tokenURI(uint256)\":{\"details\":\"Get URI for given token id\",\"params\":{\"tokenId\":\"token id to get uri for\"},\"returns\":{\"_0\":\"base64-encoded json metadata object\"}},\"totalSupply()\":{\"details\":\"returns the number of minted tokens within the drop\"},\"transferFrom(address,address,uint256)\":{\"details\":\"See {IERC721-transferFrom}.\"},\"transferOwnership(address)\":{\"details\":\"Transfers ownership of the contract to a new account (`newOwner`). Can only be called by the current owner.\"},\"updateEditionURLs(uint256,string,string)\":{\"details\":\"Allows for updates of edition urls by the owner of the edition. Only URLs can be updated (data-uris are supported), hashes cannot be updated.\"},\"withdraw()\":{\"details\":\"This withdraws ETH from the contract to the contract owner.\"}},\"version\":1},\"userdoc\":{\"kind\":\"user\",\"methods\":{\"getArtistWallet()\":{\"notice\":\"return the artists wallet address\"},\"getPaymentToken()\":{\"notice\":\"return the payment tokens address\"},\"numberCanMint()\":{\"notice\":\"Returns the number of editions allowed to mint\"},\"owner()\":{\"notice\":\"Simple override for owner interface.\"},\"setArtistWallet(address)\":{\"notice\":\"set the artists wallet address\"},\"setPaymentToken(address)\":{\"notice\":\"set a new payment token address\"}},\"notice\":\"This is a smart contract for handling dynamic contract minting.\",\"version\":1}},\"settings\":{\"compilationTarget\":{\"contracts/ExpandedNFT.sol\":\"ExpandedNFT\"},\"evmVersion\":\"london\",\"libraries\":{},\"metadata\":{\"bytecodeHash\":\"ipfs\",\"useLiteralContent\":true},\"optimizer\":{\"enabled\":true,\"runs\":100},\"remappings\":[]},\"sources\":{\"@openzeppelin/contracts-upgradeable/access/OwnableUpgradeable.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n// OpenZeppelin Contracts (last updated v4.7.0) (access/Ownable.sol)\\n\\npragma solidity ^0.8.0;\\n\\nimport \\\"../utils/ContextUpgradeable.sol\\\";\\nimport \\\"../proxy/utils/Initializable.sol\\\";\\n\\n/**\\n * @dev Contract module which provides a basic access control mechanism, where\\n * there is an account (an owner) that can be granted exclusive access to\\n * specific functions.\\n *\\n * By default, the owner account will be the one that deploys the contract. This\\n * can later be changed with {transferOwnership}.\\n *\\n * This module is used through inheritance. It will make available the modifier\\n * `onlyOwner`, which can be applied to your functions to restrict their use to\\n * the owner.\\n */\\nabstract contract OwnableUpgradeable is Initializable, ContextUpgradeable {\\n address private _owner;\\n\\n event OwnershipTransferred(address indexed previousOwner, address indexed newOwner);\\n\\n /**\\n * @dev Initializes the contract setting the deployer as the initial owner.\\n */\\n function __Ownable_init() internal onlyInitializing {\\n __Ownable_init_unchained();\\n }\\n\\n function __Ownable_init_unchained() internal onlyInitializing {\\n _transferOwnership(_msgSender());\\n }\\n\\n /**\\n * @dev Throws if called by any account other than the owner.\\n */\\n modifier onlyOwner() {\\n _checkOwner();\\n _;\\n }\\n\\n /**\\n * @dev Returns the address of the current owner.\\n */\\n function owner() public view virtual returns (address) {\\n return _owner;\\n }\\n\\n /**\\n * @dev Throws if the sender is not the owner.\\n */\\n function _checkOwner() internal view virtual {\\n require(owner() == _msgSender(), \\\"Ownable: caller is not the owner\\\");\\n }\\n\\n /**\\n * @dev Leaves the contract without owner. It will not be possible to call\\n * `onlyOwner` functions anymore. Can only be called by the current owner.\\n *\\n * NOTE: Renouncing ownership will leave the contract without an owner,\\n * thereby removing any functionality that is only available to the owner.\\n */\\n function renounceOwnership() public virtual onlyOwner {\\n _transferOwnership(address(0));\\n }\\n\\n /**\\n * @dev Transfers ownership of the contract to a new account (`newOwner`).\\n * Can only be called by the current owner.\\n */\\n function transferOwnership(address newOwner) public virtual onlyOwner {\\n require(newOwner != address(0), \\\"Ownable: new owner is the zero address\\\");\\n _transferOwnership(newOwner);\\n }\\n\\n /**\\n * @dev Transfers ownership of the contract to a new account (`newOwner`).\\n * Internal function without access restriction.\\n */\\n function _transferOwnership(address newOwner) internal virtual {\\n address oldOwner = _owner;\\n _owner = newOwner;\\n emit OwnershipTransferred(oldOwner, newOwner);\\n }\\n\\n /**\\n * @dev This empty reserved space is put in place to allow future versions to add new\\n * variables without shifting down storage in the inheritance chain.\\n * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps\\n */\\n uint256[49] private __gap;\\n}\\n\",\"keccak256\":\"0x247c62047745915c0af6b955470a72d1696ebad4352d7d3011aef1a2463cd888\",\"license\":\"MIT\"},\"@openzeppelin/contracts-upgradeable/interfaces/IERC2981Upgradeable.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n// OpenZeppelin Contracts (last updated v4.6.0) (interfaces/IERC2981.sol)\\n\\npragma solidity ^0.8.0;\\n\\nimport \\\"../utils/introspection/IERC165Upgradeable.sol\\\";\\n\\n/**\\n * @dev Interface for the NFT Royalty Standard.\\n *\\n * A standardized way to retrieve royalty payment information for non-fungible tokens (NFTs) to enable universal\\n * support for royalty payments across all NFT marketplaces and ecosystem participants.\\n *\\n * _Available since v4.5._\\n */\\ninterface IERC2981Upgradeable is IERC165Upgradeable {\\n /**\\n * @dev Returns how much royalty is owed and to whom, based on a sale price that may be denominated in any unit of\\n * exchange. The royalty amount is denominated and should be paid in that same unit of exchange.\\n */\\n function royaltyInfo(uint256 tokenId, uint256 salePrice)\\n external\\n view\\n returns (address receiver, uint256 royaltyAmount);\\n}\\n\",\"keccak256\":\"0xa8ff557539dcfed5706eddde2aa929e06bb1764e71aa8c1048a78970bf3ca37d\",\"license\":\"MIT\"},\"@openzeppelin/contracts-upgradeable/proxy/utils/Initializable.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n// OpenZeppelin Contracts (last updated v4.7.0) (proxy/utils/Initializable.sol)\\n\\npragma solidity ^0.8.2;\\n\\nimport \\\"../../utils/AddressUpgradeable.sol\\\";\\n\\n/**\\n * @dev This is a base contract to aid in writing upgradeable contracts, or any kind of contract that will be deployed\\n * behind a proxy. Since proxied contracts do not make use of a constructor, it's common to move constructor logic to an\\n * external initializer function, usually called `initialize`. It then becomes necessary to protect this initializer\\n * function so it can only be called once. The {initializer} modifier provided by this contract will have this effect.\\n *\\n * The initialization functions use a version number. Once a version number is used, it is consumed and cannot be\\n * reused. This mechanism prevents re-execution of each \\\"step\\\" but allows the creation of new initialization steps in\\n * case an upgrade adds a module that needs to be initialized.\\n *\\n * For example:\\n *\\n * [.hljs-theme-light.nopadding]\\n * ```\\n * contract MyToken is ERC20Upgradeable {\\n * function initialize() initializer public {\\n * __ERC20_init(\\\"MyToken\\\", \\\"MTK\\\");\\n * }\\n * }\\n * contract MyTokenV2 is MyToken, ERC20PermitUpgradeable {\\n * function initializeV2() reinitializer(2) public {\\n * __ERC20Permit_init(\\\"MyToken\\\");\\n * }\\n * }\\n * ```\\n *\\n * TIP: To avoid leaving the proxy in an uninitialized state, the initializer function should be called as early as\\n * possible by providing the encoded function call as the `_data` argument to {ERC1967Proxy-constructor}.\\n *\\n * CAUTION: When used with inheritance, manual care must be taken to not invoke a parent initializer twice, or to ensure\\n * that all initializers are idempotent. This is not verified automatically as constructors are by Solidity.\\n *\\n * [CAUTION]\\n * ====\\n * Avoid leaving a contract uninitialized.\\n *\\n * An uninitialized contract can be taken over by an attacker. This applies to both a proxy and its implementation\\n * contract, which may impact the proxy. To prevent the implementation contract from being used, you should invoke\\n * the {_disableInitializers} function in the constructor to automatically lock it when it is deployed:\\n *\\n * [.hljs-theme-light.nopadding]\\n * ```\\n * /// @custom:oz-upgrades-unsafe-allow constructor\\n * constructor() {\\n * _disableInitializers();\\n * }\\n * ```\\n * ====\\n */\\nabstract contract Initializable {\\n /**\\n * @dev Indicates that the contract has been initialized.\\n * @custom:oz-retyped-from bool\\n */\\n uint8 private _initialized;\\n\\n /**\\n * @dev Indicates that the contract is in the process of being initialized.\\n */\\n bool private _initializing;\\n\\n /**\\n * @dev Triggered when the contract has been initialized or reinitialized.\\n */\\n event Initialized(uint8 version);\\n\\n /**\\n * @dev A modifier that defines a protected initializer function that can be invoked at most once. In its scope,\\n * `onlyInitializing` functions can be used to initialize parent contracts. Equivalent to `reinitializer(1)`.\\n */\\n modifier initializer() {\\n bool isTopLevelCall = !_initializing;\\n require(\\n (isTopLevelCall && _initialized < 1) || (!AddressUpgradeable.isContract(address(this)) && _initialized == 1),\\n \\\"Initializable: contract is already initialized\\\"\\n );\\n _initialized = 1;\\n if (isTopLevelCall) {\\n _initializing = true;\\n }\\n _;\\n if (isTopLevelCall) {\\n _initializing = false;\\n emit Initialized(1);\\n }\\n }\\n\\n /**\\n * @dev A modifier that defines a protected reinitializer function that can be invoked at most once, and only if the\\n * contract hasn't been initialized to a greater version before. In its scope, `onlyInitializing` functions can be\\n * used to initialize parent contracts.\\n *\\n * `initializer` is equivalent to `reinitializer(1)`, so a reinitializer may be used after the original\\n * initialization step. This is essential to configure modules that are added through upgrades and that require\\n * initialization.\\n *\\n * Note that versions can jump in increments greater than 1; this implies that if multiple reinitializers coexist in\\n * a contract, executing them in the right order is up to the developer or operator.\\n */\\n modifier reinitializer(uint8 version) {\\n require(!_initializing && _initialized < version, \\\"Initializable: contract is already initialized\\\");\\n _initialized = version;\\n _initializing = true;\\n _;\\n _initializing = false;\\n emit Initialized(version);\\n }\\n\\n /**\\n * @dev Modifier to protect an initialization function so that it can only be invoked by functions with the\\n * {initializer} and {reinitializer} modifiers, directly or indirectly.\\n */\\n modifier onlyInitializing() {\\n require(_initializing, \\\"Initializable: contract is not initializing\\\");\\n _;\\n }\\n\\n /**\\n * @dev Locks the contract, preventing any future reinitialization. This cannot be part of an initializer call.\\n * Calling this in the constructor of a contract will prevent that contract from being initialized or reinitialized\\n * to any version. It is recommended to use this to lock implementation contracts that are designed to be called\\n * through proxies.\\n */\\n function _disableInitializers() internal virtual {\\n require(!_initializing, \\\"Initializable: contract is initializing\\\");\\n if (_initialized < type(uint8).max) {\\n _initialized = type(uint8).max;\\n emit Initialized(type(uint8).max);\\n }\\n }\\n}\\n\",\"keccak256\":\"0x0203dcadc5737d9ef2c211d6fa15d18ebc3b30dfa51903b64870b01a062b0b4e\",\"license\":\"MIT\"},\"@openzeppelin/contracts-upgradeable/token/ERC20/IERC20Upgradeable.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n// OpenZeppelin Contracts (last updated v4.6.0) (token/ERC20/IERC20.sol)\\n\\npragma solidity ^0.8.0;\\n\\n/**\\n * @dev Interface of the ERC20 standard as defined in the EIP.\\n */\\ninterface IERC20Upgradeable {\\n /**\\n * @dev Emitted when `value` tokens are moved from one account (`from`) to\\n * another (`to`).\\n *\\n * Note that `value` may be zero.\\n */\\n event Transfer(address indexed from, address indexed to, uint256 value);\\n\\n /**\\n * @dev Emitted when the allowance of a `spender` for an `owner` is set by\\n * a call to {approve}. `value` is the new allowance.\\n */\\n event Approval(address indexed owner, address indexed spender, uint256 value);\\n\\n /**\\n * @dev Returns the amount of tokens in existence.\\n */\\n function totalSupply() external view returns (uint256);\\n\\n /**\\n * @dev Returns the amount of tokens owned by `account`.\\n */\\n function balanceOf(address account) external view returns (uint256);\\n\\n /**\\n * @dev Moves `amount` tokens from the caller's account to `to`.\\n *\\n * Returns a boolean value indicating whether the operation succeeded.\\n *\\n * Emits a {Transfer} event.\\n */\\n function transfer(address to, uint256 amount) external returns (bool);\\n\\n /**\\n * @dev Returns the remaining number of tokens that `spender` will be\\n * allowed to spend on behalf of `owner` through {transferFrom}. This is\\n * zero by default.\\n *\\n * This value changes when {approve} or {transferFrom} are called.\\n */\\n function allowance(address owner, address spender) external view returns (uint256);\\n\\n /**\\n * @dev Sets `amount` as the allowance of `spender` over the caller's tokens.\\n *\\n * Returns a boolean value indicating whether the operation succeeded.\\n *\\n * IMPORTANT: Beware that changing an allowance with this method brings the risk\\n * that someone may use both the old and the new allowance by unfortunate\\n * transaction ordering. One possible solution to mitigate this race\\n * condition is to first reduce the spender's allowance to 0 and set the\\n * desired value afterwards:\\n * https://github.com/ethereum/EIPs/issues/20#issuecomment-263524729\\n *\\n * Emits an {Approval} event.\\n */\\n function approve(address spender, uint256 amount) external returns (bool);\\n\\n /**\\n * @dev Moves `amount` tokens from `from` to `to` using the\\n * allowance mechanism. `amount` is then deducted from the caller's\\n * allowance.\\n *\\n * Returns a boolean value indicating whether the operation succeeded.\\n *\\n * Emits a {Transfer} event.\\n */\\n function transferFrom(\\n address from,\\n address to,\\n uint256 amount\\n ) external returns (bool);\\n}\\n\",\"keccak256\":\"0x4e733d3164f73f461eaf9d8087a7ad1ea180bdc8ba0d3d61b0e1ae16d8e63dff\",\"license\":\"MIT\"},\"@openzeppelin/contracts-upgradeable/token/ERC721/ERC721Upgradeable.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n// OpenZeppelin Contracts (last updated v4.7.0) (token/ERC721/ERC721.sol)\\n\\npragma solidity ^0.8.0;\\n\\nimport \\\"./IERC721Upgradeable.sol\\\";\\nimport \\\"./IERC721ReceiverUpgradeable.sol\\\";\\nimport \\\"./extensions/IERC721MetadataUpgradeable.sol\\\";\\nimport \\\"../../utils/AddressUpgradeable.sol\\\";\\nimport \\\"../../utils/ContextUpgradeable.sol\\\";\\nimport \\\"../../utils/StringsUpgradeable.sol\\\";\\nimport \\\"../../utils/introspection/ERC165Upgradeable.sol\\\";\\nimport \\\"../../proxy/utils/Initializable.sol\\\";\\n\\n/**\\n * @dev Implementation of https://eips.ethereum.org/EIPS/eip-721[ERC721] Non-Fungible Token Standard, including\\n * the Metadata extension, but not including the Enumerable extension, which is available separately as\\n * {ERC721Enumerable}.\\n */\\ncontract ERC721Upgradeable is Initializable, ContextUpgradeable, ERC165Upgradeable, IERC721Upgradeable, IERC721MetadataUpgradeable {\\n using AddressUpgradeable for address;\\n using StringsUpgradeable for uint256;\\n\\n // Token name\\n string private _name;\\n\\n // Token symbol\\n string private _symbol;\\n\\n // Mapping from token ID to owner address\\n mapping(uint256 => address) private _owners;\\n\\n // Mapping owner address to token count\\n mapping(address => uint256) private _balances;\\n\\n // Mapping from token ID to approved address\\n mapping(uint256 => address) private _tokenApprovals;\\n\\n // Mapping from owner to operator approvals\\n mapping(address => mapping(address => bool)) private _operatorApprovals;\\n\\n /**\\n * @dev Initializes the contract by setting a `name` and a `symbol` to the token collection.\\n */\\n function __ERC721_init(string memory name_, string memory symbol_) internal onlyInitializing {\\n __ERC721_init_unchained(name_, symbol_);\\n }\\n\\n function __ERC721_init_unchained(string memory name_, string memory symbol_) internal onlyInitializing {\\n _name = name_;\\n _symbol = symbol_;\\n }\\n\\n /**\\n * @dev See {IERC165-supportsInterface}.\\n */\\n function supportsInterface(bytes4 interfaceId) public view virtual override(ERC165Upgradeable, IERC165Upgradeable) returns (bool) {\\n return\\n interfaceId == type(IERC721Upgradeable).interfaceId ||\\n interfaceId == type(IERC721MetadataUpgradeable).interfaceId ||\\n super.supportsInterface(interfaceId);\\n }\\n\\n /**\\n * @dev See {IERC721-balanceOf}.\\n */\\n function balanceOf(address owner) public view virtual override returns (uint256) {\\n require(owner != address(0), \\\"ERC721: address zero is not a valid owner\\\");\\n return _balances[owner];\\n }\\n\\n /**\\n * @dev See {IERC721-ownerOf}.\\n */\\n function ownerOf(uint256 tokenId) public view virtual override returns (address) {\\n address owner = _owners[tokenId];\\n require(owner != address(0), \\\"ERC721: invalid token ID\\\");\\n return owner;\\n }\\n\\n /**\\n * @dev See {IERC721Metadata-name}.\\n */\\n function name() public view virtual override returns (string memory) {\\n return _name;\\n }\\n\\n /**\\n * @dev See {IERC721Metadata-symbol}.\\n */\\n function symbol() public view virtual override returns (string memory) {\\n return _symbol;\\n }\\n\\n /**\\n * @dev See {IERC721Metadata-tokenURI}.\\n */\\n function tokenURI(uint256 tokenId) public view virtual override returns (string memory) {\\n _requireMinted(tokenId);\\n\\n string memory baseURI = _baseURI();\\n return bytes(baseURI).length > 0 ? string(abi.encodePacked(baseURI, tokenId.toString())) : \\\"\\\";\\n }\\n\\n /**\\n * @dev Base URI for computing {tokenURI}. If set, the resulting URI for each\\n * token will be the concatenation of the `baseURI` and the `tokenId`. Empty\\n * by default, can be overridden in child contracts.\\n */\\n function _baseURI() internal view virtual returns (string memory) {\\n return \\\"\\\";\\n }\\n\\n /**\\n * @dev See {IERC721-approve}.\\n */\\n function approve(address to, uint256 tokenId) public virtual override {\\n address owner = ERC721Upgradeable.ownerOf(tokenId);\\n require(to != owner, \\\"ERC721: approval to current owner\\\");\\n\\n require(\\n _msgSender() == owner || isApprovedForAll(owner, _msgSender()),\\n \\\"ERC721: approve caller is not token owner nor approved for all\\\"\\n );\\n\\n _approve(to, tokenId);\\n }\\n\\n /**\\n * @dev See {IERC721-getApproved}.\\n */\\n function getApproved(uint256 tokenId) public view virtual override returns (address) {\\n _requireMinted(tokenId);\\n\\n return _tokenApprovals[tokenId];\\n }\\n\\n /**\\n * @dev See {IERC721-setApprovalForAll}.\\n */\\n function setApprovalForAll(address operator, bool approved) public virtual override {\\n _setApprovalForAll(_msgSender(), operator, approved);\\n }\\n\\n /**\\n * @dev See {IERC721-isApprovedForAll}.\\n */\\n function isApprovedForAll(address owner, address operator) public view virtual override returns (bool) {\\n return _operatorApprovals[owner][operator];\\n }\\n\\n /**\\n * @dev See {IERC721-transferFrom}.\\n */\\n function transferFrom(\\n address from,\\n address to,\\n uint256 tokenId\\n ) public virtual override {\\n //solhint-disable-next-line max-line-length\\n require(_isApprovedOrOwner(_msgSender(), tokenId), \\\"ERC721: caller is not token owner nor approved\\\");\\n\\n _transfer(from, to, tokenId);\\n }\\n\\n /**\\n * @dev See {IERC721-safeTransferFrom}.\\n */\\n function safeTransferFrom(\\n address from,\\n address to,\\n uint256 tokenId\\n ) public virtual override {\\n safeTransferFrom(from, to, tokenId, \\\"\\\");\\n }\\n\\n /**\\n * @dev See {IERC721-safeTransferFrom}.\\n */\\n function safeTransferFrom(\\n address from,\\n address to,\\n uint256 tokenId,\\n bytes memory data\\n ) public virtual override {\\n require(_isApprovedOrOwner(_msgSender(), tokenId), \\\"ERC721: caller is not token owner nor approved\\\");\\n _safeTransfer(from, to, tokenId, data);\\n }\\n\\n /**\\n * @dev Safely transfers `tokenId` token from `from` to `to`, checking first that contract recipients\\n * are aware of the ERC721 protocol to prevent tokens from being forever locked.\\n *\\n * `data` is additional data, it has no specified format and it is sent in call to `to`.\\n *\\n * This internal function is equivalent to {safeTransferFrom}, and can be used to e.g.\\n * implement alternative mechanisms to perform token transfer, such as signature-based.\\n *\\n * Requirements:\\n *\\n * - `from` cannot be the zero address.\\n * - `to` cannot be the zero address.\\n * - `tokenId` token must exist and be owned by `from`.\\n * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer.\\n *\\n * Emits a {Transfer} event.\\n */\\n function _safeTransfer(\\n address from,\\n address to,\\n uint256 tokenId,\\n bytes memory data\\n ) internal virtual {\\n _transfer(from, to, tokenId);\\n require(_checkOnERC721Received(from, to, tokenId, data), \\\"ERC721: transfer to non ERC721Receiver implementer\\\");\\n }\\n\\n /**\\n * @dev Returns whether `tokenId` exists.\\n *\\n * Tokens can be managed by their owner or approved accounts via {approve} or {setApprovalForAll}.\\n *\\n * Tokens start existing when they are minted (`_mint`),\\n * and stop existing when they are burned (`_burn`).\\n */\\n function _exists(uint256 tokenId) internal view virtual returns (bool) {\\n return _owners[tokenId] != address(0);\\n }\\n\\n /**\\n * @dev Returns whether `spender` is allowed to manage `tokenId`.\\n *\\n * Requirements:\\n *\\n * - `tokenId` must exist.\\n */\\n function _isApprovedOrOwner(address spender, uint256 tokenId) internal view virtual returns (bool) {\\n address owner = ERC721Upgradeable.ownerOf(tokenId);\\n return (spender == owner || isApprovedForAll(owner, spender) || getApproved(tokenId) == spender);\\n }\\n\\n /**\\n * @dev Safely mints `tokenId` and transfers it to `to`.\\n *\\n * Requirements:\\n *\\n * - `tokenId` must not exist.\\n * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer.\\n *\\n * Emits a {Transfer} event.\\n */\\n function _safeMint(address to, uint256 tokenId) internal virtual {\\n _safeMint(to, tokenId, \\\"\\\");\\n }\\n\\n /**\\n * @dev Same as {xref-ERC721-_safeMint-address-uint256-}[`_safeMint`], with an additional `data` parameter which is\\n * forwarded in {IERC721Receiver-onERC721Received} to contract recipients.\\n */\\n function _safeMint(\\n address to,\\n uint256 tokenId,\\n bytes memory data\\n ) internal virtual {\\n _mint(to, tokenId);\\n require(\\n _checkOnERC721Received(address(0), to, tokenId, data),\\n \\\"ERC721: transfer to non ERC721Receiver implementer\\\"\\n );\\n }\\n\\n /**\\n * @dev Mints `tokenId` and transfers it to `to`.\\n *\\n * WARNING: Usage of this method is discouraged, use {_safeMint} whenever possible\\n *\\n * Requirements:\\n *\\n * - `tokenId` must not exist.\\n * - `to` cannot be the zero address.\\n *\\n * Emits a {Transfer} event.\\n */\\n function _mint(address to, uint256 tokenId) internal virtual {\\n require(to != address(0), \\\"ERC721: mint to the zero address\\\");\\n require(!_exists(tokenId), \\\"ERC721: token already minted\\\");\\n\\n _beforeTokenTransfer(address(0), to, tokenId);\\n\\n _balances[to] += 1;\\n _owners[tokenId] = to;\\n\\n emit Transfer(address(0), to, tokenId);\\n\\n _afterTokenTransfer(address(0), to, tokenId);\\n }\\n\\n /**\\n * @dev Destroys `tokenId`.\\n * The approval is cleared when the token is burned.\\n *\\n * Requirements:\\n *\\n * - `tokenId` must exist.\\n *\\n * Emits a {Transfer} event.\\n */\\n function _burn(uint256 tokenId) internal virtual {\\n address owner = ERC721Upgradeable.ownerOf(tokenId);\\n\\n _beforeTokenTransfer(owner, address(0), tokenId);\\n\\n // Clear approvals\\n _approve(address(0), tokenId);\\n\\n _balances[owner] -= 1;\\n delete _owners[tokenId];\\n\\n emit Transfer(owner, address(0), tokenId);\\n\\n _afterTokenTransfer(owner, address(0), tokenId);\\n }\\n\\n /**\\n * @dev Transfers `tokenId` from `from` to `to`.\\n * As opposed to {transferFrom}, this imposes no restrictions on msg.sender.\\n *\\n * Requirements:\\n *\\n * - `to` cannot be the zero address.\\n * - `tokenId` token must be owned by `from`.\\n *\\n * Emits a {Transfer} event.\\n */\\n function _transfer(\\n address from,\\n address to,\\n uint256 tokenId\\n ) internal virtual {\\n require(ERC721Upgradeable.ownerOf(tokenId) == from, \\\"ERC721: transfer from incorrect owner\\\");\\n require(to != address(0), \\\"ERC721: transfer to the zero address\\\");\\n\\n _beforeTokenTransfer(from, to, tokenId);\\n\\n // Clear approvals from the previous owner\\n _approve(address(0), tokenId);\\n\\n _balances[from] -= 1;\\n _balances[to] += 1;\\n _owners[tokenId] = to;\\n\\n emit Transfer(from, to, tokenId);\\n\\n _afterTokenTransfer(from, to, tokenId);\\n }\\n\\n /**\\n * @dev Approve `to` to operate on `tokenId`\\n *\\n * Emits an {Approval} event.\\n */\\n function _approve(address to, uint256 tokenId) internal virtual {\\n _tokenApprovals[tokenId] = to;\\n emit Approval(ERC721Upgradeable.ownerOf(tokenId), to, tokenId);\\n }\\n\\n /**\\n * @dev Approve `operator` to operate on all of `owner` tokens\\n *\\n * Emits an {ApprovalForAll} event.\\n */\\n function _setApprovalForAll(\\n address owner,\\n address operator,\\n bool approved\\n ) internal virtual {\\n require(owner != operator, \\\"ERC721: approve to caller\\\");\\n _operatorApprovals[owner][operator] = approved;\\n emit ApprovalForAll(owner, operator, approved);\\n }\\n\\n /**\\n * @dev Reverts if the `tokenId` has not been minted yet.\\n */\\n function _requireMinted(uint256 tokenId) internal view virtual {\\n require(_exists(tokenId), \\\"ERC721: invalid token ID\\\");\\n }\\n\\n /**\\n * @dev Internal function to invoke {IERC721Receiver-onERC721Received} on a target address.\\n * The call is not executed if the target address is not a contract.\\n *\\n * @param from address representing the previous owner of the given token ID\\n * @param to target address that will receive the tokens\\n * @param tokenId uint256 ID of the token to be transferred\\n * @param data bytes optional data to send along with the call\\n * @return bool whether the call correctly returned the expected magic value\\n */\\n function _checkOnERC721Received(\\n address from,\\n address to,\\n uint256 tokenId,\\n bytes memory data\\n ) private returns (bool) {\\n if (to.isContract()) {\\n try IERC721ReceiverUpgradeable(to).onERC721Received(_msgSender(), from, tokenId, data) returns (bytes4 retval) {\\n return retval == IERC721ReceiverUpgradeable.onERC721Received.selector;\\n } catch (bytes memory reason) {\\n if (reason.length == 0) {\\n revert(\\\"ERC721: transfer to non ERC721Receiver implementer\\\");\\n } else {\\n /// @solidity memory-safe-assembly\\n assembly {\\n revert(add(32, reason), mload(reason))\\n }\\n }\\n }\\n } else {\\n return true;\\n }\\n }\\n\\n /**\\n * @dev Hook that is called before any token transfer. This includes minting\\n * and burning.\\n *\\n * Calling conditions:\\n *\\n * - When `from` and `to` are both non-zero, ``from``'s `tokenId` will be\\n * transferred to `to`.\\n * - When `from` is zero, `tokenId` will be minted for `to`.\\n * - When `to` is zero, ``from``'s `tokenId` will be burned.\\n * - `from` and `to` are never both zero.\\n *\\n * To learn more about hooks, head to xref:ROOT:extending-contracts.adoc#using-hooks[Using Hooks].\\n */\\n function _beforeTokenTransfer(\\n address from,\\n address to,\\n uint256 tokenId\\n ) internal virtual {}\\n\\n /**\\n * @dev Hook that is called after any transfer of tokens. This includes\\n * minting and burning.\\n *\\n * Calling conditions:\\n *\\n * - when `from` and `to` are both non-zero.\\n * - `from` and `to` are never both zero.\\n *\\n * To learn more about hooks, head to xref:ROOT:extending-contracts.adoc#using-hooks[Using Hooks].\\n */\\n function _afterTokenTransfer(\\n address from,\\n address to,\\n uint256 tokenId\\n ) internal virtual {}\\n\\n /**\\n * @dev This empty reserved space is put in place to allow future versions to add new\\n * variables without shifting down storage in the inheritance chain.\\n * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps\\n */\\n uint256[44] private __gap;\\n}\\n\",\"keccak256\":\"0x5331c8909221d9f9f3851cfadd5959d0873413a2c27e30e0f2fa234158c1c6cf\",\"license\":\"MIT\"},\"@openzeppelin/contracts-upgradeable/token/ERC721/IERC721ReceiverUpgradeable.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n// OpenZeppelin Contracts (last updated v4.6.0) (token/ERC721/IERC721Receiver.sol)\\n\\npragma solidity ^0.8.0;\\n\\n/**\\n * @title ERC721 token receiver interface\\n * @dev Interface for any contract that wants to support safeTransfers\\n * from ERC721 asset contracts.\\n */\\ninterface IERC721ReceiverUpgradeable {\\n /**\\n * @dev Whenever an {IERC721} `tokenId` token is transferred to this contract via {IERC721-safeTransferFrom}\\n * by `operator` from `from`, this function is called.\\n *\\n * It must return its Solidity selector to confirm the token transfer.\\n * If any other value is returned or the interface is not implemented by the recipient, the transfer will be reverted.\\n *\\n * The selector can be obtained in Solidity with `IERC721Receiver.onERC721Received.selector`.\\n */\\n function onERC721Received(\\n address operator,\\n address from,\\n uint256 tokenId,\\n bytes calldata data\\n ) external returns (bytes4);\\n}\\n\",\"keccak256\":\"0xbb2ed8106d94aeae6858e2551a1e7174df73994b77b13ebd120ccaaef80155f5\",\"license\":\"MIT\"},\"@openzeppelin/contracts-upgradeable/token/ERC721/IERC721Upgradeable.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n// OpenZeppelin Contracts (last updated v4.7.0) (token/ERC721/IERC721.sol)\\n\\npragma solidity ^0.8.0;\\n\\nimport \\\"../../utils/introspection/IERC165Upgradeable.sol\\\";\\n\\n/**\\n * @dev Required interface of an ERC721 compliant contract.\\n */\\ninterface IERC721Upgradeable is IERC165Upgradeable {\\n /**\\n * @dev Emitted when `tokenId` token is transferred from `from` to `to`.\\n */\\n event Transfer(address indexed from, address indexed to, uint256 indexed tokenId);\\n\\n /**\\n * @dev Emitted when `owner` enables `approved` to manage the `tokenId` token.\\n */\\n event Approval(address indexed owner, address indexed approved, uint256 indexed tokenId);\\n\\n /**\\n * @dev Emitted when `owner` enables or disables (`approved`) `operator` to manage all of its assets.\\n */\\n event ApprovalForAll(address indexed owner, address indexed operator, bool approved);\\n\\n /**\\n * @dev Returns the number of tokens in ``owner``'s account.\\n */\\n function balanceOf(address owner) external view returns (uint256 balance);\\n\\n /**\\n * @dev Returns the owner of the `tokenId` token.\\n *\\n * Requirements:\\n *\\n * - `tokenId` must exist.\\n */\\n function ownerOf(uint256 tokenId) external view returns (address owner);\\n\\n /**\\n * @dev Safely transfers `tokenId` token from `from` to `to`.\\n *\\n * Requirements:\\n *\\n * - `from` cannot be the zero address.\\n * - `to` cannot be the zero address.\\n * - `tokenId` token must exist and be owned by `from`.\\n * - If the caller is not `from`, it must be approved to move this token by either {approve} or {setApprovalForAll}.\\n * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer.\\n *\\n * Emits a {Transfer} event.\\n */\\n function safeTransferFrom(\\n address from,\\n address to,\\n uint256 tokenId,\\n bytes calldata data\\n ) external;\\n\\n /**\\n * @dev Safely transfers `tokenId` token from `from` to `to`, checking first that contract recipients\\n * are aware of the ERC721 protocol to prevent tokens from being forever locked.\\n *\\n * Requirements:\\n *\\n * - `from` cannot be the zero address.\\n * - `to` cannot be the zero address.\\n * - `tokenId` token must exist and be owned by `from`.\\n * - If the caller is not `from`, it must have been allowed to move this token by either {approve} or {setApprovalForAll}.\\n * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer.\\n *\\n * Emits a {Transfer} event.\\n */\\n function safeTransferFrom(\\n address from,\\n address to,\\n uint256 tokenId\\n ) external;\\n\\n /**\\n * @dev Transfers `tokenId` token from `from` to `to`.\\n *\\n * WARNING: Usage of this method is discouraged, use {safeTransferFrom} whenever possible.\\n *\\n * Requirements:\\n *\\n * - `from` cannot be the zero address.\\n * - `to` cannot be the zero address.\\n * - `tokenId` token must be owned by `from`.\\n * - If the caller is not `from`, it must be approved to move this token by either {approve} or {setApprovalForAll}.\\n *\\n * Emits a {Transfer} event.\\n */\\n function transferFrom(\\n address from,\\n address to,\\n uint256 tokenId\\n ) external;\\n\\n /**\\n * @dev Gives permission to `to` to transfer `tokenId` token to another account.\\n * The approval is cleared when the token is transferred.\\n *\\n * Only a single account can be approved at a time, so approving the zero address clears previous approvals.\\n *\\n * Requirements:\\n *\\n * - The caller must own the token or be an approved operator.\\n * - `tokenId` must exist.\\n *\\n * Emits an {Approval} event.\\n */\\n function approve(address to, uint256 tokenId) external;\\n\\n /**\\n * @dev Approve or remove `operator` as an operator for the caller.\\n * Operators can call {transferFrom} or {safeTransferFrom} for any token owned by the caller.\\n *\\n * Requirements:\\n *\\n * - The `operator` cannot be the caller.\\n *\\n * Emits an {ApprovalForAll} event.\\n */\\n function setApprovalForAll(address operator, bool _approved) external;\\n\\n /**\\n * @dev Returns the account approved for `tokenId` token.\\n *\\n * Requirements:\\n *\\n * - `tokenId` must exist.\\n */\\n function getApproved(uint256 tokenId) external view returns (address operator);\\n\\n /**\\n * @dev Returns if the `operator` is allowed to manage all of the assets of `owner`.\\n *\\n * See {setApprovalForAll}\\n */\\n function isApprovedForAll(address owner, address operator) external view returns (bool);\\n}\\n\",\"keccak256\":\"0x016298e66a5810253c6c905e61966bb31c8775c3f3517bf946ff56ee31d6c005\",\"license\":\"MIT\"},\"@openzeppelin/contracts-upgradeable/token/ERC721/extensions/IERC721MetadataUpgradeable.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n// OpenZeppelin Contracts v4.4.1 (token/ERC721/extensions/IERC721Metadata.sol)\\n\\npragma solidity ^0.8.0;\\n\\nimport \\\"../IERC721Upgradeable.sol\\\";\\n\\n/**\\n * @title ERC-721 Non-Fungible Token Standard, optional metadata extension\\n * @dev See https://eips.ethereum.org/EIPS/eip-721\\n */\\ninterface IERC721MetadataUpgradeable is IERC721Upgradeable {\\n /**\\n * @dev Returns the token collection name.\\n */\\n function name() external view returns (string memory);\\n\\n /**\\n * @dev Returns the token collection symbol.\\n */\\n function symbol() external view returns (string memory);\\n\\n /**\\n * @dev Returns the Uniform Resource Identifier (URI) for `tokenId` token.\\n */\\n function tokenURI(uint256 tokenId) external view returns (string memory);\\n}\\n\",\"keccak256\":\"0x95a471796eb5f030fdc438660bebec121ad5d063763e64d92376ffb4b5ce8b70\",\"license\":\"MIT\"},\"@openzeppelin/contracts-upgradeable/utils/AddressUpgradeable.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n// OpenZeppelin Contracts (last updated v4.7.0) (utils/Address.sol)\\n\\npragma solidity ^0.8.1;\\n\\n/**\\n * @dev Collection of functions related to the address type\\n */\\nlibrary AddressUpgradeable {\\n /**\\n * @dev Returns true if `account` is a contract.\\n *\\n * [IMPORTANT]\\n * ====\\n * It is unsafe to assume that an address for which this function returns\\n * false is an externally-owned account (EOA) and not a contract.\\n *\\n * Among others, `isContract` will return false for the following\\n * types of addresses:\\n *\\n * - an externally-owned account\\n * - a contract in construction\\n * - an address where a contract will be created\\n * - an address where a contract lived, but was destroyed\\n * ====\\n *\\n * [IMPORTANT]\\n * ====\\n * You shouldn't rely on `isContract` to protect against flash loan attacks!\\n *\\n * Preventing calls from contracts is highly discouraged. It breaks composability, breaks support for smart wallets\\n * like Gnosis Safe, and does not provide security since it can be circumvented by calling from a contract\\n * constructor.\\n * ====\\n */\\n function isContract(address account) internal view returns (bool) {\\n // This method relies on extcodesize/address.code.length, which returns 0\\n // for contracts in construction, since the code is only stored at the end\\n // of the constructor execution.\\n\\n return account.code.length > 0;\\n }\\n\\n /**\\n * @dev Replacement for Solidity's `transfer`: sends `amount` wei to\\n * `recipient`, forwarding all available gas and reverting on errors.\\n *\\n * https://eips.ethereum.org/EIPS/eip-1884[EIP1884] increases the gas cost\\n * of certain opcodes, possibly making contracts go over the 2300 gas limit\\n * imposed by `transfer`, making them unable to receive funds via\\n * `transfer`. {sendValue} removes this limitation.\\n *\\n * https://diligence.consensys.net/posts/2019/09/stop-using-soliditys-transfer-now/[Learn more].\\n *\\n * IMPORTANT: because control is transferred to `recipient`, care must be\\n * taken to not create reentrancy vulnerabilities. Consider using\\n * {ReentrancyGuard} or the\\n * https://solidity.readthedocs.io/en/v0.5.11/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern].\\n */\\n function sendValue(address payable recipient, uint256 amount) internal {\\n require(address(this).balance >= amount, \\\"Address: insufficient balance\\\");\\n\\n (bool success, ) = recipient.call{value: amount}(\\\"\\\");\\n require(success, \\\"Address: unable to send value, recipient may have reverted\\\");\\n }\\n\\n /**\\n * @dev Performs a Solidity function call using a low level `call`. A\\n * plain `call` is an unsafe replacement for a function call: use this\\n * function instead.\\n *\\n * If `target` reverts with a revert reason, it is bubbled up by this\\n * function (like regular Solidity function calls).\\n *\\n * Returns the raw returned data. To convert to the expected return value,\\n * use https://solidity.readthedocs.io/en/latest/units-and-global-variables.html?highlight=abi.decode#abi-encoding-and-decoding-functions[`abi.decode`].\\n *\\n * Requirements:\\n *\\n * - `target` must be a contract.\\n * - calling `target` with `data` must not revert.\\n *\\n * _Available since v3.1._\\n */\\n function functionCall(address target, bytes memory data) internal returns (bytes memory) {\\n return functionCall(target, data, \\\"Address: low-level call failed\\\");\\n }\\n\\n /**\\n * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], but with\\n * `errorMessage` as a fallback revert reason when `target` reverts.\\n *\\n * _Available since v3.1._\\n */\\n function functionCall(\\n address target,\\n bytes memory data,\\n string memory errorMessage\\n ) internal returns (bytes memory) {\\n return functionCallWithValue(target, data, 0, errorMessage);\\n }\\n\\n /**\\n * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],\\n * but also transferring `value` wei to `target`.\\n *\\n * Requirements:\\n *\\n * - the calling contract must have an ETH balance of at least `value`.\\n * - the called Solidity function must be `payable`.\\n *\\n * _Available since v3.1._\\n */\\n function functionCallWithValue(\\n address target,\\n bytes memory data,\\n uint256 value\\n ) internal returns (bytes memory) {\\n return functionCallWithValue(target, data, value, \\\"Address: low-level call with value failed\\\");\\n }\\n\\n /**\\n * @dev Same as {xref-Address-functionCallWithValue-address-bytes-uint256-}[`functionCallWithValue`], but\\n * with `errorMessage` as a fallback revert reason when `target` reverts.\\n *\\n * _Available since v3.1._\\n */\\n function functionCallWithValue(\\n address target,\\n bytes memory data,\\n uint256 value,\\n string memory errorMessage\\n ) internal returns (bytes memory) {\\n require(address(this).balance >= value, \\\"Address: insufficient balance for call\\\");\\n require(isContract(target), \\\"Address: call to non-contract\\\");\\n\\n (bool success, bytes memory returndata) = target.call{value: value}(data);\\n return verifyCallResult(success, returndata, errorMessage);\\n }\\n\\n /**\\n * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],\\n * but performing a static call.\\n *\\n * _Available since v3.3._\\n */\\n function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) {\\n return functionStaticCall(target, data, \\\"Address: low-level static call failed\\\");\\n }\\n\\n /**\\n * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],\\n * but performing a static call.\\n *\\n * _Available since v3.3._\\n */\\n function functionStaticCall(\\n address target,\\n bytes memory data,\\n string memory errorMessage\\n ) internal view returns (bytes memory) {\\n require(isContract(target), \\\"Address: static call to non-contract\\\");\\n\\n (bool success, bytes memory returndata) = target.staticcall(data);\\n return verifyCallResult(success, returndata, errorMessage);\\n }\\n\\n /**\\n * @dev Tool to verifies that a low level call was successful, and revert if it wasn't, either by bubbling the\\n * revert reason using the provided one.\\n *\\n * _Available since v4.3._\\n */\\n function verifyCallResult(\\n bool success,\\n bytes memory returndata,\\n string memory errorMessage\\n ) internal pure returns (bytes memory) {\\n if (success) {\\n return returndata;\\n } else {\\n // Look for revert reason and bubble it up if present\\n if (returndata.length > 0) {\\n // The easiest way to bubble the revert reason is using memory via assembly\\n /// @solidity memory-safe-assembly\\n assembly {\\n let returndata_size := mload(returndata)\\n revert(add(32, returndata), returndata_size)\\n }\\n } else {\\n revert(errorMessage);\\n }\\n }\\n }\\n}\\n\",\"keccak256\":\"0x611aa3f23e59cfdd1863c536776407b3e33d695152a266fa7cfb34440a29a8a3\",\"license\":\"MIT\"},\"@openzeppelin/contracts-upgradeable/utils/ContextUpgradeable.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n// OpenZeppelin Contracts v4.4.1 (utils/Context.sol)\\n\\npragma solidity ^0.8.0;\\nimport \\\"../proxy/utils/Initializable.sol\\\";\\n\\n/**\\n * @dev Provides information about the current execution context, including the\\n * sender of the transaction and its data. While these are generally available\\n * via msg.sender and msg.data, they should not be accessed in such a direct\\n * manner, since when dealing with meta-transactions the account sending and\\n * paying for execution may not be the actual sender (as far as an application\\n * is concerned).\\n *\\n * This contract is only required for intermediate, library-like contracts.\\n */\\nabstract contract ContextUpgradeable is Initializable {\\n function __Context_init() internal onlyInitializing {\\n }\\n\\n function __Context_init_unchained() internal onlyInitializing {\\n }\\n function _msgSender() internal view virtual returns (address) {\\n return msg.sender;\\n }\\n\\n function _msgData() internal view virtual returns (bytes calldata) {\\n return msg.data;\\n }\\n\\n /**\\n * @dev This empty reserved space is put in place to allow future versions to add new\\n * variables without shifting down storage in the inheritance chain.\\n * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps\\n */\\n uint256[50] private __gap;\\n}\\n\",\"keccak256\":\"0x963ea7f0b48b032eef72fe3a7582edf78408d6f834115b9feadd673a4d5bd149\",\"license\":\"MIT\"},\"@openzeppelin/contracts-upgradeable/utils/StringsUpgradeable.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n// OpenZeppelin Contracts (last updated v4.7.0) (utils/Strings.sol)\\n\\npragma solidity ^0.8.0;\\n\\n/**\\n * @dev String operations.\\n */\\nlibrary StringsUpgradeable {\\n bytes16 private constant _HEX_SYMBOLS = \\\"0123456789abcdef\\\";\\n uint8 private constant _ADDRESS_LENGTH = 20;\\n\\n /**\\n * @dev Converts a `uint256` to its ASCII `string` decimal representation.\\n */\\n function toString(uint256 value) internal pure returns (string memory) {\\n // Inspired by OraclizeAPI's implementation - MIT licence\\n // https://github.com/oraclize/ethereum-api/blob/b42146b063c7d6ee1358846c198246239e9360e8/oraclizeAPI_0.4.25.sol\\n\\n if (value == 0) {\\n return \\\"0\\\";\\n }\\n uint256 temp = value;\\n uint256 digits;\\n while (temp != 0) {\\n digits++;\\n temp /= 10;\\n }\\n bytes memory buffer = new bytes(digits);\\n while (value != 0) {\\n digits -= 1;\\n buffer[digits] = bytes1(uint8(48 + uint256(value % 10)));\\n value /= 10;\\n }\\n return string(buffer);\\n }\\n\\n /**\\n * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation.\\n */\\n function toHexString(uint256 value) internal pure returns (string memory) {\\n if (value == 0) {\\n return \\\"0x00\\\";\\n }\\n uint256 temp = value;\\n uint256 length = 0;\\n while (temp != 0) {\\n length++;\\n temp >>= 8;\\n }\\n return toHexString(value, length);\\n }\\n\\n /**\\n * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation with fixed length.\\n */\\n function toHexString(uint256 value, uint256 length) internal pure returns (string memory) {\\n bytes memory buffer = new bytes(2 * length + 2);\\n buffer[0] = \\\"0\\\";\\n buffer[1] = \\\"x\\\";\\n for (uint256 i = 2 * length + 1; i > 1; --i) {\\n buffer[i] = _HEX_SYMBOLS[value & 0xf];\\n value >>= 4;\\n }\\n require(value == 0, \\\"Strings: hex length insufficient\\\");\\n return string(buffer);\\n }\\n\\n /**\\n * @dev Converts an `address` with fixed length of 20 bytes to its not checksummed ASCII `string` hexadecimal representation.\\n */\\n function toHexString(address addr) internal pure returns (string memory) {\\n return toHexString(uint256(uint160(addr)), _ADDRESS_LENGTH);\\n }\\n}\\n\",\"keccak256\":\"0xea5339a7fff0ed42b45be56a88efdd0b2ddde9fa480dc99fef9a6a4c5b776863\",\"license\":\"MIT\"},\"@openzeppelin/contracts-upgradeable/utils/introspection/ERC165Upgradeable.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n// OpenZeppelin Contracts v4.4.1 (utils/introspection/ERC165.sol)\\n\\npragma solidity ^0.8.0;\\n\\nimport \\\"./IERC165Upgradeable.sol\\\";\\nimport \\\"../../proxy/utils/Initializable.sol\\\";\\n\\n/**\\n * @dev Implementation of the {IERC165} interface.\\n *\\n * Contracts that want to implement ERC165 should inherit from this contract and override {supportsInterface} to check\\n * for the additional interface id that will be supported. For example:\\n *\\n * ```solidity\\n * function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) {\\n * return interfaceId == type(MyInterface).interfaceId || super.supportsInterface(interfaceId);\\n * }\\n * ```\\n *\\n * Alternatively, {ERC165Storage} provides an easier to use but more expensive implementation.\\n */\\nabstract contract ERC165Upgradeable is Initializable, IERC165Upgradeable {\\n function __ERC165_init() internal onlyInitializing {\\n }\\n\\n function __ERC165_init_unchained() internal onlyInitializing {\\n }\\n /**\\n * @dev See {IERC165-supportsInterface}.\\n */\\n function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) {\\n return interfaceId == type(IERC165Upgradeable).interfaceId;\\n }\\n\\n /**\\n * @dev This empty reserved space is put in place to allow future versions to add new\\n * variables without shifting down storage in the inheritance chain.\\n * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps\\n */\\n uint256[50] private __gap;\\n}\\n\",\"keccak256\":\"0x9a3b990bd56d139df3e454a9edf1c64668530b5a77fc32eb063bc206f958274a\",\"license\":\"MIT\"},\"@openzeppelin/contracts-upgradeable/utils/introspection/IERC165Upgradeable.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n// OpenZeppelin Contracts v4.4.1 (utils/introspection/IERC165.sol)\\n\\npragma solidity ^0.8.0;\\n\\n/**\\n * @dev Interface of the ERC165 standard, as defined in the\\n * https://eips.ethereum.org/EIPS/eip-165[EIP].\\n *\\n * Implementers can declare support of contract interfaces, which can then be\\n * queried by others ({ERC165Checker}).\\n *\\n * For an implementation, see {ERC165}.\\n */\\ninterface IERC165Upgradeable {\\n /**\\n * @dev Returns true if this contract implements the interface defined by\\n * `interfaceId`. See the corresponding\\n * https://eips.ethereum.org/EIPS/eip-165#how-interfaces-are-identified[EIP section]\\n * to learn more about how these ids are created.\\n *\\n * This function call must use less than 30 000 gas.\\n */\\n function supportsInterface(bytes4 interfaceId) external view returns (bool);\\n}\\n\",\"keccak256\":\"0xc6cef87559d0aeffdf0a99803de655938a7779ec0a3cd5d4383483ad85565a09\",\"license\":\"MIT\"},\"base64-sol/base64.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n\\npragma solidity >=0.6.0;\\n\\n/// @title Base64\\n/// @author Brecht Devos - \\n/// @notice Provides functions for encoding/decoding base64\\nlibrary Base64 {\\n string internal constant TABLE_ENCODE = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/';\\n bytes internal constant TABLE_DECODE = hex\\\"0000000000000000000000000000000000000000000000000000000000000000\\\"\\n hex\\\"00000000000000000000003e0000003f3435363738393a3b3c3d000000000000\\\"\\n hex\\\"00000102030405060708090a0b0c0d0e0f101112131415161718190000000000\\\"\\n hex\\\"001a1b1c1d1e1f202122232425262728292a2b2c2d2e2f303132330000000000\\\";\\n\\n function encode(bytes memory data) internal pure returns (string memory) {\\n if (data.length == 0) return '';\\n\\n // load the table into memory\\n string memory table = TABLE_ENCODE;\\n\\n // multiply by 4/3 rounded up\\n uint256 encodedLen = 4 * ((data.length + 2) / 3);\\n\\n // add some extra buffer at the end required for the writing\\n string memory result = new string(encodedLen + 32);\\n\\n assembly {\\n // set the actual output length\\n mstore(result, encodedLen)\\n\\n // prepare the lookup table\\n let tablePtr := add(table, 1)\\n\\n // input ptr\\n let dataPtr := data\\n let endPtr := add(dataPtr, mload(data))\\n\\n // result ptr, jump over length\\n let resultPtr := add(result, 32)\\n\\n // run over the input, 3 bytes at a time\\n for {} lt(dataPtr, endPtr) {}\\n {\\n // read 3 bytes\\n dataPtr := add(dataPtr, 3)\\n let input := mload(dataPtr)\\n\\n // write 4 characters\\n mstore8(resultPtr, mload(add(tablePtr, and(shr(18, input), 0x3F))))\\n resultPtr := add(resultPtr, 1)\\n mstore8(resultPtr, mload(add(tablePtr, and(shr(12, input), 0x3F))))\\n resultPtr := add(resultPtr, 1)\\n mstore8(resultPtr, mload(add(tablePtr, and(shr( 6, input), 0x3F))))\\n resultPtr := add(resultPtr, 1)\\n mstore8(resultPtr, mload(add(tablePtr, and( input, 0x3F))))\\n resultPtr := add(resultPtr, 1)\\n }\\n\\n // padding with '='\\n switch mod(mload(data), 3)\\n case 1 { mstore(sub(resultPtr, 2), shl(240, 0x3d3d)) }\\n case 2 { mstore(sub(resultPtr, 1), shl(248, 0x3d)) }\\n }\\n\\n return result;\\n }\\n\\n function decode(string memory _data) internal pure returns (bytes memory) {\\n bytes memory data = bytes(_data);\\n\\n if (data.length == 0) return new bytes(0);\\n require(data.length % 4 == 0, \\\"invalid base64 decoder input\\\");\\n\\n // load the table into memory\\n bytes memory table = TABLE_DECODE;\\n\\n // every 4 characters represent 3 bytes\\n uint256 decodedLen = (data.length / 4) * 3;\\n\\n // add some extra buffer at the end required for the writing\\n bytes memory result = new bytes(decodedLen + 32);\\n\\n assembly {\\n // padding with '='\\n let lastBytes := mload(add(data, mload(data)))\\n if eq(and(lastBytes, 0xFF), 0x3d) {\\n decodedLen := sub(decodedLen, 1)\\n if eq(and(lastBytes, 0xFFFF), 0x3d3d) {\\n decodedLen := sub(decodedLen, 1)\\n }\\n }\\n\\n // set the actual output length\\n mstore(result, decodedLen)\\n\\n // prepare the lookup table\\n let tablePtr := add(table, 1)\\n\\n // input ptr\\n let dataPtr := data\\n let endPtr := add(dataPtr, mload(data))\\n\\n // result ptr, jump over length\\n let resultPtr := add(result, 32)\\n\\n // run over the input, 4 characters at a time\\n for {} lt(dataPtr, endPtr) {}\\n {\\n // read 4 characters\\n dataPtr := add(dataPtr, 4)\\n let input := mload(dataPtr)\\n\\n // write 3 bytes\\n let output := add(\\n add(\\n shl(18, and(mload(add(tablePtr, and(shr(24, input), 0xFF))), 0xFF)),\\n shl(12, and(mload(add(tablePtr, and(shr(16, input), 0xFF))), 0xFF))),\\n add(\\n shl( 6, and(mload(add(tablePtr, and(shr( 8, input), 0xFF))), 0xFF)),\\n and(mload(add(tablePtr, and( input , 0xFF))), 0xFF)\\n )\\n )\\n mstore(resultPtr, shl(232, output))\\n resultPtr := add(resultPtr, 3)\\n }\\n }\\n\\n return result;\\n }\\n}\\n\",\"keccak256\":\"0xa73959e6ef0b693e4423a562e612370160b934a75e618361ddd8c9c4b8ddbaaf\",\"license\":\"MIT\"},\"contracts/ExpandedNFT.sol\":{\"content\":\"// SPDX-License-Identifier: GPL-3.0\\n\\n/**\\n\\n ExpandedNFTs\\n\\n */\\n\\npragma solidity ^0.8.15;\\n\\nimport {ERC721Upgradeable} from \\\"@openzeppelin/contracts-upgradeable/token/ERC721/ERC721Upgradeable.sol\\\";\\nimport {IERC2981Upgradeable, IERC165Upgradeable} from \\\"@openzeppelin/contracts-upgradeable/interfaces/IERC2981Upgradeable.sol\\\";\\nimport {OwnableUpgradeable} from \\\"@openzeppelin/contracts-upgradeable/access/OwnableUpgradeable.sol\\\";\\nimport {AddressUpgradeable} from \\\"@openzeppelin/contracts-upgradeable/utils/AddressUpgradeable.sol\\\";\\nimport {IERC20Upgradeable} from \\\"@openzeppelin/contracts-upgradeable/token/ERC20/IERC20Upgradeable.sol\\\";\\n\\nimport {SharedNFTLogic} from \\\"./SharedNFTLogic.sol\\\";\\nimport {IExpandedNFT} from \\\"./IExpandedNFT.sol\\\";\\n/**\\n This is a smart contract for handling dynamic contract minting.\\n\\n @dev This allows creators to mint a unique serial drop of an expanded NFT within a custom contract\\n @author Zien\\n Repository: https://github.com/joinzien/expanded-nft\\n*/\\ncontract ExpandedNFT is\\n ERC721Upgradeable,\\n IExpandedNFT,\\n IERC2981Upgradeable,\\n OwnableUpgradeable\\n{\\n enum WhoCanMint{ ONLY_OWNER, VIPS, MEMBERS, ANYONE }\\n\\n enum ExpandedNFTStates{ UNMINTED, MINTED, REDEEM_STARTED, SET_OFFER_TERMS, ACCEPTED_OFFER, PRODUCTION_COMPLETE, REDEEMED }\\n \\n event PriceChanged(uint256 amount);\\n event EditionSold(uint256 price, address owner);\\n event WhoCanMintChanged(WhoCanMint minters);\\n\\n // State change events\\n event RedeemStarted(uint256 tokenId, address owner);\\n event RedeemAborted(uint256 tokenId, address owner); \\n event OfferTermsSet(uint256 tokenId);\\n event OfferAccepted(uint256 tokenId);\\n event OfferRejected(uint256 tokenId);\\n event ProductionComplete(uint256 tokenId);\\n event DeliveryAccepted(uint256 tokenId);\\n\\n struct PerToken { \\n // Hashmap of the Edition ID to the current \\n ExpandedNFTStates editionState;\\n\\n // Redemption price\\n uint256 editionFee; \\n\\n // Edition description\\n string description;\\n\\n // Minted\\n\\n // animation_url field in the metadata\\n string animationUrl;\\n // Hash for the associated animation\\n bytes32 animationHash;\\n // Image in the metadata\\n string imageUrl;\\n // Hash for the associated image\\n bytes32 imageHash;\\n\\n // Redeemed\\n\\n // animation_url field in the metadata\\n string redeemedAnimationUrl;\\n // Hash for the associated animation\\n bytes32 redeemedAnimationHash;\\n // Image in the metadata\\n string redeemedImageUrl;\\n // Hash for the associated image\\n bytes32 redeemedImageHash;\\n // Condition report in the metadata\\n string conditionReportUrl;\\n // Hash for the condition report\\n bytes32 conditionReportHash;\\n }\\n\\n struct Pricing { \\n // Royalty amount in bps\\n uint256 royaltyBPS;\\n\\n // Split amount to the platforms. the artist in bps\\n uint256 splitBPS;\\n\\n // Price for VIP sales\\n uint256 vipSalePrice;\\n\\n // Price for member sales\\n uint256 membersSalePrice; \\n\\n // Price for VIP sales\\n uint256 vipMintLimit;\\n\\n // Price for member sales\\n uint256 membersMintLimit;\\n\\n // Price for general sales\\n uint256 generalMintLimit; \\n\\n // Addresses allowed to mint edition\\n mapping(address => bool) allowedMinters;\\n // VIP Addresses allowed to mint edition\\n mapping(address => bool) vipAllowedMinters;\\n\\n // Who can currently mint\\n WhoCanMint whoCanMint;\\n\\n // Mint counts for each address\\n mapping(address => uint256) mintCounts; \\n }\\n\\n // metadata\\n string public description;\\n\\n // Artists wallet address\\n address private _artistWallet;\\n\\n // Per Token data\\n mapping(uint256 => PerToken) private _perTokenMetadata;\\n\\n // Total size of the drop that can be minted\\n uint256 public dropSize;\\n\\n uint256 private _loadedMetadata;\\n\\n // reservation list\\n uint256 private _reserveCount;\\n mapping(uint256 => address) private _reserveAddress;\\n mapping(uint256 => uint256) private _reserveTokenId;\\n\\n mapping(uint256 => bool) private _tokenClaimed; \\n uint256 private _claimCount; \\n uint256 private _currentIndex;\\n\\n Pricing private _pricing;\\n\\n // Price for general sales\\n uint256 public salePrice;\\n\\n // ERC20 interface for the payment token\\n IERC20Upgradeable private _paymentTokenERC20;\\n\\n // NFT rendering logic contract\\n SharedNFTLogic private immutable _sharedNFTLogic;\\n\\n // Global constructor for factory\\n constructor(SharedNFTLogic sharedNFTLogic) {\\n _sharedNFTLogic = sharedNFTLogic;\\n _pricing.whoCanMint = WhoCanMint.ONLY_OWNER;\\n }\\n\\n /**\\n @param _owner wallet addres for the user that owns and can mint the drop, gets royalty and sales payouts and can update the base url if needed.\\n @param artistWallet wallet address for thr User that created the drop\\n @param _name Name of drop, used in the title as \\\"$NAME NUMBER/TOTAL\\\"\\n @param _symbol Symbol of the new token contract\\n @param _dropSize Number of editions that can be minted in total. \\n @dev Function to create a new drop. Can only be called by the allowed creator\\n Sets the only allowed minter to the address that creates/owns the drop.\\n This can be re-assigned or updated later\\n */\\n function initialize(\\n address _owner,\\n address artistWallet,\\n string memory _name,\\n string memory _symbol,\\n uint256 _dropSize\\n ) public initializer {\\n require(_dropSize > 0, \\\"Drop size must be > 0\\\");\\n\\n __ERC721_init(_name, _symbol);\\n __Ownable_init();\\n\\n // Set ownership to original sender of contract call\\n transferOwnership(_owner);\\n\\n _artistWallet = artistWallet;\\n dropSize = _dropSize;\\n\\n // Set edition id start to be 1 not 0\\n _claimCount = 0; \\n _currentIndex = 1;\\n\\n // Set the metadata\\n description = _name;\\n _loadedMetadata = 0; \\n }\\n\\n /**\\n @param _description Description of the edition, used in the description field of the NFT\\n @param imageUrl Image URL of the the edition. Strongly encouraged to be used, if necessary, only animation URL can be used. One of animation and image url need to exist in a drop to render the NFT.\\n @param imageHash SHA256 of the given image in bytes32 format (0xHASH). If no image is included, the hash can be zero.\\n @param animationUrl Animation URL of the edition. Not required, but if omitted image URL needs to be included. This follows the opensea spec for NFTs\\n @param animationHash The associated hash of the animation in sha-256 bytes32 format. If animation is omitted the hash can be zero.\\n @dev Function to create a new drop. Can only be called by the allowed creator\\n Sets the only allowed minter to the address that creates/owns the drop.\\n This can be re-assigned or updated later\\n */\\n function loadMetadataChunk(\\n uint256 startOffset,\\n uint256 count,\\n string[] memory _description,\\n string[] memory animationUrl,\\n bytes32[] memory animationHash,\\n string[] memory imageUrl,\\n bytes32[] memory imageHash\\n\\n ) public {\\n require(_description.length == count, \\\"Data size mismatch\\\");\\n require(animationUrl.length == count, \\\"Data size mismatch\\\");\\n require(animationHash.length == count, \\\"Data size mismatch\\\");\\n require(imageUrl.length == count, \\\"Data size mismatch\\\");\\n require(imageHash.length == count, \\\"Data size mismatch\\\");\\n\\n for (uint i = 0; i < count; i++) {\\n uint index = startOffset + i + 1;\\n \\n _perTokenMetadata[index].description = _description[i];\\n _perTokenMetadata[index].imageUrl = imageUrl[i];\\n _perTokenMetadata[index].imageHash = imageHash[i];\\n _perTokenMetadata[index].animationUrl = animationUrl[i];\\n _perTokenMetadata[index].animationHash = animationHash[i];\\n\\n }\\n\\n _loadedMetadata += count;\\n }\\n\\n function metadataloaded() public view returns (bool){\\n return (_loadedMetadata >= dropSize);\\n }\\n\\n /// @dev returns the number of minted tokens within the drop\\n function totalSupply() public view returns (uint256) {\\n return _claimCount;\\n }\\n\\n /// @dev returns the royalty BPS\\n function getRoyaltyBPS() public view returns (uint256) {\\n return _pricing.royaltyBPS;\\n }\\n\\n /// @dev returns the split BPS\\n function getSplitBPS() public view returns (uint256) {\\n return _pricing.splitBPS;\\n }\\n\\n /// @dev returns the VIP sale price\\n function getVIPSalePrice() public view returns (uint256) {\\n return _pricing.vipSalePrice;\\n }\\n\\n /// @dev returns the member sale price\\n function getMembersSalePrice() public view returns (uint256) {\\n return _pricing.membersSalePrice;\\n }\\n\\n /// @dev returns the VIP mint limit\\n function getVIPMintLimit() public view returns (uint256) {\\n return _pricing.vipMintLimit;\\n }\\n\\n /// @dev returns the member mint limit\\n function getMembersMintLimit() public view returns (uint256) {\\n return _pricing.membersMintLimit;\\n }\\n\\n /// @dev returns the general mint limit\\n function getGeneralMintLimit() public view returns (uint256) {\\n return salePrice;\\n }\\n\\n /// @dev returns who can mint\\n function getWhoCanMint() public view returns (uint256) {\\n return uint256(_pricing.whoCanMint);\\n }\\n\\n /**\\n Simple eth-based sales function\\n More complex sales functions can be implemented through IExpandedNFT interface\\n */\\n\\n /**\\n @dev This allows the user to purchase an edition\\n at the given price in the contract.\\n */\\n\\n function purchase() external payable returns (uint256) {\\n uint256 currentPrice = _currentSalesPrice();\\n emit EditionSold(currentPrice, msg.sender);\\n\\n address[] memory toMint = new address[](1);\\n toMint[0] = msg.sender;\\n\\n return _mintEditionsBody(toMint); \\n }\\n\\n /**\\n @param to address to send the newly minted edition to\\n @dev This mints one edition to the given address by an allowed minter on the edition instance.\\n */\\n function mintEdition(address to) external payable override returns (uint256) {\\n address[] memory toMint = new address[](1);\\n toMint[0] = to;\\n\\n return _mintEditionsBody(toMint); \\n }\\n\\n /**\\n @param recipients list of addresses to send the newly minted editions to\\n @dev This mints multiple editions to the given list of addresses.\\n */\\n function mintEditions(address[] memory recipients)\\n external payable override returns (uint256)\\n {\\n return _mintEditionsBody(recipients);\\n } \\n\\n /**\\n @param recipients list of addresses to send the newly minted editions to\\n @dev This mints multiple editions to the given list of addresses.\\n */\\n function _mintEditionsBody(address[] memory recipients)\\n internal returns (uint256)\\n {\\n require(_loadedMetadata >= dropSize, \\\"Not all metadata loaded\\\");\\n\\n require(_isAllowedToMint(), \\\"Needs to be an allowed minter\\\");\\n\\n uint256 currentPrice = _currentSalesPrice();\\n require(currentPrice > 0, \\\"Not for sale\\\");\\n require(msg.value == (currentPrice * recipients.length), \\\"Wrong price\\\");\\n\\n require((_pricing.mintCounts[msg.sender] + recipients.length - 1) < _currentMintLimit(), \\\"Exceeded mint limit\\\");\\n\\n require(_claimCount + recipients.length <= dropSize, \\\"Over drop size\\\");\\n\\n if (_pricing.whoCanMint == WhoCanMint.VIPS) {\\n return _vipMintEditions(recipients);\\n }\\n\\n return _mintEditions(recipients);\\n } \\n\\n /**\\n @dev Private function to mint without any access checks.\\n Called by the public edition minting functions.\\n */\\n function _vipMintEditions(address[] memory recipients)\\n internal\\n returns (uint256)\\n {\\n address currentMinter = msg.sender;\\n\\n uint256 unclaimed = 0;\\n uint256 firstUnclaimed = _reserveCount;\\n\\n for (uint256 r = 0; r < _reserveCount; r++) {\\n if (_reserveAddress[r] == currentMinter) {\\n uint256 id = _reserveTokenId[r];\\n\\n if (_tokenClaimed[id] != true) {\\n if (r < firstUnclaimed) {\\n firstUnclaimed = r; \\n }\\n\\n unclaimed++;\\n }\\n }\\n }\\n\\n require(unclaimed >= recipients.length, \\\"Can not mint all editions\\\");\\n\\n uint256 idToMint = 1;\\n\\n uint256 reservationCounter = firstUnclaimed;\\n for (uint256 i = 0; i < recipients.length; i++) {\\n while (_reserveAddress[reservationCounter] != currentMinter) {\\n reservationCounter++;\\n } \\n\\n idToMint = _reserveTokenId[reservationCounter];\\n\\n _mint(recipients[i], idToMint);\\n\\n _perTokenMetadata[idToMint].editionState = ExpandedNFTStates.MINTED;\\n _tokenClaimed[idToMint] = true;\\n _pricing.mintCounts[currentMinter]++;\\n _claimCount++;\\n\\n reservationCounter++;\\n }\\n\\n return idToMint; \\n } \\n\\n /**\\n @dev Private function to mint without any access checks.\\n Called by the public edition minting functions.\\n */\\n function _mintEditions(address[] memory recipients)\\n internal\\n returns (uint256)\\n {\\n address currentMinter = msg.sender;\\n \\n for (uint256 i = 0; i < recipients.length; i++) {\\n while (_tokenClaimed[_currentIndex] == true) {\\n _currentIndex++;\\n } \\n\\n _mint(recipients[i], _currentIndex);\\n\\n _perTokenMetadata[_currentIndex].editionState = ExpandedNFTStates.MINTED;\\n _tokenClaimed[_currentIndex] = true;\\n _pricing.mintCounts[currentMinter]++;\\n _claimCount++;\\n }\\n\\n return _currentIndex; \\n } \\n\\n /**\\n @param _royaltyBPS BPS of the royalty set on the contract. Can be 0 for no royalty.\\n @param _splitBPS BPS of the royalty set on the contract. Can be 0 for no royalty. \\n @param _vipSalePrice Sale price for VIPs\\n @param _membersSalePrice SalePrice for Members \\n @param _generalSalePrice SalePrice for the general public \\n @param _vipMintLimit Mint limit for VIPs\\n @param _membersMintLimit Mint limit for Members \\n @param _generalMintLimit Mint limit for the general public \\n @dev Set various pricing related values\\n */\\n function setPricing (\\n uint256 _royaltyBPS,\\n uint256 _splitBPS,\\n uint256 _vipSalePrice,\\n uint256 _membersSalePrice, \\n uint256 _generalSalePrice,\\n uint256 _vipMintLimit,\\n uint256 _membersMintLimit,\\n uint256 _generalMintLimit \\n ) external onlyOwner { \\n _pricing.royaltyBPS = _royaltyBPS;\\n _pricing.splitBPS = _splitBPS;\\n\\n _pricing.vipSalePrice = _vipSalePrice;\\n _pricing.membersSalePrice = _membersSalePrice;\\n salePrice = _generalSalePrice;\\n\\n _pricing.vipMintLimit = _vipMintLimit;\\n _pricing.membersMintLimit = _membersMintLimit;\\n _pricing.generalMintLimit = _generalMintLimit;\\n\\n emit PriceChanged(salePrice);\\n }\\n\\n /**\\n @dev returns the current ETH sales price\\n based on who can currently mint.\\n */\\n function _currentSalesPrice() internal view returns (uint256){\\n if (_pricing.whoCanMint == WhoCanMint.VIPS) {\\n return _pricing.vipSalePrice;\\n } else if (_pricing.whoCanMint == WhoCanMint.MEMBERS) {\\n return _pricing.membersSalePrice;\\n } else if (_pricing.whoCanMint == WhoCanMint.ANYONE) {\\n return salePrice;\\n } \\n \\n return 0; \\n }\\n\\n /**\\n @param wallets A list of wallets\\n @param tokenIDs A list of tokenId to reserve \\n @dev Set various pricing related values\\n */\\n function reserve (address[] calldata wallets, uint256[] calldata tokenIDs) \\n external onlyOwner { \\n for (uint256 i = 0; i < wallets.length; i++) {\\n _reserveAddress[_reserveCount] = wallets[i]; \\n _reserveTokenId[_reserveCount] = tokenIDs[i]; \\n _reserveCount++;\\n }\\n }\\n\\n /**\\n @dev returns the current loimit on edition that \\n can be minted by one wallet\\n */\\n function _currentMintLimit() internal view returns (uint256){\\n if (_pricing.whoCanMint == WhoCanMint.VIPS) {\\n return _pricing.vipMintLimit;\\n } else if (_pricing.whoCanMint == WhoCanMint.MEMBERS) {\\n return _pricing.membersMintLimit;\\n } else if (_pricing.whoCanMint == WhoCanMint.ANYONE) {\\n return _pricing.generalMintLimit;\\n } \\n \\n return 0; \\n }\\n\\n /**\\n @param _salePrice if sale price is 0 sale is stopped, otherwise that amount \\n of ETH is needed to start the sale.\\n @dev This sets a simple ETH sales price\\n Setting a sales price allows users to mint the drop until it sells out.\\n For more granular sales, use an external sales contract.\\n */\\n function setSalePrice(uint256 _salePrice) external onlyOwner {\\n salePrice = _salePrice;\\n\\n _pricing.whoCanMint = WhoCanMint.ANYONE;\\n\\n emit WhoCanMintChanged(_pricing.whoCanMint);\\n emit PriceChanged(salePrice);\\n }\\n\\n /**\\n @param _salePrice if sale price is 0 sale is stopped, otherwise that amount \\n of ETH is needed to start the sale.\\n @dev This sets the VIP ETH sales price\\n Setting a sales price allows users to mint the drop until it sells out.\\n For more granular sales, use an external sales contract.\\n */\\n function setVIPSalePrice(uint256 _salePrice) external onlyOwner {\\n _pricing.vipSalePrice = _salePrice;\\n\\n _pricing.whoCanMint = WhoCanMint.VIPS;\\n\\n emit WhoCanMintChanged(_pricing.whoCanMint);\\n emit PriceChanged(salePrice);\\n }\\n\\n /**\\n @param _salePrice if sale price is 0 sale is stopped, otherwise that amount \\n of ETH is needed to start the sale.\\n @dev This sets the members ETH sales price\\n Setting a sales price allows users to mint the drop until it sells out.\\n For more granular sales, use an external sales contract.\\n */\\n function setMembersSalePrice(uint256 _salePrice) external onlyOwner {\\n _pricing.membersSalePrice = _salePrice;\\n\\n _pricing.whoCanMint = WhoCanMint.MEMBERS;\\n\\n emit WhoCanMintChanged(_pricing.whoCanMint);\\n emit PriceChanged(salePrice);\\n } \\n\\n\\n /**\\n @param vipSalePrice if sale price is 0 sale is stopped, otherwise that amount \\n of ETH is needed to start the sale.\\n @param membersSalePrice if sale price is 0 sale is stopped, otherwise that amount \\n of ETH is needed to start the sale.\\n @param generalSalePrice if sale price is 0 sale is stopped, otherwise that amount \\n of ETH is needed to start the sale. \\n @dev This sets the members ETH sales price\\n Setting a sales price allows users to mint the drop until it sells out.\\n For more granular sales, use an external sales contract.\\n */\\n function setSalePrices(uint256 vipSalePrice, uint256 membersSalePrice, uint256 generalSalePrice) external onlyOwner {\\n _pricing.vipSalePrice = vipSalePrice;\\n _pricing.membersSalePrice = membersSalePrice;\\n salePrice = generalSalePrice; \\n\\n emit PriceChanged(salePrice);\\n } \\n\\n /**\\n @dev This withdraws ETH from the contract to the contract owner.\\n */\\n function withdraw() external onlyOwner {\\n uint256 currentBalance = address(this).balance;\\n if (currentBalance > 0) {\\n uint256 platformFee = (currentBalance * _pricing.splitBPS) / 10000;\\n uint256 artistFee = currentBalance - platformFee;\\n\\n AddressUpgradeable.sendValue(payable(owner()), platformFee);\\n AddressUpgradeable.sendValue(payable(_artistWallet), artistFee);\\n }\\n\\n if (address(_paymentTokenERC20) != address(0x0)) {\\n uint256 currentBalanceERC20 = _paymentTokenERC20.balanceOf(address(this));\\n if (currentBalanceERC20 > 0) {\\n uint256 platformFee = (currentBalanceERC20 * _pricing.splitBPS) / 10000;\\n uint256 artistFee = currentBalanceERC20 - platformFee;\\n\\n _paymentTokenERC20.transfer(owner(), platformFee);\\n _paymentTokenERC20.transfer(_artistWallet, artistFee);\\n }\\n }\\n }\\n\\n /**\\n @dev This helper function checks if the msg.sender is allowed to mint the\\n given edition id.\\n */\\n function _isAllowedToMint() internal view returns (bool) {\\n if (_pricing.whoCanMint == WhoCanMint.ANYONE) {\\n return true;\\n }\\n\\n if (_pricing.whoCanMint == WhoCanMint.MEMBERS) {\\n if (_pricing.vipAllowedMinters[msg.sender]) {\\n return true;\\n } \\n\\n if (_pricing.allowedMinters[msg.sender]) {\\n return true;\\n } \\n }\\n\\n if (_pricing.whoCanMint == WhoCanMint.VIPS) {\\n if (_pricing.vipAllowedMinters[msg.sender]) {\\n return true;\\n } \\n }\\n\\n if (owner() == msg.sender) {\\n return true;\\n }\\n\\n return false;\\n }\\n\\n /**\\n Simple override for owner interface.\\n */\\n function owner()\\n public\\n view\\n override(OwnableUpgradeable, IExpandedNFT)\\n returns (address)\\n {\\n return super.owner();\\n }\\n\\n /**\\n return the artists wallet address\\n */\\n function getArtistWallet()\\n public\\n view\\n returns (address)\\n {\\n return _artistWallet;\\n }\\n\\n /**\\n set the artists wallet address\\n */\\n function setArtistWallet(address wallet)\\n public\\n onlyOwner\\n {\\n _artistWallet = wallet;\\n } \\n\\n /**\\n return the payment tokens address\\n */\\n function getPaymentToken()\\n public\\n view\\n returns (address)\\n {\\n return address(_paymentTokenERC20);\\n }\\n\\n /**\\n set a new payment token address\\n */\\n function setPaymentToken(address paymentToken)\\n public\\n onlyOwner\\n {\\n if (address(_paymentTokenERC20) != address(0x0)) {\\n require(_paymentTokenERC20.balanceOf(address(this)) == 0, \\\"token must have 0 balance\\\");\\n }\\n\\n _paymentTokenERC20 = IERC20Upgradeable(paymentToken);\\n } \\n\\n /**\\n @dev Sets the types of users who is allowed to mint.\\n */\\n function getAllowedMinter() public view returns (WhoCanMint){\\n return _pricing.whoCanMint;\\n }\\n\\n /**\\n @param minters WhoCanMint enum of minter types\\n @dev Sets the types of users who is allowed to mint.\\n */\\n function setAllowedMinter(WhoCanMint minters) public onlyOwner {\\n require(((minters >= WhoCanMint.ONLY_OWNER) && (minters <= WhoCanMint.ANYONE)), \\\"Needs to be a valid minter type\\\");\\n\\n _pricing.whoCanMint = minters;\\n emit WhoCanMintChanged(minters);\\n }\\n\\n /**\\n @param minter address to set approved minting status for\\n @param allowed boolean if that address is allowed to mint\\n @dev Sets the approved minting status of the given address.\\n This requires that msg.sender is the owner of the given edition id.\\n If the ZeroAddress (address(0x0)) is set as a minter,\\n anyone will be allowed to mint.\\n This setup is similar to setApprovalForAll in the ERC721 spec.\\n */\\n function setApprovedMinters(uint256 count, address[] calldata minter, bool[] calldata allowed) public onlyOwner {\\n for (uint256 i = 0; i < count; i++) {\\n _pricing.allowedMinters[minter[i]] = allowed[i];\\n }\\n }\\n\\n /**\\n @param minter address to set approved minting status for\\n @param allowed boolean if that address is allowed to mint\\n @dev Sets the approved minting status of the given address.\\n This requires that msg.sender is the owner of the given edition id.\\n If the ZeroAddress (address(0x0)) is set as a minter,\\n anyone will be allowed to mint.\\n This setup is similar to setApprovalForAll in the ERC721 spec.\\n */\\n function setApprovedVIPMinters(uint256 count, address[] calldata minter, bool[] calldata allowed) public onlyOwner {\\n for (uint256 i = 0; i < count; i++) {\\n _pricing.vipAllowedMinters[minter[i]] = allowed[i];\\n }\\n }\\n\\n /**\\n @dev Allows for updates of edition urls by the owner of the edition.\\n Only URLs can be updated (data-uris are supported), hashes cannot be updated.\\n */\\n function updateEditionURLs(\\n uint256 tokenId,\\n string memory imageUrl,\\n string memory animationUrl\\n ) public onlyOwner {\\n _perTokenMetadata[tokenId].imageUrl = imageUrl;\\n _perTokenMetadata[tokenId].animationUrl = animationUrl;\\n }\\n\\n /// Returns the number of editions allowed to mint\\n function numberCanMint() public view override returns (uint256) {\\n return dropSize - _claimCount;\\n }\\n\\n /**\\n @param tokenId Token ID to burn\\n User burn function for token id \\n */\\n function burn(uint256 tokenId) public {\\n require(_isApprovedOrOwner(_msgSender(), tokenId), \\\"Not approved\\\");\\n _burn(tokenId);\\n }\\n\\n function redeem(uint256 tokenId) public {\\n require(_exists(tokenId), \\\"No token\\\");\\n require(_isApprovedOrOwner(_msgSender(), tokenId), \\\"Not approved\\\");\\n\\n require((_perTokenMetadata[tokenId].editionState == ExpandedNFTStates.MINTED), \\\"You currently can not redeem\\\");\\n\\n _perTokenMetadata[tokenId].editionState = ExpandedNFTStates.REDEEM_STARTED;\\n emit RedeemStarted(tokenId, _msgSender());\\n }\\n\\n function abortRedemption(uint256 tokenId) public {\\n require(_exists(tokenId), \\\"No token\\\");\\n require(_isApprovedOrOwner(_msgSender(), tokenId), \\\"Not approved\\\");\\n\\n require((_perTokenMetadata[tokenId].editionState == ExpandedNFTStates.REDEEM_STARTED), \\\"You currently can not redeem\\\");\\n\\n _perTokenMetadata[tokenId].editionState = ExpandedNFTStates.MINTED;\\n emit RedeemAborted(tokenId, _msgSender());\\n }\\n\\n function setOfferTerms(uint256 tokenId, uint256 fee) public onlyOwner {\\n require(_exists(tokenId), \\\"No token\\\"); \\n require((_perTokenMetadata[tokenId].editionState == ExpandedNFTStates.REDEEM_STARTED), \\\"Wrong state\\\");\\n\\n _perTokenMetadata[tokenId].editionState = ExpandedNFTStates.SET_OFFER_TERMS;\\n _perTokenMetadata[tokenId].editionFee = fee;\\n\\n emit OfferTermsSet(tokenId);\\n }\\n\\n function rejectOfferTerms(uint256 tokenId) public {\\n require(_exists(tokenId), \\\"No token\\\"); \\n require(_isApprovedOrOwner(_msgSender(), tokenId), \\\"Not approved\\\");\\n\\n require((_perTokenMetadata[tokenId].editionState == ExpandedNFTStates.SET_OFFER_TERMS), \\\"You currently can not redeem\\\");\\n\\n _perTokenMetadata[tokenId].editionState = ExpandedNFTStates.MINTED;\\n\\n emit OfferRejected(tokenId);\\n }\\n\\n function acceptOfferTerms(uint256 tokenId, uint256 paymentAmount) external {\\n require(_exists(tokenId), \\\"No token\\\"); \\n require(_isApprovedOrOwner(_msgSender(), tokenId), \\\"Not approved\\\");\\n\\n require((_perTokenMetadata[tokenId].editionState == ExpandedNFTStates.SET_OFFER_TERMS), \\\"You currently can not redeem\\\");\\n\\n require(paymentAmount >= _perTokenMetadata[tokenId].editionFee, \\\"Wrong price\\\");\\n require(_paymentTokenERC20.allowance(_msgSender(), address(this)) >= _perTokenMetadata[tokenId].editionFee, \\\"Insufficient allowance\\\");\\n\\n bool success = _paymentTokenERC20.transferFrom(_msgSender(), address(this), _perTokenMetadata[tokenId].editionFee);\\n require(success, \\\"Could not transfer token\\\");\\n\\n _perTokenMetadata[tokenId].editionState = ExpandedNFTStates.ACCEPTED_OFFER; \\n\\n emit OfferAccepted(tokenId);\\n }\\n\\n function productionComplete(\\n uint256 tokenId,\\n string memory _description,\\n string memory animationUrl,\\n bytes32 animationHash,\\n string memory imageUrl,\\n bytes32 imageHash, \\n string memory conditionReportUrl,\\n bytes32 conditionReportHash \\n ) public onlyOwner {\\n require(_exists(tokenId), \\\"No token\\\"); \\n require((_perTokenMetadata[tokenId].editionState == ExpandedNFTStates.ACCEPTED_OFFER), \\\"You currently can not redeem\\\");\\n\\n // Set the NFT to display as redeemed\\n _perTokenMetadata[tokenId].description = _description;\\n _perTokenMetadata[tokenId].redeemedAnimationUrl = animationUrl;\\n _perTokenMetadata[tokenId].redeemedAnimationHash = animationHash;\\n _perTokenMetadata[tokenId].redeemedImageUrl = imageUrl;\\n _perTokenMetadata[tokenId].redeemedImageHash = imageHash;\\n _perTokenMetadata[tokenId].conditionReportUrl = conditionReportUrl;\\n _perTokenMetadata[tokenId].conditionReportHash = conditionReportHash;\\n\\n _perTokenMetadata[tokenId].editionState = ExpandedNFTStates.PRODUCTION_COMPLETE;\\n\\n emit ProductionComplete(tokenId);\\n }\\n\\n function acceptDelivery(uint256 tokenId) public {\\n require(_exists(tokenId), \\\"No token\\\"); \\n require(_isApprovedOrOwner(_msgSender(), tokenId), \\\"Not approved\\\");\\n\\n require((_perTokenMetadata[tokenId].editionState == ExpandedNFTStates.PRODUCTION_COMPLETE), \\\"You currently can not redeem\\\");\\n\\n _perTokenMetadata[tokenId].editionState = ExpandedNFTStates.REDEEMED;\\n\\n emit OfferRejected(tokenId);\\n }\\n\\n /**\\n @dev Get URIs for the condition report\\n @return conditionReportUrl, conditionReportHash\\n */\\n function getConditionReport(uint256 tokenId)\\n public\\n view\\n returns (\\n string memory,\\n bytes32\\n )\\n {\\n return (_perTokenMetadata[tokenId].conditionReportUrl, _perTokenMetadata[tokenId].conditionReportHash);\\n }\\n\\n /**\\n @dev Get royalty information for token\\n @param _salePrice Sale price for the token\\n */\\n function royaltyInfo(uint256, uint256 _salePrice)\\n external\\n view\\n override\\n returns (address receiver, uint256 royaltyAmount)\\n {\\n if (owner() == address(0x0)) {\\n return (owner(), 0);\\n }\\n return (owner(), (_salePrice * _pricing.royaltyBPS) / 10_000);\\n }\\n\\n /**\\n @dev Get URIs for edition NFT\\n @return _imageUrl, _imageHash, _animationUrl, _animationHash\\n */\\n function getURIs(uint256 tokenId)\\n public\\n view\\n returns (\\n string memory,\\n bytes32,\\n string memory,\\n bytes32\\n )\\n {\\n if (_perTokenMetadata[tokenId].editionState == ExpandedNFTStates.REDEEMED) { \\n return (_perTokenMetadata[tokenId].redeemedImageUrl, _perTokenMetadata[tokenId].redeemedImageHash,\\n _perTokenMetadata[tokenId].redeemedAnimationUrl, _perTokenMetadata[tokenId].redeemedAnimationHash);\\n }\\n\\n return (_perTokenMetadata[tokenId].imageUrl, _perTokenMetadata[tokenId].imageHash,\\n _perTokenMetadata[tokenId].animationUrl, _perTokenMetadata[tokenId].animationHash);\\n }\\n\\n /**\\n @dev Get URI for given token id\\n @param tokenId token id to get uri for\\n @return base64-encoded json metadata object\\n */\\n function tokenURI(uint256 tokenId)\\n public\\n view\\n override\\n returns (string memory)\\n {\\n require(_exists(tokenId), \\\"No token\\\");\\n\\n if (_perTokenMetadata[tokenId].editionState == ExpandedNFTStates.REDEEMED) {\\n return\\n _sharedNFTLogic.createMetadataEdition(\\n name(),\\n _perTokenMetadata[tokenId].description,\\n _perTokenMetadata[tokenId].redeemedImageUrl,\\n _perTokenMetadata[tokenId].redeemedAnimationUrl,\\n tokenId,\\n dropSize\\n );\\n }\\n\\n return\\n _sharedNFTLogic.createMetadataEdition(\\n name(),\\n _perTokenMetadata[tokenId].description,\\n _perTokenMetadata[tokenId].imageUrl,\\n _perTokenMetadata[tokenId].animationUrl,\\n tokenId,\\n dropSize\\n );\\n }\\n\\n function supportsInterface(bytes4 interfaceId)\\n public\\n view\\n override(ERC721Upgradeable, IERC165Upgradeable)\\n returns (bool)\\n {\\n return\\n type(IERC2981Upgradeable).interfaceId == interfaceId ||\\n ERC721Upgradeable.supportsInterface(interfaceId);\\n }\\n}\\n\",\"keccak256\":\"0x1fb27bcf9324d5b43b44cc4a97220da3b3ffddf5cce218be885e5e731e7f1a1c\",\"license\":\"GPL-3.0\"},\"contracts/IExpandedNFT.sol\":{\"content\":\"// SPDX-License-Identifier: GPL-3.0\\npragma solidity ^0.8.15;\\n\\ninterface IExpandedNFT {\\n function mintEdition(address to) external payable returns (uint256);\\n function mintEditions(address[] memory to) external payable returns (uint256);\\n function numberCanMint() external view returns (uint256);\\n function owner() external view returns (address);\\n}\\n\",\"keccak256\":\"0xe1dc9fcdfab1ec02f10203a59e3fb819707f06515abe45674c5ecc814fc9c253\",\"license\":\"GPL-3.0\"},\"contracts/IPublicSharedMetadata.sol\":{\"content\":\"// SPDX-License-Identifier: GPL-3.0\\n\\npragma solidity ^0.8.15;\\n\\n/// Shared public library for on-chain NFT functions\\ninterface IPublicSharedMetadata {\\n /// @param unencoded bytes to base64-encode\\n function base64Encode(bytes memory unencoded)\\n external\\n pure\\n returns (string memory);\\n\\n /// Encodes the argument json bytes into base64-data uri format\\n /// @param json Raw json to base64 and turn into a data-uri\\n function encodeMetadataJSON(bytes memory json)\\n external\\n pure\\n returns (string memory);\\n\\n /// Proxy to openzeppelin's toString function\\n /// @param value number to return as a string\\n function numberToString(uint256 value)\\n external\\n pure\\n returns (string memory);\\n}\\n\",\"keccak256\":\"0x76caf66cf71c1067c845ff9de580b6e886c448915997ed1d6f37fe1a1ec489de\",\"license\":\"GPL-3.0\"},\"contracts/SharedNFTLogic.sol\":{\"content\":\"// SPDX-License-Identifier: GPL-3.0\\n\\npragma solidity ^0.8.15;\\n\\nimport {StringsUpgradeable} from \\\"@openzeppelin/contracts-upgradeable/utils/StringsUpgradeable.sol\\\";\\nimport {Base64} from \\\"base64-sol/base64.sol\\\";\\nimport {IPublicSharedMetadata} from \\\"./IPublicSharedMetadata.sol\\\";\\n\\n/// Shared NFT logic for rendering metadata associated with editions\\n/// @dev Can safely be used for generic base64Encode and numberToString functions\\ncontract SharedNFTLogic is IPublicSharedMetadata {\\n /// @param unencoded bytes to base64-encode\\n function base64Encode(bytes memory unencoded)\\n public\\n pure\\n override\\n returns (string memory)\\n {\\n return Base64.encode(unencoded);\\n }\\n\\n /// Proxy to openzeppelin's toString function\\n /// @param value number to return as a string\\n function numberToString(uint256 value)\\n public\\n pure\\n override\\n returns (string memory)\\n {\\n return StringsUpgradeable.toString(value);\\n }\\n\\n /// Generate edition metadata from storage information as base64-json blob\\n /// Combines the media data and metadata\\n /// @param name Name of NFT in metadata\\n /// @param description Description of NFT in metadata\\n /// @param imageUrl URL of image to render for edition\\n /// @param animationUrl URL of animation to render for edition\\n /// @param tokenOfEdition Token ID for specific token\\n /// @param editionSize Size of entire edition to show\\n function createMetadataEdition(\\n string memory name,\\n string memory description,\\n string memory imageUrl,\\n string memory animationUrl,\\n uint256 tokenOfEdition,\\n uint256 editionSize\\n ) external pure returns (string memory) {\\n string memory _tokenMediaData = tokenMediaData(\\n imageUrl,\\n animationUrl,\\n tokenOfEdition\\n );\\n bytes memory json = createMetadataJSON(\\n name,\\n description,\\n _tokenMediaData,\\n tokenOfEdition,\\n editionSize\\n );\\n return encodeMetadataJSON(json);\\n }\\n\\n /// Function to create the metadata json string for the nft edition\\n /// @param name Name of NFT in metadata\\n /// @param description Description of NFT in metadata\\n /// @param mediaData Data for media to include in json object\\n /// @param tokenOfEdition Token ID for specific token\\n /// @param editionSize Size of entire edition to show\\n function createMetadataJSON(\\n string memory name,\\n string memory description,\\n string memory mediaData,\\n uint256 tokenOfEdition,\\n uint256 editionSize\\n ) public pure returns (bytes memory) {\\n bytes memory editionSizeText;\\n if (editionSize > 0) {\\n editionSizeText = abi.encodePacked(\\n \\\"/\\\",\\n numberToString(editionSize)\\n );\\n }\\n return\\n // solhint-disable quotes\\n abi.encodePacked(\\n '{\\\"name\\\": \\\"',\\n name,\\n \\\" \\\",\\n numberToString(tokenOfEdition),\\n editionSizeText,\\n '\\\", \\\"',\\n 'description\\\": \\\"',\\n description,\\n '\\\", \\\"',\\n mediaData,\\n 'properties\\\": {\\\"number\\\": ',\\n numberToString(tokenOfEdition),\\n '}}'\\n );\\n // solhint-enable quotes\\n }\\n\\n /// Encodes the argument json bytes into base64-data uri format\\n /// @param json Raw json to base64 and turn into a data-uri\\n function encodeMetadataJSON(bytes memory json)\\n public\\n pure\\n override\\n returns (string memory)\\n {\\n return\\n string(\\n abi.encodePacked(\\n \\\"data:application/json;base64,\\\",\\n base64Encode(json)\\n )\\n );\\n }\\n\\n /// Generates edition metadata from storage information as base64-json blob\\n /// Combines the media data and metadata\\n /// @param imageUrl URL of image to render for edition\\n /// @param animationUrl URL of animation to render for edition\\n function tokenMediaData(\\n string memory imageUrl,\\n string memory animationUrl,\\n uint256 tokenOfEdition\\n ) public pure returns (string memory) {\\n bool hasImage = bytes(imageUrl).length > 0;\\n bool hasAnimation = bytes(animationUrl).length > 0;\\n if (hasImage && hasAnimation) {\\n return\\n // solhint-disable quotes\\n string(\\n abi.encodePacked(\\n 'image\\\": \\\"',\\n imageUrl,\\n \\\"?id=\\\",\\n numberToString(tokenOfEdition),\\n '\\\", \\\"animation_url\\\": \\\"',\\n animationUrl,\\n \\\"?id=\\\",\\n numberToString(tokenOfEdition),\\n '\\\", \\\"'\\n )\\n );\\n // solhint-enable quotes\\n }\\n if (hasImage) {\\n return\\n // solhint-disable quotes\\n string(\\n abi.encodePacked(\\n 'image\\\": \\\"',\\n imageUrl,\\n \\\"?id=\\\",\\n numberToString(tokenOfEdition),\\n '\\\", \\\"'\\n )\\n );\\n // solhint-enable quotes\\n }\\n if (hasAnimation) {\\n return\\n // solhint-disable quotes\\n string(\\n abi.encodePacked(\\n 'animation_url\\\": \\\"',\\n animationUrl,\\n \\\"?id=\\\",\\n numberToString(tokenOfEdition),\\n '\\\", \\\"'\\n )\\n );\\n // solhint-enable quotes\\n }\\n\\n return \\\"\\\";\\n }\\n}\\n\",\"keccak256\":\"0x869febd1ab7ac26132dc40cc132cc527f480131b1d2b92a4a65af60ca16660a5\",\"license\":\"GPL-3.0\"}},\"version\":1}", + "bytecode": "0x60a06040523480156200001157600080fd5b5060405162004a9038038062004a90833981016040819052620000349162000050565b6001600160a01b031660805260dd805460ff1916905562000082565b6000602082840312156200006357600080fd5b81516001600160a01b03811681146200007b57600080fd5b9392505050565b6080516149eb620000a5600039600081816124ed01526125ae01526149eb6000f3fe60806040526004361061032d5760003560e01c806369e1cac1116101a7578063b88d4fde116100ed578063db006a7511610090578063db006a7514610955578063de5d62a014610975578063e985e9c514610995578063ee5df550146109b5578063f2fde38b146109d5578063f51f96dd146109f5578063fc51e0c314610a0b578063fce57fd914610a3957600080fd5b8063b88d4fde14610873578063bcda01af14610893578063bf82ae49146108b3578063c87b56dd146108d3578063c9d359d6146108f3578063d375b6fc1461090d578063d41c3a6514610922578063d5344e361461094057600080fd5b80637b422c4c116101555780637b422c4c146107a15780638c503ca4146107c15780638da5cb5b146107e157806393d23923146107f657806395d89b4114610816578063a1279cfc1461082b578063a22cb46514610840578063a66ff0af1461086057600080fd5b806369e1cac1146106d75780636a326ab1146106f757806370a0823114610717578063715018a6146107375780637284e4161461074c578063787e5efc14610761578063799b7c5f1461078157600080fd5b8063392774151161027757806342966c681161021a57806342966c68146105f557806346d9c4bf14610615578063499e16721461062a57806352c9143e1461063f5780636352211e1461065f57806364e9293e1461067f57806364edfbf0146106af578063660243d0146106b757600080fd5b8063392774151461052257806339dc13bc146105405780633ccfd60b146105605780633da165d4146105755780633e1dc1071461058b578063402635cf146105a0578063410a1095146105b557806342842e0e146105d557600080fd5b80630f6a9349116102df5780630f6a93491461042c57806318160ddd1461043f5780631919fed71461045457806323b872dd146104745780632a55205a14610494578063367de39e146104c257806336f3ec76146104e25780633892aa521461050257600080fd5b806301ffc9a714610332578063036c8ea11461036757806306fdde0314610389578063081812fc146103ab578063095ea7b3146103d857806309949abe146103f85780630b65b6e714610417575b600080fd5b34801561033e57600080fd5b5061035261034d366004613991565b610a59565b60405190151581526020015b60405180910390f35b34801561037357600080fd5b506103876103823660046139ae565b610a84565b005b34801561039557600080fd5b5061039e610ae0565b60405161035e9190613a5b565b3480156103b757600080fd5b506103cb6103c6366004613a6e565b610b72565b60405161035e9190613a87565b3480156103e457600080fd5b506103876103f3366004613ab7565b610b99565b34801561040457600080fd5b5060d4545b60405190815260200161035e565b34801561042357600080fd5b50610409610cb3565b61040961043a366004613b4a565b610cca565b34801561044b57600080fd5b5060d254610409565b34801561046057600080fd5b5061038761046f366004613a6e565b610cd5565b34801561048057600080fd5b5061038761048f366004613be6565b610d62565b3480156104a057600080fd5b506104b46104af366004613c22565b610d94565b60405161035e929190613c44565b3480156104ce57600080fd5b506103876104dd366004613c5d565b610df0565b3480156104ee57600080fd5b506103876104fd366004613c7e565b610ecb565b34801561050e57600080fd5b5061038761051d366004613cee565b610f09565b34801561052e57600080fd5b5060ca546001600160a01b03166103cb565b34801561054c57600080fd5b5061038761055b366004613a6e565b610fb0565b34801561056c57600080fd5b5061038761108c565b34801561058157600080fd5b5061040960cc5481565b34801561059757600080fd5b5060d954610409565b3480156105ac57600080fd5b5060d554610409565b3480156105c157600080fd5b5060dd5460ff1660405161035e9190613d7d565b3480156105e157600080fd5b506103876105f0366004613be6565b6112aa565b34801561060157600080fd5b50610387610610366004613a6e565b6112c5565b34801561062157600080fd5b5060df54610409565b34801561063657600080fd5b5060d754610409565b34801561064b57600080fd5b5061038761065a366004613e25565b6112f3565b34801561066b57600080fd5b506103cb61067a366004613a6e565b611431565b34801561068b57600080fd5b5061069f61069a366004613a6e565b611466565b60405161035e9493929190613ef8565b61040961172b565b3480156106c357600080fd5b506103876106d2366004613c22565b6117d7565b3480156106e357600080fd5b506103876106f2366004613f35565b611aa0565b34801561070357600080fd5b50610387610712366004613fa0565b611b60565b34801561072357600080fd5b50610409610732366004613fa0565b611c55565b34801561074357600080fd5b50610387611cdb565b34801561075857600080fd5b5061039e611cef565b34801561076d57600080fd5b5061038761077c366004613a6e565b611d7d565b34801561078d57600080fd5b5061038761079c366004613cee565b611d9d565b3480156107ad57600080fd5b506103876107bc366004613a6e565b611e3c565b3480156107cd57600080fd5b506103876107dc366004613fbb565b611ee8565b3480156107ed57600080fd5b506103cb611f2d565b34801561080257600080fd5b50610387610811366004614027565b611f41565b34801561082257600080fd5b5061039e6120ed565b34801561083757600080fd5b5060d654610409565b34801561084c57600080fd5b5061038761085b3660046140c1565b6120fc565b61040961086e366004613fa0565b612107565b34801561087f57600080fd5b5061038761088e3660046140f8565b612170565b34801561089f57600080fd5b506103876108ae366004614258565b6121a1565b3480156108bf57600080fd5b506103876108ce366004613a6e565b6123b4565b3480156108df57600080fd5b5061039e6108ee366004613a6e565b61249a565b3480156108ff57600080fd5b5060cc5460cd541015610352565b34801561091957600080fd5b50610409612628565b34801561092e57600080fd5b5060e0546001600160a01b03166103cb565b34801561094c57600080fd5b5060d854610409565b34801561096157600080fd5b50610387610970366004613a6e565b612642565b34801561098157600080fd5b50610387610990366004613a6e565b61270f565b3480156109a157600080fd5b506103526109b036600461433c565b612730565b3480156109c157600080fd5b506103876109d0366004613c22565b61275e565b3480156109e157600080fd5b506103876109f0366004613fa0565b612842565b348015610a0157600080fd5b5061040960df5481565b348015610a1757600080fd5b50610a2b610a26366004613a6e565b6128b8565b60405161035e92919061436f565b348015610a4557600080fd5b50610387610a54366004613fa0565b61296b565b600063152a902d60e11b6001600160e01b031983161480610a7e5750610a7e82612995565b92915050565b610a8c6129e5565b60d488905560d587905560d686905560d785905560df84905560d883905560d982905560da819055604051848152600080516020614996833981519152906020015b60405180910390a15050505050505050565b606060658054610aef90614391565b80601f0160208091040260200160405190810160405280929190818152602001828054610b1b90614391565b8015610b685780601f10610b3d57610100808354040283529160200191610b68565b820191906000526020600020905b815481529060010190602001808311610b4b57829003601f168201915b5050505050905090565b6000610b7d82612a44565b506000908152606960205260409020546001600160a01b031690565b6000610ba482611431565b9050806001600160a01b0316836001600160a01b031603610c165760405162461bcd60e51b815260206004820152602160248201527f4552433732313a20617070726f76616c20746f2063757272656e74206f776e656044820152603960f91b60648201526084015b60405180910390fd5b336001600160a01b0382161480610c325750610c328133612730565b610ca45760405162461bcd60e51b815260206004820152603e60248201527f4552433732313a20617070726f76652063616c6c6572206973206e6f7420746f60448201527f6b656e206f776e6572206e6f7220617070726f76656420666f7220616c6c00006064820152608401610c0d565b610cae8383612a69565b505050565b600060d25460cc54610cc591906143e1565b905090565b6000610a7e82612ad7565b610cdd6129e5565b60df81905560dd80546003919060ff19166001835b021790555060dd546040517f40cb75607c032a2aac1c3565feb3b70e354cbd56e8f8fa36773cba5750464c5191610d2e9160ff90911690613d7d565b60405180910390a160008051602061499683398151915260df54604051610d5791815260200190565b60405180910390a150565b610d6d335b82612ce4565b610d895760405162461bcd60e51b8152600401610c0d906143f8565b610cae838383612d43565b60008080610da0611f2d565b6001600160a01b031603610dc157610db6611f2d565b600091509150610de9565b610dc9611f2d565b60d45461271090610dda9086614446565b610de49190614465565b915091505b9250929050565b610df86129e5565b6000816003811115610e0c57610e0c613d67565b10158015610e2c57506003816003811115610e2957610e29613d67565b11155b610e785760405162461bcd60e51b815260206004820152601f60248201527f4e6565647320746f20626520612076616c6964206d696e7465722074797065006044820152606401610c0d565b60dd805482919060ff19166001836003811115610e9757610e97613d67565b02179055507f40cb75607c032a2aac1c3565feb3b70e354cbd56e8f8fa36773cba5750464c5181604051610d579190613d7d565b610ed36129e5565b60d683905560d782905560df819055604051818152600080516020614996833981519152906020015b60405180910390a1505050565b610f116129e5565b60005b85811015610fa857828282818110610f2e57610f2e614487565b9050602002016020810190610f43919061449d565b60dc6000878785818110610f5957610f59614487565b9050602002016020810190610f6e9190613fa0565b6001600160a01b031681526020810191909152604001600020805460ff191691151591909117905580610fa0816144ba565b915050610f14565b505050505050565b610fb981612ecd565b610fd55760405162461bcd60e51b8152600401610c0d906144d3565b610fde33610d67565b610ffa5760405162461bcd60e51b8152600401610c0d906144f5565b6003600082815260cb602052604090205460ff16600681111561101f5761101f613d67565b1461103c5760405162461bcd60e51b8152600401610c0d9061451b565b600081815260cb6020526040902080546001919060ff191682805b02179055506040518181527facb9cd98d41dc8854281172ab2a764d949b62b51776edd911f85376ed2eca64b90602001610d57565b6110946129e5565b4780156110f35760d554600090612710906110af9084614446565b6110b99190614465565b905060006110c782846143e1565b90506110da6110d4611f2d565b83612eea565b60ca546110f0906001600160a01b031682612eea565b50505b60e0546001600160a01b0316156112a75760e0546040516370a0823160e01b81526000916001600160a01b0316906370a0823190611135903090600401613a87565b602060405180830381865afa158015611152573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906111769190614552565b905080156112a55760d554600090612710906111929084614446565b61119c9190614465565b905060006111aa82846143e1565b60e0549091506001600160a01b031663a9059cbb6111c6611f2d565b846040518363ffffffff1660e01b81526004016111e4929190613c44565b6020604051808303816000875af1158015611203573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611227919061456b565b5060e05460ca5460405163a9059cbb60e01b81526001600160a01b039283169263a9059cbb9261125e929116908590600401613c44565b6020604051808303816000875af115801561127d573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906112a1919061456b565b5050505b505b50565b610cae83838360405180602001604052806000815250612170565b6112ce33610d67565b6112ea5760405162461bcd60e51b8152600401610c0d906144f5565b6112a781613003565b6112fb6129e5565b61130488612ecd565b6113205760405162461bcd60e51b8152600401610c0d906144d3565b6004600089815260cb602052604090205460ff16600681111561134557611345613d67565b146113625760405162461bcd60e51b8152600401610c0d9061451b565b600088815260cb6020526040902060020161137d88826145ce565b50600088815260cb6020526040902060070161139987826145ce565b50600088815260cb60205260409020600881018690556009016113bc85826145ce565b50600088815260cb60205260409020600a8101849055600b016113df83826145ce565b50600088815260cb6020908152604091829020600c8101849055805460ff1916600517905590518981527fadb6d4f651f6fbf8b37026019ee48fdb86abf9ec94eb4a5f0acf72ad8c7872a39101610ace565b6000818152606760205260408120546001600160a01b031680610a7e5760405162461bcd60e51b8152600401610c0d9061468d565b6060600081816006600086815260cb602052604090205460ff16600681111561149157611491613d67565b036115df57600085815260cb60205260409020600a81015460088201546009830180549093600701919084906114c690614391565b80601f01602080910402602001604051908101604052809291908181526020018280546114f290614391565b801561153f5780601f106115145761010080835404028352916020019161153f565b820191906000526020600020905b81548152906001019060200180831161152257829003601f168201915b5050505050935081805461155290614391565b80601f016020809104026020016040519081016040528092919081815260200182805461157e90614391565b80156115cb5780601f106115a0576101008083540402835291602001916115cb565b820191906000526020600020905b8154815290600101906020018083116115ae57829003601f168201915b505050505091509350935093509350611724565b600085815260cb602052604090206006810154600482015460058301805490936003019190849061160f90614391565b80601f016020809104026020016040519081016040528092919081815260200182805461163b90614391565b80156116885780601f1061165d57610100808354040283529160200191611688565b820191906000526020600020905b81548152906001019060200180831161166b57829003601f168201915b5050505050935081805461169b90614391565b80601f01602080910402602001604051908101604052809291908181526020018280546116c790614391565b80156117145780601f106116e957610100808354040283529160200191611714565b820191906000526020600020905b8154815290600101906020018083116116f757829003601f168201915b5050505050915093509350935093505b9193509193565b60008061173661308d565b604080518281523360208201529192507f60a6c75698fadb72223808131f9f9bb9db3afa32122db6d94fb8fc985a504baa910160405180910390a16040805160018082528183019092526000916020808301908036833701905050905033816000815181106117a7576117a7614487565b60200260200101906001600160a01b031690816001600160a01b0316815250506117d081612ad7565b9250505090565b6117e082612ecd565b6117fc5760405162461bcd60e51b8152600401610c0d906144d3565b611807335b83612ce4565b6118235760405162461bcd60e51b8152600401610c0d906144f5565b6003600083815260cb602052604090205460ff16600681111561184857611848613d67565b146118655760405162461bcd60e51b8152600401610c0d9061451b565b600082815260cb60205260409020600101548110156118965760405162461bcd60e51b8152600401610c0d906146bf565b600082815260cb602052604090206001015460e0546001600160a01b031663dd62ed3e336040516001600160e01b031960e084901b1681526001600160a01b039091166004820152306024820152604401602060405180830381865afa158015611904573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906119289190614552565b101561196f5760405162461bcd60e51b8152602060048201526016602482015275496e73756666696369656e7420616c6c6f77616e636560501b6044820152606401610c0d565b60e0546000906001600160a01b03166323b872dd33600086815260cb60205260409081902060010154905160e084901b6001600160e01b03191681526001600160a01b03909216600483015230602483015260448201526064016020604051808303816000875af11580156119e8573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611a0c919061456b565b905080611a565760405162461bcd60e51b815260206004820152601860248201527721b7bab632103737ba103a3930b739b332b9103a37b5b2b760411b6044820152606401610c0d565b600083815260cb6020908152604091829020805460ff1916600417905590518481527f9bd6b4fd288008520fd788a93304e5688a401aea817ea8140ecf1fb8648f31919101610efc565b611aa86129e5565b60005b838110156112a157848482818110611ac557611ac5614487565b9050602002016020810190611ada9190613fa0565b60ce54600090815260cf6020526040902080546001600160a01b0319166001600160a01b0392909216919091179055828282818110611b1b57611b1b614487565b60ce8054600090815260d060209081526040822093029490940135909155805492909150611b48836144ba565b91905055508080611b58906144ba565b915050611aab565b611b686129e5565b60e0546001600160a01b031615611c335760e0546040516370a0823160e01b81526001600160a01b03909116906370a0823190611ba9903090600401613a87565b602060405180830381865afa158015611bc6573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611bea9190614552565b15611c335760405162461bcd60e51b8152602060048201526019602482015278746f6b656e206d757374206861766520302062616c616e636560381b6044820152606401610c0d565b60e080546001600160a01b0319166001600160a01b0392909216919091179055565b60006001600160a01b038216611cbf5760405162461bcd60e51b815260206004820152602960248201527f4552433732313a2061646472657373207a65726f206973206e6f7420612076616044820152683634b21037bbb732b960b91b6064820152608401610c0d565b506001600160a01b031660009081526068602052604090205490565b611ce36129e5565b611ced6000613104565b565b60c98054611cfc90614391565b80601f0160208091040260200160405190810160405280929190818152602001828054611d2890614391565b8015611d755780601f10611d4a57610100808354040283529160200191611d75565b820191906000526020600020905b815481529060010190602001808311611d5857829003601f168201915b505050505081565b611d856129e5565b60d681905560dd80546001919060ff19168280610cf2565b611da56129e5565b60005b85811015610fa857828282818110611dc257611dc2614487565b9050602002016020810190611dd7919061449d565b60db6000878785818110611ded57611ded614487565b9050602002016020810190611e029190613fa0565b6001600160a01b031681526020810191909152604001600020805460ff191691151591909117905580611e34816144ba565b915050611da8565b611e4581612ecd565b611e615760405162461bcd60e51b8152600401610c0d906144d3565b611e6a33610d67565b611e865760405162461bcd60e51b8152600401610c0d906144f5565b6005600082815260cb602052604090205460ff166006811115611eab57611eab613d67565b14611ec85760405162461bcd60e51b8152600401610c0d9061451b565b600081815260cb6020526040902080546006919060ff1916600183611057565b611ef06129e5565b600083815260cb60205260409020600501611f0b83826145ce565b50600083815260cb60205260409020600301611f2782826145ce565b50505050565b6000610cc56097546001600160a01b031690565b600054610100900460ff1615808015611f615750600054600160ff909116105b80611f7b5750303b158015611f7b575060005460ff166001145b611fde5760405162461bcd60e51b815260206004820152602e60248201527f496e697469616c697a61626c653a20636f6e747261637420697320616c72656160448201526d191e481a5b9a5d1a585b1a5e995960921b6064820152608401610c0d565b6000805460ff191660011790558015612001576000805461ff0019166101001790555b600082116120495760405162461bcd60e51b8152602060048201526015602482015274044726f702073697a65206d757374206265203e203605c1b6044820152606401610c0d565b6120538484613156565b61205b613187565b61206486612842565b60ca80546001600160a01b0319166001600160a01b03871617905560cc829055600060d255600160d35560c961209a85826145ce565b50600060cd558015610fa8576000805461ff0019169055604051600181527f7f26b83ff96e1f2b6a682f133852f6798a09c465da95921460cefb38474024989060200160405180910390a1505050505050565b606060668054610aef90614391565b6112a53383836131b6565b60408051600180825281830190925260009182919060208083019080368337019050509050828160008151811061214057612140614487565b60200260200101906001600160a01b031690816001600160a01b03168152505061216981612ad7565b9392505050565b61217933611801565b6121955760405162461bcd60e51b8152600401610c0d906143f8565b611f2784848484613280565b858551146121c15760405162461bcd60e51b8152600401610c0d906146e4565b858451146121e15760405162461bcd60e51b8152600401610c0d906146e4565b858351146122015760405162461bcd60e51b8152600401610c0d906146e4565b858251146122215760405162461bcd60e51b8152600401610c0d906146e4565b858151146122415760405162461bcd60e51b8152600401610c0d906146e4565b60005b86811015612393576000612258828a614710565b612263906001614710565b905086828151811061227757612277614487565b602002602001015160cb600083815260200190815260200160002060020190816122a191906145ce565b508382815181106122b4576122b4614487565b602002602001015160cb600083815260200190815260200160002060050190816122de91906145ce565b508282815181106122f1576122f1614487565b602002602001015160cb60008381526020019081526020016000206006018190555085828151811061232557612325614487565b602002602001015160cb6000838152602001908152602001600020600301908161234f91906145ce565b5084828151811061236257612362614487565b602090810291909101810151600092835260cb9091526040909120600401558061238b816144ba565b915050612244565b508560cd60008282546123a69190614710565b909155505050505050505050565b6123bd81612ecd565b6123d95760405162461bcd60e51b8152600401610c0d906144d3565b6123e233610d67565b6123fe5760405162461bcd60e51b8152600401610c0d906144f5565b6002600082815260cb602052604090205460ff16600681111561242357612423613d67565b146124405760405162461bcd60e51b8152600401610c0d9061451b565b600081815260cb60205260409020805460ff191660011790557f6c7c9f699aacfa5bacd2758023ab209b2d1e1c237bfa752bdc94cb87878cf1b781335b604080519283526001600160a01b03909116602083015201610d57565b60606124a582612ecd565b6124c15760405162461bcd60e51b8152600401610c0d906144d3565b6006600083815260cb602052604090205460ff1660068111156124e6576124e6613d67565b036125ac577f00000000000000000000000000000000000000000000000000000000000000006001600160a01b031663df30dba0612522610ae0565b600085815260cb60205260409081902060cc5491516001600160e01b031960e086901b1681526125679392600283019260098101926007909101918a916004016147a5565b600060405180830381865afa158015612584573d6000803e3d6000fd5b505050506040513d6000823e601f3d908101601f19168201604052610a7e9190810190614808565b7f00000000000000000000000000000000000000000000000000000000000000006001600160a01b031663df30dba06125e3610ae0565b600085815260cb60205260409081902060cc5491516001600160e01b031960e086901b1681526125679392600283019260058101926003909101918a916004016147a5565b60dd5460009060ff166003811115610cc557610cc5613d67565b61264b81612ecd565b6126675760405162461bcd60e51b8152600401610c0d906144d3565b61267033610d67565b61268c5760405162461bcd60e51b8152600401610c0d906144f5565b6001600082815260cb602052604090205460ff1660068111156126b1576126b1613d67565b146126ce5760405162461bcd60e51b8152600401610c0d9061451b565b600081815260cb60205260409020805460ff191660021790557fc290e40808ef06721539ad0d926f8d5f53d2dd6d5e28334308a012676f867416813361247d565b6127176129e5565b60d781905560dd80546002919060ff1916600183610cf2565b6001600160a01b039182166000908152606a6020908152604080832093909416825291909152205460ff1690565b6127666129e5565b61276f82612ecd565b61278b5760405162461bcd60e51b8152600401610c0d906144d3565b6002600083815260cb602052604090205460ff1660068111156127b0576127b0613d67565b146127eb5760405162461bcd60e51b815260206004820152600b60248201526a57726f6e6720737461746560a81b6044820152606401610c0d565b600082815260cb6020908152604091829020805460ff1916600317815560010183905590518381527f316b408b8cf323e8f20ef7a8c267eedc89bef19ca5db7fc30be938e55bf068f9910160405180910390a15050565b61284a6129e5565b6001600160a01b0381166128af5760405162461bcd60e51b815260206004820152602660248201527f4f776e61626c653a206e6577206f776e657220697320746865207a65726f206160448201526564647265737360d01b6064820152608401610c0d565b6112a781613104565b600081815260cb60205260408120600c810154600b9091018054606093929082906128e290614391565b80601f016020809104026020016040519081016040528092919081815260200182805461290e90614391565b801561295b5780601f106129305761010080835404028352916020019161295b565b820191906000526020600020905b81548152906001019060200180831161293e57829003601f168201915b5050505050915091509150915091565b6129736129e5565b60ca80546001600160a01b0319166001600160a01b0392909216919091179055565b60006001600160e01b031982166380ac58cd60e01b14806129c657506001600160e01b03198216635b5e139f60e01b145b80610a7e57506301ffc9a760e01b6001600160e01b0319831614610a7e565b336129ee611f2d565b6001600160a01b031614611ced5760405162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e65726044820152606401610c0d565b612a4d81612ecd565b6112a75760405162461bcd60e51b8152600401610c0d9061468d565b600081815260696020526040902080546001600160a01b0319166001600160a01b0384169081179091558190612a9e82611431565b6001600160a01b03167f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b92560405160405180910390a45050565b600060cc5460cd541015612b275760405162461bcd60e51b8152602060048201526017602482015276139bdd08185b1b081b595d1859185d18481b1bd8591959604a1b6044820152606401610c0d565b612b2f6132b3565b612b7b5760405162461bcd60e51b815260206004820152601d60248201527f4e6565647320746f20626520616e20616c6c6f776564206d696e7465720000006044820152606401610c0d565b6000612b8561308d565b905060008111612bc65760405162461bcd60e51b815260206004820152600c60248201526b4e6f7420666f722073616c6560a01b6044820152606401610c0d565b8251612bd29082614446565b3414612bf05760405162461bcd60e51b8152600401610c0d906146bf565b612bf861338c565b835133600090815260de6020526040902054600191612c1691614710565b612c2091906143e1565b10612c635760405162461bcd60e51b8152602060048201526013602482015272115e18d959591959081b5a5b9d081b1a5b5a5d606a1b6044820152606401610c0d565b60cc54835160d254612c759190614710565b1115612cb45760405162461bcd60e51b815260206004820152600e60248201526d4f7665722064726f702073697a6560901b6044820152606401610c0d565b600160dd5460ff166003811115612ccd57612ccd613d67565b03612cdb57612169836133fd565b612169836135ed565b600080612cf083611431565b9050806001600160a01b0316846001600160a01b03161480612d175750612d178185612730565b80612d3b5750836001600160a01b0316612d3084610b72565b6001600160a01b0316145b949350505050565b826001600160a01b0316612d5682611431565b6001600160a01b031614612dba5760405162461bcd60e51b815260206004820152602560248201527f4552433732313a207472616e736665722066726f6d20696e636f72726563742060448201526437bbb732b960d91b6064820152608401610c0d565b6001600160a01b038216612e1c5760405162461bcd60e51b8152602060048201526024808201527f4552433732313a207472616e7366657220746f20746865207a65726f206164646044820152637265737360e01b6064820152608401610c0d565b612e27600082612a69565b6001600160a01b0383166000908152606860205260408120805460019290612e509084906143e1565b90915550506001600160a01b0382166000908152606860205260408120805460019290612e7e908490614710565b909155505060008181526067602052604080822080546001600160a01b0319166001600160a01b03868116918217909255915184939187169160008051602061497683398151915291a4505050565b6000908152606760205260409020546001600160a01b0316151590565b80471015612f3a5760405162461bcd60e51b815260206004820152601d60248201527f416464726573733a20696e73756666696369656e742062616c616e63650000006044820152606401610c0d565b6000826001600160a01b03168260405160006040518083038185875af1925050503d8060008114612f87576040519150601f19603f3d011682016040523d82523d6000602084013e612f8c565b606091505b5050905080610cae5760405162461bcd60e51b815260206004820152603a60248201527f416464726573733a20756e61626c6520746f2073656e642076616c75652c207260448201527f6563697069656e74206d617920686176652072657665727465640000000000006064820152608401610c0d565b600061300e82611431565b905061301b600083612a69565b6001600160a01b03811660009081526068602052604081208054600192906130449084906143e1565b909155505060008281526067602052604080822080546001600160a01b0319169055518391906001600160a01b03841690600080516020614976833981519152908390a46112a5565b6000600160dd5460ff1660038111156130a8576130a8613d67565b036130b4575060d65490565b600260dd5460ff1660038111156130cd576130cd613d67565b036130d9575060d75490565b600360dd5460ff1660038111156130f2576130f2613d67565b036130fe575060df5490565b50600090565b609780546001600160a01b038381166001600160a01b0319831681179093556040519116919082907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e090600090a35050565b600054610100900460ff1661317d5760405162461bcd60e51b8152600401610c0d9061487e565b6112a582826136e8565b600054610100900460ff166131ae5760405162461bcd60e51b8152600401610c0d9061487e565b611ced613728565b816001600160a01b0316836001600160a01b0316036132135760405162461bcd60e51b815260206004820152601960248201527822a9219b99189d1030b8383937bb32903a379031b0b63632b960391b6044820152606401610c0d565b6001600160a01b038381166000818152606a6020908152604080832094871680845294825291829020805460ff191686151590811790915591519182527f17307eab39ab6107e8899845ad3d59bd9653f200f220920489ca2b5937696c31910160405180910390a3505050565b61328b848484612d43565b61329784848484613758565b611f275760405162461bcd60e51b8152600401610c0d906148c9565b6000600360dd5460ff1660038111156132ce576132ce613d67565b036132d95750600190565b600260dd5460ff1660038111156132f2576132f2613d67565b036133335733600090815260dc602052604090205460ff16156133155750600190565b33600090815260db602052604090205460ff16156133335750600190565b600160dd5460ff16600381111561334c5761334c613d67565b0361336f5733600090815260dc602052604090205460ff161561336f5750600190565b33613378611f2d565b6001600160a01b0316036130fe5750600190565b6000600160dd5460ff1660038111156133a7576133a7613d67565b036133b3575060d85490565b600260dd5460ff1660038111156133cc576133cc613d67565b036133d8575060d95490565b600360dd5460ff1660038111156133f1576133f1613d67565b036130fe575060da5490565b60ce5460009033908290815b60ce5481101561348c57600081815260cf60205260409020546001600160a01b0380861691160361347a57600081815260d0602090815260408083205480845260d19092529091205460ff161515600114613478578282101561346a578192505b83613474816144ba565b9450505b505b80613484816144ba565b915050613409565b5084518210156134da5760405162461bcd60e51b815260206004820152601960248201527843616e206e6f74206d696e7420616c6c2065646974696f6e7360381b6044820152606401610c0d565b60018160005b87518110156135e1575b600082815260cf60205260409020546001600160a01b0387811691161461351d5781613515816144ba565b9250506134ea565b60d0600083815260200190815260200160002054925061355688828151811061354857613548614487565b602002602001015184613859565b600083815260cb602090815260408083208054600160ff19918216811790925560d18452828520805490911690911790556001600160a01b038916835260de90915281208054916135a6836144ba565b909155505060d280549060006135bb836144ba565b919050555081806135cb906144ba565b92505080806135d9906144ba565b9150506134e0565b50909695505050505050565b600033815b83518110156136dd575b60d354600090815260d1602052604090205460ff1615156001036136345760d3805490600061362a836144ba565b91905055506135fc565b61365984828151811061364957613649614487565b602002602001015160d354613859565b60d38054600090815260cb602090815260408083208054600160ff1991821681179092559454845260d183528184208054909516179093556001600160a01b038516825260de90529081208054916136b0836144ba565b909155505060d280549060006136c5836144ba565b919050555080806136d5906144ba565b9150506135f2565b505060d35492915050565b600054610100900460ff1661370f5760405162461bcd60e51b8152600401610c0d9061487e565b606561371b83826145ce565b506066610cae82826145ce565b600054610100900460ff1661374f5760405162461bcd60e51b8152600401610c0d9061487e565b611ced33613104565b60006001600160a01b0384163b1561384e57604051630a85bd0160e11b81526001600160a01b0385169063150b7a029061379c90339089908890889060040161491b565b6020604051808303816000875af19250505080156137d7575060408051601f3d908101601f191682019092526137d491810190614958565b60015b613834573d808015613805576040519150601f19603f3d011682016040523d82523d6000602084013e61380a565b606091505b50805160000361382c5760405162461bcd60e51b8152600401610c0d906148c9565b805181602001fd5b6001600160e01b031916630a85bd0160e11b149050612d3b565b506001949350505050565b6001600160a01b0382166138af5760405162461bcd60e51b815260206004820181905260248201527f4552433732313a206d696e7420746f20746865207a65726f20616464726573736044820152606401610c0d565b6138b881612ecd565b156139055760405162461bcd60e51b815260206004820152601c60248201527f4552433732313a20746f6b656e20616c7265616479206d696e746564000000006044820152606401610c0d565b6001600160a01b038216600090815260686020526040812080546001929061392e908490614710565b909155505060008181526067602052604080822080546001600160a01b0319166001600160a01b0386169081179091559051839290600080516020614976833981519152908290a46112a5565b6001600160e01b0319811681146112a757600080fd5b6000602082840312156139a357600080fd5b81356121698161397b565b600080600080600080600080610100898b0312156139cb57600080fd5b505086359860208801359850604088013597606081013597506080810135965060a0810135955060c0810135945060e0013592509050565b60005b83811015613a1e578181015183820152602001613a06565b83811115611f275750506000910152565b60008151808452613a47816020860160208601613a03565b601f01601f19169290920160200192915050565b6020815260006121696020830184613a2f565b600060208284031215613a8057600080fd5b5035919050565b6001600160a01b0391909116815260200190565b80356001600160a01b0381168114613ab257600080fd5b919050565b60008060408385031215613aca57600080fd5b613ad383613a9b565b946020939093013593505050565b634e487b7160e01b600052604160045260246000fd5b604051601f8201601f191681016001600160401b0381118282101715613b1f57613b1f613ae1565b604052919050565b60006001600160401b03821115613b4057613b40613ae1565b5060051b60200190565b60006020808385031215613b5d57600080fd5b82356001600160401b03811115613b7357600080fd5b8301601f81018513613b8457600080fd5b8035613b97613b9282613b27565b613af7565b81815260059190911b82018301908381019087831115613bb657600080fd5b928401925b82841015613bdb57613bcc84613a9b565b82529284019290840190613bbb565b979650505050505050565b600080600060608486031215613bfb57600080fd5b613c0484613a9b565b9250613c1260208501613a9b565b9150604084013590509250925092565b60008060408385031215613c3557600080fd5b50508035926020909101359150565b6001600160a01b03929092168252602082015260400190565b600060208284031215613c6f57600080fd5b81356004811061216957600080fd5b600080600060608486031215613c9357600080fd5b505081359360208301359350604090920135919050565b60008083601f840112613cbc57600080fd5b5081356001600160401b03811115613cd357600080fd5b6020830191508360208260051b8501011115610de957600080fd5b600080600080600060608688031215613d0657600080fd5b8535945060208601356001600160401b0380821115613d2457600080fd5b613d3089838a01613caa565b90965094506040880135915080821115613d4957600080fd5b50613d5688828901613caa565b969995985093965092949392505050565b634e487b7160e01b600052602160045260246000fd5b6020810160048310613d9f57634e487b7160e01b600052602160045260246000fd5b91905290565b60006001600160401b03821115613dbe57613dbe613ae1565b50601f01601f191660200190565b6000613dda613b9284613da5565b9050828152838383011115613dee57600080fd5b828260208301376000602084830101529392505050565b600082601f830112613e1657600080fd5b61216983833560208501613dcc565b600080600080600080600080610100898b031215613e4257600080fd5b8835975060208901356001600160401b0380821115613e6057600080fd5b613e6c8c838d01613e05565b985060408b0135915080821115613e8257600080fd5b613e8e8c838d01613e05565b975060608b0135965060808b0135915080821115613eab57600080fd5b613eb78c838d01613e05565b955060a08b0135945060c08b0135915080821115613ed457600080fd5b50613ee18b828c01613e05565b92505060e089013590509295985092959890939650565b608081526000613f0b6080830187613a2f565b8560208401528281036040840152613f238186613a2f565b91505082606083015295945050505050565b60008060008060408587031215613f4b57600080fd5b84356001600160401b0380821115613f6257600080fd5b613f6e88838901613caa565b90965094506020870135915080821115613f8757600080fd5b50613f9487828801613caa565b95989497509550505050565b600060208284031215613fb257600080fd5b61216982613a9b565b600080600060608486031215613fd057600080fd5b8335925060208401356001600160401b0380821115613fee57600080fd5b613ffa87838801613e05565b9350604086013591508082111561401057600080fd5b5061401d86828701613e05565b9150509250925092565b600080600080600060a0868803121561403f57600080fd5b61404886613a9b565b945061405660208701613a9b565b935060408601356001600160401b038082111561407257600080fd5b61407e89838a01613e05565b9450606088013591508082111561409457600080fd5b506140a188828901613e05565b95989497509295608001359392505050565b80151581146112a757600080fd5b600080604083850312156140d457600080fd5b6140dd83613a9b565b915060208301356140ed816140b3565b809150509250929050565b6000806000806080858703121561410e57600080fd5b61411785613a9b565b935061412560208601613a9b565b92506040850135915060608501356001600160401b0381111561414757600080fd5b8501601f8101871361415857600080fd5b61416787823560208401613dcc565b91505092959194509250565b600082601f83011261418457600080fd5b81356020614194613b9283613b27565b82815260059290921b840181019181810190868411156141b357600080fd5b8286015b848110156141f25780356001600160401b038111156141d65760008081fd5b6141e48986838b0101613e05565b8452509183019183016141b7565b509695505050505050565b600082601f83011261420e57600080fd5b8135602061421e613b9283613b27565b82815260059290921b8401810191818101908684111561423d57600080fd5b8286015b848110156141f25780358352918301918301614241565b600080600080600080600060e0888a03121561427357600080fd5b873596506020880135955060408801356001600160401b038082111561429857600080fd5b6142a48b838c01614173565b965060608a01359150808211156142ba57600080fd5b6142c68b838c01614173565b955060808a01359150808211156142dc57600080fd5b6142e88b838c016141fd565b945060a08a01359150808211156142fe57600080fd5b61430a8b838c01614173565b935060c08a013591508082111561432057600080fd5b5061432d8a828b016141fd565b91505092959891949750929550565b6000806040838503121561434f57600080fd5b61435883613a9b565b915061436660208401613a9b565b90509250929050565b6040815260006143826040830185613a2f565b90508260208301529392505050565b600181811c908216806143a557607f821691505b6020821081036143c557634e487b7160e01b600052602260045260246000fd5b50919050565b634e487b7160e01b600052601160045260246000fd5b6000828210156143f3576143f36143cb565b500390565b6020808252602e908201527f4552433732313a2063616c6c6572206973206e6f7420746f6b656e206f776e6560408201526d1c881b9bdc88185c1c1c9bdd995960921b606082015260800190565b6000816000190483118215151615614460576144606143cb565b500290565b60008261448257634e487b7160e01b600052601260045260246000fd5b500490565b634e487b7160e01b600052603260045260246000fd5b6000602082840312156144af57600080fd5b8135612169816140b3565b6000600182016144cc576144cc6143cb565b5060010190565b6020808252600890820152672737903a37b5b2b760c11b604082015260600190565b6020808252600c908201526b139bdd08185c1c1c9bdd995960a21b604082015260600190565b6020808252601c908201527f596f752063757272656e746c792063616e206e6f742072656465656d00000000604082015260600190565b60006020828403121561456457600080fd5b5051919050565b60006020828403121561457d57600080fd5b8151612169816140b3565b601f821115610cae57600081815260208120601f850160051c810160208610156145af5750805b601f850160051c820191505b81811015610fa8578281556001016145bb565b81516001600160401b038111156145e7576145e7613ae1565b6145fb816145f58454614391565b84614588565b602080601f83116001811461463057600084156146185750858301515b600019600386901b1c1916600185901b178555610fa8565b600085815260208120601f198616915b8281101561465f57888601518255948401946001909101908401614640565b508582101561467d5787850151600019600388901b60f8161c191681555b5050505050600190811b01905550565b602080825260189082015277115490cdcc8c4e881a5b9d985b1a59081d1bdad95b88125160421b604082015260600190565b6020808252600b908201526a57726f6e6720707269636560a81b604082015260600190565b602080825260129082015271088c2e8c240e6d2f4ca40dad2e6dac2e8c6d60731b604082015260600190565b60008219821115614723576147236143cb565b500190565b6000815461473581614391565b808552602060018381168015614752576001811461476c5761479a565b60ff1985168884015283151560051b88018301955061479a565b866000528260002060005b858110156147925781548a8201860152908301908401614777565b890184019650505b505050505092915050565b60c0815260006147b860c0830189613a2f565b82810360208401526147ca8189614728565b905082810360408401526147de8188614728565b905082810360608401526147f28187614728565b6080840195909552505060a00152949350505050565b60006020828403121561481a57600080fd5b81516001600160401b0381111561483057600080fd5b8201601f8101841361484157600080fd5b805161484f613b9282613da5565b81815285602083850101111561486457600080fd5b614875826020830160208601613a03565b95945050505050565b6020808252602b908201527f496e697469616c697a61626c653a20636f6e7472616374206973206e6f74206960408201526a6e697469616c697a696e6760a81b606082015260800190565b60208082526032908201527f4552433732313a207472616e7366657220746f206e6f6e20455243373231526560408201527131b2b4bb32b91034b6b83632b6b2b73a32b960711b606082015260800190565b6001600160a01b038581168252841660208201526040810183905260806060820181905260009061494e90830184613a2f565b9695505050505050565b60006020828403121561496a57600080fd5b81516121698161397b56feddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3efa6dc15bdb68da224c66db4b3838d9a2b205138e8cff6774e57d0af91e196d622a2646970667358221220824fd54c64454ccba32c84242855be80fe2cf8bacdb68f0174b3d2bca951cc0a64736f6c634300080f0033", + "deployedBytecode": "0x60806040526004361061032d5760003560e01c806369e1cac1116101a7578063b88d4fde116100ed578063db006a7511610090578063db006a7514610955578063de5d62a014610975578063e985e9c514610995578063ee5df550146109b5578063f2fde38b146109d5578063f51f96dd146109f5578063fc51e0c314610a0b578063fce57fd914610a3957600080fd5b8063b88d4fde14610873578063bcda01af14610893578063bf82ae49146108b3578063c87b56dd146108d3578063c9d359d6146108f3578063d375b6fc1461090d578063d41c3a6514610922578063d5344e361461094057600080fd5b80637b422c4c116101555780637b422c4c146107a15780638c503ca4146107c15780638da5cb5b146107e157806393d23923146107f657806395d89b4114610816578063a1279cfc1461082b578063a22cb46514610840578063a66ff0af1461086057600080fd5b806369e1cac1146106d75780636a326ab1146106f757806370a0823114610717578063715018a6146107375780637284e4161461074c578063787e5efc14610761578063799b7c5f1461078157600080fd5b8063392774151161027757806342966c681161021a57806342966c68146105f557806346d9c4bf14610615578063499e16721461062a57806352c9143e1461063f5780636352211e1461065f57806364e9293e1461067f57806364edfbf0146106af578063660243d0146106b757600080fd5b8063392774151461052257806339dc13bc146105405780633ccfd60b146105605780633da165d4146105755780633e1dc1071461058b578063402635cf146105a0578063410a1095146105b557806342842e0e146105d557600080fd5b80630f6a9349116102df5780630f6a93491461042c57806318160ddd1461043f5780631919fed71461045457806323b872dd146104745780632a55205a14610494578063367de39e146104c257806336f3ec76146104e25780633892aa521461050257600080fd5b806301ffc9a714610332578063036c8ea11461036757806306fdde0314610389578063081812fc146103ab578063095ea7b3146103d857806309949abe146103f85780630b65b6e714610417575b600080fd5b34801561033e57600080fd5b5061035261034d366004613991565b610a59565b60405190151581526020015b60405180910390f35b34801561037357600080fd5b506103876103823660046139ae565b610a84565b005b34801561039557600080fd5b5061039e610ae0565b60405161035e9190613a5b565b3480156103b757600080fd5b506103cb6103c6366004613a6e565b610b72565b60405161035e9190613a87565b3480156103e457600080fd5b506103876103f3366004613ab7565b610b99565b34801561040457600080fd5b5060d4545b60405190815260200161035e565b34801561042357600080fd5b50610409610cb3565b61040961043a366004613b4a565b610cca565b34801561044b57600080fd5b5060d254610409565b34801561046057600080fd5b5061038761046f366004613a6e565b610cd5565b34801561048057600080fd5b5061038761048f366004613be6565b610d62565b3480156104a057600080fd5b506104b46104af366004613c22565b610d94565b60405161035e929190613c44565b3480156104ce57600080fd5b506103876104dd366004613c5d565b610df0565b3480156104ee57600080fd5b506103876104fd366004613c7e565b610ecb565b34801561050e57600080fd5b5061038761051d366004613cee565b610f09565b34801561052e57600080fd5b5060ca546001600160a01b03166103cb565b34801561054c57600080fd5b5061038761055b366004613a6e565b610fb0565b34801561056c57600080fd5b5061038761108c565b34801561058157600080fd5b5061040960cc5481565b34801561059757600080fd5b5060d954610409565b3480156105ac57600080fd5b5060d554610409565b3480156105c157600080fd5b5060dd5460ff1660405161035e9190613d7d565b3480156105e157600080fd5b506103876105f0366004613be6565b6112aa565b34801561060157600080fd5b50610387610610366004613a6e565b6112c5565b34801561062157600080fd5b5060df54610409565b34801561063657600080fd5b5060d754610409565b34801561064b57600080fd5b5061038761065a366004613e25565b6112f3565b34801561066b57600080fd5b506103cb61067a366004613a6e565b611431565b34801561068b57600080fd5b5061069f61069a366004613a6e565b611466565b60405161035e9493929190613ef8565b61040961172b565b3480156106c357600080fd5b506103876106d2366004613c22565b6117d7565b3480156106e357600080fd5b506103876106f2366004613f35565b611aa0565b34801561070357600080fd5b50610387610712366004613fa0565b611b60565b34801561072357600080fd5b50610409610732366004613fa0565b611c55565b34801561074357600080fd5b50610387611cdb565b34801561075857600080fd5b5061039e611cef565b34801561076d57600080fd5b5061038761077c366004613a6e565b611d7d565b34801561078d57600080fd5b5061038761079c366004613cee565b611d9d565b3480156107ad57600080fd5b506103876107bc366004613a6e565b611e3c565b3480156107cd57600080fd5b506103876107dc366004613fbb565b611ee8565b3480156107ed57600080fd5b506103cb611f2d565b34801561080257600080fd5b50610387610811366004614027565b611f41565b34801561082257600080fd5b5061039e6120ed565b34801561083757600080fd5b5060d654610409565b34801561084c57600080fd5b5061038761085b3660046140c1565b6120fc565b61040961086e366004613fa0565b612107565b34801561087f57600080fd5b5061038761088e3660046140f8565b612170565b34801561089f57600080fd5b506103876108ae366004614258565b6121a1565b3480156108bf57600080fd5b506103876108ce366004613a6e565b6123b4565b3480156108df57600080fd5b5061039e6108ee366004613a6e565b61249a565b3480156108ff57600080fd5b5060cc5460cd541015610352565b34801561091957600080fd5b50610409612628565b34801561092e57600080fd5b5060e0546001600160a01b03166103cb565b34801561094c57600080fd5b5060d854610409565b34801561096157600080fd5b50610387610970366004613a6e565b612642565b34801561098157600080fd5b50610387610990366004613a6e565b61270f565b3480156109a157600080fd5b506103526109b036600461433c565b612730565b3480156109c157600080fd5b506103876109d0366004613c22565b61275e565b3480156109e157600080fd5b506103876109f0366004613fa0565b612842565b348015610a0157600080fd5b5061040960df5481565b348015610a1757600080fd5b50610a2b610a26366004613a6e565b6128b8565b60405161035e92919061436f565b348015610a4557600080fd5b50610387610a54366004613fa0565b61296b565b600063152a902d60e11b6001600160e01b031983161480610a7e5750610a7e82612995565b92915050565b610a8c6129e5565b60d488905560d587905560d686905560d785905560df84905560d883905560d982905560da819055604051848152600080516020614996833981519152906020015b60405180910390a15050505050505050565b606060658054610aef90614391565b80601f0160208091040260200160405190810160405280929190818152602001828054610b1b90614391565b8015610b685780601f10610b3d57610100808354040283529160200191610b68565b820191906000526020600020905b815481529060010190602001808311610b4b57829003601f168201915b5050505050905090565b6000610b7d82612a44565b506000908152606960205260409020546001600160a01b031690565b6000610ba482611431565b9050806001600160a01b0316836001600160a01b031603610c165760405162461bcd60e51b815260206004820152602160248201527f4552433732313a20617070726f76616c20746f2063757272656e74206f776e656044820152603960f91b60648201526084015b60405180910390fd5b336001600160a01b0382161480610c325750610c328133612730565b610ca45760405162461bcd60e51b815260206004820152603e60248201527f4552433732313a20617070726f76652063616c6c6572206973206e6f7420746f60448201527f6b656e206f776e6572206e6f7220617070726f76656420666f7220616c6c00006064820152608401610c0d565b610cae8383612a69565b505050565b600060d25460cc54610cc591906143e1565b905090565b6000610a7e82612ad7565b610cdd6129e5565b60df81905560dd80546003919060ff19166001835b021790555060dd546040517f40cb75607c032a2aac1c3565feb3b70e354cbd56e8f8fa36773cba5750464c5191610d2e9160ff90911690613d7d565b60405180910390a160008051602061499683398151915260df54604051610d5791815260200190565b60405180910390a150565b610d6d335b82612ce4565b610d895760405162461bcd60e51b8152600401610c0d906143f8565b610cae838383612d43565b60008080610da0611f2d565b6001600160a01b031603610dc157610db6611f2d565b600091509150610de9565b610dc9611f2d565b60d45461271090610dda9086614446565b610de49190614465565b915091505b9250929050565b610df86129e5565b6000816003811115610e0c57610e0c613d67565b10158015610e2c57506003816003811115610e2957610e29613d67565b11155b610e785760405162461bcd60e51b815260206004820152601f60248201527f4e6565647320746f20626520612076616c6964206d696e7465722074797065006044820152606401610c0d565b60dd805482919060ff19166001836003811115610e9757610e97613d67565b02179055507f40cb75607c032a2aac1c3565feb3b70e354cbd56e8f8fa36773cba5750464c5181604051610d579190613d7d565b610ed36129e5565b60d683905560d782905560df819055604051818152600080516020614996833981519152906020015b60405180910390a1505050565b610f116129e5565b60005b85811015610fa857828282818110610f2e57610f2e614487565b9050602002016020810190610f43919061449d565b60dc6000878785818110610f5957610f59614487565b9050602002016020810190610f6e9190613fa0565b6001600160a01b031681526020810191909152604001600020805460ff191691151591909117905580610fa0816144ba565b915050610f14565b505050505050565b610fb981612ecd565b610fd55760405162461bcd60e51b8152600401610c0d906144d3565b610fde33610d67565b610ffa5760405162461bcd60e51b8152600401610c0d906144f5565b6003600082815260cb602052604090205460ff16600681111561101f5761101f613d67565b1461103c5760405162461bcd60e51b8152600401610c0d9061451b565b600081815260cb6020526040902080546001919060ff191682805b02179055506040518181527facb9cd98d41dc8854281172ab2a764d949b62b51776edd911f85376ed2eca64b90602001610d57565b6110946129e5565b4780156110f35760d554600090612710906110af9084614446565b6110b99190614465565b905060006110c782846143e1565b90506110da6110d4611f2d565b83612eea565b60ca546110f0906001600160a01b031682612eea565b50505b60e0546001600160a01b0316156112a75760e0546040516370a0823160e01b81526000916001600160a01b0316906370a0823190611135903090600401613a87565b602060405180830381865afa158015611152573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906111769190614552565b905080156112a55760d554600090612710906111929084614446565b61119c9190614465565b905060006111aa82846143e1565b60e0549091506001600160a01b031663a9059cbb6111c6611f2d565b846040518363ffffffff1660e01b81526004016111e4929190613c44565b6020604051808303816000875af1158015611203573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611227919061456b565b5060e05460ca5460405163a9059cbb60e01b81526001600160a01b039283169263a9059cbb9261125e929116908590600401613c44565b6020604051808303816000875af115801561127d573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906112a1919061456b565b5050505b505b50565b610cae83838360405180602001604052806000815250612170565b6112ce33610d67565b6112ea5760405162461bcd60e51b8152600401610c0d906144f5565b6112a781613003565b6112fb6129e5565b61130488612ecd565b6113205760405162461bcd60e51b8152600401610c0d906144d3565b6004600089815260cb602052604090205460ff16600681111561134557611345613d67565b146113625760405162461bcd60e51b8152600401610c0d9061451b565b600088815260cb6020526040902060020161137d88826145ce565b50600088815260cb6020526040902060070161139987826145ce565b50600088815260cb60205260409020600881018690556009016113bc85826145ce565b50600088815260cb60205260409020600a8101849055600b016113df83826145ce565b50600088815260cb6020908152604091829020600c8101849055805460ff1916600517905590518981527fadb6d4f651f6fbf8b37026019ee48fdb86abf9ec94eb4a5f0acf72ad8c7872a39101610ace565b6000818152606760205260408120546001600160a01b031680610a7e5760405162461bcd60e51b8152600401610c0d9061468d565b6060600081816006600086815260cb602052604090205460ff16600681111561149157611491613d67565b036115df57600085815260cb60205260409020600a81015460088201546009830180549093600701919084906114c690614391565b80601f01602080910402602001604051908101604052809291908181526020018280546114f290614391565b801561153f5780601f106115145761010080835404028352916020019161153f565b820191906000526020600020905b81548152906001019060200180831161152257829003601f168201915b5050505050935081805461155290614391565b80601f016020809104026020016040519081016040528092919081815260200182805461157e90614391565b80156115cb5780601f106115a0576101008083540402835291602001916115cb565b820191906000526020600020905b8154815290600101906020018083116115ae57829003601f168201915b505050505091509350935093509350611724565b600085815260cb602052604090206006810154600482015460058301805490936003019190849061160f90614391565b80601f016020809104026020016040519081016040528092919081815260200182805461163b90614391565b80156116885780601f1061165d57610100808354040283529160200191611688565b820191906000526020600020905b81548152906001019060200180831161166b57829003601f168201915b5050505050935081805461169b90614391565b80601f01602080910402602001604051908101604052809291908181526020018280546116c790614391565b80156117145780601f106116e957610100808354040283529160200191611714565b820191906000526020600020905b8154815290600101906020018083116116f757829003601f168201915b5050505050915093509350935093505b9193509193565b60008061173661308d565b604080518281523360208201529192507f60a6c75698fadb72223808131f9f9bb9db3afa32122db6d94fb8fc985a504baa910160405180910390a16040805160018082528183019092526000916020808301908036833701905050905033816000815181106117a7576117a7614487565b60200260200101906001600160a01b031690816001600160a01b0316815250506117d081612ad7565b9250505090565b6117e082612ecd565b6117fc5760405162461bcd60e51b8152600401610c0d906144d3565b611807335b83612ce4565b6118235760405162461bcd60e51b8152600401610c0d906144f5565b6003600083815260cb602052604090205460ff16600681111561184857611848613d67565b146118655760405162461bcd60e51b8152600401610c0d9061451b565b600082815260cb60205260409020600101548110156118965760405162461bcd60e51b8152600401610c0d906146bf565b600082815260cb602052604090206001015460e0546001600160a01b031663dd62ed3e336040516001600160e01b031960e084901b1681526001600160a01b039091166004820152306024820152604401602060405180830381865afa158015611904573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906119289190614552565b101561196f5760405162461bcd60e51b8152602060048201526016602482015275496e73756666696369656e7420616c6c6f77616e636560501b6044820152606401610c0d565b60e0546000906001600160a01b03166323b872dd33600086815260cb60205260409081902060010154905160e084901b6001600160e01b03191681526001600160a01b03909216600483015230602483015260448201526064016020604051808303816000875af11580156119e8573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611a0c919061456b565b905080611a565760405162461bcd60e51b815260206004820152601860248201527721b7bab632103737ba103a3930b739b332b9103a37b5b2b760411b6044820152606401610c0d565b600083815260cb6020908152604091829020805460ff1916600417905590518481527f9bd6b4fd288008520fd788a93304e5688a401aea817ea8140ecf1fb8648f31919101610efc565b611aa86129e5565b60005b838110156112a157848482818110611ac557611ac5614487565b9050602002016020810190611ada9190613fa0565b60ce54600090815260cf6020526040902080546001600160a01b0319166001600160a01b0392909216919091179055828282818110611b1b57611b1b614487565b60ce8054600090815260d060209081526040822093029490940135909155805492909150611b48836144ba565b91905055508080611b58906144ba565b915050611aab565b611b686129e5565b60e0546001600160a01b031615611c335760e0546040516370a0823160e01b81526001600160a01b03909116906370a0823190611ba9903090600401613a87565b602060405180830381865afa158015611bc6573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611bea9190614552565b15611c335760405162461bcd60e51b8152602060048201526019602482015278746f6b656e206d757374206861766520302062616c616e636560381b6044820152606401610c0d565b60e080546001600160a01b0319166001600160a01b0392909216919091179055565b60006001600160a01b038216611cbf5760405162461bcd60e51b815260206004820152602960248201527f4552433732313a2061646472657373207a65726f206973206e6f7420612076616044820152683634b21037bbb732b960b91b6064820152608401610c0d565b506001600160a01b031660009081526068602052604090205490565b611ce36129e5565b611ced6000613104565b565b60c98054611cfc90614391565b80601f0160208091040260200160405190810160405280929190818152602001828054611d2890614391565b8015611d755780601f10611d4a57610100808354040283529160200191611d75565b820191906000526020600020905b815481529060010190602001808311611d5857829003601f168201915b505050505081565b611d856129e5565b60d681905560dd80546001919060ff19168280610cf2565b611da56129e5565b60005b85811015610fa857828282818110611dc257611dc2614487565b9050602002016020810190611dd7919061449d565b60db6000878785818110611ded57611ded614487565b9050602002016020810190611e029190613fa0565b6001600160a01b031681526020810191909152604001600020805460ff191691151591909117905580611e34816144ba565b915050611da8565b611e4581612ecd565b611e615760405162461bcd60e51b8152600401610c0d906144d3565b611e6a33610d67565b611e865760405162461bcd60e51b8152600401610c0d906144f5565b6005600082815260cb602052604090205460ff166006811115611eab57611eab613d67565b14611ec85760405162461bcd60e51b8152600401610c0d9061451b565b600081815260cb6020526040902080546006919060ff1916600183611057565b611ef06129e5565b600083815260cb60205260409020600501611f0b83826145ce565b50600083815260cb60205260409020600301611f2782826145ce565b50505050565b6000610cc56097546001600160a01b031690565b600054610100900460ff1615808015611f615750600054600160ff909116105b80611f7b5750303b158015611f7b575060005460ff166001145b611fde5760405162461bcd60e51b815260206004820152602e60248201527f496e697469616c697a61626c653a20636f6e747261637420697320616c72656160448201526d191e481a5b9a5d1a585b1a5e995960921b6064820152608401610c0d565b6000805460ff191660011790558015612001576000805461ff0019166101001790555b600082116120495760405162461bcd60e51b8152602060048201526015602482015274044726f702073697a65206d757374206265203e203605c1b6044820152606401610c0d565b6120538484613156565b61205b613187565b61206486612842565b60ca80546001600160a01b0319166001600160a01b03871617905560cc829055600060d255600160d35560c961209a85826145ce565b50600060cd558015610fa8576000805461ff0019169055604051600181527f7f26b83ff96e1f2b6a682f133852f6798a09c465da95921460cefb38474024989060200160405180910390a1505050505050565b606060668054610aef90614391565b6112a53383836131b6565b60408051600180825281830190925260009182919060208083019080368337019050509050828160008151811061214057612140614487565b60200260200101906001600160a01b031690816001600160a01b03168152505061216981612ad7565b9392505050565b61217933611801565b6121955760405162461bcd60e51b8152600401610c0d906143f8565b611f2784848484613280565b858551146121c15760405162461bcd60e51b8152600401610c0d906146e4565b858451146121e15760405162461bcd60e51b8152600401610c0d906146e4565b858351146122015760405162461bcd60e51b8152600401610c0d906146e4565b858251146122215760405162461bcd60e51b8152600401610c0d906146e4565b858151146122415760405162461bcd60e51b8152600401610c0d906146e4565b60005b86811015612393576000612258828a614710565b612263906001614710565b905086828151811061227757612277614487565b602002602001015160cb600083815260200190815260200160002060020190816122a191906145ce565b508382815181106122b4576122b4614487565b602002602001015160cb600083815260200190815260200160002060050190816122de91906145ce565b508282815181106122f1576122f1614487565b602002602001015160cb60008381526020019081526020016000206006018190555085828151811061232557612325614487565b602002602001015160cb6000838152602001908152602001600020600301908161234f91906145ce565b5084828151811061236257612362614487565b602090810291909101810151600092835260cb9091526040909120600401558061238b816144ba565b915050612244565b508560cd60008282546123a69190614710565b909155505050505050505050565b6123bd81612ecd565b6123d95760405162461bcd60e51b8152600401610c0d906144d3565b6123e233610d67565b6123fe5760405162461bcd60e51b8152600401610c0d906144f5565b6002600082815260cb602052604090205460ff16600681111561242357612423613d67565b146124405760405162461bcd60e51b8152600401610c0d9061451b565b600081815260cb60205260409020805460ff191660011790557f6c7c9f699aacfa5bacd2758023ab209b2d1e1c237bfa752bdc94cb87878cf1b781335b604080519283526001600160a01b03909116602083015201610d57565b60606124a582612ecd565b6124c15760405162461bcd60e51b8152600401610c0d906144d3565b6006600083815260cb602052604090205460ff1660068111156124e6576124e6613d67565b036125ac577f00000000000000000000000000000000000000000000000000000000000000006001600160a01b031663df30dba0612522610ae0565b600085815260cb60205260409081902060cc5491516001600160e01b031960e086901b1681526125679392600283019260098101926007909101918a916004016147a5565b600060405180830381865afa158015612584573d6000803e3d6000fd5b505050506040513d6000823e601f3d908101601f19168201604052610a7e9190810190614808565b7f00000000000000000000000000000000000000000000000000000000000000006001600160a01b031663df30dba06125e3610ae0565b600085815260cb60205260409081902060cc5491516001600160e01b031960e086901b1681526125679392600283019260058101926003909101918a916004016147a5565b60dd5460009060ff166003811115610cc557610cc5613d67565b61264b81612ecd565b6126675760405162461bcd60e51b8152600401610c0d906144d3565b61267033610d67565b61268c5760405162461bcd60e51b8152600401610c0d906144f5565b6001600082815260cb602052604090205460ff1660068111156126b1576126b1613d67565b146126ce5760405162461bcd60e51b8152600401610c0d9061451b565b600081815260cb60205260409020805460ff191660021790557fc290e40808ef06721539ad0d926f8d5f53d2dd6d5e28334308a012676f867416813361247d565b6127176129e5565b60d781905560dd80546002919060ff1916600183610cf2565b6001600160a01b039182166000908152606a6020908152604080832093909416825291909152205460ff1690565b6127666129e5565b61276f82612ecd565b61278b5760405162461bcd60e51b8152600401610c0d906144d3565b6002600083815260cb602052604090205460ff1660068111156127b0576127b0613d67565b146127eb5760405162461bcd60e51b815260206004820152600b60248201526a57726f6e6720737461746560a81b6044820152606401610c0d565b600082815260cb6020908152604091829020805460ff1916600317815560010183905590518381527f316b408b8cf323e8f20ef7a8c267eedc89bef19ca5db7fc30be938e55bf068f9910160405180910390a15050565b61284a6129e5565b6001600160a01b0381166128af5760405162461bcd60e51b815260206004820152602660248201527f4f776e61626c653a206e6577206f776e657220697320746865207a65726f206160448201526564647265737360d01b6064820152608401610c0d565b6112a781613104565b600081815260cb60205260408120600c810154600b9091018054606093929082906128e290614391565b80601f016020809104026020016040519081016040528092919081815260200182805461290e90614391565b801561295b5780601f106129305761010080835404028352916020019161295b565b820191906000526020600020905b81548152906001019060200180831161293e57829003601f168201915b5050505050915091509150915091565b6129736129e5565b60ca80546001600160a01b0319166001600160a01b0392909216919091179055565b60006001600160e01b031982166380ac58cd60e01b14806129c657506001600160e01b03198216635b5e139f60e01b145b80610a7e57506301ffc9a760e01b6001600160e01b0319831614610a7e565b336129ee611f2d565b6001600160a01b031614611ced5760405162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e65726044820152606401610c0d565b612a4d81612ecd565b6112a75760405162461bcd60e51b8152600401610c0d9061468d565b600081815260696020526040902080546001600160a01b0319166001600160a01b0384169081179091558190612a9e82611431565b6001600160a01b03167f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b92560405160405180910390a45050565b600060cc5460cd541015612b275760405162461bcd60e51b8152602060048201526017602482015276139bdd08185b1b081b595d1859185d18481b1bd8591959604a1b6044820152606401610c0d565b612b2f6132b3565b612b7b5760405162461bcd60e51b815260206004820152601d60248201527f4e6565647320746f20626520616e20616c6c6f776564206d696e7465720000006044820152606401610c0d565b6000612b8561308d565b905060008111612bc65760405162461bcd60e51b815260206004820152600c60248201526b4e6f7420666f722073616c6560a01b6044820152606401610c0d565b8251612bd29082614446565b3414612bf05760405162461bcd60e51b8152600401610c0d906146bf565b612bf861338c565b835133600090815260de6020526040902054600191612c1691614710565b612c2091906143e1565b10612c635760405162461bcd60e51b8152602060048201526013602482015272115e18d959591959081b5a5b9d081b1a5b5a5d606a1b6044820152606401610c0d565b60cc54835160d254612c759190614710565b1115612cb45760405162461bcd60e51b815260206004820152600e60248201526d4f7665722064726f702073697a6560901b6044820152606401610c0d565b600160dd5460ff166003811115612ccd57612ccd613d67565b03612cdb57612169836133fd565b612169836135ed565b600080612cf083611431565b9050806001600160a01b0316846001600160a01b03161480612d175750612d178185612730565b80612d3b5750836001600160a01b0316612d3084610b72565b6001600160a01b0316145b949350505050565b826001600160a01b0316612d5682611431565b6001600160a01b031614612dba5760405162461bcd60e51b815260206004820152602560248201527f4552433732313a207472616e736665722066726f6d20696e636f72726563742060448201526437bbb732b960d91b6064820152608401610c0d565b6001600160a01b038216612e1c5760405162461bcd60e51b8152602060048201526024808201527f4552433732313a207472616e7366657220746f20746865207a65726f206164646044820152637265737360e01b6064820152608401610c0d565b612e27600082612a69565b6001600160a01b0383166000908152606860205260408120805460019290612e509084906143e1565b90915550506001600160a01b0382166000908152606860205260408120805460019290612e7e908490614710565b909155505060008181526067602052604080822080546001600160a01b0319166001600160a01b03868116918217909255915184939187169160008051602061497683398151915291a4505050565b6000908152606760205260409020546001600160a01b0316151590565b80471015612f3a5760405162461bcd60e51b815260206004820152601d60248201527f416464726573733a20696e73756666696369656e742062616c616e63650000006044820152606401610c0d565b6000826001600160a01b03168260405160006040518083038185875af1925050503d8060008114612f87576040519150601f19603f3d011682016040523d82523d6000602084013e612f8c565b606091505b5050905080610cae5760405162461bcd60e51b815260206004820152603a60248201527f416464726573733a20756e61626c6520746f2073656e642076616c75652c207260448201527f6563697069656e74206d617920686176652072657665727465640000000000006064820152608401610c0d565b600061300e82611431565b905061301b600083612a69565b6001600160a01b03811660009081526068602052604081208054600192906130449084906143e1565b909155505060008281526067602052604080822080546001600160a01b0319169055518391906001600160a01b03841690600080516020614976833981519152908390a46112a5565b6000600160dd5460ff1660038111156130a8576130a8613d67565b036130b4575060d65490565b600260dd5460ff1660038111156130cd576130cd613d67565b036130d9575060d75490565b600360dd5460ff1660038111156130f2576130f2613d67565b036130fe575060df5490565b50600090565b609780546001600160a01b038381166001600160a01b0319831681179093556040519116919082907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e090600090a35050565b600054610100900460ff1661317d5760405162461bcd60e51b8152600401610c0d9061487e565b6112a582826136e8565b600054610100900460ff166131ae5760405162461bcd60e51b8152600401610c0d9061487e565b611ced613728565b816001600160a01b0316836001600160a01b0316036132135760405162461bcd60e51b815260206004820152601960248201527822a9219b99189d1030b8383937bb32903a379031b0b63632b960391b6044820152606401610c0d565b6001600160a01b038381166000818152606a6020908152604080832094871680845294825291829020805460ff191686151590811790915591519182527f17307eab39ab6107e8899845ad3d59bd9653f200f220920489ca2b5937696c31910160405180910390a3505050565b61328b848484612d43565b61329784848484613758565b611f275760405162461bcd60e51b8152600401610c0d906148c9565b6000600360dd5460ff1660038111156132ce576132ce613d67565b036132d95750600190565b600260dd5460ff1660038111156132f2576132f2613d67565b036133335733600090815260dc602052604090205460ff16156133155750600190565b33600090815260db602052604090205460ff16156133335750600190565b600160dd5460ff16600381111561334c5761334c613d67565b0361336f5733600090815260dc602052604090205460ff161561336f5750600190565b33613378611f2d565b6001600160a01b0316036130fe5750600190565b6000600160dd5460ff1660038111156133a7576133a7613d67565b036133b3575060d85490565b600260dd5460ff1660038111156133cc576133cc613d67565b036133d8575060d95490565b600360dd5460ff1660038111156133f1576133f1613d67565b036130fe575060da5490565b60ce5460009033908290815b60ce5481101561348c57600081815260cf60205260409020546001600160a01b0380861691160361347a57600081815260d0602090815260408083205480845260d19092529091205460ff161515600114613478578282101561346a578192505b83613474816144ba565b9450505b505b80613484816144ba565b915050613409565b5084518210156134da5760405162461bcd60e51b815260206004820152601960248201527843616e206e6f74206d696e7420616c6c2065646974696f6e7360381b6044820152606401610c0d565b60018160005b87518110156135e1575b600082815260cf60205260409020546001600160a01b0387811691161461351d5781613515816144ba565b9250506134ea565b60d0600083815260200190815260200160002054925061355688828151811061354857613548614487565b602002602001015184613859565b600083815260cb602090815260408083208054600160ff19918216811790925560d18452828520805490911690911790556001600160a01b038916835260de90915281208054916135a6836144ba565b909155505060d280549060006135bb836144ba565b919050555081806135cb906144ba565b92505080806135d9906144ba565b9150506134e0565b50909695505050505050565b600033815b83518110156136dd575b60d354600090815260d1602052604090205460ff1615156001036136345760d3805490600061362a836144ba565b91905055506135fc565b61365984828151811061364957613649614487565b602002602001015160d354613859565b60d38054600090815260cb602090815260408083208054600160ff1991821681179092559454845260d183528184208054909516179093556001600160a01b038516825260de90529081208054916136b0836144ba565b909155505060d280549060006136c5836144ba565b919050555080806136d5906144ba565b9150506135f2565b505060d35492915050565b600054610100900460ff1661370f5760405162461bcd60e51b8152600401610c0d9061487e565b606561371b83826145ce565b506066610cae82826145ce565b600054610100900460ff1661374f5760405162461bcd60e51b8152600401610c0d9061487e565b611ced33613104565b60006001600160a01b0384163b1561384e57604051630a85bd0160e11b81526001600160a01b0385169063150b7a029061379c90339089908890889060040161491b565b6020604051808303816000875af19250505080156137d7575060408051601f3d908101601f191682019092526137d491810190614958565b60015b613834573d808015613805576040519150601f19603f3d011682016040523d82523d6000602084013e61380a565b606091505b50805160000361382c5760405162461bcd60e51b8152600401610c0d906148c9565b805181602001fd5b6001600160e01b031916630a85bd0160e11b149050612d3b565b506001949350505050565b6001600160a01b0382166138af5760405162461bcd60e51b815260206004820181905260248201527f4552433732313a206d696e7420746f20746865207a65726f20616464726573736044820152606401610c0d565b6138b881612ecd565b156139055760405162461bcd60e51b815260206004820152601c60248201527f4552433732313a20746f6b656e20616c7265616479206d696e746564000000006044820152606401610c0d565b6001600160a01b038216600090815260686020526040812080546001929061392e908490614710565b909155505060008181526067602052604080822080546001600160a01b0319166001600160a01b0386169081179091559051839290600080516020614976833981519152908290a46112a5565b6001600160e01b0319811681146112a757600080fd5b6000602082840312156139a357600080fd5b81356121698161397b565b600080600080600080600080610100898b0312156139cb57600080fd5b505086359860208801359850604088013597606081013597506080810135965060a0810135955060c0810135945060e0013592509050565b60005b83811015613a1e578181015183820152602001613a06565b83811115611f275750506000910152565b60008151808452613a47816020860160208601613a03565b601f01601f19169290920160200192915050565b6020815260006121696020830184613a2f565b600060208284031215613a8057600080fd5b5035919050565b6001600160a01b0391909116815260200190565b80356001600160a01b0381168114613ab257600080fd5b919050565b60008060408385031215613aca57600080fd5b613ad383613a9b565b946020939093013593505050565b634e487b7160e01b600052604160045260246000fd5b604051601f8201601f191681016001600160401b0381118282101715613b1f57613b1f613ae1565b604052919050565b60006001600160401b03821115613b4057613b40613ae1565b5060051b60200190565b60006020808385031215613b5d57600080fd5b82356001600160401b03811115613b7357600080fd5b8301601f81018513613b8457600080fd5b8035613b97613b9282613b27565b613af7565b81815260059190911b82018301908381019087831115613bb657600080fd5b928401925b82841015613bdb57613bcc84613a9b565b82529284019290840190613bbb565b979650505050505050565b600080600060608486031215613bfb57600080fd5b613c0484613a9b565b9250613c1260208501613a9b565b9150604084013590509250925092565b60008060408385031215613c3557600080fd5b50508035926020909101359150565b6001600160a01b03929092168252602082015260400190565b600060208284031215613c6f57600080fd5b81356004811061216957600080fd5b600080600060608486031215613c9357600080fd5b505081359360208301359350604090920135919050565b60008083601f840112613cbc57600080fd5b5081356001600160401b03811115613cd357600080fd5b6020830191508360208260051b8501011115610de957600080fd5b600080600080600060608688031215613d0657600080fd5b8535945060208601356001600160401b0380821115613d2457600080fd5b613d3089838a01613caa565b90965094506040880135915080821115613d4957600080fd5b50613d5688828901613caa565b969995985093965092949392505050565b634e487b7160e01b600052602160045260246000fd5b6020810160048310613d9f57634e487b7160e01b600052602160045260246000fd5b91905290565b60006001600160401b03821115613dbe57613dbe613ae1565b50601f01601f191660200190565b6000613dda613b9284613da5565b9050828152838383011115613dee57600080fd5b828260208301376000602084830101529392505050565b600082601f830112613e1657600080fd5b61216983833560208501613dcc565b600080600080600080600080610100898b031215613e4257600080fd5b8835975060208901356001600160401b0380821115613e6057600080fd5b613e6c8c838d01613e05565b985060408b0135915080821115613e8257600080fd5b613e8e8c838d01613e05565b975060608b0135965060808b0135915080821115613eab57600080fd5b613eb78c838d01613e05565b955060a08b0135945060c08b0135915080821115613ed457600080fd5b50613ee18b828c01613e05565b92505060e089013590509295985092959890939650565b608081526000613f0b6080830187613a2f565b8560208401528281036040840152613f238186613a2f565b91505082606083015295945050505050565b60008060008060408587031215613f4b57600080fd5b84356001600160401b0380821115613f6257600080fd5b613f6e88838901613caa565b90965094506020870135915080821115613f8757600080fd5b50613f9487828801613caa565b95989497509550505050565b600060208284031215613fb257600080fd5b61216982613a9b565b600080600060608486031215613fd057600080fd5b8335925060208401356001600160401b0380821115613fee57600080fd5b613ffa87838801613e05565b9350604086013591508082111561401057600080fd5b5061401d86828701613e05565b9150509250925092565b600080600080600060a0868803121561403f57600080fd5b61404886613a9b565b945061405660208701613a9b565b935060408601356001600160401b038082111561407257600080fd5b61407e89838a01613e05565b9450606088013591508082111561409457600080fd5b506140a188828901613e05565b95989497509295608001359392505050565b80151581146112a757600080fd5b600080604083850312156140d457600080fd5b6140dd83613a9b565b915060208301356140ed816140b3565b809150509250929050565b6000806000806080858703121561410e57600080fd5b61411785613a9b565b935061412560208601613a9b565b92506040850135915060608501356001600160401b0381111561414757600080fd5b8501601f8101871361415857600080fd5b61416787823560208401613dcc565b91505092959194509250565b600082601f83011261418457600080fd5b81356020614194613b9283613b27565b82815260059290921b840181019181810190868411156141b357600080fd5b8286015b848110156141f25780356001600160401b038111156141d65760008081fd5b6141e48986838b0101613e05565b8452509183019183016141b7565b509695505050505050565b600082601f83011261420e57600080fd5b8135602061421e613b9283613b27565b82815260059290921b8401810191818101908684111561423d57600080fd5b8286015b848110156141f25780358352918301918301614241565b600080600080600080600060e0888a03121561427357600080fd5b873596506020880135955060408801356001600160401b038082111561429857600080fd5b6142a48b838c01614173565b965060608a01359150808211156142ba57600080fd5b6142c68b838c01614173565b955060808a01359150808211156142dc57600080fd5b6142e88b838c016141fd565b945060a08a01359150808211156142fe57600080fd5b61430a8b838c01614173565b935060c08a013591508082111561432057600080fd5b5061432d8a828b016141fd565b91505092959891949750929550565b6000806040838503121561434f57600080fd5b61435883613a9b565b915061436660208401613a9b565b90509250929050565b6040815260006143826040830185613a2f565b90508260208301529392505050565b600181811c908216806143a557607f821691505b6020821081036143c557634e487b7160e01b600052602260045260246000fd5b50919050565b634e487b7160e01b600052601160045260246000fd5b6000828210156143f3576143f36143cb565b500390565b6020808252602e908201527f4552433732313a2063616c6c6572206973206e6f7420746f6b656e206f776e6560408201526d1c881b9bdc88185c1c1c9bdd995960921b606082015260800190565b6000816000190483118215151615614460576144606143cb565b500290565b60008261448257634e487b7160e01b600052601260045260246000fd5b500490565b634e487b7160e01b600052603260045260246000fd5b6000602082840312156144af57600080fd5b8135612169816140b3565b6000600182016144cc576144cc6143cb565b5060010190565b6020808252600890820152672737903a37b5b2b760c11b604082015260600190565b6020808252600c908201526b139bdd08185c1c1c9bdd995960a21b604082015260600190565b6020808252601c908201527f596f752063757272656e746c792063616e206e6f742072656465656d00000000604082015260600190565b60006020828403121561456457600080fd5b5051919050565b60006020828403121561457d57600080fd5b8151612169816140b3565b601f821115610cae57600081815260208120601f850160051c810160208610156145af5750805b601f850160051c820191505b81811015610fa8578281556001016145bb565b81516001600160401b038111156145e7576145e7613ae1565b6145fb816145f58454614391565b84614588565b602080601f83116001811461463057600084156146185750858301515b600019600386901b1c1916600185901b178555610fa8565b600085815260208120601f198616915b8281101561465f57888601518255948401946001909101908401614640565b508582101561467d5787850151600019600388901b60f8161c191681555b5050505050600190811b01905550565b602080825260189082015277115490cdcc8c4e881a5b9d985b1a59081d1bdad95b88125160421b604082015260600190565b6020808252600b908201526a57726f6e6720707269636560a81b604082015260600190565b602080825260129082015271088c2e8c240e6d2f4ca40dad2e6dac2e8c6d60731b604082015260600190565b60008219821115614723576147236143cb565b500190565b6000815461473581614391565b808552602060018381168015614752576001811461476c5761479a565b60ff1985168884015283151560051b88018301955061479a565b866000528260002060005b858110156147925781548a8201860152908301908401614777565b890184019650505b505050505092915050565b60c0815260006147b860c0830189613a2f565b82810360208401526147ca8189614728565b905082810360408401526147de8188614728565b905082810360608401526147f28187614728565b6080840195909552505060a00152949350505050565b60006020828403121561481a57600080fd5b81516001600160401b0381111561483057600080fd5b8201601f8101841361484157600080fd5b805161484f613b9282613da5565b81815285602083850101111561486457600080fd5b614875826020830160208601613a03565b95945050505050565b6020808252602b908201527f496e697469616c697a61626c653a20636f6e7472616374206973206e6f74206960408201526a6e697469616c697a696e6760a81b606082015260800190565b60208082526032908201527f4552433732313a207472616e7366657220746f206e6f6e20455243373231526560408201527131b2b4bb32b91034b6b83632b6b2b73a32b960711b606082015260800190565b6001600160a01b038581168252841660208201526040810183905260806060820181905260009061494e90830184613a2f565b9695505050505050565b60006020828403121561496a57600080fd5b81516121698161397b56feddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3efa6dc15bdb68da224c66db4b3838d9a2b205138e8cff6774e57d0af91e196d622a2646970667358221220824fd54c64454ccba32c84242855be80fe2cf8bacdb68f0174b3d2bca951cc0a64736f6c634300080f0033", "devdoc": { "author": "Zien Repository: https://github.com/joinzien/expanded-nft", "details": "This allows creators to mint a unique serial drop of an expanded NFT within a custom contract", @@ -1431,7 +1431,7 @@ "getConditionReport(uint256)": { "details": "Get URIs for the condition report", "returns": { - "_0": "_imageUrl, _imageHash" + "_0": "conditionReportUrl, conditionReportHash" } }, "getGeneralMintLimit()": { diff --git a/deployments/rinkeby/SharedNFTLogic.json b/deployments/rinkeby/SharedNFTLogic.json index 928e33d3..2c288efa 100644 --- a/deployments/rinkeby/SharedNFTLogic.json +++ b/deployments/rinkeby/SharedNFTLogic.json @@ -1,5 +1,5 @@ { - "address": "0x952dc1F837Bae76512cBBC744B4e341E8b22a048", + "address": "0x691fB71440A5d5A4A48d8F65020E67F011e413b7", "abi": [ { "inputs": [ @@ -171,27 +171,27 @@ "type": "function" } ], - "transactionHash": "0x07f78591b1c94af22ef222bd59c6be8611c34e160d808191054c09020cdbc914", + "transactionHash": "0xbad239c942e5c92dba37fc4487f1aa95aa0c99229087de6a886716e895c49bff", "receipt": { "to": null, "from": "0xaD1fcD83DE77518d3D1b769F22B0A169eD55A919", - "contractAddress": "0x952dc1F837Bae76512cBBC744B4e341E8b22a048", + "contractAddress": "0x691fB71440A5d5A4A48d8F65020E67F011e413b7", "transactionIndex": 13, - "gasUsed": "760857", + "gasUsed": "757179", "logsBloom": "0x00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000", - "blockHash": "0x6c41176cba0438e0095af7acfc54170ea40ff091d1e51a0569613d3c0935d7a2", - "transactionHash": "0x07f78591b1c94af22ef222bd59c6be8611c34e160d808191054c09020cdbc914", + "blockHash": "0x532e06648db2c36b19c5fed98a412423aef353ffc8caee310760ab82359f9e91", + "transactionHash": "0xbad239c942e5c92dba37fc4487f1aa95aa0c99229087de6a886716e895c49bff", "logs": [], - "blockNumber": 11091888, - "cumulativeGasUsed": "7142842", + "blockNumber": 11092051, + "cumulativeGasUsed": "5209250", "status": 1, "byzantium": true }, "args": [], - "solcInputHash": "f22aeb84b70c3f3bdbb59fc583a836f8", - "metadata": "{\"compiler\":{\"version\":\"0.8.15+commit.e14f2714\"},\"language\":\"Solidity\",\"output\":{\"abi\":[{\"inputs\":[{\"internalType\":\"bytes\",\"name\":\"unencoded\",\"type\":\"bytes\"}],\"name\":\"base64Encode\",\"outputs\":[{\"internalType\":\"string\",\"name\":\"\",\"type\":\"string\"}],\"stateMutability\":\"pure\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"string\",\"name\":\"name\",\"type\":\"string\"},{\"internalType\":\"string\",\"name\":\"description\",\"type\":\"string\"},{\"internalType\":\"string\",\"name\":\"imageUrl\",\"type\":\"string\"},{\"internalType\":\"string\",\"name\":\"animationUrl\",\"type\":\"string\"},{\"internalType\":\"uint256\",\"name\":\"tokenOfEdition\",\"type\":\"uint256\"},{\"internalType\":\"uint256\",\"name\":\"editionSize\",\"type\":\"uint256\"}],\"name\":\"createMetadataEdition\",\"outputs\":[{\"internalType\":\"string\",\"name\":\"\",\"type\":\"string\"}],\"stateMutability\":\"pure\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"string\",\"name\":\"name\",\"type\":\"string\"},{\"internalType\":\"string\",\"name\":\"description\",\"type\":\"string\"},{\"internalType\":\"string\",\"name\":\"mediaData\",\"type\":\"string\"},{\"internalType\":\"uint256\",\"name\":\"tokenOfEdition\",\"type\":\"uint256\"},{\"internalType\":\"uint256\",\"name\":\"editionSize\",\"type\":\"uint256\"}],\"name\":\"createMetadataJSON\",\"outputs\":[{\"internalType\":\"bytes\",\"name\":\"\",\"type\":\"bytes\"}],\"stateMutability\":\"pure\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"bytes\",\"name\":\"json\",\"type\":\"bytes\"}],\"name\":\"encodeMetadataJSON\",\"outputs\":[{\"internalType\":\"string\",\"name\":\"\",\"type\":\"string\"}],\"stateMutability\":\"pure\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"uint256\",\"name\":\"value\",\"type\":\"uint256\"}],\"name\":\"numberToString\",\"outputs\":[{\"internalType\":\"string\",\"name\":\"\",\"type\":\"string\"}],\"stateMutability\":\"pure\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"string\",\"name\":\"imageUrl\",\"type\":\"string\"},{\"internalType\":\"string\",\"name\":\"animationUrl\",\"type\":\"string\"},{\"internalType\":\"uint256\",\"name\":\"tokenOfEdition\",\"type\":\"uint256\"}],\"name\":\"tokenMediaData\",\"outputs\":[{\"internalType\":\"string\",\"name\":\"\",\"type\":\"string\"}],\"stateMutability\":\"pure\",\"type\":\"function\"}],\"devdoc\":{\"details\":\"Can safely be used for generic base64Encode and numberToString functions\",\"kind\":\"dev\",\"methods\":{\"base64Encode(bytes)\":{\"params\":{\"unencoded\":\"bytes to base64-encode\"}},\"createMetadataEdition(string,string,string,string,uint256,uint256)\":{\"params\":{\"animationUrl\":\"URL of animation to render for edition\",\"description\":\"Description of NFT in metadata\",\"editionSize\":\"Size of entire edition to show\",\"imageUrl\":\"URL of image to render for edition\",\"name\":\"Name of NFT in metadata\",\"tokenOfEdition\":\"Token ID for specific token\"}},\"createMetadataJSON(string,string,string,uint256,uint256)\":{\"params\":{\"description\":\"Description of NFT in metadata\",\"editionSize\":\"Size of entire edition to show\",\"mediaData\":\"Data for media to include in json object\",\"name\":\"Name of NFT in metadata\",\"tokenOfEdition\":\"Token ID for specific token\"}},\"encodeMetadataJSON(bytes)\":{\"params\":{\"json\":\"Raw json to base64 and turn into a data-uri\"}},\"numberToString(uint256)\":{\"params\":{\"value\":\"number to return as a string\"}},\"tokenMediaData(string,string,uint256)\":{\"params\":{\"animationUrl\":\"URL of animation to render for edition\",\"imageUrl\":\"URL of image to render for edition\"}}},\"version\":1},\"userdoc\":{\"kind\":\"user\",\"methods\":{\"createMetadataEdition(string,string,string,string,uint256,uint256)\":{\"notice\":\"Generate edition metadata from storage information as base64-json blob Combines the media data and metadata\"},\"createMetadataJSON(string,string,string,uint256,uint256)\":{\"notice\":\"Function to create the metadata json string for the nft edition\"},\"encodeMetadataJSON(bytes)\":{\"notice\":\"Encodes the argument json bytes into base64-data uri format\"},\"numberToString(uint256)\":{\"notice\":\"Proxy to openzeppelin's toString function\"},\"tokenMediaData(string,string,uint256)\":{\"notice\":\"Generates edition metadata from storage information as base64-json blob Combines the media data and metadata\"}},\"notice\":\"Shared NFT logic for rendering metadata associated with editions\",\"version\":1}},\"settings\":{\"compilationTarget\":{\"contracts/SharedNFTLogic.sol\":\"SharedNFTLogic\"},\"evmVersion\":\"london\",\"libraries\":{},\"metadata\":{\"bytecodeHash\":\"ipfs\",\"useLiteralContent\":true},\"optimizer\":{\"enabled\":true,\"runs\":100},\"remappings\":[]},\"sources\":{\"@openzeppelin/contracts-upgradeable/utils/StringsUpgradeable.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n// OpenZeppelin Contracts (last updated v4.7.0) (utils/Strings.sol)\\n\\npragma solidity ^0.8.0;\\n\\n/**\\n * @dev String operations.\\n */\\nlibrary StringsUpgradeable {\\n bytes16 private constant _HEX_SYMBOLS = \\\"0123456789abcdef\\\";\\n uint8 private constant _ADDRESS_LENGTH = 20;\\n\\n /**\\n * @dev Converts a `uint256` to its ASCII `string` decimal representation.\\n */\\n function toString(uint256 value) internal pure returns (string memory) {\\n // Inspired by OraclizeAPI's implementation - MIT licence\\n // https://github.com/oraclize/ethereum-api/blob/b42146b063c7d6ee1358846c198246239e9360e8/oraclizeAPI_0.4.25.sol\\n\\n if (value == 0) {\\n return \\\"0\\\";\\n }\\n uint256 temp = value;\\n uint256 digits;\\n while (temp != 0) {\\n digits++;\\n temp /= 10;\\n }\\n bytes memory buffer = new bytes(digits);\\n while (value != 0) {\\n digits -= 1;\\n buffer[digits] = bytes1(uint8(48 + uint256(value % 10)));\\n value /= 10;\\n }\\n return string(buffer);\\n }\\n\\n /**\\n * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation.\\n */\\n function toHexString(uint256 value) internal pure returns (string memory) {\\n if (value == 0) {\\n return \\\"0x00\\\";\\n }\\n uint256 temp = value;\\n uint256 length = 0;\\n while (temp != 0) {\\n length++;\\n temp >>= 8;\\n }\\n return toHexString(value, length);\\n }\\n\\n /**\\n * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation with fixed length.\\n */\\n function toHexString(uint256 value, uint256 length) internal pure returns (string memory) {\\n bytes memory buffer = new bytes(2 * length + 2);\\n buffer[0] = \\\"0\\\";\\n buffer[1] = \\\"x\\\";\\n for (uint256 i = 2 * length + 1; i > 1; --i) {\\n buffer[i] = _HEX_SYMBOLS[value & 0xf];\\n value >>= 4;\\n }\\n require(value == 0, \\\"Strings: hex length insufficient\\\");\\n return string(buffer);\\n }\\n\\n /**\\n * @dev Converts an `address` with fixed length of 20 bytes to its not checksummed ASCII `string` hexadecimal representation.\\n */\\n function toHexString(address addr) internal pure returns (string memory) {\\n return toHexString(uint256(uint160(addr)), _ADDRESS_LENGTH);\\n }\\n}\\n\",\"keccak256\":\"0xea5339a7fff0ed42b45be56a88efdd0b2ddde9fa480dc99fef9a6a4c5b776863\",\"license\":\"MIT\"},\"base64-sol/base64.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n\\npragma solidity >=0.6.0;\\n\\n/// @title Base64\\n/// @author Brecht Devos - \\n/// @notice Provides functions for encoding/decoding base64\\nlibrary Base64 {\\n string internal constant TABLE_ENCODE = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/';\\n bytes internal constant TABLE_DECODE = hex\\\"0000000000000000000000000000000000000000000000000000000000000000\\\"\\n hex\\\"00000000000000000000003e0000003f3435363738393a3b3c3d000000000000\\\"\\n hex\\\"00000102030405060708090a0b0c0d0e0f101112131415161718190000000000\\\"\\n hex\\\"001a1b1c1d1e1f202122232425262728292a2b2c2d2e2f303132330000000000\\\";\\n\\n function encode(bytes memory data) internal pure returns (string memory) {\\n if (data.length == 0) return '';\\n\\n // load the table into memory\\n string memory table = TABLE_ENCODE;\\n\\n // multiply by 4/3 rounded up\\n uint256 encodedLen = 4 * ((data.length + 2) / 3);\\n\\n // add some extra buffer at the end required for the writing\\n string memory result = new string(encodedLen + 32);\\n\\n assembly {\\n // set the actual output length\\n mstore(result, encodedLen)\\n\\n // prepare the lookup table\\n let tablePtr := add(table, 1)\\n\\n // input ptr\\n let dataPtr := data\\n let endPtr := add(dataPtr, mload(data))\\n\\n // result ptr, jump over length\\n let resultPtr := add(result, 32)\\n\\n // run over the input, 3 bytes at a time\\n for {} lt(dataPtr, endPtr) {}\\n {\\n // read 3 bytes\\n dataPtr := add(dataPtr, 3)\\n let input := mload(dataPtr)\\n\\n // write 4 characters\\n mstore8(resultPtr, mload(add(tablePtr, and(shr(18, input), 0x3F))))\\n resultPtr := add(resultPtr, 1)\\n mstore8(resultPtr, mload(add(tablePtr, and(shr(12, input), 0x3F))))\\n resultPtr := add(resultPtr, 1)\\n mstore8(resultPtr, mload(add(tablePtr, and(shr( 6, input), 0x3F))))\\n resultPtr := add(resultPtr, 1)\\n mstore8(resultPtr, mload(add(tablePtr, and( input, 0x3F))))\\n resultPtr := add(resultPtr, 1)\\n }\\n\\n // padding with '='\\n switch mod(mload(data), 3)\\n case 1 { mstore(sub(resultPtr, 2), shl(240, 0x3d3d)) }\\n case 2 { mstore(sub(resultPtr, 1), shl(248, 0x3d)) }\\n }\\n\\n return result;\\n }\\n\\n function decode(string memory _data) internal pure returns (bytes memory) {\\n bytes memory data = bytes(_data);\\n\\n if (data.length == 0) return new bytes(0);\\n require(data.length % 4 == 0, \\\"invalid base64 decoder input\\\");\\n\\n // load the table into memory\\n bytes memory table = TABLE_DECODE;\\n\\n // every 4 characters represent 3 bytes\\n uint256 decodedLen = (data.length / 4) * 3;\\n\\n // add some extra buffer at the end required for the writing\\n bytes memory result = new bytes(decodedLen + 32);\\n\\n assembly {\\n // padding with '='\\n let lastBytes := mload(add(data, mload(data)))\\n if eq(and(lastBytes, 0xFF), 0x3d) {\\n decodedLen := sub(decodedLen, 1)\\n if eq(and(lastBytes, 0xFFFF), 0x3d3d) {\\n decodedLen := sub(decodedLen, 1)\\n }\\n }\\n\\n // set the actual output length\\n mstore(result, decodedLen)\\n\\n // prepare the lookup table\\n let tablePtr := add(table, 1)\\n\\n // input ptr\\n let dataPtr := data\\n let endPtr := add(dataPtr, mload(data))\\n\\n // result ptr, jump over length\\n let resultPtr := add(result, 32)\\n\\n // run over the input, 4 characters at a time\\n for {} lt(dataPtr, endPtr) {}\\n {\\n // read 4 characters\\n dataPtr := add(dataPtr, 4)\\n let input := mload(dataPtr)\\n\\n // write 3 bytes\\n let output := add(\\n add(\\n shl(18, and(mload(add(tablePtr, and(shr(24, input), 0xFF))), 0xFF)),\\n shl(12, and(mload(add(tablePtr, and(shr(16, input), 0xFF))), 0xFF))),\\n add(\\n shl( 6, and(mload(add(tablePtr, and(shr( 8, input), 0xFF))), 0xFF)),\\n and(mload(add(tablePtr, and( input , 0xFF))), 0xFF)\\n )\\n )\\n mstore(resultPtr, shl(232, output))\\n resultPtr := add(resultPtr, 3)\\n }\\n }\\n\\n return result;\\n }\\n}\\n\",\"keccak256\":\"0xa73959e6ef0b693e4423a562e612370160b934a75e618361ddd8c9c4b8ddbaaf\",\"license\":\"MIT\"},\"contracts/IPublicSharedMetadata.sol\":{\"content\":\"// SPDX-License-Identifier: GPL-3.0\\n\\npragma solidity ^0.8.15;\\n\\n/// Shared public library for on-chain NFT functions\\ninterface IPublicSharedMetadata {\\n /// @param unencoded bytes to base64-encode\\n function base64Encode(bytes memory unencoded)\\n external\\n pure\\n returns (string memory);\\n\\n /// Encodes the argument json bytes into base64-data uri format\\n /// @param json Raw json to base64 and turn into a data-uri\\n function encodeMetadataJSON(bytes memory json)\\n external\\n pure\\n returns (string memory);\\n\\n /// Proxy to openzeppelin's toString function\\n /// @param value number to return as a string\\n function numberToString(uint256 value)\\n external\\n pure\\n returns (string memory);\\n}\\n\",\"keccak256\":\"0x76caf66cf71c1067c845ff9de580b6e886c448915997ed1d6f37fe1a1ec489de\",\"license\":\"GPL-3.0\"},\"contracts/SharedNFTLogic.sol\":{\"content\":\"// SPDX-License-Identifier: GPL-3.0\\n\\npragma solidity ^0.8.15;\\n\\nimport {StringsUpgradeable} from \\\"@openzeppelin/contracts-upgradeable/utils/StringsUpgradeable.sol\\\";\\nimport {Base64} from \\\"base64-sol/base64.sol\\\";\\nimport {IPublicSharedMetadata} from \\\"./IPublicSharedMetadata.sol\\\";\\n\\n/// Shared NFT logic for rendering metadata associated with editions\\n/// @dev Can safely be used for generic base64Encode and numberToString functions\\ncontract SharedNFTLogic is IPublicSharedMetadata {\\n /// @param unencoded bytes to base64-encode\\n function base64Encode(bytes memory unencoded)\\n public\\n pure\\n override\\n returns (string memory)\\n {\\n return Base64.encode(unencoded);\\n }\\n\\n /// Proxy to openzeppelin's toString function\\n /// @param value number to return as a string\\n function numberToString(uint256 value)\\n public\\n pure\\n override\\n returns (string memory)\\n {\\n return StringsUpgradeable.toString(value);\\n }\\n\\n /// Generate edition metadata from storage information as base64-json blob\\n /// Combines the media data and metadata\\n /// @param name Name of NFT in metadata\\n /// @param description Description of NFT in metadata\\n /// @param imageUrl URL of image to render for edition\\n /// @param animationUrl URL of animation to render for edition\\n /// @param tokenOfEdition Token ID for specific token\\n /// @param editionSize Size of entire edition to show\\n function createMetadataEdition(\\n string memory name,\\n string memory description,\\n string memory imageUrl,\\n string memory animationUrl,\\n uint256 tokenOfEdition,\\n uint256 editionSize\\n ) external pure returns (string memory) {\\n string memory _tokenMediaData = tokenMediaData(\\n imageUrl,\\n animationUrl,\\n tokenOfEdition\\n );\\n bytes memory json = createMetadataJSON(\\n name,\\n description,\\n _tokenMediaData,\\n tokenOfEdition,\\n editionSize\\n );\\n return encodeMetadataJSON(json);\\n }\\n\\n /// Function to create the metadata json string for the nft edition\\n /// @param name Name of NFT in metadata\\n /// @param description Description of NFT in metadata\\n /// @param mediaData Data for media to include in json object\\n /// @param tokenOfEdition Token ID for specific token\\n /// @param editionSize Size of entire edition to show\\n function createMetadataJSON(\\n string memory name,\\n string memory description,\\n string memory mediaData,\\n uint256 tokenOfEdition,\\n uint256 editionSize\\n ) public pure returns (bytes memory) {\\n bytes memory editionSizeText;\\n if (editionSize > 0) {\\n editionSizeText = abi.encodePacked(\\n \\\"/\\\",\\n numberToString(editionSize)\\n );\\n }\\n return\\n // solhint-disable quotes\\n abi.encodePacked(\\n '{\\\"name\\\": \\\"',\\n name,\\n \\\" \\\",\\n numberToString(tokenOfEdition),\\n editionSizeText,\\n '\\\", \\\"',\\n 'description\\\": \\\"',\\n description,\\n '\\\", \\\"',\\n mediaData,\\n 'properties\\\": {\\\"number\\\": ',\\n numberToString(tokenOfEdition),\\n ',',\\n '\\\"}}'\\n );\\n // solhint-enable quotes\\n }\\n\\n /// Encodes the argument json bytes into base64-data uri format\\n /// @param json Raw json to base64 and turn into a data-uri\\n function encodeMetadataJSON(bytes memory json)\\n public\\n pure\\n override\\n returns (string memory)\\n {\\n return\\n string(\\n abi.encodePacked(\\n \\\"data:application/json;base64,\\\",\\n base64Encode(json)\\n )\\n );\\n }\\n\\n /// Generates edition metadata from storage information as base64-json blob\\n /// Combines the media data and metadata\\n /// @param imageUrl URL of image to render for edition\\n /// @param animationUrl URL of animation to render for edition\\n function tokenMediaData(\\n string memory imageUrl,\\n string memory animationUrl,\\n uint256 tokenOfEdition\\n ) public pure returns (string memory) {\\n bool hasImage = bytes(imageUrl).length > 0;\\n bool hasAnimation = bytes(animationUrl).length > 0;\\n if (hasImage && hasAnimation) {\\n return\\n // solhint-disable quotes\\n string(\\n abi.encodePacked(\\n 'image\\\": \\\"',\\n imageUrl,\\n \\\"?id=\\\",\\n numberToString(tokenOfEdition),\\n '\\\", \\\"animation_url\\\": \\\"',\\n animationUrl,\\n \\\"?id=\\\",\\n numberToString(tokenOfEdition),\\n '\\\", \\\"'\\n )\\n );\\n // solhint-enable quotes\\n }\\n if (hasImage) {\\n return\\n // solhint-disable quotes\\n string(\\n abi.encodePacked(\\n 'image\\\": \\\"',\\n imageUrl,\\n \\\"?id=\\\",\\n numberToString(tokenOfEdition),\\n '\\\", \\\"'\\n )\\n );\\n // solhint-enable quotes\\n }\\n if (hasAnimation) {\\n return\\n // solhint-disable quotes\\n string(\\n abi.encodePacked(\\n 'animation_url\\\": \\\"',\\n animationUrl,\\n \\\"?id=\\\",\\n numberToString(tokenOfEdition),\\n '\\\", \\\"'\\n )\\n );\\n // solhint-enable quotes\\n }\\n\\n return \\\"\\\";\\n }\\n}\\n\",\"keccak256\":\"0xb1c53433d1f74e358472e64cbb82db20af76022599c34147ce42d458fc4647b8\",\"license\":\"GPL-3.0\"}},\"version\":1}", - "bytecode": "0x608060405234801561001057600080fd5b50610cce806100206000396000f3fe608060405234801561001057600080fd5b50600436106100625760003560e01c8063170dc6601461006757806358464f3014610090578063d01fde8c146100a3578063d5fb1b19146100b6578063df30dba0146100c9578063eb111ab6146100dc575b600080fd5b61007a610075366004610513565b6100ef565b6040516100879190610588565b60405180910390f35b61007a61009e366004610646565b610100565b61007a6100b13660046106df565b61017a565b61007a6100c43660046106df565b6101ab565b61007a6100d7366004610727565b6101b6565b61007a6100ea3660046107e5565b6101ee565b60606100fa826102a7565b92915050565b606080821561013457610112836100ef565b604051602001610122919061086d565b60405160208183030381529060405290505b8661013e856100ef565b82888861014a896100ef565b60405160200161015f96959493929190610896565b60405160208183030381529060405291505095945050505050565b6060610185826101ab565b60405160200161019591906109b3565b6040516020818303038152906040529050919050565b60606100fa826103af565b606060006101c58686866101ee565b905060006101d68989848888610100565b90506101e18161017a565b9998505050505050505050565b825182516060911580159115159082906102055750805b156102495785610214856100ef565b8661021e876100ef565b60405160200161023194939291906109f8565b604051602081830303815290604052925050506102a0565b811561026a5785610259856100ef565b604051602001610231929190610ab3565b801561028b578461027a856100ef565b604051602001610231929190610b19565b60405180602001604052806000815250925050505b9392505050565b6060816000036102ce5750506040805180820190915260018152600360fc1b602082015290565b8160005b81156102f857806102e281610b9d565b91506102f19050600a83610bcc565b91506102d2565b6000816001600160401b038111156103125761031261059b565b6040519080825280601f01601f19166020018201604052801561033c576020820181803683370190505b5090505b84156103a757610351600183610be0565b915061035e600a86610bf7565b610369906030610c0b565b60f81b81838151811061037e5761037e610c23565b60200101906001600160f81b031916908160001a9053506103a0600a86610bcc565b9450610340565b949350505050565b606081516000036103ce57505060408051602081019091526000815290565b6000604051806060016040528060408152602001610c5960409139905060006003845160026103fd9190610c0b565b6104079190610bcc565b610412906004610c39565b90506000610421826020610c0b565b6001600160401b038111156104385761043861059b565b6040519080825280601f01601f191660200182016040528015610462576020820181803683370190505b509050818152600183018586518101602084015b818310156104ce576003830192508251603f8160121c168501518253600182019150603f81600c1c168501518253600182019150603f8160061c168501518253600182019150603f8116850151825350600101610476565b6003895106600181146104e857600281146104f957610505565b613d3d60f01b600119830152610505565b603d60f81b6000198301525b509398975050505050505050565b60006020828403121561052557600080fd5b5035919050565b60005b8381101561054757818101518382015260200161052f565b83811115610556576000848401525b50505050565b6000815180845261057481602086016020860161052c565b601f01601f19169290920160200192915050565b6020815260006102a0602083018461055c565b634e487b7160e01b600052604160045260246000fd5b60006001600160401b03808411156105cb576105cb61059b565b604051601f8501601f19908116603f011681019082821181831017156105f3576105f361059b565b8160405280935085815286868601111561060c57600080fd5b858560208301376000602087830101525050509392505050565b600082601f83011261063757600080fd5b6102a0838335602085016105b1565b600080600080600060a0868803121561065e57600080fd5b85356001600160401b038082111561067557600080fd5b61068189838a01610626565b9650602088013591508082111561069757600080fd5b6106a389838a01610626565b955060408801359150808211156106b957600080fd5b506106c688828901610626565b9598949750949560608101359550608001359392505050565b6000602082840312156106f157600080fd5b81356001600160401b0381111561070757600080fd5b8201601f8101841361071857600080fd5b6103a7848235602084016105b1565b60008060008060008060c0878903121561074057600080fd5b86356001600160401b038082111561075757600080fd5b6107638a838b01610626565b9750602089013591508082111561077957600080fd5b6107858a838b01610626565b9650604089013591508082111561079b57600080fd5b6107a78a838b01610626565b955060608901359150808211156107bd57600080fd5b506107ca89828a01610626565b9350506080870135915060a087013590509295509295509295565b6000806000606084860312156107fa57600080fd5b83356001600160401b038082111561081157600080fd5b61081d87838801610626565b9450602086013591508082111561083357600080fd5b5061084086828701610626565b925050604084013590509250925092565b6000815161086381856020860161052c565b9290920192915050565b602f60f81b81526000825161088981600185016020870161052c565b9190910160010192915050565b693d913730b6b2911d101160b11b815286516000906108bc81600a850160208c0161052c565b600160fd1b600a9184019182015287516108dd81600b840160208c0161052c565b87519101906108f381600b840160208b0161052c565b631116101160e11b600b929091019182018190526e3232b9b1b934b83a34b7b7111d101160891b600f830152865161093281601e850160208b0161052c565b601e920191820152845161094d81602284016020890161052c565b6109a5610996610989610983602285870101770383937b832b93a34b2b9911d103d91373ab6b132b9111d160451b815260180190565b88610851565b600b60fa1b815260010190565b62227d7d60e81b815260030190565b9a9950505050505050505050565b7f646174613a6170706c69636174696f6e2f6a736f6e3b6261736536342c0000008152600082516109eb81601d85016020870161052c565b91909101601d0192915050565b6834b6b0b3b2911d101160b91b81528451600090610a1d816009850160208a0161052c565b8083019050633f69643d60e01b8060098301528651610a4381600d850160208b0161052c565b741116101130b734b6b0ba34b7b72fbab936111d101160591b600d93909101928301528551610a79816022850160208a0161052c565b60229201918201528351610a9481602684016020880161052c565b631116101160e11b60269290910191820152602a019695505050505050565b6834b6b0b3b2911d101160b91b81528251600090610ad881600985016020880161052c565b633f69643d60e01b6009918401918201528351610afc81600d84016020880161052c565b631116101160e11b600d9290910191820152601101949350505050565b7030b734b6b0ba34b7b72fbab936111d101160791b81528251600090610b4681601185016020880161052c565b633f69643d60e01b6011918401918201528351610b6a81601584016020880161052c565b631116101160e11b60159290910191820152601901949350505050565b634e487b7160e01b600052601160045260246000fd5b600060018201610baf57610baf610b87565b5060010190565b634e487b7160e01b600052601260045260246000fd5b600082610bdb57610bdb610bb6565b500490565b600082821015610bf257610bf2610b87565b500390565b600082610c0657610c06610bb6565b500690565b60008219821115610c1e57610c1e610b87565b500190565b634e487b7160e01b600052603260045260246000fd5b6000816000190483118215151615610c5357610c53610b87565b50029056fe4142434445464748494a4b4c4d4e4f505152535455565758595a6162636465666768696a6b6c6d6e6f707172737475767778797a303132333435363738392b2fa2646970667358221220c4b6434eda1d17c01bf162df9a7a6f5e2690e45231ad658eaf40277d6dd15aa764736f6c634300080f0033", - "deployedBytecode": "0x608060405234801561001057600080fd5b50600436106100625760003560e01c8063170dc6601461006757806358464f3014610090578063d01fde8c146100a3578063d5fb1b19146100b6578063df30dba0146100c9578063eb111ab6146100dc575b600080fd5b61007a610075366004610513565b6100ef565b6040516100879190610588565b60405180910390f35b61007a61009e366004610646565b610100565b61007a6100b13660046106df565b61017a565b61007a6100c43660046106df565b6101ab565b61007a6100d7366004610727565b6101b6565b61007a6100ea3660046107e5565b6101ee565b60606100fa826102a7565b92915050565b606080821561013457610112836100ef565b604051602001610122919061086d565b60405160208183030381529060405290505b8661013e856100ef565b82888861014a896100ef565b60405160200161015f96959493929190610896565b60405160208183030381529060405291505095945050505050565b6060610185826101ab565b60405160200161019591906109b3565b6040516020818303038152906040529050919050565b60606100fa826103af565b606060006101c58686866101ee565b905060006101d68989848888610100565b90506101e18161017a565b9998505050505050505050565b825182516060911580159115159082906102055750805b156102495785610214856100ef565b8661021e876100ef565b60405160200161023194939291906109f8565b604051602081830303815290604052925050506102a0565b811561026a5785610259856100ef565b604051602001610231929190610ab3565b801561028b578461027a856100ef565b604051602001610231929190610b19565b60405180602001604052806000815250925050505b9392505050565b6060816000036102ce5750506040805180820190915260018152600360fc1b602082015290565b8160005b81156102f857806102e281610b9d565b91506102f19050600a83610bcc565b91506102d2565b6000816001600160401b038111156103125761031261059b565b6040519080825280601f01601f19166020018201604052801561033c576020820181803683370190505b5090505b84156103a757610351600183610be0565b915061035e600a86610bf7565b610369906030610c0b565b60f81b81838151811061037e5761037e610c23565b60200101906001600160f81b031916908160001a9053506103a0600a86610bcc565b9450610340565b949350505050565b606081516000036103ce57505060408051602081019091526000815290565b6000604051806060016040528060408152602001610c5960409139905060006003845160026103fd9190610c0b565b6104079190610bcc565b610412906004610c39565b90506000610421826020610c0b565b6001600160401b038111156104385761043861059b565b6040519080825280601f01601f191660200182016040528015610462576020820181803683370190505b509050818152600183018586518101602084015b818310156104ce576003830192508251603f8160121c168501518253600182019150603f81600c1c168501518253600182019150603f8160061c168501518253600182019150603f8116850151825350600101610476565b6003895106600181146104e857600281146104f957610505565b613d3d60f01b600119830152610505565b603d60f81b6000198301525b509398975050505050505050565b60006020828403121561052557600080fd5b5035919050565b60005b8381101561054757818101518382015260200161052f565b83811115610556576000848401525b50505050565b6000815180845261057481602086016020860161052c565b601f01601f19169290920160200192915050565b6020815260006102a0602083018461055c565b634e487b7160e01b600052604160045260246000fd5b60006001600160401b03808411156105cb576105cb61059b565b604051601f8501601f19908116603f011681019082821181831017156105f3576105f361059b565b8160405280935085815286868601111561060c57600080fd5b858560208301376000602087830101525050509392505050565b600082601f83011261063757600080fd5b6102a0838335602085016105b1565b600080600080600060a0868803121561065e57600080fd5b85356001600160401b038082111561067557600080fd5b61068189838a01610626565b9650602088013591508082111561069757600080fd5b6106a389838a01610626565b955060408801359150808211156106b957600080fd5b506106c688828901610626565b9598949750949560608101359550608001359392505050565b6000602082840312156106f157600080fd5b81356001600160401b0381111561070757600080fd5b8201601f8101841361071857600080fd5b6103a7848235602084016105b1565b60008060008060008060c0878903121561074057600080fd5b86356001600160401b038082111561075757600080fd5b6107638a838b01610626565b9750602089013591508082111561077957600080fd5b6107858a838b01610626565b9650604089013591508082111561079b57600080fd5b6107a78a838b01610626565b955060608901359150808211156107bd57600080fd5b506107ca89828a01610626565b9350506080870135915060a087013590509295509295509295565b6000806000606084860312156107fa57600080fd5b83356001600160401b038082111561081157600080fd5b61081d87838801610626565b9450602086013591508082111561083357600080fd5b5061084086828701610626565b925050604084013590509250925092565b6000815161086381856020860161052c565b9290920192915050565b602f60f81b81526000825161088981600185016020870161052c565b9190910160010192915050565b693d913730b6b2911d101160b11b815286516000906108bc81600a850160208c0161052c565b600160fd1b600a9184019182015287516108dd81600b840160208c0161052c565b87519101906108f381600b840160208b0161052c565b631116101160e11b600b929091019182018190526e3232b9b1b934b83a34b7b7111d101160891b600f830152865161093281601e850160208b0161052c565b601e920191820152845161094d81602284016020890161052c565b6109a5610996610989610983602285870101770383937b832b93a34b2b9911d103d91373ab6b132b9111d160451b815260180190565b88610851565b600b60fa1b815260010190565b62227d7d60e81b815260030190565b9a9950505050505050505050565b7f646174613a6170706c69636174696f6e2f6a736f6e3b6261736536342c0000008152600082516109eb81601d85016020870161052c565b91909101601d0192915050565b6834b6b0b3b2911d101160b91b81528451600090610a1d816009850160208a0161052c565b8083019050633f69643d60e01b8060098301528651610a4381600d850160208b0161052c565b741116101130b734b6b0ba34b7b72fbab936111d101160591b600d93909101928301528551610a79816022850160208a0161052c565b60229201918201528351610a9481602684016020880161052c565b631116101160e11b60269290910191820152602a019695505050505050565b6834b6b0b3b2911d101160b91b81528251600090610ad881600985016020880161052c565b633f69643d60e01b6009918401918201528351610afc81600d84016020880161052c565b631116101160e11b600d9290910191820152601101949350505050565b7030b734b6b0ba34b7b72fbab936111d101160791b81528251600090610b4681601185016020880161052c565b633f69643d60e01b6011918401918201528351610b6a81601584016020880161052c565b631116101160e11b60159290910191820152601901949350505050565b634e487b7160e01b600052601160045260246000fd5b600060018201610baf57610baf610b87565b5060010190565b634e487b7160e01b600052601260045260246000fd5b600082610bdb57610bdb610bb6565b500490565b600082821015610bf257610bf2610b87565b500390565b600082610c0657610c06610bb6565b500690565b60008219821115610c1e57610c1e610b87565b500190565b634e487b7160e01b600052603260045260246000fd5b6000816000190483118215151615610c5357610c53610b87565b50029056fe4142434445464748494a4b4c4d4e4f505152535455565758595a6162636465666768696a6b6c6d6e6f707172737475767778797a303132333435363738392b2fa2646970667358221220c4b6434eda1d17c01bf162df9a7a6f5e2690e45231ad658eaf40277d6dd15aa764736f6c634300080f0033", + "solcInputHash": "e6a2ffa4bb274580347e73637376a7e0", + "metadata": "{\"compiler\":{\"version\":\"0.8.15+commit.e14f2714\"},\"language\":\"Solidity\",\"output\":{\"abi\":[{\"inputs\":[{\"internalType\":\"bytes\",\"name\":\"unencoded\",\"type\":\"bytes\"}],\"name\":\"base64Encode\",\"outputs\":[{\"internalType\":\"string\",\"name\":\"\",\"type\":\"string\"}],\"stateMutability\":\"pure\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"string\",\"name\":\"name\",\"type\":\"string\"},{\"internalType\":\"string\",\"name\":\"description\",\"type\":\"string\"},{\"internalType\":\"string\",\"name\":\"imageUrl\",\"type\":\"string\"},{\"internalType\":\"string\",\"name\":\"animationUrl\",\"type\":\"string\"},{\"internalType\":\"uint256\",\"name\":\"tokenOfEdition\",\"type\":\"uint256\"},{\"internalType\":\"uint256\",\"name\":\"editionSize\",\"type\":\"uint256\"}],\"name\":\"createMetadataEdition\",\"outputs\":[{\"internalType\":\"string\",\"name\":\"\",\"type\":\"string\"}],\"stateMutability\":\"pure\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"string\",\"name\":\"name\",\"type\":\"string\"},{\"internalType\":\"string\",\"name\":\"description\",\"type\":\"string\"},{\"internalType\":\"string\",\"name\":\"mediaData\",\"type\":\"string\"},{\"internalType\":\"uint256\",\"name\":\"tokenOfEdition\",\"type\":\"uint256\"},{\"internalType\":\"uint256\",\"name\":\"editionSize\",\"type\":\"uint256\"}],\"name\":\"createMetadataJSON\",\"outputs\":[{\"internalType\":\"bytes\",\"name\":\"\",\"type\":\"bytes\"}],\"stateMutability\":\"pure\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"bytes\",\"name\":\"json\",\"type\":\"bytes\"}],\"name\":\"encodeMetadataJSON\",\"outputs\":[{\"internalType\":\"string\",\"name\":\"\",\"type\":\"string\"}],\"stateMutability\":\"pure\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"uint256\",\"name\":\"value\",\"type\":\"uint256\"}],\"name\":\"numberToString\",\"outputs\":[{\"internalType\":\"string\",\"name\":\"\",\"type\":\"string\"}],\"stateMutability\":\"pure\",\"type\":\"function\"},{\"inputs\":[{\"internalType\":\"string\",\"name\":\"imageUrl\",\"type\":\"string\"},{\"internalType\":\"string\",\"name\":\"animationUrl\",\"type\":\"string\"},{\"internalType\":\"uint256\",\"name\":\"tokenOfEdition\",\"type\":\"uint256\"}],\"name\":\"tokenMediaData\",\"outputs\":[{\"internalType\":\"string\",\"name\":\"\",\"type\":\"string\"}],\"stateMutability\":\"pure\",\"type\":\"function\"}],\"devdoc\":{\"details\":\"Can safely be used for generic base64Encode and numberToString functions\",\"kind\":\"dev\",\"methods\":{\"base64Encode(bytes)\":{\"params\":{\"unencoded\":\"bytes to base64-encode\"}},\"createMetadataEdition(string,string,string,string,uint256,uint256)\":{\"params\":{\"animationUrl\":\"URL of animation to render for edition\",\"description\":\"Description of NFT in metadata\",\"editionSize\":\"Size of entire edition to show\",\"imageUrl\":\"URL of image to render for edition\",\"name\":\"Name of NFT in metadata\",\"tokenOfEdition\":\"Token ID for specific token\"}},\"createMetadataJSON(string,string,string,uint256,uint256)\":{\"params\":{\"description\":\"Description of NFT in metadata\",\"editionSize\":\"Size of entire edition to show\",\"mediaData\":\"Data for media to include in json object\",\"name\":\"Name of NFT in metadata\",\"tokenOfEdition\":\"Token ID for specific token\"}},\"encodeMetadataJSON(bytes)\":{\"params\":{\"json\":\"Raw json to base64 and turn into a data-uri\"}},\"numberToString(uint256)\":{\"params\":{\"value\":\"number to return as a string\"}},\"tokenMediaData(string,string,uint256)\":{\"params\":{\"animationUrl\":\"URL of animation to render for edition\",\"imageUrl\":\"URL of image to render for edition\"}}},\"version\":1},\"userdoc\":{\"kind\":\"user\",\"methods\":{\"createMetadataEdition(string,string,string,string,uint256,uint256)\":{\"notice\":\"Generate edition metadata from storage information as base64-json blob Combines the media data and metadata\"},\"createMetadataJSON(string,string,string,uint256,uint256)\":{\"notice\":\"Function to create the metadata json string for the nft edition\"},\"encodeMetadataJSON(bytes)\":{\"notice\":\"Encodes the argument json bytes into base64-data uri format\"},\"numberToString(uint256)\":{\"notice\":\"Proxy to openzeppelin's toString function\"},\"tokenMediaData(string,string,uint256)\":{\"notice\":\"Generates edition metadata from storage information as base64-json blob Combines the media data and metadata\"}},\"notice\":\"Shared NFT logic for rendering metadata associated with editions\",\"version\":1}},\"settings\":{\"compilationTarget\":{\"contracts/SharedNFTLogic.sol\":\"SharedNFTLogic\"},\"evmVersion\":\"london\",\"libraries\":{},\"metadata\":{\"bytecodeHash\":\"ipfs\",\"useLiteralContent\":true},\"optimizer\":{\"enabled\":true,\"runs\":100},\"remappings\":[]},\"sources\":{\"@openzeppelin/contracts-upgradeable/utils/StringsUpgradeable.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n// OpenZeppelin Contracts (last updated v4.7.0) (utils/Strings.sol)\\n\\npragma solidity ^0.8.0;\\n\\n/**\\n * @dev String operations.\\n */\\nlibrary StringsUpgradeable {\\n bytes16 private constant _HEX_SYMBOLS = \\\"0123456789abcdef\\\";\\n uint8 private constant _ADDRESS_LENGTH = 20;\\n\\n /**\\n * @dev Converts a `uint256` to its ASCII `string` decimal representation.\\n */\\n function toString(uint256 value) internal pure returns (string memory) {\\n // Inspired by OraclizeAPI's implementation - MIT licence\\n // https://github.com/oraclize/ethereum-api/blob/b42146b063c7d6ee1358846c198246239e9360e8/oraclizeAPI_0.4.25.sol\\n\\n if (value == 0) {\\n return \\\"0\\\";\\n }\\n uint256 temp = value;\\n uint256 digits;\\n while (temp != 0) {\\n digits++;\\n temp /= 10;\\n }\\n bytes memory buffer = new bytes(digits);\\n while (value != 0) {\\n digits -= 1;\\n buffer[digits] = bytes1(uint8(48 + uint256(value % 10)));\\n value /= 10;\\n }\\n return string(buffer);\\n }\\n\\n /**\\n * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation.\\n */\\n function toHexString(uint256 value) internal pure returns (string memory) {\\n if (value == 0) {\\n return \\\"0x00\\\";\\n }\\n uint256 temp = value;\\n uint256 length = 0;\\n while (temp != 0) {\\n length++;\\n temp >>= 8;\\n }\\n return toHexString(value, length);\\n }\\n\\n /**\\n * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation with fixed length.\\n */\\n function toHexString(uint256 value, uint256 length) internal pure returns (string memory) {\\n bytes memory buffer = new bytes(2 * length + 2);\\n buffer[0] = \\\"0\\\";\\n buffer[1] = \\\"x\\\";\\n for (uint256 i = 2 * length + 1; i > 1; --i) {\\n buffer[i] = _HEX_SYMBOLS[value & 0xf];\\n value >>= 4;\\n }\\n require(value == 0, \\\"Strings: hex length insufficient\\\");\\n return string(buffer);\\n }\\n\\n /**\\n * @dev Converts an `address` with fixed length of 20 bytes to its not checksummed ASCII `string` hexadecimal representation.\\n */\\n function toHexString(address addr) internal pure returns (string memory) {\\n return toHexString(uint256(uint160(addr)), _ADDRESS_LENGTH);\\n }\\n}\\n\",\"keccak256\":\"0xea5339a7fff0ed42b45be56a88efdd0b2ddde9fa480dc99fef9a6a4c5b776863\",\"license\":\"MIT\"},\"base64-sol/base64.sol\":{\"content\":\"// SPDX-License-Identifier: MIT\\n\\npragma solidity >=0.6.0;\\n\\n/// @title Base64\\n/// @author Brecht Devos - \\n/// @notice Provides functions for encoding/decoding base64\\nlibrary Base64 {\\n string internal constant TABLE_ENCODE = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/';\\n bytes internal constant TABLE_DECODE = hex\\\"0000000000000000000000000000000000000000000000000000000000000000\\\"\\n hex\\\"00000000000000000000003e0000003f3435363738393a3b3c3d000000000000\\\"\\n hex\\\"00000102030405060708090a0b0c0d0e0f101112131415161718190000000000\\\"\\n hex\\\"001a1b1c1d1e1f202122232425262728292a2b2c2d2e2f303132330000000000\\\";\\n\\n function encode(bytes memory data) internal pure returns (string memory) {\\n if (data.length == 0) return '';\\n\\n // load the table into memory\\n string memory table = TABLE_ENCODE;\\n\\n // multiply by 4/3 rounded up\\n uint256 encodedLen = 4 * ((data.length + 2) / 3);\\n\\n // add some extra buffer at the end required for the writing\\n string memory result = new string(encodedLen + 32);\\n\\n assembly {\\n // set the actual output length\\n mstore(result, encodedLen)\\n\\n // prepare the lookup table\\n let tablePtr := add(table, 1)\\n\\n // input ptr\\n let dataPtr := data\\n let endPtr := add(dataPtr, mload(data))\\n\\n // result ptr, jump over length\\n let resultPtr := add(result, 32)\\n\\n // run over the input, 3 bytes at a time\\n for {} lt(dataPtr, endPtr) {}\\n {\\n // read 3 bytes\\n dataPtr := add(dataPtr, 3)\\n let input := mload(dataPtr)\\n\\n // write 4 characters\\n mstore8(resultPtr, mload(add(tablePtr, and(shr(18, input), 0x3F))))\\n resultPtr := add(resultPtr, 1)\\n mstore8(resultPtr, mload(add(tablePtr, and(shr(12, input), 0x3F))))\\n resultPtr := add(resultPtr, 1)\\n mstore8(resultPtr, mload(add(tablePtr, and(shr( 6, input), 0x3F))))\\n resultPtr := add(resultPtr, 1)\\n mstore8(resultPtr, mload(add(tablePtr, and( input, 0x3F))))\\n resultPtr := add(resultPtr, 1)\\n }\\n\\n // padding with '='\\n switch mod(mload(data), 3)\\n case 1 { mstore(sub(resultPtr, 2), shl(240, 0x3d3d)) }\\n case 2 { mstore(sub(resultPtr, 1), shl(248, 0x3d)) }\\n }\\n\\n return result;\\n }\\n\\n function decode(string memory _data) internal pure returns (bytes memory) {\\n bytes memory data = bytes(_data);\\n\\n if (data.length == 0) return new bytes(0);\\n require(data.length % 4 == 0, \\\"invalid base64 decoder input\\\");\\n\\n // load the table into memory\\n bytes memory table = TABLE_DECODE;\\n\\n // every 4 characters represent 3 bytes\\n uint256 decodedLen = (data.length / 4) * 3;\\n\\n // add some extra buffer at the end required for the writing\\n bytes memory result = new bytes(decodedLen + 32);\\n\\n assembly {\\n // padding with '='\\n let lastBytes := mload(add(data, mload(data)))\\n if eq(and(lastBytes, 0xFF), 0x3d) {\\n decodedLen := sub(decodedLen, 1)\\n if eq(and(lastBytes, 0xFFFF), 0x3d3d) {\\n decodedLen := sub(decodedLen, 1)\\n }\\n }\\n\\n // set the actual output length\\n mstore(result, decodedLen)\\n\\n // prepare the lookup table\\n let tablePtr := add(table, 1)\\n\\n // input ptr\\n let dataPtr := data\\n let endPtr := add(dataPtr, mload(data))\\n\\n // result ptr, jump over length\\n let resultPtr := add(result, 32)\\n\\n // run over the input, 4 characters at a time\\n for {} lt(dataPtr, endPtr) {}\\n {\\n // read 4 characters\\n dataPtr := add(dataPtr, 4)\\n let input := mload(dataPtr)\\n\\n // write 3 bytes\\n let output := add(\\n add(\\n shl(18, and(mload(add(tablePtr, and(shr(24, input), 0xFF))), 0xFF)),\\n shl(12, and(mload(add(tablePtr, and(shr(16, input), 0xFF))), 0xFF))),\\n add(\\n shl( 6, and(mload(add(tablePtr, and(shr( 8, input), 0xFF))), 0xFF)),\\n and(mload(add(tablePtr, and( input , 0xFF))), 0xFF)\\n )\\n )\\n mstore(resultPtr, shl(232, output))\\n resultPtr := add(resultPtr, 3)\\n }\\n }\\n\\n return result;\\n }\\n}\\n\",\"keccak256\":\"0xa73959e6ef0b693e4423a562e612370160b934a75e618361ddd8c9c4b8ddbaaf\",\"license\":\"MIT\"},\"contracts/IPublicSharedMetadata.sol\":{\"content\":\"// SPDX-License-Identifier: GPL-3.0\\n\\npragma solidity ^0.8.15;\\n\\n/// Shared public library for on-chain NFT functions\\ninterface IPublicSharedMetadata {\\n /// @param unencoded bytes to base64-encode\\n function base64Encode(bytes memory unencoded)\\n external\\n pure\\n returns (string memory);\\n\\n /// Encodes the argument json bytes into base64-data uri format\\n /// @param json Raw json to base64 and turn into a data-uri\\n function encodeMetadataJSON(bytes memory json)\\n external\\n pure\\n returns (string memory);\\n\\n /// Proxy to openzeppelin's toString function\\n /// @param value number to return as a string\\n function numberToString(uint256 value)\\n external\\n pure\\n returns (string memory);\\n}\\n\",\"keccak256\":\"0x76caf66cf71c1067c845ff9de580b6e886c448915997ed1d6f37fe1a1ec489de\",\"license\":\"GPL-3.0\"},\"contracts/SharedNFTLogic.sol\":{\"content\":\"// SPDX-License-Identifier: GPL-3.0\\n\\npragma solidity ^0.8.15;\\n\\nimport {StringsUpgradeable} from \\\"@openzeppelin/contracts-upgradeable/utils/StringsUpgradeable.sol\\\";\\nimport {Base64} from \\\"base64-sol/base64.sol\\\";\\nimport {IPublicSharedMetadata} from \\\"./IPublicSharedMetadata.sol\\\";\\n\\n/// Shared NFT logic for rendering metadata associated with editions\\n/// @dev Can safely be used for generic base64Encode and numberToString functions\\ncontract SharedNFTLogic is IPublicSharedMetadata {\\n /// @param unencoded bytes to base64-encode\\n function base64Encode(bytes memory unencoded)\\n public\\n pure\\n override\\n returns (string memory)\\n {\\n return Base64.encode(unencoded);\\n }\\n\\n /// Proxy to openzeppelin's toString function\\n /// @param value number to return as a string\\n function numberToString(uint256 value)\\n public\\n pure\\n override\\n returns (string memory)\\n {\\n return StringsUpgradeable.toString(value);\\n }\\n\\n /// Generate edition metadata from storage information as base64-json blob\\n /// Combines the media data and metadata\\n /// @param name Name of NFT in metadata\\n /// @param description Description of NFT in metadata\\n /// @param imageUrl URL of image to render for edition\\n /// @param animationUrl URL of animation to render for edition\\n /// @param tokenOfEdition Token ID for specific token\\n /// @param editionSize Size of entire edition to show\\n function createMetadataEdition(\\n string memory name,\\n string memory description,\\n string memory imageUrl,\\n string memory animationUrl,\\n uint256 tokenOfEdition,\\n uint256 editionSize\\n ) external pure returns (string memory) {\\n string memory _tokenMediaData = tokenMediaData(\\n imageUrl,\\n animationUrl,\\n tokenOfEdition\\n );\\n bytes memory json = createMetadataJSON(\\n name,\\n description,\\n _tokenMediaData,\\n tokenOfEdition,\\n editionSize\\n );\\n return encodeMetadataJSON(json);\\n }\\n\\n /// Function to create the metadata json string for the nft edition\\n /// @param name Name of NFT in metadata\\n /// @param description Description of NFT in metadata\\n /// @param mediaData Data for media to include in json object\\n /// @param tokenOfEdition Token ID for specific token\\n /// @param editionSize Size of entire edition to show\\n function createMetadataJSON(\\n string memory name,\\n string memory description,\\n string memory mediaData,\\n uint256 tokenOfEdition,\\n uint256 editionSize\\n ) public pure returns (bytes memory) {\\n bytes memory editionSizeText;\\n if (editionSize > 0) {\\n editionSizeText = abi.encodePacked(\\n \\\"/\\\",\\n numberToString(editionSize)\\n );\\n }\\n return\\n // solhint-disable quotes\\n abi.encodePacked(\\n '{\\\"name\\\": \\\"',\\n name,\\n \\\" \\\",\\n numberToString(tokenOfEdition),\\n editionSizeText,\\n '\\\", \\\"',\\n 'description\\\": \\\"',\\n description,\\n '\\\", \\\"',\\n mediaData,\\n 'properties\\\": {\\\"number\\\": ',\\n numberToString(tokenOfEdition),\\n '}}'\\n );\\n // solhint-enable quotes\\n }\\n\\n /// Encodes the argument json bytes into base64-data uri format\\n /// @param json Raw json to base64 and turn into a data-uri\\n function encodeMetadataJSON(bytes memory json)\\n public\\n pure\\n override\\n returns (string memory)\\n {\\n return\\n string(\\n abi.encodePacked(\\n \\\"data:application/json;base64,\\\",\\n base64Encode(json)\\n )\\n );\\n }\\n\\n /// Generates edition metadata from storage information as base64-json blob\\n /// Combines the media data and metadata\\n /// @param imageUrl URL of image to render for edition\\n /// @param animationUrl URL of animation to render for edition\\n function tokenMediaData(\\n string memory imageUrl,\\n string memory animationUrl,\\n uint256 tokenOfEdition\\n ) public pure returns (string memory) {\\n bool hasImage = bytes(imageUrl).length > 0;\\n bool hasAnimation = bytes(animationUrl).length > 0;\\n if (hasImage && hasAnimation) {\\n return\\n // solhint-disable quotes\\n string(\\n abi.encodePacked(\\n 'image\\\": \\\"',\\n imageUrl,\\n \\\"?id=\\\",\\n numberToString(tokenOfEdition),\\n '\\\", \\\"animation_url\\\": \\\"',\\n animationUrl,\\n \\\"?id=\\\",\\n numberToString(tokenOfEdition),\\n '\\\", \\\"'\\n )\\n );\\n // solhint-enable quotes\\n }\\n if (hasImage) {\\n return\\n // solhint-disable quotes\\n string(\\n abi.encodePacked(\\n 'image\\\": \\\"',\\n imageUrl,\\n \\\"?id=\\\",\\n numberToString(tokenOfEdition),\\n '\\\", \\\"'\\n )\\n );\\n // solhint-enable quotes\\n }\\n if (hasAnimation) {\\n return\\n // solhint-disable quotes\\n string(\\n abi.encodePacked(\\n 'animation_url\\\": \\\"',\\n animationUrl,\\n \\\"?id=\\\",\\n numberToString(tokenOfEdition),\\n '\\\", \\\"'\\n )\\n );\\n // solhint-enable quotes\\n }\\n\\n return \\\"\\\";\\n }\\n}\\n\",\"keccak256\":\"0x869febd1ab7ac26132dc40cc132cc527f480131b1d2b92a4a65af60ca16660a5\",\"license\":\"GPL-3.0\"}},\"version\":1}", + "bytecode": "0x608060405234801561001057600080fd5b50610cbd806100206000396000f3fe608060405234801561001057600080fd5b50600436106100625760003560e01c8063170dc6601461006757806358464f3014610090578063d01fde8c146100a3578063d5fb1b19146100b6578063df30dba0146100c9578063eb111ab6146100dc575b600080fd5b61007a610075366004610513565b6100ef565b6040516100879190610588565b60405180910390f35b61007a61009e366004610646565b610100565b61007a6100b13660046106df565b61017a565b61007a6100c43660046106df565b6101ab565b61007a6100d7366004610727565b6101b6565b61007a6100ea3660046107e5565b6101ee565b60606100fa826102a7565b92915050565b606080821561013457610112836100ef565b604051602001610122919061086d565b60405160208183030381529060405290505b8661013e856100ef565b82888861014a896100ef565b60405160200161015f96959493929190610896565b60405160208183030381529060405291505095945050505050565b6060610185826101ab565b60405160200161019591906109a2565b6040516020818303038152906040529050919050565b60606100fa826103af565b606060006101c58686866101ee565b905060006101d68989848888610100565b90506101e18161017a565b9998505050505050505050565b825182516060911580159115159082906102055750805b156102495785610214856100ef565b8661021e876100ef565b60405160200161023194939291906109e7565b604051602081830303815290604052925050506102a0565b811561026a5785610259856100ef565b604051602001610231929190610aa2565b801561028b578461027a856100ef565b604051602001610231929190610b08565b60405180602001604052806000815250925050505b9392505050565b6060816000036102ce5750506040805180820190915260018152600360fc1b602082015290565b8160005b81156102f857806102e281610b8c565b91506102f19050600a83610bbb565b91506102d2565b6000816001600160401b038111156103125761031261059b565b6040519080825280601f01601f19166020018201604052801561033c576020820181803683370190505b5090505b84156103a757610351600183610bcf565b915061035e600a86610be6565b610369906030610bfa565b60f81b81838151811061037e5761037e610c12565b60200101906001600160f81b031916908160001a9053506103a0600a86610bbb565b9450610340565b949350505050565b606081516000036103ce57505060408051602081019091526000815290565b6000604051806060016040528060408152602001610c4860409139905060006003845160026103fd9190610bfa565b6104079190610bbb565b610412906004610c28565b90506000610421826020610bfa565b6001600160401b038111156104385761043861059b565b6040519080825280601f01601f191660200182016040528015610462576020820181803683370190505b509050818152600183018586518101602084015b818310156104ce576003830192508251603f8160121c168501518253600182019150603f81600c1c168501518253600182019150603f8160061c168501518253600182019150603f8116850151825350600101610476565b6003895106600181146104e857600281146104f957610505565b613d3d60f01b600119830152610505565b603d60f81b6000198301525b509398975050505050505050565b60006020828403121561052557600080fd5b5035919050565b60005b8381101561054757818101518382015260200161052f565b83811115610556576000848401525b50505050565b6000815180845261057481602086016020860161052c565b601f01601f19169290920160200192915050565b6020815260006102a0602083018461055c565b634e487b7160e01b600052604160045260246000fd5b60006001600160401b03808411156105cb576105cb61059b565b604051601f8501601f19908116603f011681019082821181831017156105f3576105f361059b565b8160405280935085815286868601111561060c57600080fd5b858560208301376000602087830101525050509392505050565b600082601f83011261063757600080fd5b6102a0838335602085016105b1565b600080600080600060a0868803121561065e57600080fd5b85356001600160401b038082111561067557600080fd5b61068189838a01610626565b9650602088013591508082111561069757600080fd5b6106a389838a01610626565b955060408801359150808211156106b957600080fd5b506106c688828901610626565b9598949750949560608101359550608001359392505050565b6000602082840312156106f157600080fd5b81356001600160401b0381111561070757600080fd5b8201601f8101841361071857600080fd5b6103a7848235602084016105b1565b60008060008060008060c0878903121561074057600080fd5b86356001600160401b038082111561075757600080fd5b6107638a838b01610626565b9750602089013591508082111561077957600080fd5b6107858a838b01610626565b9650604089013591508082111561079b57600080fd5b6107a78a838b01610626565b955060608901359150808211156107bd57600080fd5b506107ca89828a01610626565b9350506080870135915060a087013590509295509295509295565b6000806000606084860312156107fa57600080fd5b83356001600160401b038082111561081157600080fd5b61081d87838801610626565b9450602086013591508082111561083357600080fd5b5061084086828701610626565b925050604084013590509250925092565b6000815161086381856020860161052c565b9290920192915050565b602f60f81b81526000825161088981600185016020870161052c565b9190910160010192915050565b693d913730b6b2911d101160b11b815286516000906108bc81600a850160208c0161052c565b600160fd1b600a9184019182015287516108dd81600b840160208c0161052c565b87519101906108f381600b840160208b0161052c565b631116101160e11b600b929091019182018190526e3232b9b1b934b83a34b7b7111d101160891b600f830152865161093281601e850160208b0161052c565b601e920191820152845161094d81602284016020890161052c565b610994610986610980602284860101770383937b832b93a34b2b9911d103d91373ab6b132b9111d160451b815260180190565b87610851565b617d7d60f01b815260020190565b9a9950505050505050505050565b7f646174613a6170706c69636174696f6e2f6a736f6e3b6261736536342c0000008152600082516109da81601d85016020870161052c565b91909101601d0192915050565b6834b6b0b3b2911d101160b91b81528451600090610a0c816009850160208a0161052c565b8083019050633f69643d60e01b8060098301528651610a3281600d850160208b0161052c565b741116101130b734b6b0ba34b7b72fbab936111d101160591b600d93909101928301528551610a68816022850160208a0161052c565b60229201918201528351610a8381602684016020880161052c565b631116101160e11b60269290910191820152602a019695505050505050565b6834b6b0b3b2911d101160b91b81528251600090610ac781600985016020880161052c565b633f69643d60e01b6009918401918201528351610aeb81600d84016020880161052c565b631116101160e11b600d9290910191820152601101949350505050565b7030b734b6b0ba34b7b72fbab936111d101160791b81528251600090610b3581601185016020880161052c565b633f69643d60e01b6011918401918201528351610b5981601584016020880161052c565b631116101160e11b60159290910191820152601901949350505050565b634e487b7160e01b600052601160045260246000fd5b600060018201610b9e57610b9e610b76565b5060010190565b634e487b7160e01b600052601260045260246000fd5b600082610bca57610bca610ba5565b500490565b600082821015610be157610be1610b76565b500390565b600082610bf557610bf5610ba5565b500690565b60008219821115610c0d57610c0d610b76565b500190565b634e487b7160e01b600052603260045260246000fd5b6000816000190483118215151615610c4257610c42610b76565b50029056fe4142434445464748494a4b4c4d4e4f505152535455565758595a6162636465666768696a6b6c6d6e6f707172737475767778797a303132333435363738392b2fa264697066735822122014071f6bfa44b6d77ac7549b1c45d00426536a719f467ad011f213b1f646920e64736f6c634300080f0033", + "deployedBytecode": "0x608060405234801561001057600080fd5b50600436106100625760003560e01c8063170dc6601461006757806358464f3014610090578063d01fde8c146100a3578063d5fb1b19146100b6578063df30dba0146100c9578063eb111ab6146100dc575b600080fd5b61007a610075366004610513565b6100ef565b6040516100879190610588565b60405180910390f35b61007a61009e366004610646565b610100565b61007a6100b13660046106df565b61017a565b61007a6100c43660046106df565b6101ab565b61007a6100d7366004610727565b6101b6565b61007a6100ea3660046107e5565b6101ee565b60606100fa826102a7565b92915050565b606080821561013457610112836100ef565b604051602001610122919061086d565b60405160208183030381529060405290505b8661013e856100ef565b82888861014a896100ef565b60405160200161015f96959493929190610896565b60405160208183030381529060405291505095945050505050565b6060610185826101ab565b60405160200161019591906109a2565b6040516020818303038152906040529050919050565b60606100fa826103af565b606060006101c58686866101ee565b905060006101d68989848888610100565b90506101e18161017a565b9998505050505050505050565b825182516060911580159115159082906102055750805b156102495785610214856100ef565b8661021e876100ef565b60405160200161023194939291906109e7565b604051602081830303815290604052925050506102a0565b811561026a5785610259856100ef565b604051602001610231929190610aa2565b801561028b578461027a856100ef565b604051602001610231929190610b08565b60405180602001604052806000815250925050505b9392505050565b6060816000036102ce5750506040805180820190915260018152600360fc1b602082015290565b8160005b81156102f857806102e281610b8c565b91506102f19050600a83610bbb565b91506102d2565b6000816001600160401b038111156103125761031261059b565b6040519080825280601f01601f19166020018201604052801561033c576020820181803683370190505b5090505b84156103a757610351600183610bcf565b915061035e600a86610be6565b610369906030610bfa565b60f81b81838151811061037e5761037e610c12565b60200101906001600160f81b031916908160001a9053506103a0600a86610bbb565b9450610340565b949350505050565b606081516000036103ce57505060408051602081019091526000815290565b6000604051806060016040528060408152602001610c4860409139905060006003845160026103fd9190610bfa565b6104079190610bbb565b610412906004610c28565b90506000610421826020610bfa565b6001600160401b038111156104385761043861059b565b6040519080825280601f01601f191660200182016040528015610462576020820181803683370190505b509050818152600183018586518101602084015b818310156104ce576003830192508251603f8160121c168501518253600182019150603f81600c1c168501518253600182019150603f8160061c168501518253600182019150603f8116850151825350600101610476565b6003895106600181146104e857600281146104f957610505565b613d3d60f01b600119830152610505565b603d60f81b6000198301525b509398975050505050505050565b60006020828403121561052557600080fd5b5035919050565b60005b8381101561054757818101518382015260200161052f565b83811115610556576000848401525b50505050565b6000815180845261057481602086016020860161052c565b601f01601f19169290920160200192915050565b6020815260006102a0602083018461055c565b634e487b7160e01b600052604160045260246000fd5b60006001600160401b03808411156105cb576105cb61059b565b604051601f8501601f19908116603f011681019082821181831017156105f3576105f361059b565b8160405280935085815286868601111561060c57600080fd5b858560208301376000602087830101525050509392505050565b600082601f83011261063757600080fd5b6102a0838335602085016105b1565b600080600080600060a0868803121561065e57600080fd5b85356001600160401b038082111561067557600080fd5b61068189838a01610626565b9650602088013591508082111561069757600080fd5b6106a389838a01610626565b955060408801359150808211156106b957600080fd5b506106c688828901610626565b9598949750949560608101359550608001359392505050565b6000602082840312156106f157600080fd5b81356001600160401b0381111561070757600080fd5b8201601f8101841361071857600080fd5b6103a7848235602084016105b1565b60008060008060008060c0878903121561074057600080fd5b86356001600160401b038082111561075757600080fd5b6107638a838b01610626565b9750602089013591508082111561077957600080fd5b6107858a838b01610626565b9650604089013591508082111561079b57600080fd5b6107a78a838b01610626565b955060608901359150808211156107bd57600080fd5b506107ca89828a01610626565b9350506080870135915060a087013590509295509295509295565b6000806000606084860312156107fa57600080fd5b83356001600160401b038082111561081157600080fd5b61081d87838801610626565b9450602086013591508082111561083357600080fd5b5061084086828701610626565b925050604084013590509250925092565b6000815161086381856020860161052c565b9290920192915050565b602f60f81b81526000825161088981600185016020870161052c565b9190910160010192915050565b693d913730b6b2911d101160b11b815286516000906108bc81600a850160208c0161052c565b600160fd1b600a9184019182015287516108dd81600b840160208c0161052c565b87519101906108f381600b840160208b0161052c565b631116101160e11b600b929091019182018190526e3232b9b1b934b83a34b7b7111d101160891b600f830152865161093281601e850160208b0161052c565b601e920191820152845161094d81602284016020890161052c565b610994610986610980602284860101770383937b832b93a34b2b9911d103d91373ab6b132b9111d160451b815260180190565b87610851565b617d7d60f01b815260020190565b9a9950505050505050505050565b7f646174613a6170706c69636174696f6e2f6a736f6e3b6261736536342c0000008152600082516109da81601d85016020870161052c565b91909101601d0192915050565b6834b6b0b3b2911d101160b91b81528451600090610a0c816009850160208a0161052c565b8083019050633f69643d60e01b8060098301528651610a3281600d850160208b0161052c565b741116101130b734b6b0ba34b7b72fbab936111d101160591b600d93909101928301528551610a68816022850160208a0161052c565b60229201918201528351610a8381602684016020880161052c565b631116101160e11b60269290910191820152602a019695505050505050565b6834b6b0b3b2911d101160b91b81528251600090610ac781600985016020880161052c565b633f69643d60e01b6009918401918201528351610aeb81600d84016020880161052c565b631116101160e11b600d9290910191820152601101949350505050565b7030b734b6b0ba34b7b72fbab936111d101160791b81528251600090610b3581601185016020880161052c565b633f69643d60e01b6011918401918201528351610b5981601584016020880161052c565b631116101160e11b60159290910191820152601901949350505050565b634e487b7160e01b600052601160045260246000fd5b600060018201610b9e57610b9e610b76565b5060010190565b634e487b7160e01b600052601260045260246000fd5b600082610bca57610bca610ba5565b500490565b600082821015610be157610be1610b76565b500390565b600082610bf557610bf5610ba5565b500690565b60008219821115610c0d57610c0d610b76565b500190565b634e487b7160e01b600052603260045260246000fd5b6000816000190483118215151615610c4257610c42610b76565b50029056fe4142434445464748494a4b4c4d4e4f505152535455565758595a6162636465666768696a6b6c6d6e6f707172737475767778797a303132333435363738392b2fa264697066735822122014071f6bfa44b6d77ac7549b1c45d00426536a719f467ad011f213b1f646920e64736f6c634300080f0033", "devdoc": { "details": "Can safely be used for generic base64Encode and numberToString functions", "kind": "dev",