Source Code
Overview
ETH Balance
0 ETH
More Info
ContractCreator
TokenTracker
Multichain Info
N/A
Latest 25 from a total of 37 transactions
Transaction Hash |
Method
|
Block
|
From
|
To
|
|||||
---|---|---|---|---|---|---|---|---|---|
Set Approval For... | 5863636 | 28 hrs ago | IN | 0 ETH | 0.00004577 | ||||
Set Approval For... | 5448618 | 7 days ago | IN | 0 ETH | 0.0000021 | ||||
Set Approval For... | 5448605 | 7 days ago | IN | 0 ETH | 0.00000213 | ||||
Set Approval For... | 5448586 | 7 days ago | IN | 0 ETH | 0.0000021 | ||||
Set Approval For... | 5448567 | 7 days ago | IN | 0 ETH | 0.00000213 | ||||
Set Approval For... | 5448534 | 7 days ago | IN | 0 ETH | 0.0000021 | ||||
Set Approval For... | 5448495 | 7 days ago | IN | 0 ETH | 0.00000327 | ||||
Set Approval For... | 5389949 | 8 days ago | IN | 0 ETH | 0.0000021 | ||||
Set Approval For... | 5389816 | 8 days ago | IN | 0 ETH | 0.0000021 | ||||
Set Approval For... | 5389786 | 8 days ago | IN | 0 ETH | 0.0000021 | ||||
Set Approval For... | 5389736 | 8 days ago | IN | 0 ETH | 0.0000021 | ||||
Set Approval For... | 5389582 | 8 days ago | IN | 0 ETH | 0.00000364 | ||||
Set Approval For... | 5381178 | 8 days ago | IN | 0 ETH | 0.0000021 | ||||
Set Approval For... | 5381163 | 8 days ago | IN | 0 ETH | 0.00000334 | ||||
Set Approval For... | 5380836 | 8 days ago | IN | 0 ETH | 0.0000021 | ||||
Set Approval For... | 5380821 | 8 days ago | IN | 0 ETH | 0.00000336 | ||||
Set Approval For... | 5380688 | 8 days ago | IN | 0 ETH | 0.0000021 | ||||
Set Approval For... | 5380650 | 8 days ago | IN | 0 ETH | 0.00000337 | ||||
Set Approval For... | 5380387 | 8 days ago | IN | 0 ETH | 0.00000338 | ||||
Set Approval For... | 5380020 | 8 days ago | IN | 0 ETH | 0.0000021 | ||||
Set Approval For... | 5379992 | 8 days ago | IN | 0 ETH | 0.00000217 | ||||
Set Approval For... | 5379896 | 8 days ago | IN | 0 ETH | 0.00000333 | ||||
Set Approval For... | 5379880 | 8 days ago | IN | 0 ETH | 0.00000216 | ||||
Set Approval For... | 5379848 | 8 days ago | IN | 0 ETH | 0.0000021 | ||||
Set Approval For... | 5379824 | 8 days ago | IN | 0 ETH | 0.00000216 |
Latest 25 internal transactions (View All)
Parent Transaction Hash | Block | From | To | |||
---|---|---|---|---|---|---|
5863636 | 28 hrs ago | 0 ETH | ||||
5863636 | 28 hrs ago | 0 ETH | ||||
5863636 | 28 hrs ago | 0 ETH | ||||
5863636 | 28 hrs ago | 0 ETH | ||||
5448618 | 7 days ago | 0 ETH | ||||
5448618 | 7 days ago | 0 ETH | ||||
5448618 | 7 days ago | 0 ETH | ||||
5448618 | 7 days ago | 0 ETH | ||||
5448605 | 7 days ago | 0 ETH | ||||
5448605 | 7 days ago | 0 ETH | ||||
5448605 | 7 days ago | 0 ETH | ||||
5448605 | 7 days ago | 0 ETH | ||||
5448586 | 7 days ago | 0 ETH | ||||
5448586 | 7 days ago | 0 ETH | ||||
5448586 | 7 days ago | 0 ETH | ||||
5448586 | 7 days ago | 0 ETH | ||||
5448567 | 7 days ago | 0 ETH | ||||
5448567 | 7 days ago | 0 ETH | ||||
5448567 | 7 days ago | 0 ETH | ||||
5448567 | 7 days ago | 0 ETH | ||||
5448534 | 7 days ago | 0 ETH | ||||
5448534 | 7 days ago | 0 ETH | ||||
5448534 | 7 days ago | 0 ETH | ||||
5448534 | 7 days ago | 0 ETH | ||||
5448495 | 7 days ago | 0 ETH |
Loading...
Loading
This contract may be a proxy contract. Click on More Options and select Is this a proxy? to confirm and enable the "Read as Proxy" & "Write as Proxy" tabs.
Contract Source Code Verified (Exact Match)
Contract Name:
ERC721Merkle
Compiler Version
v0.8.24+commit.e11b9ed9
ZkSolc Version
v1.5.7
Optimization Enabled:
Yes with Mode 3
Other Settings:
paris EvmVersion
Contract Source Code (Solidity Standard Json-Input format)
//SPDX-License-Identifier: MIT pragma solidity ^0.8.23; import "@openzeppelin/contracts/utils/cryptography/MerkleProof.sol"; import "@openzeppelin/contracts/access/Ownable.sol"; import "./ERC721Template.sol"; contract ERC721Merkle is ERC721Template { using SafeERC20 for IERC20; struct Tier { string title; bytes32 merkleRoot; uint256 price; uint256 erc20Price; uint128 maxMintAmount; uint128 saleStartTime; mapping(address => uint256) mints; } mapping(uint256 => Tier) public tiers; uint256[] public tierIds; constructor( string memory _name, string memory _symbol, string memory _contractURI, uint256 _maxSupply, uint256 _publicPrice, string memory _defaultBaseURI, string memory _notRevealedURI, address payable _withdrawalRecipientAddress, address payable _commissionRecipientAddress, uint256 _fixedCommisionThreshold, uint256 _commissionPercentageIn10000, address payable _defaultRoyaltyRecipient, // separate from withdrawal recipient to enhance security uint256 _defaultRoyaltyPercentageIn10000 ) ERC721Template( _name, _symbol, _contractURI, _maxSupply, _publicPrice, _defaultBaseURI, _notRevealedURI, _withdrawalRecipientAddress, _commissionRecipientAddress, _fixedCommisionThreshold, _commissionPercentageIn10000, _defaultRoyaltyRecipient, _defaultRoyaltyPercentageIn10000 ) { // add code here if you want to do something specific during contract deployment } /** * @notice Get how many more the user is eligible to mint * @param tierId Id of the tier minting from * @param user Address of the user * @param proof Merkle proof * @return Amount left to mint */ function getMintEligibility(uint256 tierId, address user, bytes32[] calldata proof) external view returns (uint256) { //require(MerkleProof.verify(proof, tier.merkleRoot, keccak256(abi.encodePacked(msg.sender))), "Not in presale list for this tier"); // return 0 if user is not in the merkleRoot if (!MerkleProof.verify(proof, tiers[tierId].merkleRoot, keccak256(abi.encodePacked(user)))) { return 0; } if (tiers[tierId].mints[user] >= tiers[tierId].maxMintAmount) { return 0; } return tiers[tierId].maxMintAmount - tiers[tierId].mints[user]; } /** * @notice Get the mint's tier details * @param tierId Id of the tier to get the details for * @return merkleRoot Merkle root * @return price Price in ETH * @return maxMintAmount Max amount mintable * @return saleStartTime Mint's start timestamp * @return title Tier's title * @return ERC20Price Price in ERC20 */ function getTierDetails(uint256 tierId) external view returns (bytes32 merkleRoot, uint256 price, uint256 maxMintAmount, uint256 saleStartTime, string memory title, uint256 ERC20Price) { Tier storage tier = tiers[tierId]; uint256 requiredERC20Tokens = 0; if (ethPriceFeedAddress != address(0) && ERC20PriceFeedAddress != address(0)) { requiredERC20Tokens = getRequiredERC20TokensChainlink(publicPrice); } else { requiredERC20Tokens = tier.erc20Price; } return (tier.merkleRoot, tier.price, tier.maxMintAmount, tier.saleStartTime, tier.title, requiredERC20Tokens); } /** * @notice Get all the tier ids */ function getTierIds() external view returns (uint256[] memory) { return tierIds; } /** * @notice Mint tokens in a specific tier using ETH * @param tierId Id of the tier * @param amount Amount of tokens * @param proof Merkle proof */ function whitelistMint(uint256 tierId, uint256 amount, bytes32[] calldata proof) external payable { Tier storage tier = tiers[tierId]; checkWhitelistMintRequirements(amount, tier, proof); require(msg.value >= amount * tier.price, "Insufficient funds for mint"); tier.mints[msg.sender] += amount; _safeMint(msg.sender, amount); } /** * @notice Mint tokens in a specific tier using ERC20 and Chainlink * @param tierId Id of the tier * @param amount Amount of tokens * @param proof Merkle proof */ function whitelistMintWithERC20ChainlinkPrice(uint256 tierId, uint256 amount, bytes32[] calldata proof) external { Tier storage tier = tiers[tierId]; checkWhitelistMintRequirements(amount, tier, proof); // Let's make sure price feed contract address exists require(ERC20TokenAddress != address(0), "Payment token address not set"); // Calculate the cost in ERC20 tokens uint256 requiredTokenAmount = getRequiredERC20TokensChainlink(tier.price * amount); tier.mints[msg.sender] += amount; IERC20(ERC20TokenAddress).safeTransferFrom(msg.sender, address(this), requiredTokenAmount); _safeMint(msg.sender, amount); } /** * @notice Mint tokens in a specific tier using ERC20 and fixed price * @param tierId Id of the tier * @param amount Amount of tokens * @param proof Merkle proof */ function whitelistMintWithFixedERC20Price(uint256 tierId, uint256 amount, bytes32[] calldata proof) external { Tier storage tier = tiers[tierId]; checkWhitelistMintRequirements(amount, tier, proof); uint256 erc20Price = tier.erc20Price; require(ERC20TokenAddress != address(0), "Payment token address not set"); require(erc20Price > 0, "Price per token not set"); // Calculate the cost in ERC20 tokens uint256 requiredTokenAmount = erc20Price * amount; tier.mints[msg.sender] += amount; IERC20(ERC20TokenAddress).safeTransferFrom(msg.sender, address(this), requiredTokenAmount); _safeMint(msg.sender, amount); } /** * @notice Add or Set existing tier details * @param tierId Id of the tier * @param title Title * @param merkleRoot Merkle root * @param price Price in ETH * @param erc20Price Price in ERC20 * @param maxMintAmount Max amount mintable * @param saleStartTime Mint's start timestamp */ function setTier(uint256 tierId, string calldata title, bytes32 merkleRoot, uint256 price, uint256 erc20Price, uint256 maxMintAmount, uint256 saleStartTime) external onlyOwner { Tier storage tier = tiers[tierId]; tier.merkleRoot = merkleRoot; tier.title = title; tier.price = price; tier.erc20Price = erc20Price; tier.maxMintAmount = uint128(maxMintAmount); tier.saleStartTime = uint128(saleStartTime); // type(uint256).max; is used to disable the tier // check if tierId is already in the array bool isNewTierId = true; for (uint256 i = 0; i < tierIds.length; i++) { if (tierIds[i] == tierId) { isNewTierId = false; break; } } if (isNewTierId) { tierIds.push(tierId); } } /** * @notice Enable a tier by setting its mint's start timestamp to 0 * @param tierId Id of the tier */ function enableTier(uint256 tierId) external onlyOwner { tiers[tierId].saleStartTime = 0; } /** * @notice Disable a tier by setting its mint's start timestamp to uint128.max * @param tierId Id of the tier */ function disableTier(uint256 tierId) external onlyOwner { tiers[tierId].saleStartTime = type(uint128).max; } /** * @notice Set mint's start timestamp for a tier * @param tierId Id of the tier * @param saleStartTime Mint's start timestamp */ function setTierSaleStartTime(uint256 tierId, uint256 saleStartTime) external onlyOwner { require(tiers[tierId].merkleRoot != bytes32(0), "Tier does not exist"); tiers[tierId].saleStartTime = uint128(saleStartTime); } /** * @notice Set the prices for a tier * @param tierId Id of the tier * @param price Price in ETH * @param erc20Price Price in ERC20 */ function setTierPrice(uint256 tierId, uint256 price, uint256 erc20Price) external onlyOwner { require(tiers[tierId].merkleRoot != bytes32(0), "Tier does not exist"); tiers[tierId].price = price; tiers[tierId].erc20Price = erc20Price; } /** * @notice Set tier's max mintable amount * @param tierId Id of the tier * @param maxMintAmount Max mintable amount */ function setTierMaxMintAmount(uint256 tierId, uint256 maxMintAmount) external onlyOwner { require(tiers[tierId].merkleRoot != bytes32(0), "Tier does not exist"); tiers[tierId].maxMintAmount = uint128(maxMintAmount); } /** * @notice Set the merkle root for a tier * @param tierId Id of the tier * @param merkleRoot Merkle root */ function setTierMerkleRoot(uint256 tierId, bytes32 merkleRoot) external onlyOwner { require(tiers[tierId].merkleRoot != bytes32(0), "Tier does not exist"); tiers[tierId].merkleRoot = merkleRoot; } /** * @notice Enforce tier's mint requirements * @param _mintAmount Amount of tokens to mint * @param tier Id of the tier * @param _proof Merkle proof */ function checkWhitelistMintRequirements(uint256 _mintAmount, Tier storage tier, bytes32[] calldata _proof) internal view { bytes32 merkleRoot = tier.merkleRoot; require(merkleRoot != bytes32(0), "Tier does not exist"); require(block.timestamp >= tier.saleStartTime, "Tier sale not started"); require(MerkleProof.verify(_proof, merkleRoot, keccak256(abi.encodePacked(msg.sender))), "Not in presale list for this tier"); require(_mintAmount <= tier.maxMintAmount - tier.mints[msg.sender], "Exceeds tier max mint amount"); require(totalSupply() + _mintAmount <= maxSupply, "Exceeds max supply"); } }
//SPDX-License-Identifier: Unlicense pragma solidity ^0.8.23; import "@openzeppelin/contracts/access/Ownable.sol"; import "@openzeppelin/contracts/interfaces/IERC2981.sol"; import "@openzeppelin/contracts/utils/Strings.sol"; import "./ERC721A.sol"; import "@openzeppelin/contracts/token/ERC20/ERC20.sol"; import "@openzeppelin/contracts/token/ERC20/utils/SafeERC20.sol"; import {AggregatorV3Interface} from "@chainlink/contracts/src/v0.8/shared/interfaces/AggregatorV3Interface.sol"; contract ERC721Template is IERC2981, Ownable, ERC721A { using Strings for uint256; using SafeERC20 for ERC20; string private baseURI; string public notRevealedURI; uint256 public maxSupply; uint256 public publicPrice; uint256 public publicMaxMintAmount; uint256 public publicSaleStartTime = type(uint256).max; bool public isRevealed; address payable public immutable withdrawalRecipientAddress; // address that will receive revenue address payable public immutable commissionRecipientAddress;// address that will receive a part of revenue on withdrawal uint256 public immutable commissionPercentageIn10000; // percentage of revenue to be sent to commissionRecipientAddress uint256 private immutable fixedCommissionThreshold; uint256 private totalCommissionWithdrawn; uint256 private commissionToWithdraw; uint256 private ownerToWithdraw; uint256 immutable deployTimestamp = block.timestamp; string public contractURI; //presale price is set after // Default royalty info address payable public defaultRoyaltyRecipient; uint256 public defaultRoyaltyPercentageIn10000; // Add new state variables to handle ERC20 payments address public ERC20TokenAddress; uint256 public ERC20FixedPricePerToken; // Fixed price per token in ERC20 uint256 public ERC20DiscountIn10000; // Discount percentage for ERC20 payments address public ERC20PriceFeedAddress; uint32 private ERC20PriceFeedStaleness; uint32 private ERC20PriceFeedDecimals; address public ethPriceFeedAddress; uint32 private ethPriceFeedStaleness; uint32 private ethPriceFeedDecimals; // Per-token royalty info mapping(uint256 => address payable) public tokenRoyaltyRecipient; mapping(uint256 => uint256) public tokenRoyaltyPercentage; event ContractURIUpdated(); bool public tradingEnabled = true; mapping(address => bool) public blacklist; constructor( string memory _name, string memory _symbol, string memory _contractURI, uint256 _maxSupply, uint256 _publicPrice, string memory _defaultBaseURI, string memory _notRevealedURI, address payable _withdrawalRecipientAddress, address payable _commissionRecipientAddress, uint256 _fixedCommisionThreshold, uint256 _commissionPercentageIn10000, address payable _defaultRoyaltyRecipient, // separate from withdrawal recipient to enhance security uint256 _defaultRoyaltyPercentageIn10000 // set max mint amount after deployment ) ERC721A(_name, _symbol) Ownable(msg.sender) { maxSupply = _maxSupply; publicPrice = _publicPrice; contractURI = _contractURI; // no need to emit event here, as it's set in the constructor baseURI = _defaultBaseURI; notRevealedURI =_notRevealedURI; publicMaxMintAmount = 10000; withdrawalRecipientAddress = _withdrawalRecipientAddress; commissionRecipientAddress = _commissionRecipientAddress; fixedCommissionThreshold = _fixedCommisionThreshold; // Ensure commission percentage is between 0 and 10000 (0-100%) require(_commissionPercentageIn10000 <= 10000, "Invalid commission percentage"); commissionPercentageIn10000 = _commissionPercentageIn10000; defaultRoyaltyRecipient = _defaultRoyaltyRecipient; defaultRoyaltyPercentageIn10000 = _defaultRoyaltyPercentageIn10000; isRevealed = bytes(_notRevealedURI).length == 0; } /** * @notice Implement EIP * @param interfaceId bytes to check if EIP compatible */ function supportsInterface(bytes4 interfaceId) public view virtual override(ERC721A, IERC165) returns (bool) { return interfaceId == type(IERC2981).interfaceId || ERC721A.supportsInterface(interfaceId) || super.supportsInterface(interfaceId); } /** * @notice implement ERC2981 * @param _tokenId Id of the token * @param _salePrice Price of the token * @return recipient address * @return royalty amount */ function royaltyInfo(uint256 _tokenId, uint256 _salePrice) external view override returns (address, uint256) { uint256 royaltyPercentage = tokenRoyaltyPercentage[_tokenId] != 0 ? tokenRoyaltyPercentage[_tokenId] : defaultRoyaltyPercentageIn10000; address royaltyRecipient = tokenRoyaltyRecipient[_tokenId] != address(0) ? tokenRoyaltyRecipient[_tokenId] : defaultRoyaltyRecipient; return (royaltyRecipient, (_salePrice * royaltyPercentage) / 10000); } /** * @notice Returns tokenURI, override to enable reveal/notRevealed * @param tokenId Id of the token to get URI for * @return URI */ function tokenURI( uint256 tokenId ) public view virtual override returns (string memory) { require( ownerOf(tokenId) != address(0), "ERC721Metadata: URI query for nonexistent token" ); if (isRevealed == false) { return notRevealedURI; } string memory identifier = tokenId.toString(); return bytes(baseURI).length != 0 ? string(abi.encodePacked(baseURI, identifier, ".json")) : ""; } /** * @notice Get the amount of mint available for msg.sender * @return Amount of mint available */ function getPublicMintEligibility() public view returns (uint256) { uint256 balance = balanceOf(msg.sender); uint256 maxMint = publicMaxMintAmount; if (balance >= maxMint) { return 0; } return maxMint - balance; } /** * @notice Get the mint details when minting with ETH * @return Mint's max supply * @return Mint's Public price * @return Current total supply * @return Mint's start timestamp */ function getLaunchpadDetails() external view returns (uint256, uint256, uint256, uint256) { return (maxSupply, publicPrice, totalSupply(), publicSaleStartTime); } /** * @notice Get the mint details when minting with ERC20 * @return Mint's ERC20 address * @return Mint's ERC20 price * @return Mint's ERC20 price if using Chainlink */ function getLaunchpadDetailsERC20() external view returns (address, uint256, uint256) { uint256 requiredTokens = 0; if (ethPriceFeedAddress != address(0) && ERC20PriceFeedAddress != address(0)) { requiredTokens = getRequiredERC20TokensChainlink(publicPrice); } return (ERC20TokenAddress, ERC20FixedPricePerToken, requiredTokens); } /** * @notice Get the amount of ERC20 needed to mint at ETH price using Chainlink * @param ethPrice Total ETH needed for the mint * @return Amount of ERC20 needed */ function getRequiredERC20TokensChainlink(uint256 ethPrice) public view returns (uint256) { address ethPriceFeed = ethPriceFeedAddress; address ERC20PriceFeed = ERC20PriceFeedAddress; require(ethPriceFeed != address(0) && ERC20PriceFeed != address(0), "Price feed addresses not set"); // Get the latest prices from Chainlink uint256 ethPriceInUsd = getLatestPrice(ethPriceFeed, ethPriceFeedStaleness); uint256 ERC20PriceInUsd = getLatestPrice(ERC20PriceFeed, ERC20PriceFeedStaleness); // Prices from Chainlink are usually returned with 8 decimals uint256 ethPriceInUsdScaled = ethPriceInUsd * 10**(18 - ethPriceFeedDecimals); // Scale to 18 decimals uint256 ERC20PriceInUsdScaled = ERC20PriceInUsd * 10**(18 - ERC20PriceFeedDecimals); // Scale to 18 decimals // Calculate the equivalent cost in ERC20 tokens uint256 decimalsDiff = 10 ** (18 - ERC20(ERC20TokenAddress).decimals()); //most tokens don't go over 18 decimals uint256 totalERC20Cost = (ethPrice * ethPriceInUsdScaled) / ERC20PriceInUsdScaled / decimalsDiff; // Apply discount if set uint256 ERC20Discount = ERC20DiscountIn10000; if (ERC20Discount > 0) { totalERC20Cost = (totalERC20Cost * (10000 - ERC20Discount)) / 10000; } return totalERC20Cost; } /** * @notice Mint tokens with ETH for the msg.sender * @param _mintAmount Amount of token to mint */ function mint(uint256 _mintAmount) external payable { checkMintRequirements(_mintAmount); require(msg.value >= publicPrice * _mintAmount, "Cost is higher than the amount sent"); _safeMint(msg.sender, _mintAmount); } /** * @notice Mint tokens using ERC20 and Chainlink * @param _mintAmount Amount of tokens to mint */ function mintWithERC20ChainlinkPrice(uint256 _mintAmount) external { checkMintRequirements(_mintAmount); // Let's make sure price feed contract address exists address ERC20Token = ERC20TokenAddress; require(ERC20Token != address(0), "Payment token address not set"); // Calculate the cost in ERC20 tokens uint256 requiredTokenAmount = getRequiredERC20TokensChainlink(publicPrice * _mintAmount); ERC20(ERC20Token).safeTransferFrom(msg.sender, address(this), requiredTokenAmount); _safeMint(msg.sender, _mintAmount); } /** * @notice Mint tokens using ERC20 and a fixed price set by the owner * @param _mintAmount Amount of tokens to mint */ function mintWithFixedERC20Price(uint256 _mintAmount) external { checkMintRequirements(_mintAmount); address ERC20Token = ERC20TokenAddress; uint256 ERC20FixedPrice = ERC20FixedPricePerToken; require(ERC20Token != address(0), "Payment token address not set"); require(ERC20FixedPrice > 0, "Price per token not set"); // Calculate the cost in ERC20 tokens uint256 requiredTokenAmount = ERC20FixedPrice * _mintAmount; ERC20(ERC20Token).safeTransferFrom(msg.sender, address(this), requiredTokenAmount); _safeMint(msg.sender, _mintAmount); } /** * @notice Mint tokens for free as admin to one address * @param _to Address receiving the tokens * @param _mintAmount Amount of tokens to mint */ function adminMint(address _to, uint256 _mintAmount) public onlyOwner { require(totalSupply() + _mintAmount <= maxSupply, "Total supply exceeded"); _safeMint(_to, _mintAmount); } /** * @notice Mint tokens for free as admin to multiple addresses * @param recipients Array of addressed receiving the tokens * @param amounts Array of amount of tokens to mint for each address */ function batchAdminMint(address[] calldata recipients, uint256[] calldata amounts) external onlyOwner { for (uint256 i = 0; i < recipients.length; i++) { adminMint(recipients[i], amounts[i]); } } /** * @notice Set the public price in ETH * @param _newPrice Price in ETH */ function setPublicPrice(uint256 _newPrice) external onlyOwner { publicPrice = _newPrice; } /** * @notice Set the base URI * @param _newBaseURI Base URI */ function setBaseURI(string memory _newBaseURI) external onlyOwner { baseURI = _newBaseURI; } /** * @notice Set the not revealed URI * @param _notRevealedURI Not revealed URI */ function setNotRevealedURI(string memory _notRevealedURI) external onlyOwner { notRevealedURI = _notRevealedURI; } /** * @notice Set the Max supply possible * @param _newMaxSupply Max supply amount */ function setMaxSupply(uint256 _newMaxSupply) external onlyOwner { maxSupply = _newMaxSupply; } /** * @notice Sets the start time of the public sale to a specific timestamp * @param _publicSaleStartTime Start timestamp */ function setPublicSaleStartTime(uint256 _publicSaleStartTime) external onlyOwner { publicSaleStartTime = _publicSaleStartTime; } /** * @notice Set the max amount mintable per address * @param _publicMaxMintAmount Amount mintable */ function setPublicMaxMintAmount(uint256 _publicMaxMintAmount) external onlyOwner { publicMaxMintAmount = _publicMaxMintAmount; } /** * @notice Set the default royalty recipient and BPS * @param _defaultRoyaltyRecipient Address of the recipient * @param _defaultRoyaltyPercentageIn10000 Amount of royalty in BPS */ function setDefaultRoyaltyInfo(address payable _defaultRoyaltyRecipient, uint256 _defaultRoyaltyPercentageIn10000) external onlyOwner { defaultRoyaltyRecipient = _defaultRoyaltyRecipient; defaultRoyaltyPercentageIn10000 = _defaultRoyaltyPercentageIn10000; } /** * @notice Set the royalty info for a specific id * @param _tokenId Id of the token * @param _royaltyRecipient Address of the recipient * @param _royaltyPercentage Amount of royalty in BPS */ function setTokenRoyaltyInfo(uint256 _tokenId, address payable _royaltyRecipient, uint256 _royaltyPercentage) external onlyOwner { require(ownerOf(_tokenId) != address(0), "Token does not exist"); tokenRoyaltyRecipient[_tokenId] = _royaltyRecipient; tokenRoyaltyPercentage[_tokenId] = _royaltyPercentage; } /** * @notice Set the contract URI * @param newURI Contract URI */ function setContractURI(string memory newURI) external onlyOwner { contractURI = newURI; emit ContractURIUpdated(); } /** * @notice Set the ERC20 that can be used to mint * @param _ERC20TokenAddress Address of the ERC20 */ function setERC20TokenAddress(address _ERC20TokenAddress) external onlyOwner { ERC20TokenAddress = _ERC20TokenAddress; } /** * @notice Set the fixed price to mint with ERC20 * @param _erc20FixedPricePerToken Price in ERC20 */ function setErc20FixedPricePerToken(uint256 _erc20FixedPricePerToken) external onlyOwner { ERC20FixedPricePerToken = _erc20FixedPricePerToken; } /** * @notice Set the ERC20 priceFeed details * @param _ERC20PriceFeedAddress Address of the pricefeed * @param _maxStaleness Max staleness */ function setERC20PriceFeedAddress(address _ERC20PriceFeedAddress, uint256 _maxStaleness) external onlyOwner { ERC20PriceFeedAddress = _ERC20PriceFeedAddress; ERC20PriceFeedDecimals = uint32(AggregatorV3Interface(_ERC20PriceFeedAddress).decimals()); ERC20PriceFeedStaleness = uint32(_maxStaleness); } /** * @notice Set the ERC20 priceFeed details * @param _ethPriceFeedAddress Address of the pricefeed * @param _maxStaleness Max staleness */ function setETHPriceFeedAddress(address _ethPriceFeedAddress, uint256 _maxStaleness) external onlyOwner { ethPriceFeedAddress = _ethPriceFeedAddress; ethPriceFeedDecimals = uint32(AggregatorV3Interface(_ethPriceFeedAddress).decimals()); ethPriceFeedStaleness = uint32(_maxStaleness); } /** * @notice Set the discount when minting with ERC20 * @param _erc20DiscountIn10000 Discount in BPS */ function setERC20DiscountIn10000(uint256 _erc20DiscountIn10000) external onlyOwner { require(_erc20DiscountIn10000 <= 10000, "Invalid discount percentage"); ERC20DiscountIn10000 = _erc20DiscountIn10000; } /** * Set trading enabled for the token * @param _tradingEnabled Boolean to enable trading or not */ function setTradingEnabled(bool _tradingEnabled) external onlyOwner { tradingEnabled = _tradingEnabled; } /** * @notice Add an address to the blacklist, not allowing them to transfer tokens anymore * @param _address Address to add to the blacklist */ function addToBlacklist(address _address) external onlyOwner { blacklist[_address] = true; } /** * Remove an address from the blacklist * @param _address Address to remove from the blacklist */ function removeFromBlacklist(address _address) external onlyOwner { blacklist[_address] = false; } /** * @notice Toggle the sale on or off by modifying the publicSaleStartTime variable */ function togglePublicSaleActive() external onlyOwner { if (block.timestamp < publicSaleStartTime) { publicSaleStartTime = block.timestamp; } else { // This effectively disables the public sale by setting the start time to a far future publicSaleStartTime = type(uint256).max; } } /** * @notice Toggle reveal on or off */ function toggleReveal() external onlyOwner { isRevealed = !isRevealed; } /** * @notice Withdraw the fixed commission for the commissionRecipientAddress */ function withdrawFixedCommission() external { require( msg.sender == owner() || msg.sender == commissionRecipientAddress, "Only owner or commission recipient can withdraw" ); uint256 withdrawn = totalCommissionWithdrawn; uint256 remainingCommission = fixedCommissionThreshold - withdrawn; uint256 amount = remainingCommission > address(this).balance ? address(this).balance : remainingCommission; // Updating the total withdrawn by A before making the transfer totalCommissionWithdrawn += amount; (bool success, ) = commissionRecipientAddress.call{value: amount}(""); require(success, "Transfer failed"); } /** * @notice Withdraw ETH for the actor calling and saves the other actors due in storage to limit DOS from one actor */ function withdraw() external virtual { require( msg.sender == owner() || msg.sender == commissionRecipientAddress || msg.sender == withdrawalRecipientAddress, "Only owner or commission recipient can withdraw" ); uint256 _commissionToWithdraw = commissionToWithdraw; uint256 _ownerToWithdraw = ownerToWithdraw; //This is ok if fixedCommissionThreshold makes it underflow, we don't allow eth withdraws until fixedCommissionThreshold can be paid fully uint256 available = address(this).balance - (fixedCommissionThreshold - totalCommissionWithdrawn) - _commissionToWithdraw - _ownerToWithdraw; uint256 newCommission = available * commissionPercentageIn10000 / 10000; uint256 newOwnerAmount = available - newCommission; if (msg.sender == commissionRecipientAddress) { ownerToWithdraw += newOwnerAmount; _commissionToWithdraw += newCommission; commissionToWithdraw = 0; (bool success, ) = commissionRecipientAddress.call{value: _commissionToWithdraw}(""); require(success); } else if (msg.sender == withdrawalRecipientAddress || msg.sender == owner()) { commissionToWithdraw += newCommission; _ownerToWithdraw += newOwnerAmount; ownerToWithdraw = 0; (bool success, ) = withdrawalRecipientAddress.call{value: _ownerToWithdraw}(""); require(success); } } /** * @notice Withdraw ERC20 for every actors * @param erc20Token Address of ther ERC20 to withdraw */ function withdrawERC20(ERC20 erc20Token) external { require( msg.sender == owner() || msg.sender == commissionRecipientAddress || msg.sender == withdrawalRecipientAddress, "Only owner or commission recipient can withdraw" ); uint256 erc20Balance = erc20Token.balanceOf(address(this)); uint256 commission = (erc20Balance * commissionPercentageIn10000) / 10000; uint256 withdrawalAddressAmount = erc20Balance - commission; //withdrawalRecipientAddress if(commission > 0) { erc20Token.safeTransfer(commissionRecipientAddress, commission); } if(withdrawalAddressAmount > 0) { erc20Token.safeTransfer(withdrawalRecipientAddress, withdrawalAddressAmount); } } /** * @notice Allows the owner to withdraw all ETH 28 weeks after deploy time */ function emergencyWithdraw() external onlyOwner { require(block.timestamp > deployTimestamp + 28 weeks, "Too early to emergency withdraw"); uint256 balance = address(this).balance; (bool success,) = payable(owner()).call{value: balance}(""); require(success); } /** * @notice Allows the owner to withdraw all ERC20 28 weeks after deploy time */ function emergencyWithdrawERC20(ERC20 erc20Token) external onlyOwner { require(block.timestamp > deployTimestamp + 28 weeks, "Too early to emergency withdraw"); uint256 balance = erc20Token.balanceOf(address(this)); erc20Token.safeTransfer(owner(), balance); } /** * @notice Get the price to mint with an ERC20 using Chainlink * @param priceFeedAddress Address of the pricefeed * @param maxStaleness Max staleness accepted for the pricefeed * @return Price received from Chainlink */ function getLatestPrice(address priceFeedAddress, uint256 maxStaleness) internal view returns (uint256) { AggregatorV3Interface priceFeed = AggregatorV3Interface(priceFeedAddress); (, int256 price,,uint256 updatedAt,) = priceFeed.latestRoundData(); require(price > 0, "Invalid price from Chainlink"); require(updatedAt > block.timestamp - maxStaleness, "Invalid price from Chainlink"); return uint256(price); } /** * @notice Enforce mint requirements * @param _mintAmount Amount of token to mint */ function checkMintRequirements(uint256 _mintAmount) internal view { require(totalSupply() + _mintAmount <= maxSupply, "Total supply exceeded"); require(block.timestamp >= publicSaleStartTime, "Public sale not active"); require(getPublicMintEligibility() >= _mintAmount, "Invalid amount to be minted"); } /** * @notice Override to start from 1 instead of 0 */ function _startTokenId() internal view virtual override returns (uint256) { return 1; } /** * @notice Override to enable blacklist and pause */ function _beforeTokenTransfers( address from, address to, uint256 tokenId, uint256 batchSize ) internal override { if (from != address(0)) { require(tradingEnabled, "Trading is disabled"); require(!blacklist[msg.sender], "Blacklisted entities cannot execute trades"); } super._beforeTokenTransfers(from, to, tokenId, batchSize); } }
// SPDX-License-Identifier: MIT // ERC721A Contracts v4.3.0 // Creator: Chiru Labs pragma solidity ^0.8.4; import './IERC721A.sol'; /** * @dev Interface of ERC721 token receiver. */ interface ERC721A__IERC721Receiver { function onERC721Received( address operator, address from, uint256 tokenId, bytes calldata data ) external returns (bytes4); } /** * @title ERC721A * * @dev Implementation of the [ERC721](https://eips.ethereum.org/EIPS/eip-721) * Non-Fungible Token Standard, including the Metadata extension. * Optimized for lower gas during batch mints. * * Token IDs are minted in sequential order (e.g. 0, 1, 2, 3, ...) * starting from `_startTokenId()`. * * The `_sequentialUpTo()` function can be overriden to enable spot mints * (i.e. non-consecutive mints) for `tokenId`s greater than `_sequentialUpTo()`. * * Assumptions: * * - An owner cannot have more than 2**64 - 1 (max value of uint64) of supply. * - The maximum token ID cannot exceed 2**256 - 1 (max value of uint256). */ contract ERC721A is IERC721A { // Bypass for a `--via-ir` bug (https://github.com/chiru-labs/ERC721A/pull/364). struct TokenApprovalRef { address value; } // ============================================================= // CONSTANTS // ============================================================= // Mask of an entry in packed address data. uint256 private constant _BITMASK_ADDRESS_DATA_ENTRY = (1 << 64) - 1; // The bit position of `numberMinted` in packed address data. uint256 private constant _BITPOS_NUMBER_MINTED = 64; // The bit position of `numberBurned` in packed address data. uint256 private constant _BITPOS_NUMBER_BURNED = 128; // The bit position of `aux` in packed address data. uint256 private constant _BITPOS_AUX = 192; // Mask of all 256 bits in packed address data except the 64 bits for `aux`. uint256 private constant _BITMASK_AUX_COMPLEMENT = (1 << 192) - 1; // The bit position of `startTimestamp` in packed ownership. uint256 private constant _BITPOS_START_TIMESTAMP = 160; // The bit mask of the `burned` bit in packed ownership. uint256 private constant _BITMASK_BURNED = 1 << 224; // The bit position of the `nextInitialized` bit in packed ownership. uint256 private constant _BITPOS_NEXT_INITIALIZED = 225; // The bit mask of the `nextInitialized` bit in packed ownership. uint256 private constant _BITMASK_NEXT_INITIALIZED = 1 << 225; // The bit position of `extraData` in packed ownership. uint256 private constant _BITPOS_EXTRA_DATA = 232; // Mask of all 256 bits in a packed ownership except the 24 bits for `extraData`. uint256 private constant _BITMASK_EXTRA_DATA_COMPLEMENT = (1 << 232) - 1; // The mask of the lower 160 bits for addresses. uint256 private constant _BITMASK_ADDRESS = (1 << 160) - 1; // The maximum `quantity` that can be minted with {_mintERC2309}. // This limit is to prevent overflows on the address data entries. // For a limit of 5000, a total of 3.689e15 calls to {_mintERC2309} // is required to cause an overflow, which is unrealistic. uint256 private constant _MAX_MINT_ERC2309_QUANTITY_LIMIT = 5000; // The `Transfer` event signature is given by: // `keccak256(bytes("Transfer(address,address,uint256)"))`. bytes32 private constant _TRANSFER_EVENT_SIGNATURE = 0xddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef; // ============================================================= // STORAGE // ============================================================= // The next token ID to be minted. uint256 private _currentIndex; // The number of tokens burned. uint256 private _burnCounter; // Token name string private _name; // Token symbol string private _symbol; // Mapping from token ID to ownership details // An empty struct value does not necessarily mean the token is unowned. // See {_packedOwnershipOf} implementation for details. // // Bits Layout: // - [0..159] `addr` // - [160..223] `startTimestamp` // - [224] `burned` // - [225] `nextInitialized` // - [232..255] `extraData` mapping(uint256 => uint256) private _packedOwnerships; // Mapping owner address to address data. // // Bits Layout: // - [0..63] `balance` // - [64..127] `numberMinted` // - [128..191] `numberBurned` // - [192..255] `aux` mapping(address => uint256) private _packedAddressData; // Mapping from token ID to approved address. mapping(uint256 => TokenApprovalRef) private _tokenApprovals; // Mapping from owner to operator approvals mapping(address => mapping(address => bool)) private _operatorApprovals; // The amount of tokens minted above `_sequentialUpTo()`. // We call these spot mints (i.e. non-sequential mints). uint256 private _spotMinted; // ============================================================= // CONSTRUCTOR // ============================================================= constructor(string memory name_, string memory symbol_) { _name = name_; _symbol = symbol_; _currentIndex = _startTokenId(); if (_sequentialUpTo() < _startTokenId()) _revert(SequentialUpToTooSmall.selector); } // ============================================================= // TOKEN COUNTING OPERATIONS // ============================================================= /** * @dev Returns the starting token ID for sequential mints. * * Override this function to change the starting token ID for sequential mints. * * Note: The value returned must never change after any tokens have been minted. */ function _startTokenId() internal view virtual returns (uint256) { return 0; } /** * @dev Returns the maximum token ID (inclusive) for sequential mints. * * Override this function to return a value less than 2**256 - 1, * but greater than `_startTokenId()`, to enable spot (non-sequential) mints. * * Note: The value returned must never change after any tokens have been minted. */ function _sequentialUpTo() internal view virtual returns (uint256) { return type(uint256).max; } /** * @dev Returns the next token ID to be minted. */ function _nextTokenId() internal view virtual returns (uint256) { return _currentIndex; } /** * @dev Returns the total number of tokens in existence. * Burned tokens will reduce the count. * To get the total number of tokens minted, please see {_totalMinted}. */ function totalSupply() public view virtual override returns (uint256 result) { // Counter underflow is impossible as `_burnCounter` cannot be incremented // more than `_currentIndex + _spotMinted - _startTokenId()` times. unchecked { // With spot minting, the intermediate `result` can be temporarily negative, // and the computation must be unchecked. result = _currentIndex - _burnCounter - _startTokenId(); if (_sequentialUpTo() != type(uint256).max) result += _spotMinted; } } /** * @dev Returns the total amount of tokens minted in the contract. */ function _totalMinted() internal view virtual returns (uint256 result) { // Counter underflow is impossible as `_currentIndex` does not decrement, // and it is initialized to `_startTokenId()`. unchecked { result = _currentIndex - _startTokenId(); if (_sequentialUpTo() != type(uint256).max) result += _spotMinted; } } /** * @dev Returns the total number of tokens burned. */ function _totalBurned() internal view virtual returns (uint256) { return _burnCounter; } /** * @dev Returns the total number of tokens that are spot-minted. */ function _totalSpotMinted() internal view virtual returns (uint256) { return _spotMinted; } // ============================================================= // ADDRESS DATA OPERATIONS // ============================================================= /** * @dev Returns the number of tokens in `owner`'s account. */ function balanceOf(address owner) public view virtual override returns (uint256) { if (owner == address(0)) _revert(BalanceQueryForZeroAddress.selector); return _packedAddressData[owner] & _BITMASK_ADDRESS_DATA_ENTRY; } /** * Returns the number of tokens minted by `owner`. */ function _numberMinted(address owner) internal view returns (uint256) { return (_packedAddressData[owner] >> _BITPOS_NUMBER_MINTED) & _BITMASK_ADDRESS_DATA_ENTRY; } /** * Returns the number of tokens burned by or on behalf of `owner`. */ function _numberBurned(address owner) internal view returns (uint256) { return (_packedAddressData[owner] >> _BITPOS_NUMBER_BURNED) & _BITMASK_ADDRESS_DATA_ENTRY; } /** * Returns the auxiliary data for `owner`. (e.g. number of whitelist mint slots used). */ function _getAux(address owner) internal view returns (uint64) { return uint64(_packedAddressData[owner] >> _BITPOS_AUX); } /** * Sets the auxiliary data for `owner`. (e.g. number of whitelist mint slots used). * If there are multiple variables, please pack them into a uint64. */ function _setAux(address owner, uint64 aux) internal virtual { uint256 packed = _packedAddressData[owner]; uint256 auxCasted; // Cast `aux` with assembly to avoid redundant masking. assembly { auxCasted := aux } packed = (packed & _BITMASK_AUX_COMPLEMENT) | (auxCasted << _BITPOS_AUX); _packedAddressData[owner] = packed; } // ============================================================= // IERC165 // ============================================================= /** * @dev Returns true if this contract implements the interface defined by * `interfaceId`. See the corresponding * [EIP section](https://eips.ethereum.org/EIPS/eip-165#how-interfaces-are-identified) * to learn more about how these ids are created. * * This function call must use less than 30000 gas. */ function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) { // The interface IDs are constants representing the first 4 bytes // of the XOR of all function selectors in the interface. // See: [ERC165](https://eips.ethereum.org/EIPS/eip-165) // (e.g. `bytes4(i.functionA.selector ^ i.functionB.selector ^ ...)`) return interfaceId == 0x01ffc9a7 || // ERC165 interface ID for ERC165. interfaceId == 0x80ac58cd || // ERC165 interface ID for ERC721. interfaceId == 0x5b5e139f; // ERC165 interface ID for ERC721Metadata. } // ============================================================= // IERC721Metadata // ============================================================= /** * @dev Returns the token collection name. */ function name() public view virtual override returns (string memory) { return _name; } /** * @dev Returns the token collection symbol. */ function symbol() public view virtual override returns (string memory) { return _symbol; } /** * @dev Returns the Uniform Resource Identifier (URI) for `tokenId` token. */ function tokenURI(uint256 tokenId) public view virtual override returns (string memory) { if (!_exists(tokenId)) _revert(URIQueryForNonexistentToken.selector); string memory baseURI = _baseURI(); return bytes(baseURI).length != 0 ? string(abi.encodePacked(baseURI, _toString(tokenId))) : ''; } /** * @dev Base URI for computing {tokenURI}. If set, the resulting URI for each * token will be the concatenation of the `baseURI` and the `tokenId`. Empty * by default, it can be overridden in child contracts. */ function _baseURI() internal view virtual returns (string memory) { return ''; } // ============================================================= // OWNERSHIPS OPERATIONS // ============================================================= /** * @dev Returns the owner of the `tokenId` token. * * Requirements: * * - `tokenId` must exist. */ function ownerOf(uint256 tokenId) public view virtual override returns (address) { return address(uint160(_packedOwnershipOf(tokenId))); } /** * @dev Gas spent here starts off proportional to the maximum mint batch size. * It gradually moves to O(1) as tokens get transferred around over time. */ function _ownershipOf(uint256 tokenId) internal view virtual returns (TokenOwnership memory) { return _unpackedOwnership(_packedOwnershipOf(tokenId)); } /** * @dev Returns the unpacked `TokenOwnership` struct at `index`. */ function _ownershipAt(uint256 index) internal view virtual returns (TokenOwnership memory) { return _unpackedOwnership(_packedOwnerships[index]); } /** * @dev Returns whether the ownership slot at `index` is initialized. * An uninitialized slot does not necessarily mean that the slot has no owner. */ function _ownershipIsInitialized(uint256 index) internal view virtual returns (bool) { return _packedOwnerships[index] != 0; } /** * @dev Initializes the ownership slot minted at `index` for efficiency purposes. */ function _initializeOwnershipAt(uint256 index) internal virtual { if (_packedOwnerships[index] == 0) { _packedOwnerships[index] = _packedOwnershipOf(index); } } /** * @dev Returns the packed ownership data of `tokenId`. */ function _packedOwnershipOf(uint256 tokenId) private view returns (uint256 packed) { if (_startTokenId() <= tokenId) { packed = _packedOwnerships[tokenId]; if (tokenId > _sequentialUpTo()) { if (_packedOwnershipExists(packed)) return packed; _revert(OwnerQueryForNonexistentToken.selector); } // If the data at the starting slot does not exist, start the scan. if (packed == 0) { if (tokenId >= _currentIndex) _revert(OwnerQueryForNonexistentToken.selector); // Invariant: // There will always be an initialized ownership slot // (i.e. `ownership.addr != address(0) && ownership.burned == false`) // before an unintialized ownership slot // (i.e. `ownership.addr == address(0) && ownership.burned == false`) // Hence, `tokenId` will not underflow. // // We can directly compare the packed value. // If the address is zero, packed will be zero. for (;;) { unchecked { packed = _packedOwnerships[--tokenId]; } if (packed == 0) continue; if (packed & _BITMASK_BURNED == 0) return packed; // Otherwise, the token is burned, and we must revert. // This handles the case of batch burned tokens, where only the burned bit // of the starting slot is set, and remaining slots are left uninitialized. _revert(OwnerQueryForNonexistentToken.selector); } } // Otherwise, the data exists and we can skip the scan. // This is possible because we have already achieved the target condition. // This saves 2143 gas on transfers of initialized tokens. // If the token is not burned, return `packed`. Otherwise, revert. if (packed & _BITMASK_BURNED == 0) return packed; } _revert(OwnerQueryForNonexistentToken.selector); } /** * @dev Returns the unpacked `TokenOwnership` struct from `packed`. */ function _unpackedOwnership(uint256 packed) private pure returns (TokenOwnership memory ownership) { ownership.addr = address(uint160(packed)); ownership.startTimestamp = uint64(packed >> _BITPOS_START_TIMESTAMP); ownership.burned = packed & _BITMASK_BURNED != 0; ownership.extraData = uint24(packed >> _BITPOS_EXTRA_DATA); } /** * @dev Packs ownership data into a single uint256. */ function _packOwnershipData(address owner, uint256 flags) private view returns (uint256 result) { assembly { // Mask `owner` to the lower 160 bits, in case the upper bits somehow aren't clean. owner := and(owner, _BITMASK_ADDRESS) // `owner | (block.timestamp << _BITPOS_START_TIMESTAMP) | flags`. result := or(owner, or(shl(_BITPOS_START_TIMESTAMP, timestamp()), flags)) } } /** * @dev Returns the `nextInitialized` flag set if `quantity` equals 1. */ function _nextInitializedFlag(uint256 quantity) private pure returns (uint256 result) { // For branchless setting of the `nextInitialized` flag. assembly { // `(quantity == 1) << _BITPOS_NEXT_INITIALIZED`. result := shl(_BITPOS_NEXT_INITIALIZED, eq(quantity, 1)) } } // ============================================================= // APPROVAL OPERATIONS // ============================================================= /** * @dev Gives permission to `to` to transfer `tokenId` token to another account. See {ERC721A-_approve}. * * Requirements: * * - The caller must own the token or be an approved operator. */ function approve(address to, uint256 tokenId) public payable virtual override { _approve(to, tokenId, true); } /** * @dev Returns the account approved for `tokenId` token. * * Requirements: * * - `tokenId` must exist. */ function getApproved(uint256 tokenId) public view virtual override returns (address) { if (!_exists(tokenId)) _revert(ApprovalQueryForNonexistentToken.selector); return _tokenApprovals[tokenId].value; } /** * @dev Approve or remove `operator` as an operator for the caller. * Operators can call {transferFrom} or {safeTransferFrom} * for any token owned by the caller. * * Requirements: * * - The `operator` cannot be the caller. * * Emits an {ApprovalForAll} event. */ function setApprovalForAll(address operator, bool approved) public virtual override { _operatorApprovals[_msgSenderERC721A()][operator] = approved; emit ApprovalForAll(_msgSenderERC721A(), operator, approved); } /** * @dev Returns if the `operator` is allowed to manage all of the assets of `owner`. * * See {setApprovalForAll}. */ function isApprovedForAll(address owner, address operator) public view virtual override returns (bool) { return _operatorApprovals[owner][operator]; } /** * @dev Returns whether `tokenId` exists. * * Tokens can be managed by their owner or approved accounts via {approve} or {setApprovalForAll}. * * Tokens start existing when they are minted. See {_mint}. */ function _exists(uint256 tokenId) internal view virtual returns (bool result) { if (_startTokenId() <= tokenId) { if (tokenId > _sequentialUpTo()) return _packedOwnershipExists(_packedOwnerships[tokenId]); if (tokenId < _currentIndex) { uint256 packed; while ((packed = _packedOwnerships[tokenId]) == 0) --tokenId; result = packed & _BITMASK_BURNED == 0; } } } /** * @dev Returns whether `packed` represents a token that exists. */ function _packedOwnershipExists(uint256 packed) private pure returns (bool result) { assembly { // The following is equivalent to `owner != address(0) && burned == false`. // Symbolically tested. result := gt(and(packed, _BITMASK_ADDRESS), and(packed, _BITMASK_BURNED)) } } /** * @dev Returns whether `msgSender` is equal to `approvedAddress` or `owner`. */ function _isSenderApprovedOrOwner( address approvedAddress, address owner, address msgSender ) private pure returns (bool result) { assembly { // Mask `owner` to the lower 160 bits, in case the upper bits somehow aren't clean. owner := and(owner, _BITMASK_ADDRESS) // Mask `msgSender` to the lower 160 bits, in case the upper bits somehow aren't clean. msgSender := and(msgSender, _BITMASK_ADDRESS) // `msgSender == owner || msgSender == approvedAddress`. result := or(eq(msgSender, owner), eq(msgSender, approvedAddress)) } } /** * @dev Returns the storage slot and value for the approved address of `tokenId`. */ function _getApprovedSlotAndAddress(uint256 tokenId) private view returns (uint256 approvedAddressSlot, address approvedAddress) { TokenApprovalRef storage tokenApproval = _tokenApprovals[tokenId]; // The following is equivalent to `approvedAddress = _tokenApprovals[tokenId].value`. assembly { approvedAddressSlot := tokenApproval.slot approvedAddress := sload(approvedAddressSlot) } } // ============================================================= // TRANSFER OPERATIONS // ============================================================= /** * @dev Transfers `tokenId` from `from` to `to`. * * Requirements: * * - `from` cannot be the zero address. * - `to` cannot be the zero address. * - `tokenId` token must be owned by `from`. * - If the caller is not `from`, it must be approved to move this token * by either {approve} or {setApprovalForAll}. * * Emits a {Transfer} event. */ function transferFrom( address from, address to, uint256 tokenId ) public payable virtual override { uint256 prevOwnershipPacked = _packedOwnershipOf(tokenId); // Mask `from` to the lower 160 bits, in case the upper bits somehow aren't clean. from = address(uint160(uint256(uint160(from)) & _BITMASK_ADDRESS)); if (address(uint160(prevOwnershipPacked)) != from) _revert(TransferFromIncorrectOwner.selector); (uint256 approvedAddressSlot, address approvedAddress) = _getApprovedSlotAndAddress(tokenId); // The nested ifs save around 20+ gas over a compound boolean condition. if (!_isSenderApprovedOrOwner(approvedAddress, from, _msgSenderERC721A())) if (!isApprovedForAll(from, _msgSenderERC721A())) _revert(TransferCallerNotOwnerNorApproved.selector); _beforeTokenTransfers(from, to, tokenId, 1); // Clear approvals from the previous owner. assembly { if approvedAddress { // This is equivalent to `delete _tokenApprovals[tokenId]`. sstore(approvedAddressSlot, 0) } } // Underflow of the sender's balance is impossible because we check for // ownership above and the recipient's balance can't realistically overflow. // Counter overflow is incredibly unrealistic as `tokenId` would have to be 2**256. unchecked { // We can directly increment and decrement the balances. --_packedAddressData[from]; // Updates: `balance -= 1`. ++_packedAddressData[to]; // Updates: `balance += 1`. // Updates: // - `address` to the next owner. // - `startTimestamp` to the timestamp of transfering. // - `burned` to `false`. // - `nextInitialized` to `true`. _packedOwnerships[tokenId] = _packOwnershipData( to, _BITMASK_NEXT_INITIALIZED | _nextExtraData(from, to, prevOwnershipPacked) ); // If the next slot may not have been initialized (i.e. `nextInitialized == false`) . if (prevOwnershipPacked & _BITMASK_NEXT_INITIALIZED == 0) { uint256 nextTokenId = tokenId + 1; // If the next slot's address is zero and not burned (i.e. packed value is zero). if (_packedOwnerships[nextTokenId] == 0) { // If the next slot is within bounds. if (nextTokenId != _currentIndex) { // Initialize the next slot to maintain correctness for `ownerOf(tokenId + 1)`. _packedOwnerships[nextTokenId] = prevOwnershipPacked; } } } } // Mask `to` to the lower 160 bits, in case the upper bits somehow aren't clean. uint256 toMasked = uint256(uint160(to)) & _BITMASK_ADDRESS; assembly { // Emit the `Transfer` event. log4( 0, // Start of data (0, since no data). 0, // End of data (0, since no data). _TRANSFER_EVENT_SIGNATURE, // Signature. from, // `from`. toMasked, // `to`. tokenId // `tokenId`. ) } if (toMasked == 0) _revert(TransferToZeroAddress.selector); _afterTokenTransfers(from, to, tokenId, 1); } /** * @dev Equivalent to `safeTransferFrom(from, to, tokenId, '')`. */ function safeTransferFrom( address from, address to, uint256 tokenId ) public payable virtual override { safeTransferFrom(from, to, tokenId, ''); } /** * @dev Safely transfers `tokenId` token from `from` to `to`. * * Requirements: * * - `from` cannot be the zero address. * - `to` cannot be the zero address. * - `tokenId` token must exist and be owned by `from`. * - If the caller is not `from`, it must be approved to move this token * by either {approve} or {setApprovalForAll}. * - If `to` refers to a smart contract, it must implement * {IERC721Receiver-onERC721Received}, which is called upon a safe transfer. * * Emits a {Transfer} event. */ function safeTransferFrom( address from, address to, uint256 tokenId, bytes memory _data ) public payable virtual override { transferFrom(from, to, tokenId); if (to.code.length != 0) if (!_checkContractOnERC721Received(from, to, tokenId, _data)) { _revert(TransferToNonERC721ReceiverImplementer.selector); } } /** * @dev Hook that is called before a set of serially-ordered token IDs * are about to be transferred. This includes minting. * And also called before burning one token. * * `startTokenId` - the first token ID to be transferred. * `quantity` - the amount to be transferred. * * Calling conditions: * * - When `from` and `to` are both non-zero, `from`'s `tokenId` will be * transferred to `to`. * - When `from` is zero, `tokenId` will be minted for `to`. * - When `to` is zero, `tokenId` will be burned by `from`. * - `from` and `to` are never both zero. */ function _beforeTokenTransfers( address from, address to, uint256 startTokenId, uint256 quantity ) internal virtual {} /** * @dev Hook that is called after a set of serially-ordered token IDs * have been transferred. This includes minting. * And also called after one token has been burned. * * `startTokenId` - the first token ID to be transferred. * `quantity` - the amount to be transferred. * * Calling conditions: * * - When `from` and `to` are both non-zero, `from`'s `tokenId` has been * transferred to `to`. * - When `from` is zero, `tokenId` has been minted for `to`. * - When `to` is zero, `tokenId` has been burned by `from`. * - `from` and `to` are never both zero. */ function _afterTokenTransfers( address from, address to, uint256 startTokenId, uint256 quantity ) internal virtual {} /** * @dev Private function to invoke {IERC721Receiver-onERC721Received} on a target contract. * * `from` - Previous owner of the given token ID. * `to` - Target address that will receive the token. * `tokenId` - Token ID to be transferred. * `_data` - Optional data to send along with the call. * * Returns whether the call correctly returned the expected magic value. */ function _checkContractOnERC721Received( address from, address to, uint256 tokenId, bytes memory _data ) private returns (bool) { try ERC721A__IERC721Receiver(to).onERC721Received(_msgSenderERC721A(), from, tokenId, _data) returns ( bytes4 retval ) { return retval == ERC721A__IERC721Receiver(to).onERC721Received.selector; } catch (bytes memory reason) { if (reason.length == 0) { _revert(TransferToNonERC721ReceiverImplementer.selector); } assembly { revert(add(32, reason), mload(reason)) } } } // ============================================================= // MINT OPERATIONS // ============================================================= /** * @dev Mints `quantity` tokens and transfers them to `to`. * * Requirements: * * - `to` cannot be the zero address. * - `quantity` must be greater than 0. * * Emits a {Transfer} event for each mint. */ function _mint(address to, uint256 quantity) internal virtual { uint256 startTokenId = _currentIndex; if (quantity == 0) _revert(MintZeroQuantity.selector); _beforeTokenTransfers(address(0), to, startTokenId, quantity); // Overflows are incredibly unrealistic. // `balance` and `numberMinted` have a maximum limit of 2**64. // `tokenId` has a maximum limit of 2**256. unchecked { // Updates: // - `address` to the owner. // - `startTimestamp` to the timestamp of minting. // - `burned` to `false`. // - `nextInitialized` to `quantity == 1`. _packedOwnerships[startTokenId] = _packOwnershipData( to, _nextInitializedFlag(quantity) | _nextExtraData(address(0), to, 0) ); // Updates: // - `balance += quantity`. // - `numberMinted += quantity`. // // We can directly add to the `balance` and `numberMinted`. _packedAddressData[to] += quantity * ((1 << _BITPOS_NUMBER_MINTED) | 1); // Mask `to` to the lower 160 bits, in case the upper bits somehow aren't clean. uint256 toMasked = uint256(uint160(to)) & _BITMASK_ADDRESS; if (toMasked == 0) _revert(MintToZeroAddress.selector); uint256 end = startTokenId + quantity; uint256 tokenId = startTokenId; if (end - 1 > _sequentialUpTo()) _revert(SequentialMintExceedsLimit.selector); do { assembly { // Emit the `Transfer` event. log4( 0, // Start of data (0, since no data). 0, // End of data (0, since no data). _TRANSFER_EVENT_SIGNATURE, // Signature. 0, // `address(0)`. toMasked, // `to`. tokenId // `tokenId`. ) } // The `!=` check ensures that large values of `quantity` // that overflows uint256 will make the loop run out of gas. } while (++tokenId != end); _currentIndex = end; } _afterTokenTransfers(address(0), to, startTokenId, quantity); } /** * @dev Mints `quantity` tokens and transfers them to `to`. * * This function is intended for efficient minting only during contract creation. * * It emits only one {ConsecutiveTransfer} as defined in * [ERC2309](https://eips.ethereum.org/EIPS/eip-2309), * instead of a sequence of {Transfer} event(s). * * Calling this function outside of contract creation WILL make your contract * non-compliant with the ERC721 standard. * For full ERC721 compliance, substituting ERC721 {Transfer} event(s) with the ERC2309 * {ConsecutiveTransfer} event is only permissible during contract creation. * * Requirements: * * - `to` cannot be the zero address. * - `quantity` must be greater than 0. * * Emits a {ConsecutiveTransfer} event. */ function _mintERC2309(address to, uint256 quantity) internal virtual { uint256 startTokenId = _currentIndex; if (to == address(0)) _revert(MintToZeroAddress.selector); if (quantity == 0) _revert(MintZeroQuantity.selector); if (quantity > _MAX_MINT_ERC2309_QUANTITY_LIMIT) _revert(MintERC2309QuantityExceedsLimit.selector); _beforeTokenTransfers(address(0), to, startTokenId, quantity); // Overflows are unrealistic due to the above check for `quantity` to be below the limit. unchecked { // Updates: // - `balance += quantity`. // - `numberMinted += quantity`. // // We can directly add to the `balance` and `numberMinted`. _packedAddressData[to] += quantity * ((1 << _BITPOS_NUMBER_MINTED) | 1); // Updates: // - `address` to the owner. // - `startTimestamp` to the timestamp of minting. // - `burned` to `false`. // - `nextInitialized` to `quantity == 1`. _packedOwnerships[startTokenId] = _packOwnershipData( to, _nextInitializedFlag(quantity) | _nextExtraData(address(0), to, 0) ); if (startTokenId + quantity - 1 > _sequentialUpTo()) _revert(SequentialMintExceedsLimit.selector); emit ConsecutiveTransfer(startTokenId, startTokenId + quantity - 1, address(0), to); _currentIndex = startTokenId + quantity; } _afterTokenTransfers(address(0), to, startTokenId, quantity); } /** * @dev Safely mints `quantity` tokens and transfers them to `to`. * * Requirements: * * - If `to` refers to a smart contract, it must implement * {IERC721Receiver-onERC721Received}, which is called for each safe transfer. * - `quantity` must be greater than 0. * * See {_mint}. * * Emits a {Transfer} event for each mint. */ function _safeMint( address to, uint256 quantity, bytes memory _data ) internal virtual { _mint(to, quantity); unchecked { if (to.code.length != 0) { uint256 end = _currentIndex; uint256 index = end - quantity; do { if (!_checkContractOnERC721Received(address(0), to, index++, _data)) { _revert(TransferToNonERC721ReceiverImplementer.selector); } } while (index < end); // This prevents reentrancy to `_safeMint`. // It does not prevent reentrancy to `_safeMintSpot`. if (_currentIndex != end) revert(); } } } /** * @dev Equivalent to `_safeMint(to, quantity, '')`. */ function _safeMint(address to, uint256 quantity) internal virtual { _safeMint(to, quantity, ''); } /** * @dev Mints a single token at `tokenId`. * * Note: A spot-minted `tokenId` that has been burned can be re-minted again. * * Requirements: * * - `to` cannot be the zero address. * - `tokenId` must be greater than `_sequentialUpTo()`. * - `tokenId` must not exist. * * Emits a {Transfer} event for each mint. */ function _mintSpot(address to, uint256 tokenId) internal virtual { if (tokenId <= _sequentialUpTo()) _revert(SpotMintTokenIdTooSmall.selector); uint256 prevOwnershipPacked = _packedOwnerships[tokenId]; if (_packedOwnershipExists(prevOwnershipPacked)) _revert(TokenAlreadyExists.selector); _beforeTokenTransfers(address(0), to, tokenId, 1); // Overflows are incredibly unrealistic. // The `numberMinted` for `to` is incremented by 1, and has a max limit of 2**64 - 1. // `_spotMinted` is incremented by 1, and has a max limit of 2**256 - 1. unchecked { // Updates: // - `address` to the owner. // - `startTimestamp` to the timestamp of minting. // - `burned` to `false`. // - `nextInitialized` to `true` (as `quantity == 1`). _packedOwnerships[tokenId] = _packOwnershipData( to, _nextInitializedFlag(1) | _nextExtraData(address(0), to, prevOwnershipPacked) ); // Updates: // - `balance += 1`. // - `numberMinted += 1`. // // We can directly add to the `balance` and `numberMinted`. _packedAddressData[to] += (1 << _BITPOS_NUMBER_MINTED) | 1; // Mask `to` to the lower 160 bits, in case the upper bits somehow aren't clean. uint256 toMasked = uint256(uint160(to)) & _BITMASK_ADDRESS; if (toMasked == 0) _revert(MintToZeroAddress.selector); assembly { // Emit the `Transfer` event. log4( 0, // Start of data (0, since no data). 0, // End of data (0, since no data). _TRANSFER_EVENT_SIGNATURE, // Signature. 0, // `address(0)`. toMasked, // `to`. tokenId // `tokenId`. ) } ++_spotMinted; } _afterTokenTransfers(address(0), to, tokenId, 1); } /** * @dev Safely mints a single token at `tokenId`. * * Note: A spot-minted `tokenId` that has been burned can be re-minted again. * * Requirements: * * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}. * - `tokenId` must be greater than `_sequentialUpTo()`. * - `tokenId` must not exist. * * See {_mintSpot}. * * Emits a {Transfer} event. */ function _safeMintSpot( address to, uint256 tokenId, bytes memory _data ) internal virtual { _mintSpot(to, tokenId); unchecked { if (to.code.length != 0) { uint256 currentSpotMinted = _spotMinted; if (!_checkContractOnERC721Received(address(0), to, tokenId, _data)) { _revert(TransferToNonERC721ReceiverImplementer.selector); } // This prevents reentrancy to `_safeMintSpot`. // It does not prevent reentrancy to `_safeMint`. if (_spotMinted != currentSpotMinted) revert(); } } } /** * @dev Equivalent to `_safeMintSpot(to, tokenId, '')`. */ function _safeMintSpot(address to, uint256 tokenId) internal virtual { _safeMintSpot(to, tokenId, ''); } // ============================================================= // APPROVAL OPERATIONS // ============================================================= /** * @dev Equivalent to `_approve(to, tokenId, false)`. */ function _approve(address to, uint256 tokenId) internal virtual { _approve(to, tokenId, false); } /** * @dev Gives permission to `to` to transfer `tokenId` token to another account. * The approval is cleared when the token is transferred. * * Only a single account can be approved at a time, so approving the * zero address clears previous approvals. * * Requirements: * * - `tokenId` must exist. * * Emits an {Approval} event. */ function _approve( address to, uint256 tokenId, bool approvalCheck ) internal virtual { address owner = ownerOf(tokenId); if (approvalCheck && _msgSenderERC721A() != owner) if (!isApprovedForAll(owner, _msgSenderERC721A())) { _revert(ApprovalCallerNotOwnerNorApproved.selector); } _tokenApprovals[tokenId].value = to; emit Approval(owner, to, tokenId); } // ============================================================= // BURN OPERATIONS // ============================================================= /** * @dev Equivalent to `_burn(tokenId, false)`. */ function _burn(uint256 tokenId) internal virtual { _burn(tokenId, false); } /** * @dev Destroys `tokenId`. * The approval is cleared when the token is burned. * * Requirements: * * - `tokenId` must exist. * * Emits a {Transfer} event. */ function _burn(uint256 tokenId, bool approvalCheck) internal virtual { uint256 prevOwnershipPacked = _packedOwnershipOf(tokenId); address from = address(uint160(prevOwnershipPacked)); (uint256 approvedAddressSlot, address approvedAddress) = _getApprovedSlotAndAddress(tokenId); if (approvalCheck) { // The nested ifs save around 20+ gas over a compound boolean condition. if (!_isSenderApprovedOrOwner(approvedAddress, from, _msgSenderERC721A())) if (!isApprovedForAll(from, _msgSenderERC721A())) _revert(TransferCallerNotOwnerNorApproved.selector); } _beforeTokenTransfers(from, address(0), tokenId, 1); // Clear approvals from the previous owner. assembly { if approvedAddress { // This is equivalent to `delete _tokenApprovals[tokenId]`. sstore(approvedAddressSlot, 0) } } // Underflow of the sender's balance is impossible because we check for // ownership above and the recipient's balance can't realistically overflow. // Counter overflow is incredibly unrealistic as `tokenId` would have to be 2**256. unchecked { // Updates: // - `balance -= 1`. // - `numberBurned += 1`. // // We can directly decrement the balance, and increment the number burned. // This is equivalent to `packed -= 1; packed += 1 << _BITPOS_NUMBER_BURNED;`. _packedAddressData[from] += (1 << _BITPOS_NUMBER_BURNED) - 1; // Updates: // - `address` to the last owner. // - `startTimestamp` to the timestamp of burning. // - `burned` to `true`. // - `nextInitialized` to `true`. _packedOwnerships[tokenId] = _packOwnershipData( from, (_BITMASK_BURNED | _BITMASK_NEXT_INITIALIZED) | _nextExtraData(from, address(0), prevOwnershipPacked) ); // If the next slot may not have been initialized (i.e. `nextInitialized == false`) . if (prevOwnershipPacked & _BITMASK_NEXT_INITIALIZED == 0) { uint256 nextTokenId = tokenId + 1; // If the next slot's address is zero and not burned (i.e. packed value is zero). if (_packedOwnerships[nextTokenId] == 0) { // If the next slot is within bounds. if (nextTokenId != _currentIndex) { // Initialize the next slot to maintain correctness for `ownerOf(tokenId + 1)`. _packedOwnerships[nextTokenId] = prevOwnershipPacked; } } } } emit Transfer(from, address(0), tokenId); _afterTokenTransfers(from, address(0), tokenId, 1); // Overflow not possible, as `_burnCounter` cannot be exceed `_currentIndex + _spotMinted` times. unchecked { _burnCounter++; } } // ============================================================= // EXTRA DATA OPERATIONS // ============================================================= /** * @dev Directly sets the extra data for the ownership data `index`. */ function _setExtraDataAt(uint256 index, uint24 extraData) internal virtual { uint256 packed = _packedOwnerships[index]; if (packed == 0) _revert(OwnershipNotInitializedForExtraData.selector); uint256 extraDataCasted; // Cast `extraData` with assembly to avoid redundant masking. assembly { extraDataCasted := extraData } packed = (packed & _BITMASK_EXTRA_DATA_COMPLEMENT) | (extraDataCasted << _BITPOS_EXTRA_DATA); _packedOwnerships[index] = packed; } /** * @dev Called during each token transfer to set the 24bit `extraData` field. * Intended to be overridden by the cosumer contract. * * `previousExtraData` - the value of `extraData` before transfer. * * Calling conditions: * * - When `from` and `to` are both non-zero, `from`'s `tokenId` will be * transferred to `to`. * - When `from` is zero, `tokenId` will be minted for `to`. * - When `to` is zero, `tokenId` will be burned by `from`. * - `from` and `to` are never both zero. */ function _extraData( address from, address to, uint24 previousExtraData ) internal view virtual returns (uint24) {} /** * @dev Returns the next extra data for the packed ownership data. * The returned result is shifted into position. */ function _nextExtraData( address from, address to, uint256 prevOwnershipPacked ) private view returns (uint256) { uint24 extraData = uint24(prevOwnershipPacked >> _BITPOS_EXTRA_DATA); return uint256(_extraData(from, to, extraData)) << _BITPOS_EXTRA_DATA; } // ============================================================= // OTHER OPERATIONS // ============================================================= /** * @dev Returns the message sender (defaults to `msg.sender`). * * If you are writing GSN compatible contracts, you need to override this function. */ function _msgSenderERC721A() internal view virtual returns (address) { return msg.sender; } /** * @dev Converts a uint256 to its ASCII string decimal representation. */ function _toString(uint256 value) internal pure virtual returns (string memory str) { assembly { // The maximum value of a uint256 contains 78 digits (1 byte per digit), but // we allocate 0xa0 bytes to keep the free memory pointer 32-byte word aligned. // We will need 1 word for the trailing zeros padding, 1 word for the length, // and 3 words for a maximum of 78 digits. Total: 5 * 0x20 = 0xa0. let m := add(mload(0x40), 0xa0) // Update the free memory pointer to allocate. mstore(0x40, m) // Assign the `str` to the end. str := sub(m, 0x20) // Zeroize the slot after the string. mstore(str, 0) // Cache the end of the memory to calculate the length later. let end := str // We write the string from rightmost digit to leftmost digit. // The following is essentially a do-while loop that also handles the zero case. // prettier-ignore for { let temp := value } 1 {} { str := sub(str, 1) // Write the character to the pointer. // The ASCII index of the '0' character is 48. mstore8(str, add(48, mod(temp, 10))) // Keep dividing `temp` until zero. temp := div(temp, 10) // prettier-ignore if iszero(temp) { break } } let length := sub(end, str) // Move the pointer 32 bytes leftwards to make room for the length. str := sub(str, 0x20) // Store the length. mstore(str, length) } } /** * @dev For more efficient reverts. */ function _revert(bytes4 errorSelector) internal pure { assembly { mstore(0x00, errorSelector) revert(0x00, 0x04) } } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v5.0.0) (utils/Strings.sol) pragma solidity ^0.8.20; import {Math} from "./math/Math.sol"; import {SignedMath} from "./math/SignedMath.sol"; /** * @dev String operations. */ library Strings { bytes16 private constant HEX_DIGITS = "0123456789abcdef"; uint8 private constant ADDRESS_LENGTH = 20; /** * @dev The `value` string doesn't fit in the specified `length`. */ error StringsInsufficientHexLength(uint256 value, uint256 length); /** * @dev Converts a `uint256` to its ASCII `string` decimal representation. */ function toString(uint256 value) internal pure returns (string memory) { unchecked { uint256 length = Math.log10(value) + 1; string memory buffer = new string(length); uint256 ptr; /// @solidity memory-safe-assembly assembly { ptr := add(buffer, add(32, length)) } while (true) { ptr--; /// @solidity memory-safe-assembly assembly { mstore8(ptr, byte(mod(value, 10), HEX_DIGITS)) } value /= 10; if (value == 0) break; } return buffer; } } /** * @dev Converts a `int256` to its ASCII `string` decimal representation. */ function toStringSigned(int256 value) internal pure returns (string memory) { return string.concat(value < 0 ? "-" : "", toString(SignedMath.abs(value))); } /** * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation. */ function toHexString(uint256 value) internal pure returns (string memory) { unchecked { return toHexString(value, Math.log256(value) + 1); } } /** * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation with fixed length. */ function toHexString(uint256 value, uint256 length) internal pure returns (string memory) { uint256 localValue = value; bytes memory buffer = new bytes(2 * length + 2); buffer[0] = "0"; buffer[1] = "x"; for (uint256 i = 2 * length + 1; i > 1; --i) { buffer[i] = HEX_DIGITS[localValue & 0xf]; localValue >>= 4; } if (localValue != 0) { revert StringsInsufficientHexLength(value, length); } return string(buffer); } /** * @dev Converts an `address` with fixed length of 20 bytes to its not checksummed ASCII `string` hexadecimal * representation. */ function toHexString(address addr) internal pure returns (string memory) { return toHexString(uint256(uint160(addr)), ADDRESS_LENGTH); } /** * @dev Returns true if the two strings are equal. */ function equal(string memory a, string memory b) internal pure returns (bool) { return bytes(a).length == bytes(b).length && keccak256(bytes(a)) == keccak256(bytes(b)); } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v5.0.0) (access/Ownable.sol) pragma solidity ^0.8.20; import {Context} from "../utils/Context.sol"; /** * @dev Contract module which provides a basic access control mechanism, where * there is an account (an owner) that can be granted exclusive access to * specific functions. * * The initial owner is set to the address provided by the deployer. This can * later be changed with {transferOwnership}. * * This module is used through inheritance. It will make available the modifier * `onlyOwner`, which can be applied to your functions to restrict their use to * the owner. */ abstract contract Ownable is Context { address private _owner; /** * @dev The caller account is not authorized to perform an operation. */ error OwnableUnauthorizedAccount(address account); /** * @dev The owner is not a valid owner account. (eg. `address(0)`) */ error OwnableInvalidOwner(address owner); event OwnershipTransferred(address indexed previousOwner, address indexed newOwner); /** * @dev Initializes the contract setting the address provided by the deployer as the initial owner. */ constructor(address initialOwner) { if (initialOwner == address(0)) { revert OwnableInvalidOwner(address(0)); } _transferOwnership(initialOwner); } /** * @dev Throws if called by any account other than the owner. */ modifier onlyOwner() { _checkOwner(); _; } /** * @dev Returns the address of the current owner. */ function owner() public view virtual returns (address) { return _owner; } /** * @dev Throws if the sender is not the owner. */ function _checkOwner() internal view virtual { if (owner() != _msgSender()) { revert OwnableUnauthorizedAccount(_msgSender()); } } /** * @dev Leaves the contract without owner. It will not be possible to call * `onlyOwner` functions. Can only be called by the current owner. * * NOTE: Renouncing ownership will leave the contract without an owner, * thereby disabling any functionality that is only available to the owner. */ function renounceOwnership() public virtual onlyOwner { _transferOwnership(address(0)); } /** * @dev Transfers ownership of the contract to a new account (`newOwner`). * Can only be called by the current owner. */ function transferOwnership(address newOwner) public virtual onlyOwner { if (newOwner == address(0)) { revert OwnableInvalidOwner(address(0)); } _transferOwnership(newOwner); } /** * @dev Transfers ownership of the contract to a new account (`newOwner`). * Internal function without access restriction. */ function _transferOwnership(address newOwner) internal virtual { address oldOwner = _owner; _owner = newOwner; emit OwnershipTransferred(oldOwner, newOwner); } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v5.0.0) (interfaces/IERC2981.sol) pragma solidity ^0.8.20; import {IERC165} from "../utils/introspection/IERC165.sol"; /** * @dev Interface for the NFT Royalty Standard. * * A standardized way to retrieve royalty payment information for non-fungible tokens (NFTs) to enable universal * support for royalty payments across all NFT marketplaces and ecosystem participants. */ interface IERC2981 is IERC165 { /** * @dev Returns how much royalty is owed and to whom, based on a sale price that may be denominated in any unit of * exchange. The royalty amount is denominated and should be paid in that same unit of exchange. */ function royaltyInfo( uint256 tokenId, uint256 salePrice ) external view returns (address receiver, uint256 royaltyAmount); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v5.0.0) (token/ERC20/ERC20.sol) pragma solidity ^0.8.20; import {IERC20} from "./IERC20.sol"; import {IERC20Metadata} from "./extensions/IERC20Metadata.sol"; import {Context} from "../../utils/Context.sol"; import {IERC20Errors} from "../../interfaces/draft-IERC6093.sol"; /** * @dev Implementation of the {IERC20} interface. * * This implementation is agnostic to the way tokens are created. This means * that a supply mechanism has to be added in a derived contract using {_mint}. * * TIP: For a detailed writeup see our guide * https://forum.openzeppelin.com/t/how-to-implement-erc20-supply-mechanisms/226[How * to implement supply mechanisms]. * * The default value of {decimals} is 18. To change this, you should override * this function so it returns a different value. * * We have followed general OpenZeppelin Contracts guidelines: functions revert * instead returning `false` on failure. This behavior is nonetheless * conventional and does not conflict with the expectations of ERC20 * applications. * * Additionally, an {Approval} event is emitted on calls to {transferFrom}. * This allows applications to reconstruct the allowance for all accounts just * by listening to said events. Other implementations of the EIP may not emit * these events, as it isn't required by the specification. */ abstract contract ERC20 is Context, IERC20, IERC20Metadata, IERC20Errors { mapping(address account => uint256) private _balances; mapping(address account => mapping(address spender => uint256)) private _allowances; uint256 private _totalSupply; string private _name; string private _symbol; /** * @dev Sets the values for {name} and {symbol}. * * All two of these values are immutable: they can only be set once during * construction. */ constructor(string memory name_, string memory symbol_) { _name = name_; _symbol = symbol_; } /** * @dev Returns the name of the token. */ function name() public view virtual returns (string memory) { return _name; } /** * @dev Returns the symbol of the token, usually a shorter version of the * name. */ function symbol() public view virtual returns (string memory) { return _symbol; } /** * @dev Returns the number of decimals used to get its user representation. * For example, if `decimals` equals `2`, a balance of `505` tokens should * be displayed to a user as `5.05` (`505 / 10 ** 2`). * * Tokens usually opt for a value of 18, imitating the relationship between * Ether and Wei. This is the default value returned by this function, unless * it's overridden. * * NOTE: This information is only used for _display_ purposes: it in * no way affects any of the arithmetic of the contract, including * {IERC20-balanceOf} and {IERC20-transfer}. */ function decimals() public view virtual returns (uint8) { return 18; } /** * @dev See {IERC20-totalSupply}. */ function totalSupply() public view virtual returns (uint256) { return _totalSupply; } /** * @dev See {IERC20-balanceOf}. */ function balanceOf(address account) public view virtual returns (uint256) { return _balances[account]; } /** * @dev See {IERC20-transfer}. * * Requirements: * * - `to` cannot be the zero address. * - the caller must have a balance of at least `value`. */ function transfer(address to, uint256 value) public virtual returns (bool) { address owner = _msgSender(); _transfer(owner, to, value); return true; } /** * @dev See {IERC20-allowance}. */ function allowance(address owner, address spender) public view virtual returns (uint256) { return _allowances[owner][spender]; } /** * @dev See {IERC20-approve}. * * NOTE: If `value` is the maximum `uint256`, the allowance is not updated on * `transferFrom`. This is semantically equivalent to an infinite approval. * * Requirements: * * - `spender` cannot be the zero address. */ function approve(address spender, uint256 value) public virtual returns (bool) { address owner = _msgSender(); _approve(owner, spender, value); return true; } /** * @dev See {IERC20-transferFrom}. * * Emits an {Approval} event indicating the updated allowance. This is not * required by the EIP. See the note at the beginning of {ERC20}. * * NOTE: Does not update the allowance if the current allowance * is the maximum `uint256`. * * Requirements: * * - `from` and `to` cannot be the zero address. * - `from` must have a balance of at least `value`. * - the caller must have allowance for ``from``'s tokens of at least * `value`. */ function transferFrom(address from, address to, uint256 value) public virtual returns (bool) { address spender = _msgSender(); _spendAllowance(from, spender, value); _transfer(from, to, value); return true; } /** * @dev Moves a `value` amount of tokens from `from` to `to`. * * This internal function is equivalent to {transfer}, and can be used to * e.g. implement automatic token fees, slashing mechanisms, etc. * * Emits a {Transfer} event. * * NOTE: This function is not virtual, {_update} should be overridden instead. */ function _transfer(address from, address to, uint256 value) internal { if (from == address(0)) { revert ERC20InvalidSender(address(0)); } if (to == address(0)) { revert ERC20InvalidReceiver(address(0)); } _update(from, to, value); } /** * @dev Transfers a `value` amount of tokens from `from` to `to`, or alternatively mints (or burns) if `from` * (or `to`) is the zero address. All customizations to transfers, mints, and burns should be done by overriding * this function. * * Emits a {Transfer} event. */ function _update(address from, address to, uint256 value) internal virtual { if (from == address(0)) { // Overflow check required: The rest of the code assumes that totalSupply never overflows _totalSupply += value; } else { uint256 fromBalance = _balances[from]; if (fromBalance < value) { revert ERC20InsufficientBalance(from, fromBalance, value); } unchecked { // Overflow not possible: value <= fromBalance <= totalSupply. _balances[from] = fromBalance - value; } } if (to == address(0)) { unchecked { // Overflow not possible: value <= totalSupply or value <= fromBalance <= totalSupply. _totalSupply -= value; } } else { unchecked { // Overflow not possible: balance + value is at most totalSupply, which we know fits into a uint256. _balances[to] += value; } } emit Transfer(from, to, value); } /** * @dev Creates a `value` amount of tokens and assigns them to `account`, by transferring it from address(0). * Relies on the `_update` mechanism * * Emits a {Transfer} event with `from` set to the zero address. * * NOTE: This function is not virtual, {_update} should be overridden instead. */ function _mint(address account, uint256 value) internal { if (account == address(0)) { revert ERC20InvalidReceiver(address(0)); } _update(address(0), account, value); } /** * @dev Destroys a `value` amount of tokens from `account`, lowering the total supply. * Relies on the `_update` mechanism. * * Emits a {Transfer} event with `to` set to the zero address. * * NOTE: This function is not virtual, {_update} should be overridden instead */ function _burn(address account, uint256 value) internal { if (account == address(0)) { revert ERC20InvalidSender(address(0)); } _update(account, address(0), value); } /** * @dev Sets `value` as the allowance of `spender` over the `owner` s tokens. * * This internal function is equivalent to `approve`, and can be used to * e.g. set automatic allowances for certain subsystems, etc. * * Emits an {Approval} event. * * Requirements: * * - `owner` cannot be the zero address. * - `spender` cannot be the zero address. * * Overrides to this logic should be done to the variant with an additional `bool emitEvent` argument. */ function _approve(address owner, address spender, uint256 value) internal { _approve(owner, spender, value, true); } /** * @dev Variant of {_approve} with an optional flag to enable or disable the {Approval} event. * * By default (when calling {_approve}) the flag is set to true. On the other hand, approval changes made by * `_spendAllowance` during the `transferFrom` operation set the flag to false. This saves gas by not emitting any * `Approval` event during `transferFrom` operations. * * Anyone who wishes to continue emitting `Approval` events on the`transferFrom` operation can force the flag to * true using the following override: * ``` * function _approve(address owner, address spender, uint256 value, bool) internal virtual override { * super._approve(owner, spender, value, true); * } * ``` * * Requirements are the same as {_approve}. */ function _approve(address owner, address spender, uint256 value, bool emitEvent) internal virtual { if (owner == address(0)) { revert ERC20InvalidApprover(address(0)); } if (spender == address(0)) { revert ERC20InvalidSpender(address(0)); } _allowances[owner][spender] = value; if (emitEvent) { emit Approval(owner, spender, value); } } /** * @dev Updates `owner` s allowance for `spender` based on spent `value`. * * Does not update the allowance value in case of infinite allowance. * Revert if not enough allowance is available. * * Does not emit an {Approval} event. */ function _spendAllowance(address owner, address spender, uint256 value) internal virtual { uint256 currentAllowance = allowance(owner, spender); if (currentAllowance != type(uint256).max) { if (currentAllowance < value) { revert ERC20InsufficientAllowance(spender, currentAllowance, value); } unchecked { _approve(owner, spender, currentAllowance - value, false); } } } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v5.0.0) (utils/cryptography/MerkleProof.sol) pragma solidity ^0.8.20; /** * @dev These functions deal with verification of Merkle Tree proofs. * * The tree and the proofs can be generated using our * https://github.com/OpenZeppelin/merkle-tree[JavaScript library]. * You will find a quickstart guide in the readme. * * WARNING: You should avoid using leaf values that are 64 bytes long prior to * hashing, or use a hash function other than keccak256 for hashing leaves. * This is because the concatenation of a sorted pair of internal nodes in * the Merkle tree could be reinterpreted as a leaf value. * OpenZeppelin's JavaScript library generates Merkle trees that are safe * against this attack out of the box. */ library MerkleProof { /** *@dev The multiproof provided is not valid. */ error MerkleProofInvalidMultiproof(); /** * @dev Returns true if a `leaf` can be proved to be a part of a Merkle tree * defined by `root`. For this, a `proof` must be provided, containing * sibling hashes on the branch from the leaf to the root of the tree. Each * pair of leaves and each pair of pre-images are assumed to be sorted. */ function verify(bytes32[] memory proof, bytes32 root, bytes32 leaf) internal pure returns (bool) { return processProof(proof, leaf) == root; } /** * @dev Calldata version of {verify} */ function verifyCalldata(bytes32[] calldata proof, bytes32 root, bytes32 leaf) internal pure returns (bool) { return processProofCalldata(proof, leaf) == root; } /** * @dev Returns the rebuilt hash obtained by traversing a Merkle tree up * from `leaf` using `proof`. A `proof` is valid if and only if the rebuilt * hash matches the root of the tree. When processing the proof, the pairs * of leafs & pre-images are assumed to be sorted. */ function processProof(bytes32[] memory proof, bytes32 leaf) internal pure returns (bytes32) { bytes32 computedHash = leaf; for (uint256 i = 0; i < proof.length; i++) { computedHash = _hashPair(computedHash, proof[i]); } return computedHash; } /** * @dev Calldata version of {processProof} */ function processProofCalldata(bytes32[] calldata proof, bytes32 leaf) internal pure returns (bytes32) { bytes32 computedHash = leaf; for (uint256 i = 0; i < proof.length; i++) { computedHash = _hashPair(computedHash, proof[i]); } return computedHash; } /** * @dev Returns true if the `leaves` can be simultaneously proven to be a part of a Merkle tree defined by * `root`, according to `proof` and `proofFlags` as described in {processMultiProof}. * * CAUTION: Not all Merkle trees admit multiproofs. See {processMultiProof} for details. */ function multiProofVerify( bytes32[] memory proof, bool[] memory proofFlags, bytes32 root, bytes32[] memory leaves ) internal pure returns (bool) { return processMultiProof(proof, proofFlags, leaves) == root; } /** * @dev Calldata version of {multiProofVerify} * * CAUTION: Not all Merkle trees admit multiproofs. See {processMultiProof} for details. */ function multiProofVerifyCalldata( bytes32[] calldata proof, bool[] calldata proofFlags, bytes32 root, bytes32[] memory leaves ) internal pure returns (bool) { return processMultiProofCalldata(proof, proofFlags, leaves) == root; } /** * @dev Returns the root of a tree reconstructed from `leaves` and sibling nodes in `proof`. The reconstruction * proceeds by incrementally reconstructing all inner nodes by combining a leaf/inner node with either another * leaf/inner node or a proof sibling node, depending on whether each `proofFlags` item is true or false * respectively. * * CAUTION: Not all Merkle trees admit multiproofs. To use multiproofs, it is sufficient to ensure that: 1) the tree * is complete (but not necessarily perfect), 2) the leaves to be proven are in the opposite order they are in the * tree (i.e., as seen from right to left starting at the deepest layer and continuing at the next layer). */ function processMultiProof( bytes32[] memory proof, bool[] memory proofFlags, bytes32[] memory leaves ) internal pure returns (bytes32 merkleRoot) { // This function rebuilds the root hash by traversing the tree up from the leaves. The root is rebuilt by // consuming and producing values on a queue. The queue starts with the `leaves` array, then goes onto the // `hashes` array. At the end of the process, the last hash in the `hashes` array should contain the root of // the Merkle tree. uint256 leavesLen = leaves.length; uint256 proofLen = proof.length; uint256 totalHashes = proofFlags.length; // Check proof validity. if (leavesLen + proofLen != totalHashes + 1) { revert MerkleProofInvalidMultiproof(); } // The xxxPos values are "pointers" to the next value to consume in each array. All accesses are done using // `xxx[xxxPos++]`, which return the current value and increment the pointer, thus mimicking a queue's "pop". bytes32[] memory hashes = new bytes32[](totalHashes); uint256 leafPos = 0; uint256 hashPos = 0; uint256 proofPos = 0; // At each step, we compute the next hash using two values: // - a value from the "main queue". If not all leaves have been consumed, we get the next leaf, otherwise we // get the next hash. // - depending on the flag, either another value from the "main queue" (merging branches) or an element from the // `proof` array. for (uint256 i = 0; i < totalHashes; i++) { bytes32 a = leafPos < leavesLen ? leaves[leafPos++] : hashes[hashPos++]; bytes32 b = proofFlags[i] ? (leafPos < leavesLen ? leaves[leafPos++] : hashes[hashPos++]) : proof[proofPos++]; hashes[i] = _hashPair(a, b); } if (totalHashes > 0) { if (proofPos != proofLen) { revert MerkleProofInvalidMultiproof(); } unchecked { return hashes[totalHashes - 1]; } } else if (leavesLen > 0) { return leaves[0]; } else { return proof[0]; } } /** * @dev Calldata version of {processMultiProof}. * * CAUTION: Not all Merkle trees admit multiproofs. See {processMultiProof} for details. */ function processMultiProofCalldata( bytes32[] calldata proof, bool[] calldata proofFlags, bytes32[] memory leaves ) internal pure returns (bytes32 merkleRoot) { // This function rebuilds the root hash by traversing the tree up from the leaves. The root is rebuilt by // consuming and producing values on a queue. The queue starts with the `leaves` array, then goes onto the // `hashes` array. At the end of the process, the last hash in the `hashes` array should contain the root of // the Merkle tree. uint256 leavesLen = leaves.length; uint256 proofLen = proof.length; uint256 totalHashes = proofFlags.length; // Check proof validity. if (leavesLen + proofLen != totalHashes + 1) { revert MerkleProofInvalidMultiproof(); } // The xxxPos values are "pointers" to the next value to consume in each array. All accesses are done using // `xxx[xxxPos++]`, which return the current value and increment the pointer, thus mimicking a queue's "pop". bytes32[] memory hashes = new bytes32[](totalHashes); uint256 leafPos = 0; uint256 hashPos = 0; uint256 proofPos = 0; // At each step, we compute the next hash using two values: // - a value from the "main queue". If not all leaves have been consumed, we get the next leaf, otherwise we // get the next hash. // - depending on the flag, either another value from the "main queue" (merging branches) or an element from the // `proof` array. for (uint256 i = 0; i < totalHashes; i++) { bytes32 a = leafPos < leavesLen ? leaves[leafPos++] : hashes[hashPos++]; bytes32 b = proofFlags[i] ? (leafPos < leavesLen ? leaves[leafPos++] : hashes[hashPos++]) : proof[proofPos++]; hashes[i] = _hashPair(a, b); } if (totalHashes > 0) { if (proofPos != proofLen) { revert MerkleProofInvalidMultiproof(); } unchecked { return hashes[totalHashes - 1]; } } else if (leavesLen > 0) { return leaves[0]; } else { return proof[0]; } } /** * @dev Sorts the pair (a, b) and hashes the result. */ function _hashPair(bytes32 a, bytes32 b) private pure returns (bytes32) { return a < b ? _efficientHash(a, b) : _efficientHash(b, a); } /** * @dev Implementation of keccak256(abi.encode(a, b)) that doesn't allocate or expand memory. */ function _efficientHash(bytes32 a, bytes32 b) private pure returns (bytes32 value) { /// @solidity memory-safe-assembly assembly { mstore(0x00, a) mstore(0x20, b) value := keccak256(0x00, 0x40) } } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v5.0.0) (token/ERC20/utils/SafeERC20.sol) pragma solidity ^0.8.20; import {IERC20} from "../IERC20.sol"; import {IERC20Permit} from "../extensions/IERC20Permit.sol"; import {Address} from "../../../utils/Address.sol"; /** * @title SafeERC20 * @dev Wrappers around ERC20 operations that throw on failure (when the token * contract returns false). Tokens that return no value (and instead revert or * throw on failure) are also supported, non-reverting calls are assumed to be * successful. * To use this library you can add a `using SafeERC20 for IERC20;` statement to your contract, * which allows you to call the safe operations as `token.safeTransfer(...)`, etc. */ library SafeERC20 { using Address for address; /** * @dev An operation with an ERC20 token failed. */ error SafeERC20FailedOperation(address token); /** * @dev Indicates a failed `decreaseAllowance` request. */ error SafeERC20FailedDecreaseAllowance(address spender, uint256 currentAllowance, uint256 requestedDecrease); /** * @dev Transfer `value` amount of `token` from the calling contract to `to`. If `token` returns no value, * non-reverting calls are assumed to be successful. */ function safeTransfer(IERC20 token, address to, uint256 value) internal { _callOptionalReturn(token, abi.encodeCall(token.transfer, (to, value))); } /** * @dev Transfer `value` amount of `token` from `from` to `to`, spending the approval given by `from` to the * calling contract. If `token` returns no value, non-reverting calls are assumed to be successful. */ function safeTransferFrom(IERC20 token, address from, address to, uint256 value) internal { _callOptionalReturn(token, abi.encodeCall(token.transferFrom, (from, to, value))); } /** * @dev Increase the calling contract's allowance toward `spender` by `value`. If `token` returns no value, * non-reverting calls are assumed to be successful. */ function safeIncreaseAllowance(IERC20 token, address spender, uint256 value) internal { uint256 oldAllowance = token.allowance(address(this), spender); forceApprove(token, spender, oldAllowance + value); } /** * @dev Decrease the calling contract's allowance toward `spender` by `requestedDecrease`. If `token` returns no * value, non-reverting calls are assumed to be successful. */ function safeDecreaseAllowance(IERC20 token, address spender, uint256 requestedDecrease) internal { unchecked { uint256 currentAllowance = token.allowance(address(this), spender); if (currentAllowance < requestedDecrease) { revert SafeERC20FailedDecreaseAllowance(spender, currentAllowance, requestedDecrease); } forceApprove(token, spender, currentAllowance - requestedDecrease); } } /** * @dev Set the calling contract's allowance toward `spender` to `value`. If `token` returns no value, * non-reverting calls are assumed to be successful. Meant to be used with tokens that require the approval * to be set to zero before setting it to a non-zero value, such as USDT. */ function forceApprove(IERC20 token, address spender, uint256 value) internal { bytes memory approvalCall = abi.encodeCall(token.approve, (spender, value)); if (!_callOptionalReturnBool(token, approvalCall)) { _callOptionalReturn(token, abi.encodeCall(token.approve, (spender, 0))); _callOptionalReturn(token, approvalCall); } } /** * @dev Imitates a Solidity high-level call (i.e. a regular function call to a contract), relaxing the requirement * on the return value: the return value is optional (but if data is returned, it must not be false). * @param token The token targeted by the call. * @param data The call data (encoded using abi.encode or one of its variants). */ function _callOptionalReturn(IERC20 token, bytes memory data) private { // We need to perform a low level call here, to bypass Solidity's return data size checking mechanism, since // we're implementing it ourselves. We use {Address-functionCall} to perform this call, which verifies that // the target address contains contract code and also asserts for success in the low-level call. bytes memory returndata = address(token).functionCall(data); if (returndata.length != 0 && !abi.decode(returndata, (bool))) { revert SafeERC20FailedOperation(address(token)); } } /** * @dev Imitates a Solidity high-level call (i.e. a regular function call to a contract), relaxing the requirement * on the return value: the return value is optional (but if data is returned, it must not be false). * @param token The token targeted by the call. * @param data The call data (encoded using abi.encode or one of its variants). * * This is a variant of {_callOptionalReturn} that silents catches all reverts and returns a bool instead. */ function _callOptionalReturnBool(IERC20 token, bytes memory data) private returns (bool) { // We need to perform a low level call here, to bypass Solidity's return data size checking mechanism, since // we're implementing it ourselves. We cannot use {Address-functionCall} here since this should return false // and not revert is the subcall reverts. (bool success, bytes memory returndata) = address(token).call(data); return success && (returndata.length == 0 || abi.decode(returndata, (bool))) && address(token).code.length > 0; } }
// SPDX-License-Identifier: MIT // ERC721A Contracts v4.2.3 // Creator: Chiru Labs pragma solidity ^0.8.4; /** * @dev Interface of ERC721A. */ interface IERC721A { /** * The caller must own the token or be an approved operator. */ error ApprovalCallerNotOwnerNorApproved(); /** * The token does not exist. */ error ApprovalQueryForNonexistentToken(); /** * Cannot query the balance for the zero address. */ error BalanceQueryForZeroAddress(); /** * Cannot mint to the zero address. */ error MintToZeroAddress(); /** * The quantity of tokens minted must be more than zero. */ error MintZeroQuantity(); /** * The token does not exist. */ error OwnerQueryForNonexistentToken(); /** * The caller must own the token or be an approved operator. */ error TransferCallerNotOwnerNorApproved(); /** * The token must be owned by `from`. */ error TransferFromIncorrectOwner(); /** * Cannot safely transfer to a contract that does not implement the * ERC721Receiver interface. */ error TransferToNonERC721ReceiverImplementer(); /** * Cannot transfer to the zero address. */ error TransferToZeroAddress(); /** * The token does not exist. */ error URIQueryForNonexistentToken(); /** * The `quantity` minted with ERC2309 exceeds the safety limit. */ error MintERC2309QuantityExceedsLimit(); /** * The `extraData` cannot be set on an unintialized ownership slot. */ error OwnershipNotInitializedForExtraData(); /** * `_sequentialUpTo()` must be greater than `_startTokenId()`. */ error SequentialUpToTooSmall(); /** * The `tokenId` of a sequential mint exceeds `_sequentialUpTo()`. */ error SequentialMintExceedsLimit(); /** * Spot minting requires a `tokenId` greater than `_sequentialUpTo()`. */ error SpotMintTokenIdTooSmall(); /** * Cannot mint over a token that already exists. */ error TokenAlreadyExists(); /** * The feature is not compatible with spot mints. */ error NotCompatibleWithSpotMints(); // ============================================================= // STRUCTS // ============================================================= struct TokenOwnership { // The address of the owner. address addr; // Stores the start time of ownership with minimal overhead for tokenomics. uint64 startTimestamp; // Whether the token has been burned. bool burned; // Arbitrary data similar to `startTimestamp` that can be set via {_extraData}. uint24 extraData; } // ============================================================= // TOKEN COUNTERS // ============================================================= /** * @dev Returns the total number of tokens in existence. * Burned tokens will reduce the count. * To get the total number of tokens minted, please see {_totalMinted}. */ function totalSupply() external view returns (uint256); // ============================================================= // IERC165 // ============================================================= /** * @dev Returns true if this contract implements the interface defined by * `interfaceId`. See the corresponding * [EIP section](https://eips.ethereum.org/EIPS/eip-165#how-interfaces-are-identified) * to learn more about how these ids are created. * * This function call must use less than 30000 gas. */ function supportsInterface(bytes4 interfaceId) external view returns (bool); // ============================================================= // IERC721 // ============================================================= /** * @dev Emitted when `tokenId` token is transferred from `from` to `to`. */ event Transfer(address indexed from, address indexed to, uint256 indexed tokenId); /** * @dev Emitted when `owner` enables `approved` to manage the `tokenId` token. */ event Approval(address indexed owner, address indexed approved, uint256 indexed tokenId); /** * @dev Emitted when `owner` enables or disables * (`approved`) `operator` to manage all of its assets. */ event ApprovalForAll(address indexed owner, address indexed operator, bool approved); /** * @dev Returns the number of tokens in `owner`'s account. */ function balanceOf(address owner) external view returns (uint256 balance); /** * @dev Returns the owner of the `tokenId` token. * * Requirements: * * - `tokenId` must exist. */ function ownerOf(uint256 tokenId) external view returns (address owner); /** * @dev Safely transfers `tokenId` token from `from` to `to`, * checking first that contract recipients are aware of the ERC721 protocol * to prevent tokens from being forever locked. * * Requirements: * * - `from` cannot be the zero address. * - `to` cannot be the zero address. * - `tokenId` token must exist and be owned by `from`. * - If the caller is not `from`, it must be have been allowed to move * this token by either {approve} or {setApprovalForAll}. * - If `to` refers to a smart contract, it must implement * {IERC721Receiver-onERC721Received}, which is called upon a safe transfer. * * Emits a {Transfer} event. */ function safeTransferFrom( address from, address to, uint256 tokenId, bytes calldata data ) external payable; /** * @dev Equivalent to `safeTransferFrom(from, to, tokenId, '')`. */ function safeTransferFrom( address from, address to, uint256 tokenId ) external payable; /** * @dev Transfers `tokenId` from `from` to `to`. * * WARNING: Usage of this method is discouraged, use {safeTransferFrom} * whenever possible. * * Requirements: * * - `from` cannot be the zero address. * - `to` cannot be the zero address. * - `tokenId` token must be owned by `from`. * - If the caller is not `from`, it must be approved to move this token * by either {approve} or {setApprovalForAll}. * * Emits a {Transfer} event. */ function transferFrom( address from, address to, uint256 tokenId ) external payable; /** * @dev Gives permission to `to` to transfer `tokenId` token to another account. * The approval is cleared when the token is transferred. * * Only a single account can be approved at a time, so approving the * zero address clears previous approvals. * * Requirements: * * - The caller must own the token or be an approved operator. * - `tokenId` must exist. * * Emits an {Approval} event. */ function approve(address to, uint256 tokenId) external payable; /** * @dev Approve or remove `operator` as an operator for the caller. * Operators can call {transferFrom} or {safeTransferFrom} * for any token owned by the caller. * * Requirements: * * - The `operator` cannot be the caller. * * Emits an {ApprovalForAll} event. */ function setApprovalForAll(address operator, bool _approved) external; /** * @dev Returns the account approved for `tokenId` token. * * Requirements: * * - `tokenId` must exist. */ function getApproved(uint256 tokenId) external view returns (address operator); /** * @dev Returns if the `operator` is allowed to manage all of the assets of `owner`. * * See {setApprovalForAll}. */ function isApprovedForAll(address owner, address operator) external view returns (bool); // ============================================================= // IERC721Metadata // ============================================================= /** * @dev Returns the token collection name. */ function name() external view returns (string memory); /** * @dev Returns the token collection symbol. */ function symbol() external view returns (string memory); /** * @dev Returns the Uniform Resource Identifier (URI) for `tokenId` token. */ function tokenURI(uint256 tokenId) external view returns (string memory); // ============================================================= // IERC2309 // ============================================================= /** * @dev Emitted when tokens in `fromTokenId` to `toTokenId` * (inclusive) is transferred from `from` to `to`, as defined in the * [ERC2309](https://eips.ethereum.org/EIPS/eip-2309) standard. * * See {_mintERC2309} for more details. */ event ConsecutiveTransfer(uint256 indexed fromTokenId, uint256 toTokenId, address indexed from, address indexed to); }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; // solhint-disable-next-line interface-starts-with-i interface AggregatorV3Interface { function decimals() external view returns (uint8); function description() external view returns (string memory); function version() external view returns (uint256); function getRoundData( uint80 _roundId ) external view returns (uint80 roundId, int256 answer, uint256 startedAt, uint256 updatedAt, uint80 answeredInRound); function latestRoundData() external view returns (uint80 roundId, int256 answer, uint256 startedAt, uint256 updatedAt, uint80 answeredInRound); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v5.0.1) (utils/Context.sol) pragma solidity ^0.8.20; /** * @dev Provides information about the current execution context, including the * sender of the transaction and its data. While these are generally available * via msg.sender and msg.data, they should not be accessed in such a direct * manner, since when dealing with meta-transactions the account sending and * paying for execution may not be the actual sender (as far as an application * is concerned). * * This contract is only required for intermediate, library-like contracts. */ abstract contract Context { function _msgSender() internal view virtual returns (address) { return msg.sender; } function _msgData() internal view virtual returns (bytes calldata) { return msg.data; } function _contextSuffixLength() internal view virtual returns (uint256) { return 0; } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v5.0.0) (interfaces/draft-IERC6093.sol) pragma solidity ^0.8.20; /** * @dev Standard ERC20 Errors * Interface of the https://eips.ethereum.org/EIPS/eip-6093[ERC-6093] custom errors for ERC20 tokens. */ interface IERC20Errors { /** * @dev Indicates an error related to the current `balance` of a `sender`. Used in transfers. * @param sender Address whose tokens are being transferred. * @param balance Current balance for the interacting account. * @param needed Minimum amount required to perform a transfer. */ error ERC20InsufficientBalance(address sender, uint256 balance, uint256 needed); /** * @dev Indicates a failure with the token `sender`. Used in transfers. * @param sender Address whose tokens are being transferred. */ error ERC20InvalidSender(address sender); /** * @dev Indicates a failure with the token `receiver`. Used in transfers. * @param receiver Address to which tokens are being transferred. */ error ERC20InvalidReceiver(address receiver); /** * @dev Indicates a failure with the `spender`’s `allowance`. Used in transfers. * @param spender Address that may be allowed to operate on tokens without being their owner. * @param allowance Amount of tokens a `spender` is allowed to operate with. * @param needed Minimum amount required to perform a transfer. */ error ERC20InsufficientAllowance(address spender, uint256 allowance, uint256 needed); /** * @dev Indicates a failure with the `approver` of a token to be approved. Used in approvals. * @param approver Address initiating an approval operation. */ error ERC20InvalidApprover(address approver); /** * @dev Indicates a failure with the `spender` to be approved. Used in approvals. * @param spender Address that may be allowed to operate on tokens without being their owner. */ error ERC20InvalidSpender(address spender); } /** * @dev Standard ERC721 Errors * Interface of the https://eips.ethereum.org/EIPS/eip-6093[ERC-6093] custom errors for ERC721 tokens. */ interface IERC721Errors { /** * @dev Indicates that an address can't be an owner. For example, `address(0)` is a forbidden owner in EIP-20. * Used in balance queries. * @param owner Address of the current owner of a token. */ error ERC721InvalidOwner(address owner); /** * @dev Indicates a `tokenId` whose `owner` is the zero address. * @param tokenId Identifier number of a token. */ error ERC721NonexistentToken(uint256 tokenId); /** * @dev Indicates an error related to the ownership over a particular token. Used in transfers. * @param sender Address whose tokens are being transferred. * @param tokenId Identifier number of a token. * @param owner Address of the current owner of a token. */ error ERC721IncorrectOwner(address sender, uint256 tokenId, address owner); /** * @dev Indicates a failure with the token `sender`. Used in transfers. * @param sender Address whose tokens are being transferred. */ error ERC721InvalidSender(address sender); /** * @dev Indicates a failure with the token `receiver`. Used in transfers. * @param receiver Address to which tokens are being transferred. */ error ERC721InvalidReceiver(address receiver); /** * @dev Indicates a failure with the `operator`’s approval. Used in transfers. * @param operator Address that may be allowed to operate on tokens without being their owner. * @param tokenId Identifier number of a token. */ error ERC721InsufficientApproval(address operator, uint256 tokenId); /** * @dev Indicates a failure with the `approver` of a token to be approved. Used in approvals. * @param approver Address initiating an approval operation. */ error ERC721InvalidApprover(address approver); /** * @dev Indicates a failure with the `operator` to be approved. Used in approvals. * @param operator Address that may be allowed to operate on tokens without being their owner. */ error ERC721InvalidOperator(address operator); } /** * @dev Standard ERC1155 Errors * Interface of the https://eips.ethereum.org/EIPS/eip-6093[ERC-6093] custom errors for ERC1155 tokens. */ interface IERC1155Errors { /** * @dev Indicates an error related to the current `balance` of a `sender`. Used in transfers. * @param sender Address whose tokens are being transferred. * @param balance Current balance for the interacting account. * @param needed Minimum amount required to perform a transfer. * @param tokenId Identifier number of a token. */ error ERC1155InsufficientBalance(address sender, uint256 balance, uint256 needed, uint256 tokenId); /** * @dev Indicates a failure with the token `sender`. Used in transfers. * @param sender Address whose tokens are being transferred. */ error ERC1155InvalidSender(address sender); /** * @dev Indicates a failure with the token `receiver`. Used in transfers. * @param receiver Address to which tokens are being transferred. */ error ERC1155InvalidReceiver(address receiver); /** * @dev Indicates a failure with the `operator`’s approval. Used in transfers. * @param operator Address that may be allowed to operate on tokens without being their owner. * @param owner Address of the current owner of a token. */ error ERC1155MissingApprovalForAll(address operator, address owner); /** * @dev Indicates a failure with the `approver` of a token to be approved. Used in approvals. * @param approver Address initiating an approval operation. */ error ERC1155InvalidApprover(address approver); /** * @dev Indicates a failure with the `operator` to be approved. Used in approvals. * @param operator Address that may be allowed to operate on tokens without being their owner. */ error ERC1155InvalidOperator(address operator); /** * @dev Indicates an array length mismatch between ids and values in a safeBatchTransferFrom operation. * Used in batch transfers. * @param idsLength Length of the array of token identifiers * @param valuesLength Length of the array of token amounts */ error ERC1155InvalidArrayLength(uint256 idsLength, uint256 valuesLength); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v5.0.0) (utils/introspection/IERC165.sol) pragma solidity ^0.8.20; /** * @dev Interface of the ERC165 standard, as defined in the * https://eips.ethereum.org/EIPS/eip-165[EIP]. * * Implementers can declare support of contract interfaces, which can then be * queried by others ({ERC165Checker}). * * For an implementation, see {ERC165}. */ interface IERC165 { /** * @dev Returns true if this contract implements the interface defined by * `interfaceId`. See the corresponding * https://eips.ethereum.org/EIPS/eip-165#how-interfaces-are-identified[EIP section] * to learn more about how these ids are created. * * This function call must use less than 30 000 gas. */ function supportsInterface(bytes4 interfaceId) external view returns (bool); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v5.0.0) (utils/math/SignedMath.sol) pragma solidity ^0.8.20; /** * @dev Standard signed math utilities missing in the Solidity language. */ library SignedMath { /** * @dev Returns the largest of two signed numbers. */ function max(int256 a, int256 b) internal pure returns (int256) { return a > b ? a : b; } /** * @dev Returns the smallest of two signed numbers. */ function min(int256 a, int256 b) internal pure returns (int256) { return a < b ? a : b; } /** * @dev Returns the average of two signed numbers without overflow. * The result is rounded towards zero. */ function average(int256 a, int256 b) internal pure returns (int256) { // Formula from the book "Hacker's Delight" int256 x = (a & b) + ((a ^ b) >> 1); return x + (int256(uint256(x) >> 255) & (a ^ b)); } /** * @dev Returns the absolute unsigned value of a signed value. */ function abs(int256 n) internal pure returns (uint256) { unchecked { // must be unchecked in order to support `n = type(int256).min` return uint256(n >= 0 ? n : -n); } } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v5.0.0) (utils/Address.sol) pragma solidity ^0.8.20; /** * @dev Collection of functions related to the address type */ library Address { /** * @dev The ETH balance of the account is not enough to perform the operation. */ error AddressInsufficientBalance(address account); /** * @dev There's no code at `target` (it is not a contract). */ error AddressEmptyCode(address target); /** * @dev A call to an address target failed. The target may have reverted. */ error FailedInnerCall(); /** * @dev Replacement for Solidity's `transfer`: sends `amount` wei to * `recipient`, forwarding all available gas and reverting on errors. * * https://eips.ethereum.org/EIPS/eip-1884[EIP1884] increases the gas cost * of certain opcodes, possibly making contracts go over the 2300 gas limit * imposed by `transfer`, making them unable to receive funds via * `transfer`. {sendValue} removes this limitation. * * https://consensys.net/diligence/blog/2019/09/stop-using-soliditys-transfer-now/[Learn more]. * * IMPORTANT: because control is transferred to `recipient`, care must be * taken to not create reentrancy vulnerabilities. Consider using * {ReentrancyGuard} or the * https://solidity.readthedocs.io/en/v0.8.20/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern]. */ function sendValue(address payable recipient, uint256 amount) internal { if (address(this).balance < amount) { revert AddressInsufficientBalance(address(this)); } (bool success, ) = recipient.call{value: amount}(""); if (!success) { revert FailedInnerCall(); } } /** * @dev Performs a Solidity function call using a low level `call`. A * plain `call` is an unsafe replacement for a function call: use this * function instead. * * If `target` reverts with a revert reason or custom error, it is bubbled * up by this function (like regular Solidity function calls). However, if * the call reverted with no returned reason, this function reverts with a * {FailedInnerCall} error. * * Returns the raw returned data. To convert to the expected return value, * use https://solidity.readthedocs.io/en/latest/units-and-global-variables.html?highlight=abi.decode#abi-encoding-and-decoding-functions[`abi.decode`]. * * Requirements: * * - `target` must be a contract. * - calling `target` with `data` must not revert. */ function functionCall(address target, bytes memory data) internal returns (bytes memory) { return functionCallWithValue(target, data, 0); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], * but also transferring `value` wei to `target`. * * Requirements: * * - the calling contract must have an ETH balance of at least `value`. * - the called Solidity function must be `payable`. */ function functionCallWithValue(address target, bytes memory data, uint256 value) internal returns (bytes memory) { if (address(this).balance < value) { revert AddressInsufficientBalance(address(this)); } (bool success, bytes memory returndata) = target.call{value: value}(data); return verifyCallResultFromTarget(target, success, returndata); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], * but performing a static call. */ function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) { (bool success, bytes memory returndata) = target.staticcall(data); return verifyCallResultFromTarget(target, success, returndata); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], * but performing a delegate call. */ function functionDelegateCall(address target, bytes memory data) internal returns (bytes memory) { (bool success, bytes memory returndata) = target.delegatecall(data); return verifyCallResultFromTarget(target, success, returndata); } /** * @dev Tool to verify that a low level call to smart-contract was successful, and reverts if the target * was not a contract or bubbling up the revert reason (falling back to {FailedInnerCall}) in case of an * unsuccessful call. */ function verifyCallResultFromTarget( address target, bool success, bytes memory returndata ) internal view returns (bytes memory) { if (!success) { _revert(returndata); } else { // only check if target is a contract if the call was successful and the return data is empty // otherwise we already know that it was a contract if (returndata.length == 0 && target.code.length == 0) { revert AddressEmptyCode(target); } return returndata; } } /** * @dev Tool to verify that a low level call was successful, and reverts if it wasn't, either by bubbling the * revert reason or with a default {FailedInnerCall} error. */ function verifyCallResult(bool success, bytes memory returndata) internal pure returns (bytes memory) { if (!success) { _revert(returndata); } else { return returndata; } } /** * @dev Reverts with returndata if present. Otherwise reverts with {FailedInnerCall}. */ function _revert(bytes memory returndata) private pure { // Look for revert reason and bubble it up if present if (returndata.length > 0) { // The easiest way to bubble the revert reason is using memory via assembly /// @solidity memory-safe-assembly assembly { let returndata_size := mload(returndata) revert(add(32, returndata), returndata_size) } } else { revert FailedInnerCall(); } } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v5.0.0) (utils/math/Math.sol) pragma solidity ^0.8.20; /** * @dev Standard math utilities missing in the Solidity language. */ library Math { /** * @dev Muldiv operation overflow. */ error MathOverflowedMulDiv(); enum Rounding { Floor, // Toward negative infinity Ceil, // Toward positive infinity Trunc, // Toward zero Expand // Away from zero } /** * @dev Returns the addition of two unsigned integers, with an overflow flag. */ function tryAdd(uint256 a, uint256 b) internal pure returns (bool, uint256) { unchecked { uint256 c = a + b; if (c < a) return (false, 0); return (true, c); } } /** * @dev Returns the subtraction of two unsigned integers, with an overflow flag. */ function trySub(uint256 a, uint256 b) internal pure returns (bool, uint256) { unchecked { if (b > a) return (false, 0); return (true, a - b); } } /** * @dev Returns the multiplication of two unsigned integers, with an overflow flag. */ function tryMul(uint256 a, uint256 b) internal pure returns (bool, uint256) { unchecked { // Gas optimization: this is cheaper than requiring 'a' not being zero, but the // benefit is lost if 'b' is also tested. // See: https://github.com/OpenZeppelin/openzeppelin-contracts/pull/522 if (a == 0) return (true, 0); uint256 c = a * b; if (c / a != b) return (false, 0); return (true, c); } } /** * @dev Returns the division of two unsigned integers, with a division by zero flag. */ function tryDiv(uint256 a, uint256 b) internal pure returns (bool, uint256) { unchecked { if (b == 0) return (false, 0); return (true, a / b); } } /** * @dev Returns the remainder of dividing two unsigned integers, with a division by zero flag. */ function tryMod(uint256 a, uint256 b) internal pure returns (bool, uint256) { unchecked { if (b == 0) return (false, 0); return (true, a % b); } } /** * @dev Returns the largest of two numbers. */ function max(uint256 a, uint256 b) internal pure returns (uint256) { return a > b ? a : b; } /** * @dev Returns the smallest of two numbers. */ function min(uint256 a, uint256 b) internal pure returns (uint256) { return a < b ? a : b; } /** * @dev Returns the average of two numbers. The result is rounded towards * zero. */ function average(uint256 a, uint256 b) internal pure returns (uint256) { // (a + b) / 2 can overflow. return (a & b) + (a ^ b) / 2; } /** * @dev Returns the ceiling of the division of two numbers. * * This differs from standard division with `/` in that it rounds towards infinity instead * of rounding towards zero. */ function ceilDiv(uint256 a, uint256 b) internal pure returns (uint256) { if (b == 0) { // Guarantee the same behavior as in a regular Solidity division. return a / b; } // (a + b - 1) / b can overflow on addition, so we distribute. return a == 0 ? 0 : (a - 1) / b + 1; } /** * @notice Calculates floor(x * y / denominator) with full precision. Throws if result overflows a uint256 or * denominator == 0. * @dev Original credit to Remco Bloemen under MIT license (https://xn--2-umb.com/21/muldiv) with further edits by * Uniswap Labs also under MIT license. */ function mulDiv(uint256 x, uint256 y, uint256 denominator) internal pure returns (uint256 result) { unchecked { // 512-bit multiply [prod1 prod0] = x * y. Compute the product mod 2^256 and mod 2^256 - 1, then use // use the Chinese Remainder Theorem to reconstruct the 512 bit result. The result is stored in two 256 // variables such that product = prod1 * 2^256 + prod0. uint256 prod0 = x * y; // Least significant 256 bits of the product uint256 prod1; // Most significant 256 bits of the product assembly { let mm := mulmod(x, y, not(0)) prod1 := sub(sub(mm, prod0), lt(mm, prod0)) } // Handle non-overflow cases, 256 by 256 division. if (prod1 == 0) { // Solidity will revert if denominator == 0, unlike the div opcode on its own. // The surrounding unchecked block does not change this fact. // See https://docs.soliditylang.org/en/latest/control-structures.html#checked-or-unchecked-arithmetic. return prod0 / denominator; } // Make sure the result is less than 2^256. Also prevents denominator == 0. if (denominator <= prod1) { revert MathOverflowedMulDiv(); } /////////////////////////////////////////////// // 512 by 256 division. /////////////////////////////////////////////// // Make division exact by subtracting the remainder from [prod1 prod0]. uint256 remainder; assembly { // Compute remainder using mulmod. remainder := mulmod(x, y, denominator) // Subtract 256 bit number from 512 bit number. prod1 := sub(prod1, gt(remainder, prod0)) prod0 := sub(prod0, remainder) } // Factor powers of two out of denominator and compute largest power of two divisor of denominator. // Always >= 1. See https://cs.stackexchange.com/q/138556/92363. uint256 twos = denominator & (0 - denominator); assembly { // Divide denominator by twos. denominator := div(denominator, twos) // Divide [prod1 prod0] by twos. prod0 := div(prod0, twos) // Flip twos such that it is 2^256 / twos. If twos is zero, then it becomes one. twos := add(div(sub(0, twos), twos), 1) } // Shift in bits from prod1 into prod0. prod0 |= prod1 * twos; // Invert denominator mod 2^256. Now that denominator is an odd number, it has an inverse modulo 2^256 such // that denominator * inv = 1 mod 2^256. Compute the inverse by starting with a seed that is correct for // four bits. That is, denominator * inv = 1 mod 2^4. uint256 inverse = (3 * denominator) ^ 2; // Use the Newton-Raphson iteration to improve the precision. Thanks to Hensel's lifting lemma, this also // works in modular arithmetic, doubling the correct bits in each step. inverse *= 2 - denominator * inverse; // inverse mod 2^8 inverse *= 2 - denominator * inverse; // inverse mod 2^16 inverse *= 2 - denominator * inverse; // inverse mod 2^32 inverse *= 2 - denominator * inverse; // inverse mod 2^64 inverse *= 2 - denominator * inverse; // inverse mod 2^128 inverse *= 2 - denominator * inverse; // inverse mod 2^256 // Because the division is now exact we can divide by multiplying with the modular inverse of denominator. // This will give us the correct result modulo 2^256. Since the preconditions guarantee that the outcome is // less than 2^256, this is the final result. We don't need to compute the high bits of the result and prod1 // is no longer required. result = prod0 * inverse; return result; } } /** * @notice Calculates x * y / denominator with full precision, following the selected rounding direction. */ function mulDiv(uint256 x, uint256 y, uint256 denominator, Rounding rounding) internal pure returns (uint256) { uint256 result = mulDiv(x, y, denominator); if (unsignedRoundsUp(rounding) && mulmod(x, y, denominator) > 0) { result += 1; } return result; } /** * @dev Returns the square root of a number. If the number is not a perfect square, the value is rounded * towards zero. * * Inspired by Henry S. Warren, Jr.'s "Hacker's Delight" (Chapter 11). */ function sqrt(uint256 a) internal pure returns (uint256) { if (a == 0) { return 0; } // For our first guess, we get the biggest power of 2 which is smaller than the square root of the target. // // We know that the "msb" (most significant bit) of our target number `a` is a power of 2 such that we have // `msb(a) <= a < 2*msb(a)`. This value can be written `msb(a)=2**k` with `k=log2(a)`. // // This can be rewritten `2**log2(a) <= a < 2**(log2(a) + 1)` // → `sqrt(2**k) <= sqrt(a) < sqrt(2**(k+1))` // → `2**(k/2) <= sqrt(a) < 2**((k+1)/2) <= 2**(k/2 + 1)` // // Consequently, `2**(log2(a) / 2)` is a good first approximation of `sqrt(a)` with at least 1 correct bit. uint256 result = 1 << (log2(a) >> 1); // At this point `result` is an estimation with one bit of precision. We know the true value is a uint128, // since it is the square root of a uint256. Newton's method converges quadratically (precision doubles at // every iteration). We thus need at most 7 iteration to turn our partial result with one bit of precision // into the expected uint128 result. unchecked { result = (result + a / result) >> 1; result = (result + a / result) >> 1; result = (result + a / result) >> 1; result = (result + a / result) >> 1; result = (result + a / result) >> 1; result = (result + a / result) >> 1; result = (result + a / result) >> 1; return min(result, a / result); } } /** * @notice Calculates sqrt(a), following the selected rounding direction. */ function sqrt(uint256 a, Rounding rounding) internal pure returns (uint256) { unchecked { uint256 result = sqrt(a); return result + (unsignedRoundsUp(rounding) && result * result < a ? 1 : 0); } } /** * @dev Return the log in base 2 of a positive value rounded towards zero. * Returns 0 if given 0. */ function log2(uint256 value) internal pure returns (uint256) { uint256 result = 0; unchecked { if (value >> 128 > 0) { value >>= 128; result += 128; } if (value >> 64 > 0) { value >>= 64; result += 64; } if (value >> 32 > 0) { value >>= 32; result += 32; } if (value >> 16 > 0) { value >>= 16; result += 16; } if (value >> 8 > 0) { value >>= 8; result += 8; } if (value >> 4 > 0) { value >>= 4; result += 4; } if (value >> 2 > 0) { value >>= 2; result += 2; } if (value >> 1 > 0) { result += 1; } } return result; } /** * @dev Return the log in base 2, following the selected rounding direction, of a positive value. * Returns 0 if given 0. */ function log2(uint256 value, Rounding rounding) internal pure returns (uint256) { unchecked { uint256 result = log2(value); return result + (unsignedRoundsUp(rounding) && 1 << result < value ? 1 : 0); } } /** * @dev Return the log in base 10 of a positive value rounded towards zero. * Returns 0 if given 0. */ function log10(uint256 value) internal pure returns (uint256) { uint256 result = 0; unchecked { if (value >= 10 ** 64) { value /= 10 ** 64; result += 64; } if (value >= 10 ** 32) { value /= 10 ** 32; result += 32; } if (value >= 10 ** 16) { value /= 10 ** 16; result += 16; } if (value >= 10 ** 8) { value /= 10 ** 8; result += 8; } if (value >= 10 ** 4) { value /= 10 ** 4; result += 4; } if (value >= 10 ** 2) { value /= 10 ** 2; result += 2; } if (value >= 10 ** 1) { result += 1; } } return result; } /** * @dev Return the log in base 10, following the selected rounding direction, of a positive value. * Returns 0 if given 0. */ function log10(uint256 value, Rounding rounding) internal pure returns (uint256) { unchecked { uint256 result = log10(value); return result + (unsignedRoundsUp(rounding) && 10 ** result < value ? 1 : 0); } } /** * @dev Return the log in base 256 of a positive value rounded towards zero. * Returns 0 if given 0. * * Adding one to the result gives the number of pairs of hex symbols needed to represent `value` as a hex string. */ function log256(uint256 value) internal pure returns (uint256) { uint256 result = 0; unchecked { if (value >> 128 > 0) { value >>= 128; result += 16; } if (value >> 64 > 0) { value >>= 64; result += 8; } if (value >> 32 > 0) { value >>= 32; result += 4; } if (value >> 16 > 0) { value >>= 16; result += 2; } if (value >> 8 > 0) { result += 1; } } return result; } /** * @dev Return the log in base 256, following the selected rounding direction, of a positive value. * Returns 0 if given 0. */ function log256(uint256 value, Rounding rounding) internal pure returns (uint256) { unchecked { uint256 result = log256(value); return result + (unsignedRoundsUp(rounding) && 1 << (result << 3) < value ? 1 : 0); } } /** * @dev Returns whether a provided rounding mode is considered rounding up for unsigned integers. */ function unsignedRoundsUp(Rounding rounding) internal pure returns (bool) { return uint8(rounding) % 2 == 1; } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v5.0.0) (token/ERC20/IERC20.sol) pragma solidity ^0.8.20; /** * @dev Interface of the ERC20 standard as defined in the EIP. */ interface IERC20 { /** * @dev Emitted when `value` tokens are moved from one account (`from`) to * another (`to`). * * Note that `value` may be zero. */ event Transfer(address indexed from, address indexed to, uint256 value); /** * @dev Emitted when the allowance of a `spender` for an `owner` is set by * a call to {approve}. `value` is the new allowance. */ event Approval(address indexed owner, address indexed spender, uint256 value); /** * @dev Returns the value of tokens in existence. */ function totalSupply() external view returns (uint256); /** * @dev Returns the value of tokens owned by `account`. */ function balanceOf(address account) external view returns (uint256); /** * @dev Moves a `value` amount of tokens from the caller's account to `to`. * * Returns a boolean value indicating whether the operation succeeded. * * Emits a {Transfer} event. */ function transfer(address to, uint256 value) external returns (bool); /** * @dev Returns the remaining number of tokens that `spender` will be * allowed to spend on behalf of `owner` through {transferFrom}. This is * zero by default. * * This value changes when {approve} or {transferFrom} are called. */ function allowance(address owner, address spender) external view returns (uint256); /** * @dev Sets a `value` amount of tokens as the allowance of `spender` over the * caller's tokens. * * Returns a boolean value indicating whether the operation succeeded. * * IMPORTANT: Beware that changing an allowance with this method brings the risk * that someone may use both the old and the new allowance by unfortunate * transaction ordering. One possible solution to mitigate this race * condition is to first reduce the spender's allowance to 0 and set the * desired value afterwards: * https://github.com/ethereum/EIPs/issues/20#issuecomment-263524729 * * Emits an {Approval} event. */ function approve(address spender, uint256 value) external returns (bool); /** * @dev Moves a `value` amount of tokens from `from` to `to` using the * allowance mechanism. `value` is then deducted from the caller's * allowance. * * Returns a boolean value indicating whether the operation succeeded. * * Emits a {Transfer} event. */ function transferFrom(address from, address to, uint256 value) external returns (bool); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v5.0.0) (token/ERC20/extensions/IERC20Permit.sol) pragma solidity ^0.8.20; /** * @dev Interface of the ERC20 Permit extension allowing approvals to be made via signatures, as defined in * https://eips.ethereum.org/EIPS/eip-2612[EIP-2612]. * * Adds the {permit} method, which can be used to change an account's ERC20 allowance (see {IERC20-allowance}) by * presenting a message signed by the account. By not relying on {IERC20-approve}, the token holder account doesn't * need to send a transaction, and thus is not required to hold Ether at all. * * ==== Security Considerations * * There are two important considerations concerning the use of `permit`. The first is that a valid permit signature * expresses an allowance, and it should not be assumed to convey additional meaning. In particular, it should not be * considered as an intention to spend the allowance in any specific way. The second is that because permits have * built-in replay protection and can be submitted by anyone, they can be frontrun. A protocol that uses permits should * take this into consideration and allow a `permit` call to fail. Combining these two aspects, a pattern that may be * generally recommended is: * * ```solidity * function doThingWithPermit(..., uint256 value, uint256 deadline, uint8 v, bytes32 r, bytes32 s) public { * try token.permit(msg.sender, address(this), value, deadline, v, r, s) {} catch {} * doThing(..., value); * } * * function doThing(..., uint256 value) public { * token.safeTransferFrom(msg.sender, address(this), value); * ... * } * ``` * * Observe that: 1) `msg.sender` is used as the owner, leaving no ambiguity as to the signer intent, and 2) the use of * `try/catch` allows the permit to fail and makes the code tolerant to frontrunning. (See also * {SafeERC20-safeTransferFrom}). * * Additionally, note that smart contract wallets (such as Argent or Safe) are not able to produce permit signatures, so * contracts should have entry points that don't rely on permit. */ interface IERC20Permit { /** * @dev Sets `value` as the allowance of `spender` over ``owner``'s tokens, * given ``owner``'s signed approval. * * IMPORTANT: The same issues {IERC20-approve} has related to transaction * ordering also apply here. * * Emits an {Approval} event. * * Requirements: * * - `spender` cannot be the zero address. * - `deadline` must be a timestamp in the future. * - `v`, `r` and `s` must be a valid `secp256k1` signature from `owner` * over the EIP712-formatted function arguments. * - the signature must use ``owner``'s current nonce (see {nonces}). * * For more information on the signature format, see the * https://eips.ethereum.org/EIPS/eip-2612#specification[relevant EIP * section]. * * CAUTION: See Security Considerations above. */ function permit( address owner, address spender, uint256 value, uint256 deadline, uint8 v, bytes32 r, bytes32 s ) external; /** * @dev Returns the current nonce for `owner`. This value must be * included whenever a signature is generated for {permit}. * * Every successful call to {permit} increases ``owner``'s nonce by one. This * prevents a signature from being used multiple times. */ function nonces(address owner) external view returns (uint256); /** * @dev Returns the domain separator used in the encoding of the signature for {permit}, as defined by {EIP712}. */ // solhint-disable-next-line func-name-mixedcase function DOMAIN_SEPARATOR() external view returns (bytes32); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v5.0.0) (token/ERC20/extensions/IERC20Metadata.sol) pragma solidity ^0.8.20; import {IERC20} from "../IERC20.sol"; /** * @dev Interface for the optional metadata functions from the ERC20 standard. */ interface IERC20Metadata is IERC20 { /** * @dev Returns the name of the token. */ function name() external view returns (string memory); /** * @dev Returns the symbol of the token. */ function symbol() external view returns (string memory); /** * @dev Returns the decimals places of the token. */ function decimals() external view returns (uint8); }
{ "optimizer": { "enabled": true, "mode": "3" }, "viaIR": true, "evmVersion": "paris", "outputSelection": { "*": { "*": [ "abi", "metadata" ], "": [ "ast" ] } }, "detectMissingLibraries": false, "forceEVMLA": false, "enableEraVMExtensions": false, "libraries": {} }
[{"inputs":[{"internalType":"string","name":"_name","type":"string"},{"internalType":"string","name":"_symbol","type":"string"},{"internalType":"string","name":"_contractURI","type":"string"},{"internalType":"uint256","name":"_maxSupply","type":"uint256"},{"internalType":"uint256","name":"_publicPrice","type":"uint256"},{"internalType":"string","name":"_defaultBaseURI","type":"string"},{"internalType":"string","name":"_notRevealedURI","type":"string"},{"internalType":"address payable","name":"_withdrawalRecipientAddress","type":"address"},{"internalType":"address payable","name":"_commissionRecipientAddress","type":"address"},{"internalType":"uint256","name":"_fixedCommisionThreshold","type":"uint256"},{"internalType":"uint256","name":"_commissionPercentageIn10000","type":"uint256"},{"internalType":"address payable","name":"_defaultRoyaltyRecipient","type":"address"},{"internalType":"uint256","name":"_defaultRoyaltyPercentageIn10000","type":"uint256"}],"stateMutability":"nonpayable","type":"constructor"},{"inputs":[{"internalType":"address","name":"target","type":"address"}],"name":"AddressEmptyCode","type":"error"},{"inputs":[{"internalType":"address","name":"account","type":"address"}],"name":"AddressInsufficientBalance","type":"error"},{"inputs":[],"name":"ApprovalCallerNotOwnerNorApproved","type":"error"},{"inputs":[],"name":"ApprovalQueryForNonexistentToken","type":"error"},{"inputs":[],"name":"BalanceQueryForZeroAddress","type":"error"},{"inputs":[],"name":"FailedInnerCall","type":"error"},{"inputs":[],"name":"MintERC2309QuantityExceedsLimit","type":"error"},{"inputs":[],"name":"MintToZeroAddress","type":"error"},{"inputs":[],"name":"MintZeroQuantity","type":"error"},{"inputs":[],"name":"NotCompatibleWithSpotMints","type":"error"},{"inputs":[{"internalType":"address","name":"owner","type":"address"}],"name":"OwnableInvalidOwner","type":"error"},{"inputs":[{"internalType":"address","name":"account","type":"address"}],"name":"OwnableUnauthorizedAccount","type":"error"},{"inputs":[],"name":"OwnerQueryForNonexistentToken","type":"error"},{"inputs":[],"name":"OwnershipNotInitializedForExtraData","type":"error"},{"inputs":[{"internalType":"address","name":"token","type":"address"}],"name":"SafeERC20FailedOperation","type":"error"},{"inputs":[],"name":"SequentialMintExceedsLimit","type":"error"},{"inputs":[],"name":"SequentialUpToTooSmall","type":"error"},{"inputs":[],"name":"SpotMintTokenIdTooSmall","type":"error"},{"inputs":[],"name":"TokenAlreadyExists","type":"error"},{"inputs":[],"name":"TransferCallerNotOwnerNorApproved","type":"error"},{"inputs":[],"name":"TransferFromIncorrectOwner","type":"error"},{"inputs":[],"name":"TransferToNonERC721ReceiverImplementer","type":"error"},{"inputs":[],"name":"TransferToZeroAddress","type":"error"},{"inputs":[],"name":"URIQueryForNonexistentToken","type":"error"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"owner","type":"address"},{"indexed":true,"internalType":"address","name":"approved","type":"address"},{"indexed":true,"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"Approval","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"owner","type":"address"},{"indexed":true,"internalType":"address","name":"operator","type":"address"},{"indexed":false,"internalType":"bool","name":"approved","type":"bool"}],"name":"ApprovalForAll","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"uint256","name":"fromTokenId","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"toTokenId","type":"uint256"},{"indexed":true,"internalType":"address","name":"from","type":"address"},{"indexed":true,"internalType":"address","name":"to","type":"address"}],"name":"ConsecutiveTransfer","type":"event"},{"anonymous":false,"inputs":[],"name":"ContractURIUpdated","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"previousOwner","type":"address"},{"indexed":true,"internalType":"address","name":"newOwner","type":"address"}],"name":"OwnershipTransferred","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"from","type":"address"},{"indexed":true,"internalType":"address","name":"to","type":"address"},{"indexed":true,"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"Transfer","type":"event"},{"inputs":[],"name":"ERC20DiscountIn10000","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"ERC20FixedPricePerToken","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"ERC20PriceFeedAddress","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"ERC20TokenAddress","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_address","type":"address"}],"name":"addToBlacklist","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_to","type":"address"},{"internalType":"uint256","name":"_mintAmount","type":"uint256"}],"name":"adminMint","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"approve","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"}],"name":"balanceOf","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address[]","name":"recipients","type":"address[]"},{"internalType":"uint256[]","name":"amounts","type":"uint256[]"}],"name":"batchAdminMint","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"blacklist","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"commissionPercentageIn10000","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"commissionRecipientAddress","outputs":[{"internalType":"address payable","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"contractURI","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"defaultRoyaltyPercentageIn10000","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"defaultRoyaltyRecipient","outputs":[{"internalType":"address payable","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tierId","type":"uint256"}],"name":"disableTier","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"emergencyWithdraw","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"contract ERC20","name":"erc20Token","type":"address"}],"name":"emergencyWithdrawERC20","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"tierId","type":"uint256"}],"name":"enableTier","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"ethPriceFeedAddress","outputs":[{"internalType":"address","name":"","type":"address"}],"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":"getLaunchpadDetails","outputs":[{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getLaunchpadDetailsERC20","outputs":[{"internalType":"address","name":"","type":"address"},{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tierId","type":"uint256"},{"internalType":"address","name":"user","type":"address"},{"internalType":"bytes32[]","name":"proof","type":"bytes32[]"}],"name":"getMintEligibility","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getPublicMintEligibility","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"ethPrice","type":"uint256"}],"name":"getRequiredERC20TokensChainlink","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tierId","type":"uint256"}],"name":"getTierDetails","outputs":[{"internalType":"bytes32","name":"merkleRoot","type":"bytes32"},{"internalType":"uint256","name":"price","type":"uint256"},{"internalType":"uint256","name":"maxMintAmount","type":"uint256"},{"internalType":"uint256","name":"saleStartTime","type":"uint256"},{"internalType":"string","name":"title","type":"string"},{"internalType":"uint256","name":"ERC20Price","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getTierIds","outputs":[{"internalType":"uint256[]","name":"","type":"uint256[]"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"},{"internalType":"address","name":"operator","type":"address"}],"name":"isApprovedForAll","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"isRevealed","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"maxSupply","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_mintAmount","type":"uint256"}],"name":"mint","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_mintAmount","type":"uint256"}],"name":"mintWithERC20ChainlinkPrice","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_mintAmount","type":"uint256"}],"name":"mintWithFixedERC20Price","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"name","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"notRevealedURI","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"ownerOf","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"publicMaxMintAmount","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"publicPrice","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"publicSaleStartTime","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_address","type":"address"}],"name":"removeFromBlacklist","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_tokenId","type":"uint256"},{"internalType":"uint256","name":"_salePrice","type":"uint256"}],"name":"royaltyInfo","outputs":[{"internalType":"address","name":"","type":"address"},{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"from","type":"address"},{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"safeTransferFrom","outputs":[],"stateMutability":"payable","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":"payable","type":"function"},{"inputs":[{"internalType":"address","name":"operator","type":"address"},{"internalType":"bool","name":"approved","type":"bool"}],"name":"setApprovalForAll","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"string","name":"_newBaseURI","type":"string"}],"name":"setBaseURI","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"string","name":"newURI","type":"string"}],"name":"setContractURI","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address payable","name":"_defaultRoyaltyRecipient","type":"address"},{"internalType":"uint256","name":"_defaultRoyaltyPercentageIn10000","type":"uint256"}],"name":"setDefaultRoyaltyInfo","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_erc20DiscountIn10000","type":"uint256"}],"name":"setERC20DiscountIn10000","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_ERC20PriceFeedAddress","type":"address"},{"internalType":"uint256","name":"_maxStaleness","type":"uint256"}],"name":"setERC20PriceFeedAddress","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_ERC20TokenAddress","type":"address"}],"name":"setERC20TokenAddress","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_ethPriceFeedAddress","type":"address"},{"internalType":"uint256","name":"_maxStaleness","type":"uint256"}],"name":"setETHPriceFeedAddress","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_erc20FixedPricePerToken","type":"uint256"}],"name":"setErc20FixedPricePerToken","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_newMaxSupply","type":"uint256"}],"name":"setMaxSupply","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"string","name":"_notRevealedURI","type":"string"}],"name":"setNotRevealedURI","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_publicMaxMintAmount","type":"uint256"}],"name":"setPublicMaxMintAmount","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_newPrice","type":"uint256"}],"name":"setPublicPrice","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_publicSaleStartTime","type":"uint256"}],"name":"setPublicSaleStartTime","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"tierId","type":"uint256"},{"internalType":"string","name":"title","type":"string"},{"internalType":"bytes32","name":"merkleRoot","type":"bytes32"},{"internalType":"uint256","name":"price","type":"uint256"},{"internalType":"uint256","name":"erc20Price","type":"uint256"},{"internalType":"uint256","name":"maxMintAmount","type":"uint256"},{"internalType":"uint256","name":"saleStartTime","type":"uint256"}],"name":"setTier","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"tierId","type":"uint256"},{"internalType":"uint256","name":"maxMintAmount","type":"uint256"}],"name":"setTierMaxMintAmount","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"tierId","type":"uint256"},{"internalType":"bytes32","name":"merkleRoot","type":"bytes32"}],"name":"setTierMerkleRoot","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"tierId","type":"uint256"},{"internalType":"uint256","name":"price","type":"uint256"},{"internalType":"uint256","name":"erc20Price","type":"uint256"}],"name":"setTierPrice","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"tierId","type":"uint256"},{"internalType":"uint256","name":"saleStartTime","type":"uint256"}],"name":"setTierSaleStartTime","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_tokenId","type":"uint256"},{"internalType":"address payable","name":"_royaltyRecipient","type":"address"},{"internalType":"uint256","name":"_royaltyPercentage","type":"uint256"}],"name":"setTokenRoyaltyInfo","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bool","name":"_tradingEnabled","type":"bool"}],"name":"setTradingEnabled","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":"","type":"uint256"}],"name":"tierIds","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"tiers","outputs":[{"internalType":"string","name":"title","type":"string"},{"internalType":"bytes32","name":"merkleRoot","type":"bytes32"},{"internalType":"uint256","name":"price","type":"uint256"},{"internalType":"uint256","name":"erc20Price","type":"uint256"},{"internalType":"uint128","name":"maxMintAmount","type":"uint128"},{"internalType":"uint128","name":"saleStartTime","type":"uint128"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"togglePublicSaleActive","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"toggleReveal","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"tokenRoyaltyPercentage","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"tokenRoyaltyRecipient","outputs":[{"internalType":"address payable","name":"","type":"address"}],"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":"result","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"tradingEnabled","outputs":[{"internalType":"bool","name":"","type":"bool"}],"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":"payable","type":"function"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"tierId","type":"uint256"},{"internalType":"uint256","name":"amount","type":"uint256"},{"internalType":"bytes32[]","name":"proof","type":"bytes32[]"}],"name":"whitelistMint","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"uint256","name":"tierId","type":"uint256"},{"internalType":"uint256","name":"amount","type":"uint256"},{"internalType":"bytes32[]","name":"proof","type":"bytes32[]"}],"name":"whitelistMintWithERC20ChainlinkPrice","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"tierId","type":"uint256"},{"internalType":"uint256","name":"amount","type":"uint256"},{"internalType":"bytes32[]","name":"proof","type":"bytes32[]"}],"name":"whitelistMintWithFixedERC20Price","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"withdraw","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"contract ERC20","name":"erc20Token","type":"address"}],"name":"withdrawERC20","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"withdrawFixedCommission","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"withdrawalRecipientAddress","outputs":[{"internalType":"address payable","name":"","type":"address"}],"stateMutability":"view","type":"function"}]
Contract Creation Code
9c4d535b000000000000000000000000000000000000000000000000000000000000000001000b135bc90c09d5f1a3bb683a59393768bcd64ca5213643d2c26881715331000000000000000000000000000000000000000000000000000000000000006000000000000000000000000000000000000000000000000000000000000002c000000000000000000000000000000000000000000000000000000000000001a000000000000000000000000000000000000000000000000000000000000001e00000000000000000000000000000000000000000000000000000000000000220000000000000000000000000000000000000000000000000000000000000270f000000000000000000000000000000000000000000000000008e1bc9bf040000000000000000000000000000000000000000000000000000000000000000024000000000000000000000000000000000000000000000000000000000000002a0000000000000000000000000ff383ed09ae2d216bd37b03797dd9a3a0f75c77a0000000000000000000000008f995e8961d2ff09d444ab4ec72d67f36aa2c8cc00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000064000000000000000000000000aa4306c1b90782ce2710d9512becd03168eaf7a200000000000000000000000000000000000000000000000000000000000001f400000000000000000000000000000000000000000000000000000000000000055a65656b7300000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000055a45454b5300000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000036697066733a2f2f516d61443766374c3152505679324d726b61483662796653424141554232373172724175357a525a4e6b6d547a372f000000000000000000000000000000000000000000000000000000000000000000000000000000000000
Deployed Bytecode
0x00040000000000020018000000000002000000600410027000000a0c034001970003000000310355000200000001035500000a0c0040019d0000000100200190000000720000c13d000000800b0000390000004000b0043f000000040030008c000000930000413d000000000201043b000000e00220027000000a2c0020009c000000ae0000a13d00000a2d0020009c000000c10000213d00000a4b0020009c000001710000213d00000a5a0020009c0000035b0000a13d00000a5b0020009c000004290000213d00000a5f0020009c000009ab0000613d00000a600020009c00000c270000613d00000a610020009c000000930000c13d000000240030008c000000930000413d0000000002000416000000000002004b000000930000c13d0000000401100370000000000101043b000000000010043f0000002001000039000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f0000000100200190000000930000613d000000000401043b0000001b01000039000000000301041a00000a11023001980000003b0000613d0000001a01000039000000000101041a00000a1105100198000011650000c13d000000400500043d0000000301400039000000000601041a000000000104041a000000010210019000000001071002700000007f0770618f0000001f0070008c00000000030000390000000103002039000000000331013f000000010030019000000b2a0000c13d001200000006001d0000000403400039000000000303041a000f00000003001d0000000203400039000000000303041a001100000003001d0000000103400039000000000303041a001000000003001d0000000006750436000000000002004b000011cc0000613d000c00000007001d000d00000006001d000e00000005001d000000000040043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000ad2011001c70000801002000039282d28280000040f0000000100200190000000930000613d0000000c07000029000000000007004b00000000020000190000000e050000290000000d06000029000011d10000613d000000000101043b00000000020000190000000003260019000000000401041a000000000043043500000001011000390000002002200039000000000072004b0000006a0000413d000011d10000013d0000012004000039000000400040043f0000000002000416000000000002004b000000930000c13d0000001f0230003900000a0d022001970000012002200039000000400020043f0000001f0530018f00000a0e063001980000012002600039000000840000613d000000000701034f000000007807043c0000000004840436000000000024004b000000800000c13d000000000005004b000000910000613d000000000161034f0000000304500210000000000502043300000000054501cf000000000545022f000000000101043b0000010004400089000000000141022f00000000014101cf000000000151019f0000000000120435000001a00030008c000000950000813d00000000010000190000282f00010430000001200400043d00000a0f0040009c000000930000213d0000001f01400039000000000031004b000000000200001900000a100200804100000a1001100197000000000001004b000000000500001900000a100500404100000a100010009c000000000502c019000000000005004b000000930000c13d0000012001400039000000000201043300000a0f0020009c000002130000a13d00000acc01000041000000000010043f0000004101000039000000040010043f00000a2b010000410000282f0001043000000a680020009c000000ea0000a13d00000a690020009c000001f10000213d00000a780020009c000004070000a13d00000a790020009c000005270000213d00000a7d0020009c00000ba20000613d00000a7e0020009c00000dbd0000613d00000a7f0020009c000000930000c13d0000000001000416000000000001004b000000930000c13d00000010010000390000050a0000013d00000a2e0020009c0000018d0000213d00000a3d0020009c000003680000a13d00000a3e0020009c000004720000213d00000a420020009c00000a840000613d00000a430020009c00000c8c0000613d00000a440020009c000000930000c13d000000440030008c000000930000413d0000000002000416000000000002004b000000930000c13d0000000401100370000000000201043b00000a110020009c000000930000213d000000000100041a00000a11031001970000000001000411000000000013004b00000db80000c13d0000001a03000039000000000103041a00000a1201100197000000000121019f000000000013041b00000ab201000041000000800010043f0000000001000414000000040020008c000011790000c13d0000000103000031000000200030008c000000200400003900000000040340190000119e0000013d00000a860020009c0000032d0000a13d00000a870020009c000003d10000a13d00000a880020009c000005110000213d00000a8c0020009c00000b980000613d00000a8d0020009c00000d800000613d00000a8e0020009c000000930000c13d0000000001000416000000000001004b000000930000c13d000000000100041a00120a110010019b0000000001000411000000120010006c00000fb60000c13d0000001301000039000000000101041a001000000001001d0000001201000039000000000101041a001100000001001d00000aad01000041000000000010044300000000010004100000000400100443000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000aae011001c70000800a02000039282d28280000040f000000010020019000001c3f0000613d0000001102000039000000000202041a000f00000002001d000000000101043b000e00000001001d00000aa50100004100000000001004430000000001000412000000040010044300000060010000390000002400100443000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000aa6011001c70000800502000039282d28280000040f000000010020019000001c3f0000613d000000000101043b0000000f0110006c00000db20000413d0000000e0110006b00000db20000413d000f00110010007400000db20000413d0000000f02000029000e00100020007400000db20000413d00000aa50100004100000000001004430000000001000412000000040010044300000040010000390000002400100443000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000aa6011001c70000800502000039282d28280000040f000000010020019000001c3f0000613d000000000201043b0000000e012000b90000000f04000029000000100040006c000001460000613d0000000e031000fa000000000023004b00000db20000c13d000027100110011a000d00000001001d000f000e0010007300000db20000413d00000aa50100004100000000001004430000000001000412000000040010044300000020010000390000002400100443000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000aa6011001c70000800502000039282d28280000040f000000010020019000001c3f0000613d000000000101043b00000a11011001970000000002000411000000000012004b0000174c0000c13d0000001301000039000000000101041a0000000f0010002a00000db20000413d0000000f011000290000001302000039000000000012041b0000000d02000029000000110020002a00000db20000413d0000001201000039000000000001041b00000000010004140000000002000411000000040020008c0000177f0000c13d00000001020000390000000101000031000017fa0000013d00000a4c0020009c000003750000a13d00000a4d0020009c000004970000213d00000a510020009c00000a910000613d00000a520020009c00000cca0000613d00000a530020009c000000930000c13d000000240030008c000000930000413d0000000002000416000000000002004b000000930000c13d0000000401100370000000000201043b000000000002004b0000000001000039000000010100c039001200000002001d000000000012004b000000930000c13d282d24c70000040f0000001e01000039000000000201041a00000af602200197000005eb0000013d00000a2f0020009c0000039e0000a13d00000a300020009c000004f50000213d00000a340020009c00000ab70000613d00000a350020009c00000cf20000613d00000a360020009c000000930000c13d0000000001000416000000000001004b000000930000c13d0000000002000412000000000100041a00000a11011001970000000003000411000000000013004b00000f8e0000c13d0000001101000039000000000101041a001200000001001d00000aa5010000410000000000100443000000040020044300000060010000390000002400100443000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000aa6011001c70000800502000039282d28280000040f000000010020019000001c3f0000613d000000000101043b001100120010007400000db20000413d00000aad01000041000000000010044300000000010004100000000400100443000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000aae011001c70000800a02000039282d28280000040f000000010020019000001c3f0000613d000000000101043b0000001102000029000000000012004b000001d30000a13d00000aad01000041000000000010044300000000010004100000000400100443000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000aae011001c70000800a02000039282d28280000040f000000010020019000001c3f0000613d000000000201043b001100000002001d000000120020002a00000db20000413d000000110200002900000012012000290000001102000039000000000012041b00000aa50100004100000000001004430000000001000412000000040010044300000020010000390000002400100443000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000aa6011001c70000800502000039282d28280000040f000000010020019000001c3f0000613d000000000201043b000000000100041400000a1104200197000000040040008c000013d80000c13d00000001020000390000000101000031000015490000013d00000a6a0020009c000004140000a13d00000a6b0020009c000005440000213d00000a6f0020009c00000bab0000613d00000a700020009c00000dd40000613d00000a710020009c000000930000c13d0000000001000416000000000001004b000000930000c13d000000000100041a00000a11021001970000000005000411000000000052004b00000eee0000c13d00000a1201100197000000000010041b000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000a13011001c70000800d02000039000000030300003900000a14040000410000000006000019282d28230000040f0000000100200190000000930000613d00000000010000190000282e0001042e0000001f0120003900000af7011001970000003f0110003900000af701100197000000400600043d0000000005160019001200000006001d000000000065004b0000000001000039000000010100403900000a0f0050009c000000a80000213d0000000100100190000000a80000c13d0000012001300039000000400050043f00000012050000290000000005250436001100000005001d00000140044000390000000005420019000000000015004b000000930000213d000000000002004b0000001108000029000002350000613d000000000500001900000000065800190000000007450019000000000707043300000000007604350000002005500039000000000025004b0000022e0000413d000000120220002900000020022000390000000000020435000001400400043d00000a0f0040009c000000930000213d0000001f02400039000000000032004b000000000500001900000a100500804100000a1002200197000000000002004b000000000600001900000a100600404100000a100020009c000000000605c019000000000006004b000000930000c13d0000012002400039000000000202043300000a0f0020009c000000a80000213d0000001f0520003900000af7055001970000003f0550003900000af705500197000000400600043d0000000005560019001000000006001d000000000065004b0000000006000039000000010600403900000a0f0050009c000000a80000213d0000000100600190000000a80000c13d000000400050043f00000010050000290000000005250436000f00000005001d00000140044000390000000005420019000000000015004b000000930000213d000000000002004b0000000f080000290000026c0000613d000000000500001900000000065800190000000007450019000000000707043300000000007604350000002005500039000000000025004b000002650000413d000000100220002900000020022000390000000000020435000001600400043d00000a0f0040009c000000930000213d0000001f02400039000000000032004b000000000500001900000a100500804100000a1002200197000000000002004b000000000600001900000a100600404100000a100020009c000000000605c019000000000006004b000000930000c13d0000012002400039000000000202043300000a0f0020009c000000a80000213d0000001f0520003900000af7055001970000003f0550003900000af705500197000000400600043d0000000005560019000e00000006001d000000000065004b0000000006000039000000010600403900000a0f0050009c000000a80000213d0000000100600190000000a80000c13d000000400050043f0000000e050000290000000005250436000d00000005001d00000140044000390000000005420019000000000015004b000000930000213d000000000002004b0000000d08000029000002a30000613d000000000500001900000000065800190000000007450019000000000707043300000000007604350000002005500039000000000025004b0000029c0000413d0000000e0220002900000020022000390000000000020435000001c00400043d00000a0f0040009c000000930000213d0000001f02400039000000000032004b000000000500001900000a100500804100000a1002200197000000000002004b000000000600001900000a100600404100000a100020009c000000000605c019000000000006004b000000930000c13d0000012002400039000000000202043300000a0f0020009c000000a80000213d0000001f0520003900000af7055001970000003f0550003900000af705500197000000400600043d0000000005560019000c00000006001d000000000065004b0000000006000039000000010600403900000a0f0050009c000000a80000213d0000000100600190000000a80000c13d000001a00600043d000a00000006001d000001800600043d000900000006001d000000400050043f0000000c050000290000000005250436000b00000005001d00000140044000390000000005420019000000000015004b000000930000213d000000000002004b0000000b08000029000002de0000613d000000000500001900000000065800190000000007450019000000000707043300000000007604350000002005500039000000000025004b000002d70000413d0000000c0220002900000020022000390000000000020435000001e00400043d00000a0f0040009c000000930000213d0000001f02400039000000000032004b000000000300001900000a100300804100000a1002200197000000000002004b000000000500001900000a100500404100000a100020009c000000000503c019000000000005004b000000930000c13d0000012002400039000000000202043300000a0f0020009c000000a80000213d0000001f0320003900000af7033001970000003f0330003900000af703300197000000400500043d0000000003350019000800000005001d000000000053004b0000000005000039000000010500403900000a0f0030009c000000a80000213d0000000100500190000000a80000c13d000000400030043f00000008030000290000000003230436000700000003001d00000140034000390000000004320019000000000014004b000000930000213d000000000002004b0000000706000029000003150000613d000000000100001900000000041600190000000005310019000000000505043300000000005404350000002001100039000000000021004b0000030e0000413d000000080120002900000020011000390000000000010435000002000100043d000600000001001d00000a110010009c000000930000213d000002200100043d000500000001001d00000a110010009c000000930000213d000002800100043d000400000001001d00000a110010009c000000930000213d0000000006000411000000000006004b000018900000c13d000000400100043d00000a2a02000041000000000021043500000004021000390000000000020435000016cd0000013d00000a950020009c000003bb0000213d00000a9c0020009c000005ef0000a13d00000a9d0020009c000006d80000613d00000a9e0020009c000008570000613d00000a9f0020009c000000930000c13d000000240030008c000000930000413d0000000002000416000000000002004b000000930000c13d0000000401100370000000000201043b000000000002004b0000121b0000613d0000000101000039000000000101041a000000000012004b0000121b0000813d001100000002001d001200000002001d000000000020043f0000000501000039000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f0000000100200190000000930000613d000000000101043b000000000101041a000000000001004b0000120f0000c13d0000001202000029000000000002004b000000010220008a000003450000c13d00000db20000013d00000a620020009c000005710000a13d00000a630020009c000006990000613d00000a640020009c0000077d0000613d00000a650020009c000000930000c13d0000000001000416000000000001004b000000930000c13d000000000100041a00000ba70000013d00000a450020009c0000057e0000a13d00000a460020009c0000069e0000613d00000a470020009c0000078b0000613d00000a480020009c000000930000c13d0000000001000416000000000001004b000000930000c13d0000000c0100003900000d8d0000013d00000a540020009c000005cb0000a13d00000a550020009c000006a30000613d00000a560020009c000007ae0000613d00000a570020009c000000930000c13d000000840030008c000000930000413d0000000402100370000000000202043b001200000002001d00000a110020009c000000930000213d0000002402100370000000000202043b001100000002001d00000a110020009c000000930000213d0000006402100370000000000402043b00000a0f0040009c000000930000213d0000002302400039000000000032004b000000930000813d0000000402400039000000000121034f000000000201043b0000002401400039282d1ea40000040f00000044020000390000000202200367000000000302043b000000000401001900000012010000290000001102000029282d22a00000040f00000000010000190000282e0001042e00000a370020009c000005d90000a13d00000a380020009c000006bb0000613d00000a390020009c0000082b0000613d00000a3a0020009c000000930000c13d000000240030008c000000930000413d0000000001000416000000000001004b000000930000c13d282d24c70000040f00000004010000390000000201100367000000000101043b000000000010043f0000002001000039000000200010043f00000040020000390000000001000019282d27f00000040f0000000401100039000000000201041a00000aa4022001c7000000000021041b00000000010000190000282e0001042e00000a960020009c000006010000a13d00000a970020009c000007150000613d00000a980020009c000008760000613d00000a990020009c000000930000c13d000000240030008c000000930000413d0000000002000416000000000002004b000000930000c13d0000000401100370000000000101043b000000000010043f0000001d01000039000000200010043f00000040020000390000000001000019282d27f00000040f00000d8d0000013d00000a8f0020009c0000065f0000a13d00000a900020009c0000071a0000613d00000a910020009c000008910000613d00000a920020009c000000930000c13d000000440030008c000000930000413d0000000002000416000000000002004b000000930000c13d0000000401100370000000000101043b001200000001001d000000000010043f0000001d01000039000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f0000000100200190000000930000613d000000000101043b000000000101041a000000000001004b000003f30000c13d0000001601000039000000000101041a001100000001001d0000001201000029000000000010043f0000001c01000039000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f0000000100200190000000930000613d000000000101043b000000000101041a00000a1100100198000011440000c13d0000001501000039000011520000013d00000a800020009c0000066f0000a13d00000a810020009c0000071f0000613d00000a820020009c000009960000613d00000a830020009c000000930000c13d0000000001000416000000000001004b000000930000c13d0000001e010000390000050a0000013d00000a720020009c0000068b0000a13d00000a730020009c0000072c0000613d00000a740020009c000009a60000613d00000a750020009c000000930000c13d000000240030008c000000930000413d0000000001000416000000000001004b000000930000c13d282d24c70000040f00000004010000390000000201100367000000000101043b0000000f02000039000000000012041b00000000010000190000282e0001042e00000a5c0020009c00000b1c0000613d00000a5d0020009c00000d070000613d00000a5e0020009c000000930000c13d000000440030008c000000930000413d0000000002000416000000000002004b000000930000c13d0000000402100370000000000202043b001200000002001d00000a110020009c000000930000213d0000002401100370000000000201043b000000000002004b0000000001000039000000010100c039001100000002001d000000000012004b000000930000c13d0000000001000411000000000010043f0000000801000039000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f0000000100200190000000930000613d000000000101043b0000001202000029000000000020043f000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f0000000100200190000000930000613d000000000101043b000000000201041a00000af6022001970000001103000029000000000232019f000000000021041b000000400100043d000000000031043500000a0c0010009c00000a0c010080410000004001100210000000000200041400000a0c0020009c00000a0c02008041000000c002200210000000000112019f00000ad2011001c70000800d02000039000000030300003900000adc04000041000000000500041100000012060000290000020e0000013d00000a3f0020009c00000b300000613d00000a400020009c00000d230000613d00000a410020009c000000930000c13d0000000001000416000000000001004b000000930000c13d0000001403000039000000000203041a000000010420019000000001012002700000007f0110618f0000001f0010008c00000000050000390000000105002039000000000552013f000000010050019000000b2a0000c13d000000800010043f000000000004004b00000f780000613d000000000030043f000000000001004b000000000200001900000f7d0000613d00000a1e030000410000000002000019000000000403041a000000a005200039000000000045043500000001033000390000002002200039000000000012004b0000048f0000413d00000f7d0000013d00000a4e0020009c00000b450000613d00000a4f0020009c00000d490000613d00000a500020009c000000930000c13d000000440030008c000000930000413d0000000002000416000000000002004b000000930000c13d0000000402100370000000000202043b00000a0f0020009c000000930000213d0000002304200039000000000034004b000000930000813d0000000404200039000000000441034f000000000404043b001100000004001d00000a0f0040009c000000930000213d001000240020003d000000110200002900000005022002100000001002200029000000000032004b000000930000213d0000002402100370000000000202043b00000a0f0020009c000000930000213d0000002304200039000000000034004b000000930000813d0000000404200039000000000141034f000000000101043b000f00000001001d00000a0f0010009c000000930000213d000e00240020003d0000000f0100002900000005011002100000000e01100029000000000031004b000000930000213d000000000100041a00000a11011001970000000002000411000000000021004b000014fa0000c13d000000110000006b000002110000613d0000000005000019000000050250021000000010012000290000000203000367000000000113034f000000000101043b00000a110010009c000000930000213d0000000f0050006c000016c10000813d0000000e02200029000000000223034f000000000202043b000000000300041a00000a11033001970000000004000411000000000043004b000016c70000c13d0000000203000039000000000303041a00000af8033001670000000104000039000000000404041a0000000003340019000000000023001a00000db20000413d001200000005001d00000000032300190000000c04000039000000000404041a000000000043004b000016d20000213d282d253f0000040f00000012050000290000000105500039000000110050006c000004d00000413d000002110000013d00000a310020009c00000b7a0000613d00000a320020009c00000d560000613d00000a330020009c000000930000c13d000000240030008c000000930000413d0000000002000416000000000002004b000000930000c13d0000000401100370000000000101043b00000a110010009c000000930000213d000000000010043f0000001f01000039000000200010043f00000040020000390000000001000019282d27f00000040f000000000101041a000000ff001001900000000001000039000000010100c039000000800010043f00000aa2010000410000282e0001042e00000a890020009c00000b9d0000613d00000a8a0020009c00000d910000613d00000a8b0020009c000000930000c13d0000000001030019282d1e920000040f001200000001001d001100000002001d001000000003001d000000400100043d000f00000001001d282d1dc70000040f0000000f040000290000000000040435000000120100002900000011020000290000001003000029282d22a00000040f00000000010000190000282e0001042e00000a7a0020009c00000bb80000613d00000a7b0020009c00000de50000613d00000a7c0020009c000000930000c13d000000440030008c000000930000413d0000000002000416000000000002004b000000930000c13d0000000401100370000000000101043b001200000001001d00000a110010009c000000930000213d282d24c70000040f0000001501000039000000000201041a00000a120220019700000012022001af000000000021041b00000024010000390000000201100367000000000101043b0000001602000039000000000012041b00000000010000190000282e0001042e00000a6c0020009c00000c1d0000613d00000a6d0020009c00000df10000613d00000a6e0020009c000000930000c13d000000440030008c000000930000413d0000000002000416000000000002004b000000930000c13d0000000401100370000000000101043b001200000001001d282d24c70000040f0000001201000029000000000010043f0000002001000039000000200010043f00000040020000390000000001000019282d27f00000040f0000000101100039000000000101041a000000000001004b0000000001000039000000010100c039282d20850000040f0000001201000029000000000010043f0000002001000039000000200010043f00000024010000390000000201100367000000000101043b001200000001001d00000000010000190000004002000039282d27f00000040f000000120200002900000080022002100000000401100039000000000301041a00000aa30330019700000d7c0000013d00000a660020009c000007370000613d00000a670020009c000000930000c13d000000240030008c000000930000413d0000000002000416000000000002004b000000930000c13d0000000401100370000000000101043b282d20990000040f00000dde0000013d00000a490020009c000007500000613d00000a4a0020009c000000930000c13d000000640030008c000000930000413d0000000002000416000000000002004b000000930000c13d0000000402100370000000000202043b001100000002001d0000002401100370000000000101043b001000000001001d00000a110010009c000000930000213d000000000100041a00000a11021001970000000001000411000000000012004b00000db80000c13d0000001101000029000000000001004b000007590000613d000000000010043f0000000501000039000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f0000000100200190000000930000613d000000000101043b000000000101041a000000000001004b000005c00000c13d0000000101000039000000000101041a0000001102000029000000000021004b000007590000a13d001200000002001d0000001201000029000000010110008a001200000001001d000000000010043f0000000501000039000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f0000000100200190000000930000613d000000000101043b000000000101041a000000000001004b000005ad0000613d00000abe00100198000007590000c13d00000a1100100198000013ed0000c13d000000400100043d000000440210003900000abf0300004100000000003204350000002402100039000000140300003900000ff50000013d00000a580020009c0000075d0000613d00000a590020009c000000930000c13d0000000001000416000000000001004b000000930000c13d00000080010000390000001b02000039000000000502041a00000a110250019800000ef80000c13d000000000300001900000efd0000013d00000a3b0020009c000007620000613d00000a3c0020009c000000930000c13d000000240030008c000000930000413d0000000002000416000000000002004b000000930000c13d0000000401100370000000000101043b001200000001001d00000a110010009c000000930000213d282d24c70000040f0000001701000039000000000201041a00000a120220019700000012022001af000000000021041b00000000010000190000282e0001042e00000aa00020009c000008300000613d00000aa10020009c000000930000c13d000000240030008c000000930000413d0000000002000416000000000002004b000000930000c13d0000000401100370000000000101043b00000af000100198000000930000c13d00000af10210019700000af20020009c000010140000c13d0000000101000039000010210000013d00000a9a0020009c0000083d0000613d00000a9b0020009c000000930000c13d000000440030008c000000930000413d0000000402100370000000000202043b001100000002001d00000a110020009c000000930000213d0000002401100370000000000101043b000000000001004b000007590000613d001000000001001d000000000010043f0000000501000039000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f0000000100200190000000930000613d000000000101043b000000000101041a000000000001004b0000063a0000c13d0000000101000039000000000101041a0000001002000029000000000021004b000007590000a13d001200000002001d0000001201000029000000010110008a001200000001001d000000000010043f0000000501000039000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f0000000100200190000000930000613d000000000101043b000000000101041a000000000001004b000006270000613d00000abe00100198000007590000c13d00120a110010019b0000000002000411000000120020006c000013890000c13d0000001001000029000000000010043f0000000701000039000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f0000000100200190000000930000613d000000110200002900000a1106200197000000000101043b000000000201041a00000a1202200197000000000262019f000000000021041b000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000a13011001c70000800d02000039000000040300003900000aed04000041000000120500002900000010070000290000020e0000013d00000a930020009c000008520000613d00000a940020009c000000930000c13d0000000001000416000000000001004b000000930000c13d0000000201000039000000000101041a00000af8011001670000000102000039000000000202041a0000000001120019000000800010043f00000aa2010000410000282e0001042e00000a840020009c000008960000613d00000a850020009c000000930000c13d000000240030008c000000930000413d0000000002000416000000000002004b000000930000c13d0000000401100370000000000101043b001200000001001d00000a110010009c000000930000213d282d24c70000040f0000001201000029000000000010043f0000001f01000039000000200010043f00000040020000390000000001000019282d27f00000040f000000000301041a00000af60230019700000001022001bf000000000021041b00000000010000190000282e0001042e00000a760020009c000009880000613d00000a770020009c000000930000c13d000000240030008c000000930000413d0000000002000416000000000002004b000000930000c13d0000000401100370000000000101043b282d24d70000040f00000a110110019700000dde0000013d0000000001000416000000000001004b000000930000c13d0000001b0100003900000ba60000013d0000000001000416000000000001004b000000930000c13d0000000e0100003900000d8d0000013d000000240030008c000000930000413d0000000002000416000000000002004b000000930000c13d0000000401100370000000000101043b000000000200041a00000a11032001970000000002000411000000000023004b00000ef30000c13d000027110010008c000010000000413d00000a2801000041000000800010043f0000002001000039000000840010043f0000001b01000039000000a40010043f00000ad401000041000000c40010043f00000ad5010000410000282f00010430000000440030008c000000930000413d0000000002000416000000000002004b000000930000c13d0000000401100370000000000201043b00000a110020009c000000930000213d000000000100041a00000a11031001970000000001000411000000000013004b00000db80000c13d0000001b03000039000000000103041a00000a1201100197000000000121019f000000000013041b00000ab201000041000000800010043f0000000001000414000000040020008c000010b80000c13d0000000103000031000000200030008c00000020040000390000000004034019000010dd0000013d000000240030008c000000930000413d0000000002000416000000000002004b000000930000c13d0000000401100370000000000101043b000000000010043f0000002001000039000000200010043f00000040020000390000000001000019282d27f00000040f000f00000001001d282d1e1e0000040f0000000f030000290000000102300039000000000202041a001200000002001d0000000202300039000000000202041a001100000002001d0000000302300039000000000202041a001000000002001d0000000402300039000000000202041a000e00000002001d000000c002000039000000400300043d000d00000003001d0000000002230436000f00000002001d000000c002300039282d1e6b0000040f0000000e0500002900000080025002700000000d04000029000000a003400039000000000023043500000aa3025001970000008003400039000000000023043500000060024000390000001003000029000000000032043500000040024000390000001103000029000000000032043500000012020000290000000f030000290000000000230435000000000141004900000a0c0010009c00000a0c01008041000000600110021000000a0c0040009c00000a0c040080410000004002400210000000000121019f0000282e0001042e0000000001000416000000000001004b000000930000c13d000000160100003900000d8d0000013d0000000001000416000000000001004b000000930000c13d000000170100003900000ba60000013d000000240030008c000000930000413d0000000001000416000000000001004b000000930000c13d282d24c70000040f00000004010000390000000201100367000000000101043b0000000e02000039000000000012041b00000000010000190000282e0001042e0000000001000416000000000001004b000000930000c13d0000000001000412001600000001001d001500200000003d000080050100003900000044030000390000000004000415000000160440008a000007870000013d000000240030008c000000930000413d0000000002000416000000000002004b000000930000c13d0000000401100370000000000101043b001200000001001d282d26990000040f0000001701000039000000000101041a00110a110010019c0000000001000039000000010100c039282d20590000040f0000000d01000039000000000101041a0000001202000029282d204b0000040f282d20990000040f0000000004010019000000000200041100000000030004100000001101000029000007a80000013d000000240030008c000000930000413d0000000002000416000000000002004b000000930000c13d0000000401100370000000000101043b000000000001004b00000f0c0000c13d00000aee01000041000000000010043f00000ad9010000410000282f000104300000000001000416000000000001004b000000930000c13d0000000d0100003900000d8d0000013d000000440030008c000000930000413d0000000002000416000000000002004b000000930000c13d0000000402100370000000000202043b00000a110020009c000000930000213d0000002401100370000000000101043b001200000001001d00000a110010009c000000930000213d000000000020043f0000000801000039000000200010043f00000040020000390000000001000019282d27f00000040f0000001202000029282d1edc0000040f000000000101041a000000ff001001900000000001000039000000010100c03900000dde0000013d0000000001000416000000000001004b000000930000c13d0000000001000412001400000001001d001300000000003d000080050100003900000044030000390000000004000415000000140440008a000000050440021000000aa502000041282d28050000040f00000ba70000013d000000240030008c000000930000413d0000000002000416000000000002004b000000930000c13d0000000401100370000000000101043b001200000001001d282d26990000040f0000001801000039000000000101041a001100000001001d0000001701000039000000000101041a00100a110010019c0000000001000039000000010100c039282d20590000040f000000110000006b0000000001000039000000010100c039282d228c0000040f00000011010000290000001202000029282d204b0000040f0000000004010019000000000200041100000000030004100000001001000029282d25240000040f00000000010004110000001202000029282d253f0000040f00000000010000190000282e0001042e000000e40030008c000000930000413d0000000002000416000000000002004b000000930000c13d0000002402100370000000000202043b00000a0f0020009c000000930000213d0000002304200039000000000034004b000000930000813d001100040020003d0000001104100360000000000404043b001200000004001d00000a0f0040009c000000930000213d0000002404200039001000000004001d0000001202400029000000000032004b000000930000213d0000000402100370000000000302043b0000004402100370000000000202043b000f00000002001d0000006402100370000000000202043b000c00000002001d0000008402100370000000000202043b000d00000002001d000000a402100370000000000202043b000e00000002001d000000c401100370000000000101043b000b00000001001d000000000100041a00000a11021001970000000001000411000000000012004b00000db80000c13d000000000030043f0000002001000039000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039000a00000003001d282d28280000040f0000000100200190000000930000613d000000000201043b00000001012000390000000f03000029000000000031041b000f00000002001d000000000102041a000000010010019000000001021002700000007f0220618f000900000002001d0000001f0020008c00000000020000390000000102002039000000000121013f000000010010019000000b2a0000c13d0000000901000029000000200010008c000008170000413d0000000f01000029000000000010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000ad2011001c70000801002000039282d28280000040f0000000100200190000000930000613d00000012030000290000001f023000390000000502200270000000200030008c0000000002004019000000000301043b00000009010000290000001f01100039000000050110027000000000011300190000000002230019000000000012004b000008170000813d000000000002041b0000000102200039000000000012004b000008130000413d0000001201000029000000200010008c0000157d0000413d0000000f01000029000000000010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000ad2011001c70000801002000039282d28280000040f0000000100200190000000930000613d000000200200008a0000001202200180000000000101043b0000165d0000c13d0000000003000019000016680000013d0000000001000416000000000001004b000000930000c13d0000001a0100003900000ba60000013d000000240030008c000000930000413d0000000001000416000000000001004b000000930000c13d282d24c70000040f00000004010000390000000201100367000000000101043b0000001802000039000000000012041b00000000010000190000282e0001042e000000240030008c000000930000413d0000000001000416000000000001004b000000930000c13d282d24c70000040f00000004010000390000000201100367000000000101043b000000000010043f0000002001000039000000200010043f00000040020000390000000001000019282d27f00000040f0000000401100039000000000201041a00000aa302200197000000000021041b00000000010000190000282e0001042e0000000001000416000000000001004b000000930000c13d282d1eec0000040f00000dde0000013d0000000001000416000000000001004b000000930000c13d0000000303000039000000000203041a000000010420019000000001012002700000007f0110618f0000001f0010008c00000000050000390000000105002039000000000552013f000000010050019000000b2a0000c13d000000800010043f000000000004004b00000f780000613d000000000030043f000000000001004b000000000200001900000f7d0000613d00000a16030000410000000002000019000000000403041a000000a005200039000000000045043500000001033000390000002002200039000000000012004b0000086e0000413d00000f7d0000013d0000000001000416000000000001004b000000930000c13d000000000100041a00000a11021001970000000001000411000000000012004b00000db80000c13d0000000f01000039000000000101041a001200000001001d00000a1b010000410000000000100443000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000a1c011001c70000800b02000039282d28280000040f000000010020019000001c3f0000613d000000000101043b000000010200008a000000120010006c0000000001028019000004250000013d0000000001030019282d1e920000040f282d1f090000040f00000000010000190000282e0001042e000000640030008c000000930000413d0000000002000416000000000002004b000000930000c13d0000002402100370000000000202043b000f00000002001d0000000402100370000000000202043b0000004404100370000000000404043b00000a0f0040009c000000930000213d0000002305400039000000000035004b000000930000813d0000000405400039000000000151034f000000000101043b001200000001001d00000a0f0010009c000000930000213d000000240140003900000012040000290000000504400210000d00000001001d001100000004001d000e00000014001d0000000e0030006b000000930000213d000000000020043f0000002001000039000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f0000000100200190000000930000613d000000000201043b0000000101200039000000000101041a000c00000001001d000000000001004b00000b730000613d000a00000002001d0000000401200039000000000101041a000b00000001001d00000a1b010000410000000000100443000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000a1c011001c70000800b02000039282d28280000040f000000010020019000001c3f0000613d000000400200043d0000000b030000290000008003300270000000000101043b000000000031004b000013540000413d0000001401000039000000000112043600000000030004110000006003300210000000000031043500000ace0020009c000000a80000213d0000004003200039000000400030043f00000a0c0010009c00000a0c010080410000004001100210000000000202043300000a0c0020009c00000a0c020080410000006002200210000000000112019f000000000200041400000a0c0020009c00000a0c02008041000000c002200210000000000112019f00000a13011001c70000801002000039282d28280000040f0000000100200190000000930000613d000000000101043b00000011020000290000003f0220003900000acf02200197000000400300043d0000000002230019001100000003001d000000000032004b0000000003000039000000010300403900000a0f0020009c000000a80000213d0000000100300190000000a80000c13d000000400020043f000000120200002900000011040000290000000002240436001000000002001d0000000e02000029000000000020007c000000930000213d000000120000006b0000093a0000613d0000000d05000029000000020250036700000011030000290000000e060000290000002003300039000000002402043c00000000004304350000002005500039000000000065004b000009130000413d00000011020000290000000002020433000000000002004b0000093a0000613d0000000003000019001200000003001d000000050230021000000010022000290000000002020433000000000021004b000009280000813d000000000010043f000000200020043f00000000010004140000092b0000013d000000000020043f000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f0000000100200190000000930000613d000000000101043b0000001203000029000000010330003900000011020000290000000002020433000000000023004b0000091e0000413d0000000c0010006c000017420000c13d000000000100041100000a1101100197001200000001001d000000000010043f0000000a010000290000000501100039001100000001001d000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f0000000100200190000000930000613d0000000b0200002900000aa302200197000000000101043b000000000101041a000000000112004b00000db20000413d0000000f0010006c000014e50000413d0000000201000039000000000101041a00000af8011001670000000102000039001000000002001d000000000202041a00000000011200190000000f0010002a00000db20000413d0000000f011000290000000c02000039000000000202041a000000000021004b0000186d0000213d0000001701000039000000000101041a00000a110010019800000a7d0000613d0000000a010000290000000201100039000000000101041a000d000f001000bd000000000001004b000009700000613d0000000d011000f90000000f0010006c00000db20000c13d000000400100043d000e00000001001d0000001b01000039000000000101041a000c00000001001d00000a11021001980000190a0000613d0000001a01000039000000000101041a000a00000001001d000b0a110010019c0000190a0000613d00000ad6010000410000000e030000290000000001130436000900000001001d0000000001000414000000040020008c000019160000c13d0000000103000031000000a00030008c000000a0040000390000000004034019000019400000013d000000240030008c000000930000413d0000000002000416000000000002004b000000930000c13d0000000401100370000000000101043b000000000010043f0000001c01000039000000200010043f00000040020000390000000001000019282d27f00000040f00000ba60000013d0000000001000416000000000001004b000000930000c13d0000000001000412001800000001001d001700400000003d000080050100003900000044030000390000000004000415000000180440008a000000050440021000000aa502000041282d28050000040f000000800010043f00000aa2010000410000282e0001042e0000000001000416000000000001004b000000930000c13d0000000f0100003900000d8d0000013d000000640030008c000000930000413d0000000002000416000000000002004b000000930000c13d0000002402100370000000000202043b000f00000002001d0000000402100370000000000202043b0000004404100370000000000404043b00000a0f0040009c000000930000213d0000002305400039000000000035004b000000930000813d000e00040040003d0000000e01100360000000000101043b001200000001001d00000a0f0010009c000000930000213d000000240140003900000012040000290000000504400210000c00000001001d001100000004001d000d00000014001d0000000d0030006b000000930000213d000000000020043f0000002001000039000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f0000000100200190000000930000613d000000000201043b0000000101200039000000000101041a000b00000001001d000000000001004b00000b730000613d000900000002001d0000000401200039000000000101041a000a00000001001d00000a1b010000410000000000100443000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000a1c011001c70000800b02000039282d28280000040f000000010020019000001c3f0000613d000000400200043d0000000a030000290000008003300270000000000101043b000000000031004b000013540000413d0000001401000039000000000112043600000000030004110000006003300210000000000031043500000ace0020009c000000a80000213d0000004003200039000000400030043f00000a0c0010009c00000a0c010080410000004001100210000000000202043300000a0c0020009c00000a0c020080410000006002200210000000000112019f000000000200041400000a0c0020009c00000a0c02008041000000c002200210000000000112019f00000a13011001c70000801002000039282d28280000040f0000000100200190000000930000613d000000000101043b00000011020000290000003f0220003900000acf02200197000000400300043d0000000002230019001100000003001d000000000032004b0000000003000039000000010300403900000a0f0020009c000000a80000213d0000000100300190000000a80000c13d000000400020043f000000120200002900000011040000290000000002240436001000000002001d0000000d02000029000000000020007c000000930000213d000000120000006b00000a510000613d0000000e020000290000002002200039000000020220036700000011030000290000000c050000290000000d060000290000002003300039000000002402043c00000000004304350000002005500039000000000065004b00000a2a0000413d00000011020000290000000002020433000000000002004b00000a510000613d0000000003000019001200000003001d000000050230021000000010022000290000000002020433000000000021004b00000a3f0000813d000000000010043f000000200020043f000000000100041400000a420000013d000000000020043f000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f0000000100200190000000930000613d000000000101043b0000001203000029000000010330003900000011020000290000000002020433000000000023004b00000a350000413d0000000b0010006c000017420000c13d000000000100041100000a1101100197001200000001001d000000000010043f00000009010000290000000501100039001100000001001d000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f0000000100200190000000930000613d0000000a0200002900000aa302200197000000000101043b000000000101041a000000000112004b00000db20000413d0000000f0010006c000014e50000413d0000000201000039000000000101041a00000af8011001670000000102000039000000000202041a00000000011200190000000f0010002a00000db20000413d0000000f011000290000000c02000039000000000202041a000000000021004b0000186d0000213d0000001701000039000000000101041a00000a1100100198000018840000c13d000000400100043d000000440210003900000ae703000041000000000032043500000024021000390000001d0300003900000ff50000013d0000000001000416000000000001004b000000930000c13d0000002102000039000000000102041a000000800010043f000000000020043f0000002002000039000000000001004b00000f400000c13d000000a001000039000000000402001900000f4f0000013d000000440030008c000000930000413d0000000002000416000000000002004b000000930000c13d0000000401100370000000000101043b001200000001001d282d24c70000040f0000001201000029000000000010043f0000002001000039000000200010043f00000040020000390000000001000019282d27f00000040f0000000101100039000000000101041a000000000001004b0000000001000039000000010100c039282d20850000040f0000001201000029000000000010043f0000002001000039000000200010043f00000024010000390000000201100367000000000101043b001200000001001d00000000010000190000004002000039282d27f00000040f00000001011000390000001202000029000000000021041b00000000010000190000282e0001042e000000240030008c000000930000413d0000000002000416000000000002004b000000930000c13d0000000402100370000000000502043b00000a0f0050009c000000930000213d0000002302500039000000000032004b000000930000813d0000000406500039000000000261034f000000000202043b00000a0f0020009c000000a80000213d0000001f0720003900000af7077001970000003f0770003900000af70770019700000aac0070009c000000a80000213d00000024055000390000008007700039000000400070043f000000800020043f0000000005520019000000000035004b000000930000213d0000002003600039000000000331034f00000af7052001980000001f0620018f000000a00150003900000ae10000613d000000a007000039000000000803034f000000008908043c0000000007970436000000000017004b00000add0000c13d000000000006004b00000aee0000613d000000000353034f0000000305600210000000000601043300000000065601cf000000000656022f000000000303043b0000010005500089000000000353022f00000000035301cf000000000363019f0000000000310435000000a0012000390000000000010435000000000100041a00000a11021001970000000001000411000000000012004b000012050000c13d000000800200043d00000a0f0020009c000000a80000213d0000000b01000039000000000501041a000000010050019000000001035002700000007f0330618f0000001f0030008c00000000060000390000000106002039000000000565013f000000010050019000000b2a0000c13d000000200030008c00000b140000413d000000000010043f0000001f05200039000000050550027000000a230550009a000000200020008c00000a24050040410000001f03300039000000050330027000000a230330009a000000000035004b00000b140000813d000000000005041b0000000105500039000000000035004b00000b100000413d0000001f0020008c000014e90000a13d000000000010043f00000af704200198000015ec0000c13d000000a00500003900000a2403000041000016080000013d0000000001000416000000000001004b000000930000c13d0000000403000039000000000203041a000000010420019000000001012002700000007f0110618f0000001f0010008c00000000050000390000000105002039000000000552013f000000010050019000000f670000613d00000acc01000041000000000010043f0000002201000039000000040010043f00000a2b010000410000282f000104300000000001000416000000000001004b000000930000c13d0000000f01000039000000000101041a0000000102000039000000000202041a0000000203000039000000000303041a0000000d04000039000000000404041a0000000c05000039000000000505041a000000800050043f000000a00040043f00000af8033001670000000002320019000000c00020043f000000e00010043f00000ab8010000410000282e0001042e000000640030008c000000930000413d0000002402100370000000000202043b000f00000002001d0000000402100370000000000202043b0000004404100370000000000404043b00000a0f0040009c000000930000213d0000002305400039000000000035004b000000930000813d000e00040040003d0000000e01100360000000000101043b001200000001001d00000a0f0010009c000000930000213d000000240140003900000012040000290000000504400210000c00000001001d001100000004001d000d00000014001d0000000d0030006b000000930000213d000000000020043f0000002001000039000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f0000000100200190000000930000613d000000000201043b0000000101200039000000000101041a000b00000001001d000000000001004b0000133f0000c13d000000400100043d000000440210003900000aea0300004100000000003204350000002402100039000000130300003900000ff50000013d000000240030008c000000930000413d0000000002000416000000000002004b000000930000c13d0000000401100370000000000101043b001200000001001d00000a110010009c000000930000213d000000000100041a00000a11011001970000000002000411000000000012004b000010250000c13d00000aaa0100004100000000001b0435000000000100041000000a11011001970000000402b00039000000000012043500000000010004140000001202000029000000040020008c0000103b0000c13d0000000103000031000000200030008c00000020040000390000000004034019000010670000013d0000000001000416000000000001004b000000930000c13d000000190100003900000d8d0000013d0000000001000416000000000001004b000000930000c13d000000180100003900000d8d0000013d0000000001000416000000000001004b000000930000c13d0000001501000039000000000101041a00000a1101100197000000800010043f00000aa2010000410000282e0001042e000000240030008c000000930000413d0000000001000416000000000001004b000000930000c13d282d24c70000040f00000004010000390000000201100367000000000101043b0000000c02000039000000000012041b00000000010000190000282e0001042e000000240030008c000000930000413d0000000002000416000000000002004b000000930000c13d0000000402100370000000000502043b00000a0f0050009c000000930000213d0000002302500039000000000032004b000000930000813d0000000406500039000000000261034f000000000202043b00000a0f0020009c000000a80000213d0000001f0720003900000af7077001970000003f0770003900000af70770019700000aac0070009c000000a80000213d00000024055000390000008007700039000000400070043f000000800020043f0000000005520019000000000035004b000000930000213d0000002003600039000000000331034f00000af7052001980000001f0620018f000000a00150003900000be20000613d000000a007000039000000000803034f000000008908043c0000000007970436000000000017004b00000bde0000c13d000000000006004b00000bef0000613d000000000353034f0000000305600210000000000601043300000000065601cf000000000656022f000000000303043b0000010005500089000000000353022f00000000035301cf000000000363019f0000000000310435000000a0012000390000000000010435000000000100041a00000a11021001970000000001000411000000000012004b000012050000c13d000000800200043d00000a0f0020009c000000a80000213d0000000a01000039000000000501041a000000010050019000000001035002700000007f0330618f0000001f0030008c00000000060000390000000106002039000000000565013f000000010050019000000b2a0000c13d000000200030008c00000c150000413d000000000010043f0000001f05200039000000050550027000000a200550009a000000200020008c00000a21050040410000001f03300039000000050330027000000a200330009a000000000035004b00000c150000813d000000000005041b0000000105500039000000000035004b00000c110000413d0000001f0020008c000014e90000a13d000000000010043f00000af704200198000015fa0000c13d000000a00500003900000a2103000041000016080000013d0000000001000416000000000001004b000000930000c13d282d1de40000040f0000002002000039000000400300043d001200000003001d0000000002230436282d1e6b0000040f00000f840000013d000000240030008c000000930000413d0000000002000416000000000002004b000000930000c13d0000000402100370000000000502043b00000a0f0050009c000000930000213d0000002302500039000000000032004b000000930000813d0000000406500039000000000261034f000000000202043b00000a0f0020009c000000a80000213d0000001f0720003900000af7077001970000003f0770003900000af70770019700000aac0070009c000000a80000213d00000024055000390000008007700039000000400070043f000000800020043f0000000005520019000000000035004b000000930000213d0000002003600039000000000331034f00000af7052001980000001f0620018f000000a00150003900000c510000613d000000a007000039000000000803034f000000008908043c0000000007970436000000000017004b00000c4d0000c13d000000000006004b00000c5e0000613d000000000353034f0000000305600210000000000601043300000000065601cf000000000656022f000000000303043b0000010005500089000000000353022f00000000035301cf000000000363019f0000000000310435000000a0012000390000000000010435000000000100041a00000a11021001970000000001000411000000000012004b000012050000c13d000000800200043d00000a0f0020009c000000a80000213d0000001401000039000000000501041a000000010050019000000001035002700000007f0330618f0000001f0030008c00000000060000390000000106002039000000000565013f000000010050019000000b2a0000c13d000000200030008c00000c840000413d000000000010043f0000001f05200039000000050550027000000a1d0550009a000000200020008c00000a1e050040410000001f03300039000000050330027000000a1d0330009a000000000035004b00000c840000813d000000000005041b0000000105500039000000000035004b00000c800000413d0000001f0020008c000015c70000a13d000000000010043f00000af704200198000016170000c13d000000a00500003900000a1e03000041000016250000013d0000000001000416000000000001004b000000930000c13d000000000100041a00000a11021001970000000001000411000000000012004b00000db80000c13d00000aa50100004100000000001004430000000001000412000000040010044300000080010000390000002400100443000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000aa6011001c70000800502000039282d28280000040f000000010020019000001c3f0000613d000000000101043b00120ab9001000a400000db20000813d00000a1b010000410000000000100443000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000a1c011001c70000800b02000039282d28280000040f000000010020019000001c3f0000613d000000000101043b000000120010006c000013820000a13d00000aad01000041000000000010044300000000010004100000000400100443000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000aae011001c70000800a02000039282d28280000040f000000010020019000001c3f0000613d000000000301043b000000000200041a000000000100041400000a1104200197000000040040008c000014f30000c13d000000010200003900000001010000310000159c0000013d000000640030008c000000930000413d0000000002000416000000000002004b000000930000c13d0000000401100370000000000101043b001200000001001d282d24c70000040f0000001201000029000000000010043f0000002001000039000000200010043f00000040020000390000000001000019282d27f00000040f0000000101100039000000000101041a000000000001004b0000000001000039000000010100c039282d20850000040f0000001201000029000000000010043f0000002001000039000000200010043f00000000010000190000004002000039282d27f00000040f00000002020003670000002403200370000000000303043b0000000204100039000000000034041b0000004402200370000000000202043b0000000301100039000000000021041b00000000010000190000282e0001042e000000240030008c000000930000413d0000000002000416000000000002004b000000930000c13d0000000401100370000000000601043b00000a110060009c000000930000213d000000000100041a00000a11021001970000000005000411000000000052004b00000eee0000c13d000000000006004b000011b60000c13d00000a2a01000041000000800010043f000000840000043f00000ab1010000410000282f00010430000000240030008c000000930000413d0000000202000039000000000202041a00000af8022001670000000103000039000000000303041a00000000022300190000000401100370000000000101043b001200000001001d000000000012001a00000db20000413d00000012012000290000000c02000039000000000202041a000000000021004b00000fde0000a13d00000a2801000041000000800010043f0000002001000039000000840010043f0000001501000039000000a40010043f00000acd01000041000000c40010043f00000ad5010000410000282f00010430000000440030008c000000930000413d0000000002000416000000000002004b000000930000c13d0000000402100370000000000202043b001200000002001d00000a110020009c000000930000213d0000002401100370000000000301043b000000000100041a00000a11021001970000000001000411000000000012004b00000db80000c13d0000000201000039000000000101041a00000af8011001670000000102000039000000000202041a0000000001120019000000000031001a00000db20000413d001100000003001d00000000013100190000000c02000039000000000202041a000000000021004b0000000001000039000000010100a039282d24b30000040f00000012010000290000001102000029282d253f0000040f00000000010000190000282e0001042e000000240030008c000000930000413d0000000001000416000000000001004b000000930000c13d282d24c70000040f00000004010000390000000201100367000000000101043b0000000d02000039000000000012041b00000000010000190000282e0001042e000000440030008c000000930000413d0000000002000416000000000002004b000000930000c13d0000000401100370000000000101043b001200000001001d282d24c70000040f0000001201000029000000000010043f0000002001000039000000200010043f00000040020000390000000001000019282d27f00000040f0000000101100039000000000101041a000000000001004b0000000001000039000000010100c039282d20850000040f0000001201000029000000000010043f0000002001000039000000200010043f00000024010000390000000201100367000000000101043b001200000001001d00000000010000190000004002000039282d27f00000040f000000120200002900000aa3022001970000000401100039000000000301041a00000aa403300197000000000223019f000000000021041b00000000010000190000282e0001042e000000240030008c000000930000413d0000000002000416000000000002004b000000930000c13d0000000401100370000000000101043b0000002102000039000000000302041a000000000031004b000000930000813d000000000020043f00000ad30110009a000000000101041a000000800010043f00000aa2010000410000282e0001042e000000240030008c000000930000413d0000000002000416000000000002004b000000930000c13d0000000401100370000000000101043b001200000001001d00000a110010009c000000930000213d000000000100041a00000a11021001970000000001000411000000000012004b00000db80000c13d00000aa50100004100000000001004430000000001000412000000040010044300000080010000390000002400100443000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000aa6011001c70000800502000039282d28280000040f000000010020019000001c3f0000613d000000000101043b00110ab9001000a4000012fe0000413d00000acc01000041000000000010043f0000001101000039000000040010043f00000a2b010000410000282f0001043000000ab002000041000000800020043f000000840010043f00000ab1010000410000282f00010430000000240030008c000000930000413d0000000002000416000000000002004b000000930000c13d0000000401100370000000000101043b001200000001001d00000a110010009c000000930000213d282d24c70000040f0000001201000029000000000010043f0000001f01000039000000200010043f00000040020000390000000001000019282d27f00000040f000000000301041a00000af602300197000000000021041b00000000010000190000282e0001042e000000240030008c000000930000413d0000000002000416000000000002004b000000930000c13d0000000401100370000000000101043b00000a110010009c000000930000213d282d206d0000040f000000400200043d000000000012043500000a0c0020009c00000a0c02008041000000400120021000000ab7011001c70000282e0001042e0000000001000416000000000001004b000000930000c13d282d24c70000040f0000001001000039000000000201041a00000af603200197000000ff0020019000000001033061bf000000000031041b00000000010000190000282e0001042e000000640030008c000000930000413d0000000002000416000000000002004b000000930000c13d0000002402100370000000000202043b000f00000002001d00000a110020009c000000930000213d0000004402100370000000000202043b00000a0f0020009c000000930000213d0000002304200039000000000034004b000000930000813d000e00040020003d0000000e04100360000000000404043b001200000004001d00000a0f0040009c000000930000213d000000240220003900000012040000290000000504400210000c00000002001d001100000004001d000d00000024001d0000000d0030006b000000930000213d0000000401100370000000000101043b000b00000001001d000000000010043f0000002001000039000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f0000000100200190000000930000613d000000000101043b0000000101100039000000000101041a000a00000001001d000000400100043d000000140200003900000000022104360000000f030000290000006003300210000000000032043500000ace0010009c000000a80000213d0000004003100039000000400030043f00000a0c0020009c00000a0c020080410000004002200210000000000101043300000a0c0010009c00000a0c010080410000006001100210000000000121019f000000000200041400000a0c0020009c00000a0c02008041000000c002200210000000000112019f00000a13011001c70000801002000039282d28280000040f0000000100200190000000930000613d000000000101043b00000011020000290000003f0220003900000acf02200197000000400300043d0000000002230019001100000003001d000000000032004b0000000003000039000000010300403900000a0f0020009c000000a80000213d0000000100300190000000a80000c13d000000400020043f000000110200002900000012040000290000000002420436001000000002001d0000000d02000029000000000020007c000000930000213d000000120000006b00000e840000613d0000000e020000290000002002200039000000020220036700000011030000290000000c050000290000000d060000290000002003300039000000002402043c00000000004304350000002005500039000000000065004b00000e5d0000413d00000011020000290000000002020433000000000002004b00000e840000613d0000000003000019001200000003001d000000050230021000000010022000290000000002020433000000000021004b00000e720000813d000000000010043f000000200020043f000000000100041400000e750000013d000000000020043f000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f0000000100200190000000930000613d000000000101043b0000001203000029000000010330003900000011020000290000000002020433000000000023004b00000e680000413d0000000a0010006c000000000100001900000dde0000c13d0000000b01000029000000000010043f0000002001000039000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f0000000100200190000000930000613d000000000101043b0000000f0200002900000a1102200197001200000002001d000000000020043f0000000501100039000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f0000000100200190000000930000613d000000000101043b000000000101041a001100000001001d0000000b01000029000000000010043f0000002001000039000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f0000000100200190000000930000613d000000000101043b0000000401100039000000000101041a00000aa301100197000000110010006b000000000100001900000dde0000813d0000000b01000029000000000010043f0000002001000039000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f0000000100200190000000930000613d000000000101043b0000000401100039000000000101041a001100000001001d0000000b01000029000000000010043f0000002001000039000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f0000000100200190000000930000613d000000000101043b0000001202000029000000000020043f0000000501100039000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f0000000100200190000000930000613d000000110200002900000aa302200197000000000101043b000000000101041a000000000112004b00000db20000413d00000dde0000013d00000ab001000041000000800010043f000000840050043f00000ab1010000410000282f0001043000000ab001000041000000800010043f000000840020043f00000ab1010000410000282f000104300000001a03000039000000000403041a00000a11064001980000000003000019000010040000c13d0000001702000039000000000202041a0000001804000039000000000404041a000000400510003900000000003504350000002003100039000000000043043500000a1102200197000000000021043500000a0c0010009c00000a0c01008041000000400110021000000adb011001c70000282e0001042e001100000001001d000000000010043f0000000501000039000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f0000000100200190000000930000613d000000000101043b000000000101041a000000000001004b00000f350000c13d0000000101000039000000000101041a0000001102000029000000000021004b000007590000a13d000000010220008a001200000002001d000000000020043f0000000501000039000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f0000000100200190000000930000613d000000000101043b000000000101041a000000000001004b000000120200002900000f220000613d00000abe00100198000007590000c13d00000a11001001980000131e0000c13d000000400100043d000000640210003900000aca030000410000000000320435000000440210003900000acb0300004100000fa80000013d000000a00500003900000abc0300004100000000040000190000000006050019000000000503041a000000000556043600000001033000390000000104400039000000000014004b00000f430000413d000000410160008a00000af70410019700000aac0040009c000000a80000213d0000008001400039000000400010043f0000000000210435000000a002400039000000800300043d0000000000320435000000c002400039000000000003004b00000f5e0000613d000000a0040000390000000005000019000000004604043400000000026204360000000105500039000000000035004b00000f590000413d000000000212004900000a0c0020009c00000a0c02008041000000600220021000000a0c0010009c00000a0c010080410000004001100210000000000112019f0000282e0001042e000000800010043f000000000004004b00000f780000613d000000000030043f000000000001004b000000000200001900000f7d0000613d00000a19030000410000000002000019000000000403041a000000a005200039000000000045043500000001033000390000002002200039000000000012004b00000f700000413d00000f7d0000013d00000af602200197000000a00020043f000000000001004b0000002002000039000000000200603900000020022000390000008001000039282d1dd20000040f000000400100043d001200000001001d0000008002000039282d1e7d0000040f0000001202000029000000000121004900000a0c0010009c00000a0c01008041000000600110021000000a0c0020009c00000a0c020080410000004002200210000000000121019f0000282e0001042e00000aa5010000410000000000100443000000040020044300000020010000390000002400100443000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000aa6011001c70000800502000039282d28280000040f000000010020019000001c3f0000613d000000000101043b00000a11011001970000000002000411000000000012004b0000000002000412000001a00000613d000000400100043d000000640210003900000aa7030000410000000000320435000000440210003900000aa803000041000000000032043500000024021000390000002f03000039000000000032043500000a2802000041000000000021043500000004021000390000002003000039000000000032043500000a0c0010009c00000a0c01008041000000400110021000000aa9011001c70000282f0001043000000aa50100004100000000001004430000000001000412000000040010044300000020010000390000002400100443000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000aa6011001c70000800502000039282d28280000040f000000010020019000001c3f0000613d000000000101043b00000a11011001970000000002000411000000000012004b000000fe0000613d00000aa5010000410000000000100443000000000100041200000004001004430000002400000443000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000aa6011001c70000800502000039282d28280000040f000000010020019000001c3f0000613d000000000101043b00000a11011001970000000002000411000000000012004b000000fe0000613d00000fa20000013d0000000f01000039000000000101041a001100000001001d00000a1b010000410000000000100443000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000a1c011001c70000800b02000039282d28280000040f000000010020019000001c3f0000613d000000000101043b000000110010006c000011c50000813d000000400100043d000000440210003900000ae103000041000000000032043500000024021000390000001603000039000000000032043500000a2802000041000000000021043500000004021000390000002003000039000000000032043500000a0c0010009c00000a0c01008041000000400110021000000a29011001c70000282f000104300000001902000039000000000012041b00000000010000190000282e0001042e001000000004001d0000000d01000039000000000101041a000f00000001001d00000ad601000041000000800010043f0000000001000414000000040020008c001200000005001d001100000006001d000010ea0000c13d0000000103000031000000a00030008c000000a00400003900000000040340190000110f0000013d00000af30020009c0000000001000039000000010100603900000af40020009c00000001011061bf00000af50020009c00000001011061bf00000af40020009c000010210000613d00000af50020009c000010210000613d00000af30020009c000000000100c019000000010110018f000000800010043f00000aa2010000410000282e0001042e00000aa50100004100000000001004430000000001000412000000040010044300000020010000390000002400100443000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000aa6011001c70000800502000039282d28280000040f000000010020019000001c3f0000613d000000000101043b00000a11011001970000000002000411000000000012004b000012430000c13d000000400b00043d00000b890000013d00000a0c00b0009c00000a0c0300004100000000030b4019000000400330021000000a0c0010009c00000a0c01008041000000c001100210000000000131019f00000a2b011001c700110000000b001d282d28280000040f000000110b000029000000600310027000000a0c03300197000000200030008c000000200400003900000000040340190000001f0640018f000000200740019000000000057b0019000010560000613d000000000801034f00000000090b0019000000008a08043c0000000009a90436000000000059004b000010520000c13d000000000006004b000010630000613d000000000771034f0000000306600210000000000805043300000000086801cf000000000868022f000000000707043b0000010006600089000000000767022f00000000066701cf000000000686019f0000000000650435000100000003001f000300000001035500000001002001900000121f0000613d0000001f01400039000000600210018f0000000001b20019000000000021004b0000000002000039000000010200403900000a0f0010009c000000a80000213d0000000100200190000000a80000c13d000000400010043f000000200030008c000000930000413d00000000010b0433001100000001001d00000aa50100004100000000001004430000000001000412000000040010044300000040010000390000002400100443000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000aa6011001c70000800502000039282d28280000040f000000010020019000001c3f0000613d000000000201043b000000110000006b000002110000613d00000011012000b900000011031000fa000000000023004b00000db20000c13d000027100310011a001000000003001d000f00110030007300000db20000413d000027100010008c0000163a0000813d0000001002000029000000110020006b000002110000613d00000aa5010000410000000000100443000000000100041200000004001004430000002400000443000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000aa6011001c70000800502000039282d28280000040f000000010020019000001c3f0000613d000000000101043b000000400200043d000000200320003900000aab04000041000000000043043500000044032000390000000f04000029000000000043043500000a1101100197000000240320003900000000001304350000004401000039000000000012043500000aac0020009c000000a80000213d0000008001200039000000400010043f0000001201000029282d26fe0000040f00000000010000190000282e0001042e00000a0c0010009c00000a0c01008041000000c00110021000000ab3011001c7282d28280000040f000000600310027000000a0c03300197000000200030008c000000200400003900000000040340190000001f0640018f000000200740019000000080057001bf000010cc0000613d0000008008000039000000000901034f000000009a09043c0000000008a80436000000000058004b000010c80000c13d000000000006004b000010d90000613d000000000771034f0000000306600210000000000805043300000000086801cf000000000868022f000000000707043b0000010006600089000000000767022f00000000066701cf000000000686019f0000000000650435000100000003001f000300000001035500000001002001900000122b0000613d0000001f01400039000000600110018f00000080011001bf000000400010043f000000200030008c000000930000413d000000800100043d000000ff0010008c000000930000213d000000c00110021000000ab4011001970000001b03000039000011aa0000013d00000a0c0010009c00000a0c01008041000000c00110021000000ab3011001c7282d28280000040f000000600310027000000a0c03300197000000a00030008c000000a00400003900000000040340190000001f0640018f000000e0074001900000008005700039000010fe0000613d0000008008000039000000000901034f000000009a09043c0000000008a80436000000000058004b000010fa0000c13d000000000006004b0000110b0000613d000000000771034f0000000306600210000000000805043300000000086801cf000000000868022f000000000707043b0000010006600089000000000767022f00000000066701cf000000000686019f0000000000650435000100000003001f00030000000103550000000100200190000012370000613d0000001f01400039000001e00210018f0000008001200039000000400010043f000000a00030008c000000930000413d000000800300043d00000ad70030009c000000930000213d000001000300043d00000ad70030009c000000930000213d000000a00300043d000e00000003001d00000ad80030009c000013df0000213d0000000e0000006b000013df0000613d000000e00100043d000d00000001001d00000a1b010000410000000000100443000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000a1c011001c70000800b02000039282d28280000040f000000010020019000001c3f0000613d0000001202000029000000a00220027000000a0c02200197000000000101043b000000000121004b00000db20000413d0000000d0010006b00001c400000a13d000000400200043d00000ad601000041000d00000002001d0000000001120436000c00000001001d00000000010004140000001102000029000000040020008c000016d90000c13d0000000103000031000000a00030008c000000a0040000390000000004034019000017040000013d0000001201000029000000000010043f0000001c01000039000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f0000000100200190000000930000613d000000000101043b000000000101041a001200000001001d00000024010000390000000201100367000000000101043b0000001102000029282d204b0000040f000027100110011a000000400200043d00000020032000390000000000130435000000120100002900000a1101100197000000000012043500000a0c0020009c00000a0c02008041000000400120021000000aeb011001c70000282e0001042e000f00000005001d001000000003001d000e00000001001d000d00000004001d0000000d01000039000000000101041a000c00000001001d000000400300043d00000ad601000041001200000003001d0000000001130436001100000001001d0000000001000414000000040020008c0000126a0000c13d0000000103000031000000a00030008c000000a0040000390000000004034019000012940000013d00000a0c0010009c00000a0c01008041000000c00110021000000ab3011001c7282d28280000040f000000600310027000000a0c03300197000000200030008c000000200400003900000000040340190000001f0640018f000000200740019000000080057001bf0000118d0000613d0000008008000039000000000901034f000000009a09043c0000000008a80436000000000058004b000011890000c13d000000000006004b0000119a0000613d000000000771034f0000000306600210000000000805043300000000086801cf000000000868022f000000000707043b0000010006600089000000000767022f00000000066701cf000000000686019f0000000000650435000100000003001f00030000000103550000000100200190000012d60000613d0000001f01400039000000600110018f00000080011001bf000000400010043f000000200030008c000000930000413d000000800100043d000000ff0010008c000000930000213d000000c00110021000000ab4011001970000001a03000039000000000203041a00000ab502200197000000000112019f00000024020000390000000202200367000000000202043b000000a00220021000000ab602200197000000000121019f000000000013041b00000000010000190000282e0001042e00000a1201100197000000000161019f000000000010041b000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000a13011001c70000800d02000039000000030300003900000a1404000041282d28230000040f0000000100200190000002110000c13d000000930000013d000000000100041100000a1101100198000012e20000c13d00000ae001000041000000000010043f00000ad9010000410000282f0001043000000af6011001970000000000160435000000000007004b000000200200003900000000020060390000003f0120003900000af7031001970000000001530019000000000031004b0000000003000039000000010300403900000a0f0010009c000000a80000213d0000000100300190000000a80000c13d00000000070600190000000006050019000000400010043f0000008003100039000000c00400003900000000004304350000000f0500002900000080035002700000006004100039000000000034043500000aa3035001970000004004100039000000000034043500000020031000390000001104000029000000000043043500000010030000290000000000310435000000c00410003900000000030604330000000000340435000000e004100039000000000003004b000011fc0000613d0000000008070019000000000500001900000000064500190000000007580019000000000707043300000000007604350000002005500039000000000035004b000011f50000413d00000000044300190000000000040435000000a004100039000000120500002900000000005404350000001f0330003900000af702300197000000e00220003900000f5f0000013d000000400200043d00000ab00300004100000000003204350000000403200039000000000013043500000a0c0020009c00000a0c02008041000000400120021000000a2b011001c70000282f0001043000000abe0010019800000011010000290000121b0000c13d000000000010043f0000000701000039000000200010043f00000040020000390000000001000019282d27f00000040f000000000101041a00000a110110019700000dde0000013d00000aef01000041000000000010043f00000ad9010000410000282f000104300000001f0530018f00000a0e06300198000000400200043d00000000046200190000136f0000613d000000000701034f0000000008020019000000007907043c0000000008980436000000000048004b000012260000c13d0000136f0000013d0000001f0530018f00000a0e06300198000000400200043d00000000046200190000136f0000613d000000000701034f0000000008020019000000007907043c0000000008980436000000000048004b000012320000c13d0000136f0000013d0000001f0530018f00000a0e06300198000000400200043d00000000046200190000136f0000613d000000000701034f0000000008020019000000007907043c0000000008980436000000000048004b0000123e0000c13d0000136f0000013d00000aa5010000410000000000100443000000000100041200000004001004430000002400000443000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000aa6011001c70000800502000039282d28280000040f000000010020019000001c3f0000613d000000400b00043d000000000101043b00000a11011001970000000002000411000000000012004b00000b890000613d0000006401b0003900000aa70200004100000000002104350000004401b0003900000aa80200004100000000002104350000002401b000390000002f02000039000000000021043500000a280100004100000000001b04350000000401b000390000002002000039000000000021043500000a0c00b0009c00000a0c0b0080410000004001b0021000000aa9011001c70000282f00010430000000120300002900000a0c0030009c00000a0c03008041000000400330021000000a0c0010009c00000a0c01008041000000c001100210000000000131019f00000ad9011001c7282d28280000040f000000600310027000000a0c03300197000000a00030008c000000a00400003900000000040340190000001f0640018f000000e0074001900000001205700029000012830000613d000000000801034f0000001209000029000000008a08043c0000000009a90436000000000059004b0000127f0000c13d000000000006004b000012900000613d000000000771034f0000000306600210000000000805043300000000086801cf000000000868022f000000000707043b0000010006600089000000000767022f00000000066701cf000000000686019f0000000000650435000100000003001f00030000000103550000000100200190000013640000613d0000001f01400039000001e00210018f0000001201200029000000000021004b0000000002000039000000010200403900000a0f0010009c000000a80000213d0000000100200190000000a80000c13d000000400010043f000000a00030008c000000930000413d0000001202000029000000000202043300000ad70020009c000000930000213d00000012020000290000008002200039000000000202043300000ad70020009c000000930000213d00000011020000290000000002020433001100000002001d00000ad80020009c00001c410000213d000000110000006b00001c410000613d000000120100002900000060011000390000000001010433001200000001001d00000a1b010000410000000000100443000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000a1c011001c70000800b02000039282d28280000040f000000010020019000001c3f0000613d0000001002000029000000a00220027000000a0c02200197000000000101043b000000000121004b00000db20000413d000000120010006b00001c400000a13d000000400200043d00000ad601000041001200000002001d0000000001120436000b00000001001d00000000010004140000000f02000029000000040020008c000017880000c13d0000000103000031000000a00030008c000000a0040000390000000004034019000017b30000013d0000001f0530018f00000a0e06300198000000400200043d00000000046200190000136f0000613d000000000701034f0000000008020019000000007907043c0000000008980436000000000048004b000012dd0000c13d0000136f0000013d000000000010043f0000000601000039000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f0000000100200190000000930000613d0000000e02000039000000000202041a000000000101043b000000000101041a00000a0f01100197000000000112004b0000000001004019000000120010006c000013b30000813d000000400100043d000000440210003900000adf03000041000000000032043500000024021000390000001b0300003900000ff50000013d00000a1b010000410000000000100443000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000a1c011001c70000800b02000039282d28280000040f000000010020019000001c3f0000613d000000400200043d001000000002001d0000000402200039000000000101043b000000110010006c000013c80000a13d00000aaa0100004100000010030000290000000000130435000000000100041000000a1101100197000000000012043500000000010004140000001202000029000000040020008c000015000000c13d0000000103000031000000200030008c000000200400003900000000040340190000152b0000013d0000001001000039000000000101041a000000ff00100190000013ac0000c13d0000000b05000039000000000405041a000000010640019000000001024002700000007f0220618f0000001f0020008c00000000010000390000000101002039000000000114013f000000010010019000000b2a0000c13d000000400100043d0000000003210436000000000006004b0000158c0000613d000000000050043f000000000002004b0000000004000019000015910000613d00000a240500004100000000040000190000000006430019000000000705041a000000000076043500000001055000390000002004400039000000000024004b000013370000413d000015910000013d000900000002001d0000000401200039000000000101041a000a00000001001d00000a1b010000410000000000100443000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000a1c011001c70000800b02000039282d28280000040f000000010020019000001c3f0000613d000000400200043d0000000a030000290000008003300270000000000101043b000000000031004b0000146a0000813d000000440120003900000ad103000041000000000031043500000024012000390000001503000039000000000031043500000a2801000041000000000012043500000004012000390000002003000039000000000031043500000a0c0020009c00000a0c02008041000000400120021000000a29011001c70000282f000104300000001f0530018f00000a0e06300198000000400200043d00000000046200190000136f0000613d000000000701034f0000000008020019000000007907043c0000000008980436000000000048004b0000136b0000c13d000000000005004b0000137c0000613d000000000161034f0000000305500210000000000604043300000000065601cf000000000656022f000000000101043b0000010005500089000000000151022f00000000015101cf000000000161019f0000000000140435000000600130021000000a0c0020009c00000a0c020080410000004002200210000000000112019f0000282f00010430000000400100043d000000440210003900000abb03000041000000000032043500000024021000390000001f0300003900000ff50000013d0000001201000029000000000010043f0000000801000039000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f0000000100200190000000930000613d000000000101043b000000000200041100000a1102200197000000000020043f000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f0000000100200190000000930000613d000000000101043b000000000101041a000000ff00100190000006400000c13d00000aec01000041000000000010043f00000ad9010000410000282f00010430000000110100002900000ac00010009c000014140000413d0000004001000039000000110200002900000ac00220012a0000141d0000013d0000000d01000039000000000201041a00000012012000b9000000000002004b000013bb0000613d00000000022100d9000000120020006c00000db20000c13d0000000002000416000000000012004b000007a90000813d000000400100043d000000640210003900000add030000410000000000320435000000440210003900000ade0300004100000000003204350000002402100039000000230300003900000fab0000013d00000a28010000410000001003000029000000000013043500000020010000390000000000120435000000440130003900000abb02000041000000000021043500000024013000390000001f02000039000000000021043500000a0c0030009c00000a0c03008041000000400130021000000a29011001c70000282f0001043000000a0c0010009c00000a0c01008041000000c001100210000000110000006b000015400000c13d0000000002040019000015440000013d00000a28030000410000000000310435000000c40320003900000ada040000410000000000430435000000a4032000390000001c040000390000000000430435000000840220003900000020030000390000000000320435000000400110021000000a29011001c70000282f000104300000001101000029000000000010043f0000001c01000039000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f0000000100200190000000930000613d000000100200002900000a1102200197000000000101043b000000000301041a00000a1203300197000000000223019f000000000021041b00000044010000390000000201100367000000000101043b001200000001001d0000001101000029000000000010043f0000001d01000039000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f0000000100200190000000930000613d000000000101043b00000ab30000013d000000110200002900000ac20020009c00000ac10220212a0000000001000039000000200100203900000ac30020009c00000010011081bf00000aa30220819700000ac30220812a00000ac40020009c000000080110803900000a0f0220819700000ac40220812a000027100020008c000000040110803900000a0c02208197000027100220811a000000640020008c00000002011080390000ffff0220818f000000640220811a000000090020008c000000010110203900000af7051001970000005f0350003900000af706300197000000400400043d0000000003460019000000000063004b0000000006000039000000010600403900000a0f0030009c000000a80000213d0000000100600190000000a80000c13d000000400030043f00000001031000390000000003340436000000200550003900000af7065001980000001f0550018f000014460000613d0000000006630019000000000700003100000002077003670000000008030019000000007907043c0000000008980436000000000068004b000014420000c13d000000000005004b000000000114001900000021011000390000001107000029000000090070008c0000000a5770011a0000000305500210000000010110008a000000000601043300000ac50660019700000ac60550021f00000ac705500197000000000565019f00000000005104350000144a0000213d0000000a08000039000000000708041a000000010970019000000001057002700000007f0550618f0000001f0050008c00000000010000390000000101002039000000000117013f000000010010019000000b2a0000c13d000000400100043d000000000005004b000015de0000c13d00000ac90010009c000000a80000213d0000002002100039000000400020043f0000000000010435000000400300043d000016bf0000013d0000001401000039000000000112043600000000030004110000006003300210000000000031043500000ace0020009c000000a80000213d0000004003200039000000400030043f00000a0c0010009c00000a0c010080410000004001100210000000000202043300000a0c0020009c00000a0c020080410000006002200210000000000112019f000000000200041400000a0c0020009c00000a0c02008041000000c002200210000000000112019f00000a13011001c70000801002000039282d28280000040f0000000100200190000000930000613d000000000101043b00000011020000290000003f0220003900000acf02200197000000400300043d0000000002230019001100000003001d000000000032004b0000000003000039000000010300403900000a0f0020009c000000a80000213d0000000100300190000000a80000c13d000000400020043f000000120200002900000011040000290000000002240436001000000002001d0000000d02000029000000000020007c000000930000213d000000120000006b000014ca0000613d0000000e020000290000002002200039000000020220036700000011030000290000000c050000290000000d060000290000002003300039000000002402043c00000000004304350000002005500039000000000065004b000014a30000413d00000011020000290000000002020433000000000002004b000014ca0000613d0000000003000019001200000003001d000000050230021000000010022000290000000002020433000000000021004b000014b80000813d000000000010043f000000200020043f0000000001000414000014bb0000013d000000000020043f000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f0000000100200190000000930000613d000000000101043b0000001203000029000000010330003900000011020000290000000002020433000000000023004b000014ae0000413d0000000b0010006c000017420000c13d000000000100041100000a1101100197001200000001001d000000000010043f00000009010000290000000501100039001100000001001d000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f0000000100200190000000930000613d0000000a0200002900000aa302200197000000000101043b000000000101041a000000000112004b00000db20000413d0000000f0010006c000018600000813d000000400100043d000000440210003900000ae90300004100001c430000013d000000000002004b0000000003000019000014ed0000613d000000a00300043d000000030420021000000af80440027f00000af804400167000000000443016f0000000103200210000016130000013d00000a0c0010009c00000a0c01008041000000c001100210000000000003004b000015940000c13d0000000002040019000015970000013d00000ab001000041000000800010043f0000000001000411000000840010043f00000ab1010000410000282f00010430000000100200002900000a0c0020009c00000a0c02008041000000400220021000000a0c0010009c00000a0c01008041000000c001100210000000000121019f00000a2b011001c70000001202000029282d28280000040f000000600310027000000a0c03300197000000200030008c000000200400003900000000040340190000001f0640018f000000200740019000000010057000290000151a0000613d000000000801034f0000001009000029000000008a08043c0000000009a90436000000000059004b000015160000c13d000000000006004b000015270000613d000000000771034f0000000306600210000000000805043300000000086801cf000000000868022f000000000707043b0000010006600089000000000767022f00000000066701cf000000000686019f0000000000650435000100000003001f00030000000103550000000100200190000015d20000613d0000001f01400039000000600210018f0000001001200029000000000021004b0000000002000039000000010200403900000a0f0010009c000000a80000213d0000000100200190000000a80000c13d000000400010043f000000200030008c000000930000413d00000010010000290000000003010433000000000100041a00000a11021001970000001201000029282d250c0000040f00000000010000190000282e0001042e00000a13011001c7000080090200003900000011030000290000000005000019282d28230000040f0003000000010355000000600110027000010a0c0010019d00000a0c01100197000000000001004b000015540000c13d0000000100200190000002110000c13d000000400100043d000000440210003900000aaf03000041000000000032043500000024021000390000000f0300003900000ff50000013d00000a0f0010009c000000a80000213d0000001f0410003900000af7044001970000003f0440003900000af705400197000000400400043d0000000005540019000000000045004b0000000006000039000000010600403900000a0f0050009c000000a80000213d0000000100600190000000a80000c13d000000400050043f000000000614043600000af7031001980000001f0410018f000000000136001900000003050003670000156f0000613d000000000705034f000000007807043c0000000006860436000000000016004b0000156b0000c13d000000000004004b0000154b0000613d000000000335034f0000000304400210000000000501043300000000054501cf000000000545022f000000000303043b0000010004400089000000000343022f00000000034301cf000000000353019f00000000003104350000154b0000013d000000120000006b0000000001000019000016770000613d0000001203000029000000030130021000000af80110027f00000af801100167000000110200002900000020022000390000000202200367000000000202043b000000000112016f0000000102300210000000000121019f000016770000013d00000af6044001970000000000430435000000000002004b000000200400003900000000040060390000003f0240003900000af702200197000016b50000013d00000a13011001c700008009020000390000000005000019282d28230000040f0003000000010355000000600110027000010a0c0010019d00000a0c01100197000000000001004b000011c20000613d00000a0f0010009c000000a80000213d0000001f0410003900000af7044001970000003f0440003900000af705400197000000400400043d0000000005540019000000000045004b0000000006000039000000010600403900000a0f0050009c000000a80000213d0000000100600190000000a80000c13d000000400050043f000000000614043600000af7031001980000001f0410018f00000000013600190000000305000367000015b90000613d000000000705034f000000007807043c0000000006860436000000000016004b000015b50000c13d000000000004004b000011c20000613d000000000335034f0000000304400210000000000501043300000000054501cf000000000545022f000000000303043b0000010004400089000000000343022f00000000034301cf000000000353019f0000000000310435000011c20000013d000000000002004b0000000003000019000015cb0000613d000000a00300043d000000030420021000000af80440027f00000af804400167000000000343016f0000000102200210000000000223019f000016300000013d0000001f0530018f00000a0e06300198000000400200043d00000000046200190000136f0000613d000000000701034f0000000008020019000000007907043c0000000008980436000000000048004b000015d90000c13d0000136f0000013d0000002006100039000000000009004b0000169f0000613d000000000080043f00000a210700004100000000080000190000000009680019000000000a07041a0000000000a9043500000001077000390000002008800039000000000058004b000015e40000413d000016a10000013d00000a24030000410000002006000039000000010540008a000000050550027000000a250550009a000000000706001900000080066000390000000006060433000000000063041b00000020067000390000000103300039000000000053004b000015f10000c13d000016070000013d00000a21030000410000002006000039000000010540008a000000050550027000000a220550009a000000000706001900000080066000390000000006060433000000000063041b00000020067000390000000103300039000000000053004b000015ff0000c13d000000a005700039000000000024004b000016110000813d0000000304200210000000f80440018f00000af80440027f00000af8044001670000000005050433000000000445016f000000000043041b00000001030000390000000104200210000000000234019f000000000021041b00000000010000190000282e0001042e00000a1e030000410000002006000039000000010540008a000000050550027000000a1f0550009a000000000706001900000080066000390000000006060433000000000063041b00000020067000390000000103300039000000000053004b0000161c0000c13d000000a005700039000000000024004b0000162e0000813d0000000304200210000000f80440018f00000af80440027f00000af8044001670000000005050433000000000445016f000000000043041b000000010220021000000001022001bf000000000021041b000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000a13011001c70000800d02000039000000010300003900000ae2040000410000020e0000013d00000aa50100004100000000001004430000000001000412000000040010044300000020010000390000002400100443000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000aa6011001c70000800502000039282d28280000040f000000010020019000001c3f0000613d000000000101043b000000400200043d000000200320003900000aab04000041000000000043043500000044032000390000001004000029000000000043043500000a1101100197000000240320003900000000001304350000004401000039000000000012043500000aac0020009c000000a80000213d0000008001200039000000400010043f0000001201000029282d26fe0000040f000010920000013d0000000204000367000000000300001900000010060000290000000005630019000000000554034f000000000505043b000000000051041b00000001011000390000002003300039000000000023004b000016600000413d000000120020006c000016740000813d00000012020000290000000302200210000000f80220018f00000af80220027f00000af80220016700000010033000290000000203300367000000000303043b000000000223016f000000000021041b0000001201000029000000010110021000000001011001bf0000000f03000029000000000013041b00000002013000390000000c02000029000000000021041b00000003013000390000000d02000029000000000021041b0000000b0100002900000080011002100000000e0200002900000aa302200197000000000112019f0000000402300039000000000012041b0000002102000039000000000102041a000000000001004b000016920000c13d0000000103100039000000000032041b000000000020043f00000ad30110009a0000000a02000029000000000021041b00000000010000190000282e0001042e000000000020043f00000000030000190000000a0500002900000ad30430009a000000000404041a000000000054004b000002110000613d0000000103300039000000000013004b000016950000413d00000a0f0010009c000000a80000213d0000168a0000013d00000af607700197000000000076043500000000055600190000000004040433000000000004004b000016ad0000613d000000000600001900000000075600190000000008360019000000000808043300000000008704350000002006600039000000000046004b000016a60000413d000000000354001900000ac804000041000000000043043500000000031300490000001b0430008a0000000000410435000000240330003900000af7023001970000000004120019000000000024004b0000000002000039000000010200403900000a0f0040009c000000a80000213d0000000100200190000000a80000c13d0000000003040019000000400040043f000000200200003900000c230000013d00000acc01000041000000000010043f0000003201000039000000040010043f00000a2b010000410000282f00010430000000400100043d00000ab002000041000000000021043500000004021000390000000003000411000000000032043500000a0c0010009c00000a0c01008041000000400110021000000a2b011001c70000282f00010430000000400100043d000000440210003900000acd0300004100000000003204350000002402100039000000150300003900000ff50000013d0000000d0200002900000a0c0020009c00000a0c02008041000000400220021000000a0c0010009c00000a0c01008041000000c001100210000000000121019f00000ad9011001c70000001102000029282d28280000040f000000600310027000000a0c03300197000000a00030008c000000a00400003900000000040340190000001f0640018f000000e0074001900000000d05700029000016f30000613d000000000801034f0000000d09000029000000008a08043c0000000009a90436000000000059004b000016ef0000c13d000000000006004b000017000000613d000000000771034f0000000306600210000000000805043300000000086801cf000000000868022f000000000707043b0000010006600089000000000767022f00000000066701cf000000000686019f0000000000650435000100000003001f00030000000103550000000100200190000017730000613d0000001f01400039000001e00210018f0000000d01200029000000000021004b0000000002000039000000010200403900000a0f0010009c000000a80000213d0000000100200190000000a80000c13d000000400010043f000000a00030008c000000930000413d0000000d02000029000000000202043300000ad70020009c000000930000213d0000000d020000290000008002200039000000000202043300000ad70020009c000000930000213d0000000c020000290000000002020433001100000002001d00000ad80020009c00001c410000213d000000110000006b00001c410000613d0000000d0100002900000060011000390000000001010433000d00000001001d00000a1b010000410000000000100443000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000a1c011001c70000800b02000039282d28280000040f000000010020019000001c3f0000613d0000001002000029000000a00220027000000a0c02200197000000000101043b000000000121004b00000db20000413d0000000d0010006b00001c400000a13d0000001201000029000000c00110027000000a0c02100197000000120120008900000a0c0010009c00000db20000213d000000120020008c000019820000c13d00000001020000390000198b0000013d000000400100043d000000640210003900000ae4030000410000000000320435000000440210003900000ae50300004100000000003204350000002402100039000000210300003900000fab0000013d00000aa5010000410000000000100443000000000100041200000004001004430000002400000443000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000aa6011001c70000800502000039282d28280000040f000000010020019000001c3f0000613d000000000101043b00000a11041001970000000001000411000000120010006c000017610000613d000000000041004b000002110000c13d0000000d02000029000000110020002a00000db20000413d0000000d0200002900000011012000290000001202000039000000000012041b0000000f02000029000000100020002a00000db20000413d0000001301000039000000000001041b0000000001000414000000040040008c000018310000c13d00000001020000390000000101000031000018420000013d0000001f0530018f00000a0e06300198000000400200043d00000000046200190000136f0000613d000000000701034f0000000008020019000000007907043c0000000008980436000000000048004b0000177a0000c13d0000136f0000013d0000000d03000029000000110330002900000a0c0010009c00000a0c01008041000000c001100210000000000003004b000017f10000c13d0000000002000411000017f50000013d000000120200002900000a0c0020009c00000a0c02008041000000400220021000000a0c0010009c00000a0c01008041000000c001100210000000000121019f00000ad9011001c70000000f02000029282d28280000040f000000600310027000000a0c03300197000000a00030008c000000a00400003900000000040340190000001f0640018f000000e0074001900000001205700029000017a20000613d000000000801034f0000001209000029000000008a08043c0000000009a90436000000000059004b0000179e0000c13d000000000006004b000017af0000613d000000000771034f0000000306600210000000000805043300000000086801cf000000000868022f000000000707043b0000010006600089000000000767022f00000000066701cf000000000686019f0000000000650435000100000003001f00030000000103550000000100200190000018250000613d0000001f01400039000001e00210018f0000001201200029000000000021004b0000000002000039000000010200403900000a0f0010009c000000a80000213d0000000100200190000000a80000c13d000000400010043f000000a00030008c000000930000413d0000001202000029000000000202043300000ad70020009c000000930000213d00000012020000290000008002200039000000000202043300000ad70020009c000000930000213d0000000b020000290000000002020433000f00000002001d00000ad80020009c00001c410000213d0000000f0000006b00001c410000613d000000120100002900000060011000390000000001010433001200000001001d00000a1b010000410000000000100443000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000a1c011001c70000800b02000039282d28280000040f000000010020019000001c3f0000613d0000000e02000029000000a00220027000000a0c02200197000000000101043b000000000121004b00000db20000413d000000120010006b00001c400000a13d0000001001000029000000c00110027000000a0c02100197000000120120008900000a0c0010009c00000db20000213d000000120020008c000019b20000c13d0000000102000039000019bb0000013d00000a13011001c7000080090200003900000000040004110000000005000019282d28230000040f0003000000010355000000600110027000010a0c0010019d00000a0c01100197000000000001004b0000020f0000613d00000a0f0010009c000000a80000213d0000001f0410003900000af7044001970000003f0440003900000af705400197000000400400043d0000000005540019000000000045004b0000000006000039000000010600403900000a0f0050009c000000a80000213d0000000100600190000000a80000c13d000000400050043f000000000614043600000af7031001980000001f0410018f00000000013600190000000305000367000018170000613d000000000705034f000000007807043c0000000006860436000000000016004b000018130000c13d000000000004004b0000020f0000613d000000000335034f0000000304400210000000000501043300000000054501cf000000000545022f000000000303043b0000010004400089000000000343022f00000000034301cf000000000353019f00000000003104350000020f0000013d0000001f0530018f00000a0e06300198000000400200043d00000000046200190000136f0000613d000000000701034f0000000008020019000000007907043c0000000008980436000000000048004b0000182c0000c13d0000136f0000013d0000000f03000029000000100330002900000a0c0010009c00000a0c01008041000000c001100210000000000003004b0000183a0000c13d00000000020400190000183d0000013d00000a13011001c700008009020000390000000005000019282d28230000040f0003000000010355000000600110027000010a0c0010019d00000a0c01100197000000000001004b0000020f0000613d00000a0f0010009c000000a80000213d0000001f0410003900000af7044001970000003f0440003900000af705400197000000400400043d0000000005540019000000000045004b0000000006000039000000010600403900000a0f0050009c000000a80000213d0000000100600190000000a80000c13d000000400050043f000000000614043600000af7031001980000001f0410018f00000000013600190000000305000367000018170000613d000000000705034f000000007807043c0000000006860436000000000016004b0000185b0000c13d000018170000013d0000000201000039000000000101041a00000af8011001670000000102000039000000000202041a00000000011200190000000f0010002a00000db20000413d0000000f011000290000000c02000039000000000202041a000000000021004b000018740000a13d000000400100043d000000440210003900000ae80300004100000000003204350000002402100039000000120300003900000ff50000013d00000009010000290000000201100039000000000201041a0000000f012000b90000000f0000006b0000187d0000613d0000000f031000fa000000000023004b00000db20000c13d0000000002000416000000000012004b000018cf0000813d000000400100043d000000440210003900000ad003000041000012fa0000013d00000009010000290000000301100039000000000101041a000000000001004b000018e70000c13d000000400100043d000000440210003900000ae30300004100000000003204350000002402100039000000170300003900000ff50000013d000002600100043d000200000001001d000002400100043d000300000001001d000002a00100043d000100000001001d000000000100041a00000a1202100197000000000262019f000000000020041b000000000200041400000a110510019700000a0c0020009c00000a0c02008041000000c00120021000000a13011001c70000800d02000039000000030300003900000a1404000041282d28230000040f0000000100200190000000930000613d0000001201000029000000000101043300000a0f0010009c000000a80000213d0000000302000039000000000202041a000000010320019000000001022002700000007f0220618f0000001f0020008c00000000040000390000000104002039000000000043004b00000b2a0000c13d000000200020008c000018c60000413d0000000303000039000000000030043f0000001f03100039000000050330027000000a150330009a000000200010008c00000a16030040410000001f02200039000000050220027000000a150220009a000000000023004b000018c60000813d000000000003041b0000000103300039000000000023004b000018c20000413d000000200010008c000019a60000413d0000000302000039000000000020043f00000af704100198000019e70000c13d000000200300003900000a1602000041000019f40000013d0000001201000029000000000010043f0000001101000029000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f0000000100200190000000930000613d000000000101043b000000000301041a0000000f0030002a00000db20000413d0000000f020000290000000003230019000000000031041b0000000001000411282d253f0000040f00000000010000190000282e0001042e0010000f001000bd00000010011000f90000000f0010006c00000db20000c13d0000001201000029000000000010043f0000001101000029000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f0000000100200190000000930000613d000000000101043b000000000201041a0000000f0020002a00000db20000413d0000000f02200029000000000021041b0000001701000039000000000101041a00000a1101100197000000000300041000000000020004110000001004000029282d25240000040f00000000010004110000000f02000029282d253f0000040f00000000010000190000282e0001042e0000000e03000029000000440130003900000ae602000041000000000021043500000024013000390000001c02000039000000000021043500000a2801000041000000000013043500000004013000390000002002000039000013d20000013d0000000e0300002900000a0c0030009c00000a0c03008041000000400330021000000a0c0010009c00000a0c01008041000000c001100210000000000131019f00000ad9011001c7282d28280000040f000000600310027000000a0c03300197000000a00030008c000000a00400003900000000040340190000001f0640018f000000e0074001900000000e057000290000192f0000613d000000000801034f0000000e09000029000000008a08043c0000000009a90436000000000059004b0000192b0000c13d000000000006004b0000193c0000613d000000000771034f0000000306600210000000000805043300000000086801cf000000000868022f000000000707043b0000010006600089000000000767022f00000000066701cf000000000686019f0000000000650435000100000003001f000300000001035500000001002001900000199a0000613d0000001f01400039000001e00210018f0000000e01200029000000000021004b0000000002000039000000010200403900000a0f0010009c000000a80000213d0000000100200190000000a80000c13d000000400010043f000000a00030008c000000930000413d0000000e02000029000000000202043300000ad70020009c000000930000213d0000000e020000290000008002200039000000000202043300000ad70020009c000000930000213d00000009020000290000000002020433000900000002001d00000ad80020009c00001c410000213d000000090000006b00001c410000613d0000000e0100002900000060011000390000000001010433000e00000001001d00000a1b010000410000000000100443000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000a1c011001c70000800b02000039282d28280000040f000000010020019000001c3f0000613d0000000c02000029000000a00220027000000a0c02200197000000000101043b000000000121004b00000db20000413d0000000e0010006b00001c400000a13d000000400200043d00000ad601000041000e00000002001d0000000001120436000800000001001d00000000010004140000000b02000029000000040020008c00001bc00000c13d0000000103000031000000a00030008c000000a004000039000000000403401900001beb0000013d0000000a030000390000000102000039000000010010019000000000043300a9000000010300603900000000022300a900000001011002720000000003040019000019840000c13d0000000e032000b9001200000003001d0000000e013000fa000000000021004b00000db20000c13d0000001001000029000000c00110027000000a0c02100197000000120120008900000a0c0010009c00000db20000213d000000120020008c000019ca0000c13d0000000102000039000019d30000013d0000001f0530018f00000a0e06300198000000400200043d00000000046200190000136f0000613d000000000701034f0000000008020019000000007907043c0000000008980436000000000048004b000019a10000c13d0000136f0000013d000000000001004b000000000200001900001a000000613d000000030210021000000af80220027f00000af80220016700000011030000290000000003030433000000000223016f0000000101100210000000000212019f00001a000000013d0000000a030000390000000102000039000000010010019000000000043300a9000000010300603900000000022300a900000001011002720000000003040019000019b40000c13d00000011032000b9001200000003001d00000011013000fa000000000021004b00000db20000c13d0000000e01000029000000c00110027000000a0c02100197000000120120008900000a0c0010009c00000db20000213d000000120020008c00001a780000c13d000000010200003900001a810000013d0000000a030000390000000102000039000000010010019000000000043300a9000000010300603900000000022300a900000001011002720000000003040019000019cc0000c13d00000011012000b9001000000001001d00000011011000fa000000000021004b00000db20000c13d0000001701000039000000000201041a000000400300043d00000ab201000041001100000003001d0000000000130435000000000100041400000a1102200197000000040020008c00001a2a0000c13d0000000103000031000000200030008c0000002004000039000000000403401900001a540000013d00000a16020000410000002003000039000000010540008a000000050550027000000a170550009a000000120700002900000000067300190000000006060433000000000062041b00000020033000390000000102200039000000000052004b000019ed0000c13d000000000014004b000019fe0000813d0000000304100210000000f80440018f00000af80440027f00000af80440016700000012033000290000000003030433000000000343016f000000000032041b000000010110021000000001021001bf0000000301000039000000000021041b0000001001000029000000000201043300000a0f0020009c000000a80000213d0000000401000039000000000401041a000000010040019000000001034002700000007f0330618f0000001f0030008c00000000050000390000000105002039000000000454013f000000010040019000000b2a0000c13d000000200030008c00001a220000413d000000000010043f0000001f04200039000000050440027000000a180440009a000000200020008c00000a19040040410000001f03300039000000050330027000000a180330009a000000000034004b00001a220000813d000000000004041b0000000104400039000000000034004b00001a1e0000413d000000200020008c00001a6c0000413d000000000010043f00000af70520019800001aa10000c13d000000200400003900000a190300004100001aad0000013d000000110300002900000a0c0030009c00000a0c03008041000000400330021000000a0c0010009c00000a0c01008041000000c001100210000000000131019f00000ad9011001c7282d28280000040f000000600310027000000a0c03300197000000200030008c000000200400003900000000040340190000001f0640018f0000002007400190000000110570002900001a430000613d000000000801034f0000001109000029000000008a08043c0000000009a90436000000000059004b00001a3f0000c13d000000000006004b00001a500000613d000000000771034f0000000306600210000000000805043300000000086801cf000000000868022f000000000707043b0000010006600089000000000767022f00000000066701cf000000000686019f0000000000650435000100000003001f0003000000010355000000010020019000001a950000613d0000001f01400039000000600210018f0000001101200029000000000021004b0000000002000039000000010200403900000a0f0010009c000000a80000213d0000000100200190000000a80000c13d000000400010043f000000200030008c000000930000413d00000011020000290000000002020433000000ff0020008c000000930000213d0000001203200089000000ff0030008c00000db20000213d000000120020008c00001b5a0000c13d000000010200003900001b630000013d000000000002004b000000000300001900001ab90000613d000000030320021000000af80330027f00000af8033001670000000f040000290000000004040433000000000334016f0000000102200210000000000323019f00001ab90000013d0000000a030000390000000102000039000000010010019000000000043300a9000000010300603900000000022300a90000000101100272000000000304001900001a7a0000c13d0000000f012000b9001000000001001d0000000f011000fa000000000021004b00000db20000c13d0000001701000039000000000201041a000000400300043d00000ab201000041001100000003001d0000000000130435000000000100041400000a1102200197000000040020008c00001aff0000c13d0000000103000031000000200030008c0000002004000039000000000403401900001b290000013d0000001f0530018f00000a0e06300198000000400200043d00000000046200190000136f0000613d000000000701034f0000000008020019000000007907043c0000000008980436000000000048004b00001a9c0000c13d0000136f0000013d00000a19030000410000002004000039000000010650008a000000050660027000000a1a0660009a00000010074000290000000007070433000000000073041b00000020044000390000000103300039000000000063004b00001aa60000c13d000000000025004b00001ab70000813d0000000305200210000000f80550018f00000af80550027f00000af80550016700000010044000290000000004040433000000000454016f000000000043041b000000010220021000000001032001bf000000000031041b0000000101000039000000000011041b000000010200008a0000000f01000039000000000021041b00000a1b010000410000000000100443000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000a1c011001c70000800b02000039282d28280000040f000000010020019000001c3f0000613d000000000101043b000001000010043f0000001e03000039000000000403041a00000af60440019700000001044001bf000000000043041b0000000c030000390000000904000029000000000043041b0000000d030000390000000a04000029000000000043041b0000000e03000029000000000403043300000a0f0040009c000000a80000213d0000001403000039000000000603041a000000010060019000000001056002700000007f0550618f0000001f0050008c00000000070000390000000107002039000000000676013f000000010060019000000b2a0000c13d000000200050008c00001af70000413d000000000030043f0000001f06400039000000050660027000000a1d0660009a000000200040008c00000a1e060040410000001f05500039000000050550027000000a1d0550009a000000000056004b00001af70000813d000000000006041b0000000106600039000000000056004b00001af30000413d000000200040008c00001b4e0000413d000000000030043f00000af70740019800001b7f0000c13d000000200600003900000a1e0500004100001b8b0000013d000000110300002900000a0c0030009c00000a0c03008041000000400330021000000a0c0010009c00000a0c01008041000000c001100210000000000131019f00000ad9011001c7282d28280000040f000000600310027000000a0c03300197000000200030008c000000200400003900000000040340190000001f0640018f0000002007400190000000110570002900001b180000613d000000000801034f0000001109000029000000008a08043c0000000009a90436000000000059004b00001b140000c13d000000000006004b00001b250000613d000000000771034f0000000306600210000000000805043300000000086801cf000000000868022f000000000707043b0000010006600089000000000767022f00000000066701cf000000000686019f0000000000650435000100000003001f0003000000010355000000010020019000001b420000613d0000001f01400039000000600110018f0000001102100029000000000012004b0000000001000039000000010100403900000a0f0020009c000000a80000213d0000000100100190000000a80000c13d0000000005020019000000400020043f000000200030008c000000930000413d00000011010000290000000001010433000000ff0010008c000000930000213d0000001202100089000000ff0020008c00000db20000213d000000120010008c00001c5f0000c13d000000010100003900001c680000013d0000001f0530018f00000a0e06300198000000400200043d00000000046200190000136f0000613d000000000701034f0000000008020019000000007907043c0000000008980436000000000048004b00001b490000c13d0000136f0000013d000000000004004b000000000500001900001b970000613d000000030540021000000af80550027f00000af8055001670000000d060000290000000006060433000000000556016f0000000104400210000000000545019f00001b970000013d0000000a040000390000000102000039000000010030019000000000054400a9000000010400603900000000022400a90000000103300272000000000405001900001b5c0000c13d00000012030000290000000f033000b90000000f0000006b00001b6a0000613d0000000f043000fa000000120040006c00000db20000c13d000000100000006b00001dc10000613d000000000002004b00001dc10000613d00000010043000fa00000000032400d90000001905000039000000000505041a000000000005004b00000efd0000613d000027100050008c00000db20000213d000027100650008900000000053600a9000000000042004b00001b7d0000213d00000000023500d9000000000062004b00000db20000c13d000027100350011a00000efd0000013d00000a1e050000410000002006000039000000010870008a000000050880027000000a1f0880009a0000000e096000290000000009090433000000000095041b00000020066000390000000105500039000000000085004b00001b840000c13d000000000047004b00001b950000813d0000000307400210000000f80770018f00000af80770027f00000af8077001670000000e066000290000000006060433000000000676016f000000000065041b000000010440021000000001054001bf000000000053041b0000000c03000029000000000403043300000a0f0040009c000000a80000213d0000000a03000039000000000603041a000000010060019000000001056002700000007f0550618f0000001f0050008c00000000070000390000000107002039000000000676013f000000010060019000000b2a0000c13d000000200050008c00001bb80000413d000000000030043f0000001f06400039000000050660027000000a200660009a000000200040008c00000a21060040410000001f05500039000000050550027000000a200550009a000000000056004b00001bb80000813d000000000006041b0000000106600039000000000056004b00001bb40000413d000000200040008c00001c470000413d000000000030043f00000af70740019800001c7b0000c13d000000200600003900000a210500004100001c870000013d0000000e0200002900000a0c0020009c00000a0c02008041000000400220021000000a0c0010009c00000a0c01008041000000c001100210000000000121019f00000ad9011001c70000000b02000029282d28280000040f000000600310027000000a0c03300197000000a00030008c000000a00400003900000000040340190000001f0640018f000000e0074001900000000e0570002900001bda0000613d000000000801034f0000000e09000029000000008a08043c0000000009a90436000000000059004b00001bd60000c13d000000000006004b00001be70000613d000000000771034f0000000306600210000000000805043300000000086801cf000000000868022f000000000707043b0000010006600089000000000767022f00000000066701cf000000000686019f0000000000650435000100000003001f0003000000010355000000010020019000001c530000613d0000001f01400039000001e00210018f0000000e01200029000000000021004b0000000002000039000000010200403900000a0f0010009c000000a80000213d0000000100200190000000a80000c13d000000400010043f000000a00030008c000000930000413d0000000e02000029000000000202043300000ad70020009c000000930000213d0000000e020000290000008002200039000000000202043300000ad70020009c000000930000213d00000008020000290000000002020433000b00000002001d00000ad80020009c00001c410000213d0000000b0000006b00001c410000613d0000000e0100002900000060011000390000000001010433000e00000001001d00000a1b010000410000000000100443000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000a1c011001c70000800b02000039282d28280000040f000000010020019000001c3f0000613d0000000a02000029000000a00220027000000a0c02200197000000000101043b000000000121004b00000db20000413d0000000e0010006b00001c400000a13d0000000c01000029000000c00110027000000a0c02100197000000120120008900000a0c0010009c00000db20000213d000000120020008c00001c300000613d0000000a02000039001000010000003d000000010010019000000000032200a9000000010200603900100010002000bd0000000101100272000000000203001900001c290000c13d0000000901000029000e0010001000bd0000000e011000f9000000100010006c00000db20000c13d0000000a01000029000000c00110027000000a0c02100197000000120120008900000a0c0010009c00000db20000213d000000120020008c00001d2c0000c13d000000010200003900001d350000013d000000000001042f000000400100043d000000440210003900000ada03000041000000000032043500000024021000390000001c0300003900000ff50000013d000000000004004b000000000500001900001c930000613d000000030540021000000af80550027f00000af8055001670000000b060000290000000006060433000000000556016f0000000104400210000000000545019f00001c930000013d0000001f0530018f00000a0e06300198000000400200043d00000000046200190000136f0000613d000000000701034f0000000008020019000000007907043c0000000008980436000000000048004b00001c5a0000c13d0000136f0000013d0000000a030000390000000101000039000000010020019000000000043300a9000000010300603900000000011300a90000000102200272000000000304001900001c610000c13d00000012020000290000000c022000b90000000c0000006b00001c6f0000613d0000000c032000fa000000120030006c00000db20000c13d000000100000006b00001dc10000613d000000000001004b00001dc10000613d00000010022000fa00000000061200d90000001903000039000000000303041a000000000003004b00001d200000c13d0000000d040000290000003e0000013d00000a21050000410000002006000039000000010870008a000000050880027000000a220880009a0000000c096000290000000009090433000000000095041b00000020066000390000000105500039000000000085004b00001c800000c13d000000000047004b00001c910000813d0000000307400210000000f80770018f00000af80770027f00000af8077001670000000c066000290000000006060433000000000676016f000000000065041b000000010440021000000001054001bf000000000053041b0000000803000029000000000503043300000a0f0050009c000000a80000213d0000000b04000039000000000604041a000000010060019000000001036002700000007f0330618f0000001f0030008c00000000070000390000000107002039000000000676013f000000010060019000000b2a0000c13d000000200030008c00001cb40000413d000000000040043f0000001f06500039000000050660027000000a230660009a000000200050008c00000a24060040410000001f03300039000000050330027000000a230330009a000000000036004b00001cb40000813d000000000006041b0000000106600039000000000036004b00001cb00000413d000000200050008c000000200300003900001cd40000413d000000000040043f00000af70850019800000a2406000041000000000703001900001cc70000613d0000002007000039000000010980008a000000050990027000000a250990009a000000080a700029000000000a0a04330000000000a6041b00000020077000390000000106600039000000000096004b00001cc00000c13d000000000058004b00001cd10000813d0000000308500210000000f80880018f00000af80880027f00000af80880016700000008077000290000000007070433000000000787016f000000000076041b000000010550021000000001065001bf00001cdf0000013d000000000005004b000000000600001900001cdf0000613d000000030650021000000af80660027f00000af80660016700000007070000290000000007070433000000000667016f0000000105500210000000000656019f000000000064041b00002710040000390000000e05000039000000000045041b0000000604000029000000800040043f0000000504000029000000a00040043f0000000304000029000000e00040043f0000000204000029000027110040008c00001cf70000413d000000400100043d000000440210003900000a2704000041000000000042043500000024021000390000001d04000039000000000042043500000a28020000410000000000210435000000040210003900000ffa0000013d0000000207000029000000c00070043f000000040400002900000a11044001970000001505000039000000000605041a00000a1206600197000000000446019f000000000045041b00000016040000390000000105000029000000000054041b0000001004000039000000000504041a00000af60250019700000008050000290000000005050433000000000005004b00000001022061bf000000000024041b00000140000004430000000602000029000001600020044300000180003004430000000502000029000001a0002004430000004002000039000001c000200443000001e000700443000000600200003900000200002004430000000302000029000002200020044300000080020000390000024000200443000002600010044300000100003004430000000501000039000001200010044300000a26010000410000282e0001042e000027100030008c00000db20000213d000027100430008900000000036400a9000000000021004b00001d290000213d00000000016300d9000000000041004b00000db20000c13d000027100630011a0000000d040000290000003e0000013d0000000a030000390000000102000039000000010010019000000000043300a9000000010300603900000000022300a90000000101100272000000000304001900001d2e0000c13d0000000b012000b9000c00000001001d0000000b011000fa000000000021004b00000db20000c13d0000001701000039000000000201041a000000400300043d00000ab201000041001000000003001d0000000000130435000000000100041400000a1102200197000000040020008c00001d490000c13d0000000103000031000000200030008c0000002004000039000000000403401900001d730000013d000000100300002900000a0c0030009c00000a0c03008041000000400330021000000a0c0010009c00000a0c01008041000000c001100210000000000131019f00000ad9011001c7282d28280000040f000000600310027000000a0c03300197000000200030008c000000200400003900000000040340190000001f0640018f0000002007400190000000100570002900001d620000613d000000000801034f0000001009000029000000008a08043c0000000009a90436000000000059004b00001d5e0000c13d000000000006004b00001d6f0000613d000000000771034f0000000306600210000000000805043300000000086801cf000000000868022f000000000707043b0000010006600089000000000767022f00000000066701cf000000000686019f0000000000650435000100000003001f0003000000010355000000010020019000001d8b0000613d0000001f01400039000000600210018f0000001001200029000000000021004b0000000002000039000000010200403900000a0f0010009c000000a80000213d0000000100200190000000a80000c13d000000400010043f000000200030008c000000930000413d00000010010000290000000001010433000000ff0010008c000000930000213d0000001202100089000000ff0020008c00000db20000213d000000120010008c00001d970000c13d000000010100003900001da00000013d0000001f0530018f00000a0e06300198000000400200043d00000000046200190000136f0000613d000000000701034f0000000008020019000000007907043c0000000008980436000000000048004b00001d920000c13d0000136f0000013d0000000a030000390000000101000039000000010020019000000000043300a9000000010300603900000000011300a90000000102200272000000000304001900001d990000c13d0000000e020000290000000d022000b90000000d0000006b00001da70000613d0000000d032000fa0000000e0030006c00000db20000c13d0000000c0000006b00001dc10000613d000000000001004b00001dc10000613d0000000c022000fa00100000001200e10000001903000039000000000303041a000000000003004b00001dbb0000613d000027100030008c00000db20000213d000027100430008900000010034000b9000000000021004b00001dba0000213d00000010013000fa000000000041004b00000db20000c13d00102710003001220000001201000029000000000010043f0000001101000029000000200010043f0000000001000414000018f00000013d00000acc01000041000000000010043f0000001201000039000000040010043f00000a2b010000410000282f0001043000000af90010009c00001dcc0000813d0000002001100039000000400010043f000000000001042d00000acc01000041000000000010043f0000004101000039000000040010043f00000a2b010000410000282f000104300000001f0220003900000af7022001970000000001120019000000000021004b0000000002000039000000010200403900000a0f0010009c00001dde0000213d000000010020019000001dde0000c13d000000400010043f000000000001042d00000acc01000041000000000010043f0000004101000039000000040010043f00000a2b010000410000282f000104300000000b05000039000000000405041a000000010640019000000001024002700000007f0220618f0000001f0020008c00000000010000390000000101002039000000000016004b00001e120000c13d000000400100043d0000000003210436000000000006004b00001dff0000613d000000000050043f000000000002004b00001e050000613d00000a240500004100000000040000190000000006430019000000000705041a000000000076043500000001055000390000002004400039000000000024004b00001df70000413d00001e060000013d00000af6044001970000000000430435000000000002004b0000002004000039000000000400603900001e060000013d00000000040000190000003f0240003900000af7032001970000000002130019000000000032004b0000000003000039000000010300403900000a0f0020009c00001e180000213d000000010030019000001e180000c13d000000400020043f000000000001042d00000acc01000041000000000010043f0000002201000039000000040010043f00000a2b010000410000282f0001043000000acc01000041000000000010043f0000004101000039000000040010043f00000a2b010000410000282f000104300003000000000002000000000201041a000000010320019000000001062002700000007f0660618f0000001f0060008c00000000040000390000000104002039000000000043004b00001e5d0000c13d000000400500043d0000000004650436000000000003004b00001e480000613d000100000004001d000300000006001d000200000005001d000000000010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000ad2011001c70000801002000039282d28280000040f000000010020019000001e690000613d0000000306000029000000000006004b00001e4e0000613d000000000201043b0000000001000019000000020500002900000001070000290000000003170019000000000402041a000000000043043500000001022000390000002001100039000000000061004b00001e400000413d00001e500000013d00000af6012001970000000000140435000000000006004b0000002001000039000000000100603900001e500000013d000000000100001900000002050000290000003f0110003900000af7021001970000000001520019000000000021004b0000000002000039000000010200403900000a0f0010009c00001e630000213d000000010020019000001e630000c13d000000400010043f0000000001050019000000000001042d00000acc01000041000000000010043f0000002201000039000000040010043f00000a2b010000410000282f0001043000000acc01000041000000000010043f0000004101000039000000040010043f00000a2b010000410000282f0001043000000000010000190000282f0001043000000000430104340000000001320436000000000003004b00001e770000613d000000000200001900000000051200190000000006240019000000000606043300000000006504350000002002200039000000000032004b00001e700000413d000000000213001900000000000204350000001f0230003900000af7022001970000000001210019000000000001042d00000020030000390000000004310436000000003202043400000000002404350000004001100039000000000002004b00001e8c0000613d000000000400001900000000051400190000000006430019000000000606043300000000006504350000002004400039000000000024004b00001e850000413d000000000312001900000000000304350000001f0220003900000af7022001970000000001120019000000000001042d00000ad80010009c00001ea20000213d000000630010008c00001ea20000a13d00000002030003670000000401300370000000000101043b00000a110010009c00001ea20000213d0000002402300370000000000202043b00000a110020009c00001ea20000213d0000004403300370000000000303043b000000000001042d00000000010000190000282f0001043000000afa0020009c00001ed40000813d00000000040100190000001f0120003900000af7011001970000003f0110003900000af705100197000000400100043d0000000005510019000000000015004b0000000007000039000000010700403900000a0f0050009c00001ed40000213d000000010070019000001ed40000c13d000000400050043f00000000052104360000000007420019000000000037004b00001eda0000213d00000af7062001980000001f0720018f0000000204400367000000000365001900001ec40000613d000000000804034f0000000009050019000000008a08043c0000000009a90436000000000039004b00001ec00000c13d000000000007004b00001ed10000613d000000000464034f0000000306700210000000000703043300000000076701cf000000000767022f000000000404043b0000010006600089000000000464022f00000000046401cf000000000474019f000000000043043500000000022500190000000000020435000000000001042d00000acc01000041000000000010043f0000004101000039000000040010043f00000a2b010000410000282f0001043000000000010000190000282f0001043000000a1102200197000000000020043f000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f000000010020019000001eea0000613d000000000101043b000000000001042d00000000010000190000282f00010430000000000100041100000a110110019800001f030000613d000000000010043f0000000601000039000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f000000010020019000001f070000613d0000000e02000039000000000202041a000000000101043b000000000101041a00000a0f01100197000000000112004b0000000001004019000000000001042d00000ae001000041000000000010043f00000ad9010000410000282f0001043000000000010000190000282f000104300007000000000002000300000002001d000500000001001d000600000003001d000000000003004b000020150000613d0000000601000029000000000010043f0000000501000039000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f0000000100200190000020130000613d000000000101043b000000000101041a000000000001004b00001f380000c13d0000000101000039000000000101041a000000060010006c000020150000a13d0000000602000029000000010220008a000700000002001d000000000020043f0000000501000039000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f0000000100200190000020130000613d000000000101043b000000000101041a000000000001004b000000070200002900001f250000613d00000abe00100198000020150000c13d000000050200002900000a1102200197000400000001001d00000a1101100197000700000002001d000000000021004b000020190000c13d0000000601000029000000000010043f0000000701000039000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f0000000100200190000020130000613d000000000101043b000200000001001d000000000201041a000000000300041100000a1104300197000000070040006c000500000002001d00001f790000613d000000000024004b00001f790000613d000100000004001d0000000701000029000000000010043f0000000801000039000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f0000000100200190000020130000613d000000000101043b0000000102000029000000000020043f000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f0000000100200190000020130000613d000000000101043b000000000101041a000000ff0010019000000005020000290000000003000411000020470000613d000000070000006b00001f900000613d0000001e01000039000000000101041a000000ff00100190000020220000613d000000000030043f0000001f01000039000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f0000000100200190000020130000613d000000000101043b000000000101041a000000ff001001900000000502000029000020330000c13d000000000002004b00001f940000613d0000000201000029000000000001041b0000000701000029000000000010043f0000000601000039000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f0000000100200190000020130000613d000000000101043b000000000201041a000000010220008a000000000021041b000000030100002900000a1101100197000500000001001d000000000010043f0000000601000039000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f0000000100200190000020130000613d000000000101043b000000000201041a0000000102200039000000000021041b00000a1b010000410000000000100443000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000a1c011001c70000800b02000039282d28280000040f00000001002001900000201d0000613d000000000101043b000300000001001d0000000601000029000000000010043f0000000501000039000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f0000000100200190000020130000613d0000000302000029000000a00220021000000005022001af00000b00022001c7000000000101043b000000000021041b000000040100002900000b0000100198000020020000c13d00000006010000290000000101100039000300000001001d000000000010043f0000000501000039000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f0000000100200190000020130000613d000000000101043b000000000101041a000000000001004b000020020000c13d0000000101000039000000000101041a000000030010006b000020020000613d0000000301000029000000000010043f0000000501000039000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f0000000100200190000020130000613d000000000101043b0000000402000029000000000021041b000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000a13011001c70000800d02000039000000040300003900000b0104000041000000070500002900000005060000290000000607000029282d28230000040f0000000100200190000020130000613d000000050000006b0000201e0000613d000000000001042d00000000010000190000282f0001043000000aee01000041000000000010043f00000ad9010000410000282f0001043000000afb01000041000000000010043f00000ad9010000410000282f00010430000000000001042f00000b0201000041000000000010043f00000ad9010000410000282f00010430000000400100043d000000440210003900000aff03000041000000000032043500000024021000390000001303000039000000000032043500000a2802000041000000000021043500000004021000390000002003000039000000000032043500000a0c0010009c00000a0c01008041000000400110021000000a29011001c70000282f00010430000000400100043d000000640210003900000afd030000410000000000320435000000440210003900000afe03000041000000000032043500000024021000390000002a03000039000000000032043500000a2802000041000000000021043500000004021000390000002003000039000000000032043500000a0c0010009c00000a0c01008041000000400110021000000aa9011001c70000282f0001043000000afc01000041000000000010043f00000ad9010000410000282f00010430000000000301001900000000011200a9000000000003004b000020520000613d00000000033100d9000000000023004b000020530000c13d000000000001042d00000acc01000041000000000010043f0000001101000039000000040010043f00000a2b010000410000282f00010430000000000001004b0000205c0000613d000000000001042d000000400100043d000000440210003900000ae703000041000000000032043500000024021000390000001d03000039000000000032043500000a2802000041000000000021043500000004021000390000002003000039000000000032043500000a0c0010009c00000a0c01008041000000400110021000000a29011001c70000282f0001043000000a11011001980000207f0000613d000000000010043f0000000601000039000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f0000000100200190000020830000613d000000000101043b000000000101041a00000a0f01100197000000000001042d00000ae001000041000000000010043f00000ad9010000410000282f0001043000000000010000190000282f00010430000000000001004b000020880000613d000000000001042d000000400100043d000000440210003900000aea03000041000000000032043500000024021000390000001303000039000000000032043500000a2802000041000000000021043500000004021000390000002003000039000000000032043500000a0c0010009c00000a0c01008041000000400110021000000a29011001c70000282f000104300006000000000002000400000001001d000000400b00043d0000001b01000039000000000101041a000600000001001d00000a1102100198000022400000613d0000001a01000039000000000101041a000500000001001d00000a1101100198000022400000613d000300000001001d00000ad60100004100000000051b04360000000001000414000000040020008c000020b10000c13d0000000103000031000000a00030008c000000a0040000390000000004034019000020df0000013d000100000005001d00000a0c00b0009c00000a0c0300004100000000030b4019000000400330021000000a0c0010009c00000a0c01008041000000c001100210000000000131019f00000ad9011001c700020000000b001d282d28280000040f000000020b000029000000600310027000000a0c03300197000000a00030008c000000a00400003900000000040340190000001f0640018f000000e00740019000000000057b0019000020cd0000613d000000000801034f00000000090b0019000000008a08043c0000000009a90436000000000059004b000020c90000c13d000000000006004b000020da0000613d000000000771034f0000000306600210000000000805043300000000086801cf000000000868022f000000000707043b0000010006600089000000000767022f00000000066701cf000000000686019f0000000000650435000100000003001f00030000000103550000000100200190000022560000613d00000001050000290000001f01400039000001e00210018f0000000001b20019000000000021004b0000000002000039000000010200403900000a0f0010009c000022280000213d0000000100200190000022280000c13d000000400010043f000000a00030008c000022260000413d00000000020b043300000ad70020009c000022260000213d0000008002b00039000000000202043300000ad70020009c000022260000213d000000000205043300000ad80020009c000022300000213d000000000002004b000022300000613d000100000002001d0000006001b000390000000001010433000200000001001d00000a1b010000410000000000100443000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000a1c011001c70000800b02000039282d28280000040f00000001002001900000222e0000613d0000000602000029000000a00220027000000a0c02200197000000000101043b000000000121004b0000000302000029000022200000413d000000020010006b0000222f0000a13d000000400b00043d00000ad60100004100000000051b04360000000001000414000000040020008c0000211b0000c13d0000000103000031000000a00030008c000000a0040000390000000004034019000021490000013d000200000005001d00000a0c00b0009c00000a0c0300004100000000030b4019000000400330021000000a0c0010009c00000a0c01008041000000c001100210000000000131019f00000ad9011001c700030000000b001d282d28280000040f000000030b000029000000600310027000000a0c03300197000000a00030008c000000a00400003900000000040340190000001f0640018f000000e00740019000000000057b0019000021370000613d000000000801034f00000000090b0019000000008a08043c0000000009a90436000000000059004b000021330000c13d000000000006004b000021440000613d000000000771034f0000000306600210000000000805043300000000086801cf000000000868022f000000000707043b0000010006600089000000000767022f00000000066701cf000000000686019f0000000000650435000100000003001f00030000000103550000000100200190000022620000613d00000002050000290000001f01400039000001e00210018f0000000001b20019000000000021004b0000000002000039000000010200403900000a0f0010009c000022280000213d0000000100200190000022280000c13d000000400010043f000000a00030008c000022260000413d00000000020b043300000ad70020009c000022260000213d0000008002b00039000000000202043300000ad70020009c000022260000213d000000000205043300000ad80020009c000022300000213d000000000002004b000022300000613d000200000002001d0000006001b000390000000001010433000300000001001d00000a1b010000410000000000100443000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000a1c011001c70000800b02000039282d28280000040f00000001002001900000222e0000613d0000000502000029000000a00220027000000a0c02200197000000000101043b000000000121004b000022200000413d000000030010006b0000222f0000a13d0000000601000029000000c00110027000000a0c02100197000000120120008900000a0c0010009c000022200000213d000000120020008c000021830000c13d00000001020000390000218c0000013d0000000a030000390000000102000039000000010010019000000000043300a9000000010300603900000000022300a900000001011002720000000003040019000021850000c13d00000001052000b900000001015000fa000000000021004b000022200000c13d0000000501000029000000c00110027000000a0c02100197000000120120008900000a0c0010009c000022200000213d000000120020008c0000219a0000c13d0000000102000039000021a30000013d0000000a030000390000000102000039000000010010019000000000043300a9000000010300603900000000022300a9000000010110027200000000030400190000219c0000c13d00000002062000b900000002016000fa000000000021004b000022200000c13d0000001701000039000000000201041a000000400b00043d00000ab20100004100000000001b0435000000000100041400000a1102200197000000040020008c000021b50000c13d0000000103000031000000200030008c00000020040000390000000004034019000021e50000013d000300000006001d000500000005001d00000a0c00b0009c00000a0c0300004100000000030b4019000000400330021000000a0c0010009c00000a0c01008041000000c001100210000000000131019f00000ad9011001c700060000000b001d282d28280000040f000000060b000029000000600310027000000a0c03300197000000200030008c000000200400003900000000040340190000001f0640018f000000200740019000000000057b0019000021d20000613d000000000801034f00000000090b0019000000008a08043c0000000009a90436000000000059004b000021ce0000c13d000000000006004b000021df0000613d000000000771034f0000000306600210000000000805043300000000086801cf000000000868022f000000000707043b0000010006600089000000000767022f00000000066701cf000000000686019f0000000000650435000100000003001f000300000001035500000001002001900000226e0000613d000000050500002900000003060000290000001f01400039000000600210018f0000000001b20019000000000021004b0000000002000039000000010200403900000a0f0010009c000022280000213d0000000100200190000022280000c13d000000400010043f000000200030008c000022260000413d00000000020b0433000000ff0020008c000022260000213d0000001201200089000000ff0010008c000022200000213d000000120020008c000021fc0000c13d0000000102000039000022050000013d0000000a030000390000000102000039000000010010019000000000043300a9000000010300603900000000022300a900000001011002720000000003040019000021fe0000c13d00000004015000b9000000040000006b0000220b0000613d00000004031000fa000000000053004b000022200000c13d000000000006004b000022500000613d000000000002004b000022500000613d00000000036100d900000000012300d90000001904000039000000000404041a000000000004004b0000221f0000613d000027100040008c000022200000213d000027100540008900000000041500a9000000000032004b0000221e0000213d00000000011400d9000000000051004b000022200000c13d000027100140011a000000000001042d00000acc01000041000000000010043f0000001101000039000000040010043f00000a2b010000410000282f0001043000000000010000190000282f0001043000000acc01000041000000000010043f0000004101000039000000040010043f00000a2b010000410000282f00010430000000000001042f000000400100043d000000440210003900000ada03000041000000000032043500000024021000390000001c03000039000000000032043500000a2802000041000000000021043500000004021000390000002003000039000000000032043500000a0c0010009c00000a0c01008041000000400110021000000a29011001c70000282f000104300000004401b0003900000ae60200004100000000002104350000002401b000390000001c02000039000000000021043500000a280100004100000000001b04350000000401b000390000002002000039000000000021043500000a0c00b0009c00000a0c0b0080410000004001b0021000000a29011001c70000282f0001043000000acc01000041000000000010043f0000001201000039000000040010043f00000a2b010000410000282f000104300000001f0530018f00000a0e06300198000000400200043d0000000004620019000022790000613d000000000701034f0000000008020019000000007907043c0000000008980436000000000048004b0000225d0000c13d000022790000013d0000001f0530018f00000a0e06300198000000400200043d0000000004620019000022790000613d000000000701034f0000000008020019000000007907043c0000000008980436000000000048004b000022690000c13d000022790000013d0000001f0530018f00000a0e06300198000000400200043d0000000004620019000022790000613d000000000701034f0000000008020019000000007907043c0000000008980436000000000048004b000022750000c13d000000000005004b000022860000613d000000000161034f0000000305500210000000000604043300000000065601cf000000000656022f000000000101043b0000010005500089000000000151022f00000000015101cf000000000161019f0000000000140435000000600130021000000a0c0020009c00000a0c020080410000004002200210000000000112019f0000282f00010430000000000001004b0000228f0000613d000000000001042d000000400100043d000000440210003900000ae303000041000000000032043500000024021000390000001703000039000000000032043500000a2802000041000000000021043500000004021000390000002003000039000000000032043500000a0c0010009c00000a0c01008041000000400110021000000a29011001c70000282f00010430000a000000000002000100000004001d000500000002001d000600000001001d000700000003001d000000000003004b0000243c0000613d0000000701000029000000000010043f0000000501000039000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f00000001002001900000243a0000613d000000000101043b000000000101041a000000000001004b000022d00000c13d0000000101000039000000000101041a000000070010006c0000243c0000a13d0000000702000029000000010220008a000800000002001d000000000020043f0000000501000039000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f00000001002001900000243a0000613d000000000101043b000000000101041a000000000001004b0000000802000029000022bd0000613d00000abe001001980000243c0000c13d000000060200002900000a1102200197000300000001001d00000a1101100197000800000002001d000000000021004b000024410000c13d0000000701000029000000000010043f0000000701000039000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f00000001002001900000243a0000613d000000000101043b000200000001001d000000000301041a000000000400041100000a1102400197000400000002001d000000080020006c000600000003001d000023110000613d000000040030006b000023110000613d0000000801000029000000000010043f0000000801000039000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f00000001002001900000243a0000613d000000000101043b0000000402000029000000000020043f000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f00000001002001900000243a0000613d000000000101043b000000000101041a000000ff00100190000000060300002900000000040004110000246e0000613d000000080000006b000023280000613d0000001e01000039000000000101041a000000ff00100190000024490000613d000000000040043f0000001f01000039000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f00000001002001900000243a0000613d000000000101043b000000000101041a000000ff0010019000000006030000290000245a0000c13d000000000003004b0000232c0000613d0000000201000029000000000001041b0000000801000029000000000010043f0000000601000039000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f00000001002001900000243a0000613d000000000101043b000000000201041a000000010220008a000000000021041b000000050100002900000a1101100197000600000001001d000000000010043f0000000601000039000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f00000001002001900000243a0000613d000000000101043b000000000201041a0000000102200039000000000021041b00000a1b010000410000000000100443000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000a1c011001c70000800b02000039282d28280000040f0000000100200190000024400000613d000000000101043b000200000001001d0000000701000029000000000010043f0000000501000039000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f00000001002001900000243a0000613d0000000202000029000000a0022002100000000606000029000000000262019f00000b00022001c7000000000101043b000000000021041b000000030100002900000b00001001980000239d0000c13d00000007010000290000000101100039000200000001001d000000000010043f0000000501000039000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f00000001002001900000243a0000613d000000000101043b000000000101041a000000000001004b00000006060000290000239d0000c13d0000000101000039000000000101041a000000020010006b0000239d0000613d0000000201000029000000000010043f0000000501000039000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f00000001002001900000243a0000613d000000000101043b0000000302000029000000000021041b0000000606000029000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000a13011001c70000800d02000039000000040300003900000b010400004100000008050000290000000707000029282d28230000040f00000001002001900000243a0000613d000000060000006b000024450000613d00000b0301000041000000000010044300000005010000290000000400100443000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000aae011001c70000800202000039282d28280000040f0000000100200190000024400000613d000000000101043b000000000001004b000024390000613d0000000008000415000000400b00043d0000006401b00039000000800700003900000000007104350000004401b00039000000070200002900000000002104350000002401b000390000000802000029000000000021043500000b040100004100000000001b04350000000401b00039000000040200002900000000002104350000008403b00039000000010100002900000000210104340000000000130435000000a403b00039000000000001004b000023db0000613d000000000400001900000000053400190000000006420019000000000606043300000000006504350000002004400039000000000014004b000023d40000413d0000000002310019000000000002043500000000040004140000000602000029000000040020008c000023e90000c13d00000000050004150000000a0550008a00000005055002100000000103000031000000200030008c00000020040000390000000004034019000024210000013d000700000008001d000500000007001d0000001f0110003900000af701100197000000a40110003900000a0c0010009c00000a0c01008041000000600110021000000a0c00b0009c00000a0c0300004100000000030b40190000004003300210000000000131019f00000a0c0040009c00000a0c04008041000000c003400210000000000113019f00080000000b001d282d28230000040f000000080b000029000000600310027000000a0c03300197000000200030008c000000200400003900000000040340190000001f0640018f000000200740019000000000057b00190000240c0000613d000000000801034f00000000090b0019000000008a08043c0000000009a90436000000000059004b000024080000c13d000000000006004b000024190000613d000000000771034f0000000306600210000000000805043300000000086801cf000000000868022f000000000707043b0000010006600089000000000767022f00000000066701cf000000000686019f0000000000650435000100000003001f00030000000103550000000005000415000000090550008a00000005055002100000000100200190000024720000613d00000007080000290000001f01400039000000600210018f0000000001b20019000000000021004b0000000002000039000000010200403900000a0f0010009c000024a40000213d0000000100200190000024a40000c13d000000400010043f000000200030008c0000243a0000413d00000000010b043300000af0001001980000243a0000c13d0000000502500270000000000201001f00000000020004150000000002280049000000000200000200000af10110019700000b040010009c000024a00000c13d000000000001042d00000000010000190000282f0001043000000aee01000041000000000010043f00000ad9010000410000282f00010430000000000001042f00000afb01000041000000000010043f00000ad9010000410000282f0001043000000b0201000041000000000010043f00000ad9010000410000282f00010430000000400100043d000000440210003900000aff03000041000000000032043500000024021000390000001303000039000000000032043500000a2802000041000000000021043500000004021000390000002003000039000000000032043500000a0c0010009c00000a0c01008041000000400110021000000a29011001c70000282f00010430000000400100043d000000640210003900000afd030000410000000000320435000000440210003900000afe03000041000000000032043500000024021000390000002a03000039000000000032043500000a2802000041000000000021043500000004021000390000002003000039000000000032043500000a0c0010009c00000a0c01008041000000400110021000000aa9011001c70000282f0001043000000afc01000041000000000010043f00000ad9010000410000282f00010430000000000003004b000024760000c13d00000060020000390000249d0000013d0000001f0230003900000a0d022001970000003f0220003900000b0504200197000000400200043d0000000004420019000000000024004b0000000005000039000000010500403900000a0f0040009c000024a40000213d0000000100500190000024a40000c13d000000400040043f0000001f0430018f000000000632043600000a0e05300198000500000006001d0000000003560019000024900000613d000000000601034f0000000507000029000000006806043c0000000007870436000000000037004b0000248c0000c13d000000000004004b0000249d0000613d000000000151034f0000000304400210000000000503043300000000054501cf000000000545022f000000000101043b0000010004400089000000000141022f00000000014101cf000000000151019f00000000001304350000000001020433000000000001004b000024aa0000c13d00000b0601000041000000000010043f00000ad9010000410000282f0001043000000acc01000041000000000010043f0000004101000039000000040010043f00000a2b010000410000282f00010430000000050200002900000a0c0020009c00000a0c02008041000000400220021000000a0c0010009c00000a0c010080410000006001100210000000000121019f0000282f00010430000000000001004b000024b60000613d000000000001042d000000400100043d000000440210003900000acd03000041000000000032043500000024021000390000001503000039000000000032043500000a2802000041000000000021043500000004021000390000002003000039000000000032043500000a0c0010009c00000a0c01008041000000400110021000000a29011001c70000282f00010430000000000100041a00000a11021001970000000001000411000000000012004b000024cd0000c13d000000000001042d000000400200043d00000ab00300004100000000003204350000000403200039000000000013043500000a0c0020009c00000a0c02008041000000400120021000000a2b011001c70000282f000104300001000000000002000000000001004b000025080000613d000100000001001d000000000010043f0000000501000039000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f0000000100200190000025060000613d000000000101043b000000000101041a000000000001004b000025030000c13d0000000101000039000000000101041a0000000102000029000000000021004b000025080000a13d000000010220008a000100000002001d000000000020043f0000000501000039000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f0000000100200190000025060000613d000000000101043b000000000101041a000000000001004b0000000102000029000024f00000613d00000abe00100198000025080000c13d000000000001042d00000000010000190000282f0001043000000aee01000041000000000010043f00000ad9010000410000282f000104300000000004020019000000400200043d000000200520003900000aab0600004100000000006504350000004405200039000000000035043500000a1103400197000000240420003900000000003404350000004403000039000000000032043500000b070020009c0000251e0000813d0000008003200039000000400030043f282d26fe0000040f000000000001042d00000acc01000041000000000010043f0000004101000039000000040010043f00000a2b010000410000282f000104300000000005020019000000400200043d000000200620003900000b080700004100000000007604350000006406200039000000000046043500000a11033001970000004404200039000000000034043500000a1103500197000000240420003900000000003404350000006403000039000000000032043500000b090020009c000025390000813d000000a003200039000000400030043f282d26fe0000040f000000000001042d00000acc01000041000000000010043f0000004101000039000000040010043f00000a2b010000410000282f00010430000c000000000002000500000001001d000000400100043d000700000001001d00000af90010009c0000268a0000813d00000007010000290000002003100039000400000003001d000000400030043f0000000000010435000600000002001d000000000002004b000026910000613d0000000101000039000000000101041a000a00000001001d00000a1b010000410000000000100443000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000a1c011001c70000800b02000039282d28280000040f0000000100200190000026900000613d000000000101043b000900000001001d0000000a01000029000000000010043f0000000501000039000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f0000000100200190000026510000613d000000050200002900000a11042001970000000902000029000000a0022002100000000603000029000000010030008c000000000300001900000b0003006041000000000223019f000000000242019f000000000101043b000000000021041b000900000004001d000000000040043f0000000601000039000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f0000000100200190000026510000613d000000060400002900000b0a024000d1000000000101043b000000000301041a0000000002230019000000000021041b000000090000006b000026950000613d0008000a0040002d00000000010000190000259e0000013d0000000a07000029000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000a13011001c70000800d02000039000000040300003900000b010400004100000000050000190000000906000029000a00000007001d282d28230000040f00000001002001900000000101000039000026510000613d00000001001001900000258e0000613d0000000a070000290000000107700039000000080070006c0000258f0000c13d00000001010000390000000802000029000000000021041b00000b0301000041000000000010044300000005010000290000000400100443000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000aae011001c70000800202000039282d28280000040f0000000100200190000026900000613d000000000101043b000000000001004b0000000406000029000026500000613d0000000102000039000000000802041a000000060980006a000000800700003900000b040a000041000000000100041100000a110b100197000100000007001d000200000008001d00030000000b001d000000000089004b000000000c000039000000010c00403900000001002001900000264c0000613d000000000d000415000000400e00043d0000006401e0003900000000007104350000000000ae04350000000401e000390000000000b104350000004401e0003900000000009104350000002401e000390000000000010435000000070100002900000000010104330000008402e000390000000000120435000000a402e00039000000000001004b000025e10000613d000000000300001900000000042300190000000005630019000000000505043300000000005404350000002003300039000000000013004b000025da0000413d0000000002210019000000000002043500000000040004140000000902000029000000040020008c000025ef0000c13d00000000050004150000000c0550008a00000005055002100000000103000031000000200030008c000000200400003900000000040340190000262e0000013d00060000000d001d00080000000c001d000a00000009001d0000001f0110003900000af701100197000000a40110003900000a0c0010009c00000a0c01008041000000600110021000000a0c00e0009c00000a0c0300004100000000030e40190000004003300210000000000131019f00000a0c0040009c00000a0c04008041000000c003400210000000000113019f00050000000e001d282d28230000040f000000050e000029000000600310027000000a0c03300197000000200030008c00000020040000390000000004034019000000200640019000000000056e0019000026120000613d000000000701034f00000000080e0019000000007907043c0000000008980436000000000058004b0000260e0000c13d0000001f074001900000261f0000613d000000000661034f0000000307700210000000000805043300000000087801cf000000000878022f000000000606043b0000010007700089000000000676022f00000000067601cf000000000686019f0000000000650435000100000003001f000300000001035500000000050004150000000b0550008a000000050550021000000001002001900000000a0900002900000b040a000041000000030b000029000000080c000029000000060d000029000026530000613d0000000406000029000000800700003900000002080000290000001f01400039000000600210018f0000000001e20019000000000021004b0000000002000039000000010200403900000a0f0010009c0000268a0000213d00000001002001900000268a0000c13d000000400010043f000000200030008c000026510000413d00000000010e043300000af000100198000026510000c13d00000001099000390000000502500270000000000201001f000000000200041500000000022d0049000000000200000200000af10110019700000b040010009c00000000020c0019000025c20000613d00000b0601000041000000000010043f00000ad9010000410000282f000104300000000101000039000000000101041a000000000081004b000026510000c13d000000000001042d00000000010000190000282f00010430000000000003004b000026570000c13d00000060020000390000267e0000013d0000001f0230003900000a0d022001970000003f0220003900000b0504200197000000400200043d0000000004420019000000000024004b0000000005000039000000010500403900000a0f0040009c0000268a0000213d00000001005001900000268a0000c13d000000400040043f0000001f0430018f000000000632043600000a0e05300198000100000006001d0000000003560019000026710000613d000000000601034f0000000107000029000000006806043c0000000007870436000000000037004b0000266d0000c13d000000000004004b0000267e0000613d000000000151034f0000000304400210000000000503043300000000054501cf000000000545022f000000000101043b0000010004400089000000000141022f00000000014101cf000000000151019f00000000001304350000000001020433000000000001004b0000000102000029000026480000613d00000a0c0020009c00000a0c02008041000000400220021000000a0c0010009c00000a0c010080410000006001100210000000000121019f0000282f0001043000000acc01000041000000000010043f0000004101000039000000040010043f00000a2b010000410000282f00010430000000000001042f00000b0c01000041000000000010043f00000ad9010000410000282f0001043000000b0b01000041000000000010043f00000ad9010000410000282f0001043000020000000000020000000202000039000000000302041a00000af8033001670000000102000039000000000202041a0000000002320019000000000012001a000026f80000413d000100000001001d00000000011200190000000c02000039000000000202041a000000000021004b000026d20000213d0000000f01000039000000000101041a000200000001001d00000a1b010000410000000000100443000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000a1c011001c70000800b02000039282d28280000040f0000000100200190000026d90000613d000000000101043b000000020010006c000026da0000413d000000000100041100000a1101100198000026e10000613d000000000010043f0000000601000039000000200010043f000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000abd011001c70000801002000039282d28280000040f0000000100200190000026e50000613d0000000e02000039000000000202041a000000000101043b000000000101041a00000a0f01100197000000000112004b0000000001004019000000010010006c000026e70000413d000000000001042d000000400100043d000000440210003900000acd03000041000000000032043500000024021000390000001503000039000026ed0000013d000000000001042f000000400100043d000000440210003900000ae103000041000000000032043500000024021000390000001603000039000026ed0000013d00000ae001000041000000000010043f00000ad9010000410000282f0001043000000000010000190000282f00010430000000400100043d000000440210003900000adf03000041000000000032043500000024021000390000001b03000039000000000032043500000a2802000041000000000021043500000004021000390000002003000039000000000032043500000a0c0010009c00000a0c01008041000000400110021000000a29011001c70000282f0001043000000acc01000041000000000010043f0000001101000039000000040010043f00000a2b010000410000282f0001043000040000000000020000000034020434000000000200041400000a110a1001970000000400a0008c0000272f0000c13d00000001010000320000276a0000613d00000afa0010009c000027c50000813d0000001f0310003900000af7033001970000003f0330003900000af703300197000000400b00043d00000000033b00190000000000b3004b0000000004000039000000010400403900000a0f0030009c000027c50000213d0000000100400190000027c50000c13d000000400030043f00000000051b043600000af7021001980000001f0310018f00000000012500190000000304000367000027210000613d000000000604034f000000006706043c0000000005750436000000000015004b0000271d0000c13d000000000003004b0000276b0000613d000000000224034f0000000303300210000000000401043300000000043401cf000000000434022f000000000202043b0000010003300089000000000232022f00000000023201cf000000000242019f00000000002104350000276b0000013d00000a0c0040009c00000a0c04008041000000600140021000000a0c0030009c00000a0c030080410000004003300210000000000131019f00000a0c0020009c00000a0c02008041000000c002200210000000000121019f00020000000a001d00000000020a0019282d28230000040f0003000000010355000000600310027000010a0c0030019d00000a0c043001980000278d0000613d0000001f0340003900000a0d033001970000003f0330003900000b0503300197000000400b00043d00000000033b00190000000000b3004b0000000005000039000000010500403900000a0f0030009c000000020a000029000027c50000213d0000000100500190000027c50000c13d000000400030043f0000001f0540018f00000000034b043600000a0e0640019800000000046300190000275c0000613d000000000701034f0000000008030019000000007907043c0000000008980436000000000048004b000027580000c13d000000000005004b000027900000613d000000000161034f0000000305500210000000000604043300000000065601cf000000000656022f000000000101043b0000010005500089000000000151022f00000000015101cf000000000161019f0000000000140435000027900000013d000000600b0000390000000002000415000000040220008a000000050220021000000000010b0433000000000001004b000027980000c13d00010000000b001d00020000000a001d00000b0301000041000000000010044300000004010000390000000400100443000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000aae011001c70000800202000039282d28280000040f0000000100200190000027db0000613d0000000002000415000000040220008a0000000502200210000000000101043b000000000001004b000000010b000029000027af0000c13d000000400100043d00000b0e02000041000000000021043500000004021000390000000403000039000027e90000013d000000600b0000390000008003000039000000020a00002900000000010b04330000000100200190000027d10000613d0000000002000415000000030220008a0000000502200210000000000001004b0000279b0000613d000000050220027000000000020b001f000027b50000013d00010000000b001d00000b030100004100000000001004430000000400a00443000000000100041400000a0c0010009c00000a0c01008041000000c00110021000000aae011001c70000800202000039282d28280000040f0000000100200190000027db0000613d0000000002000415000000030220008a0000000502200210000000000101043b000000000001004b000000010b000029000027e40000613d00000000010b0433000000050220027000000000020b001f000000000001004b000000020a000029000027c20000613d00000ad80010009c000027c30000213d000000200010008c000027c30000413d0000002001b000390000000001010433000000000001004b0000000002000039000000010200c039000000000021004b000027c30000c13d000000000001004b000027cb0000613d000000000001042d00000000010000190000282f0001043000000acc01000041000000000010043f0000004101000039000000040010043f00000a2b010000410000282f00010430000000400100043d00000b0f02000041000000000021043500000004021000390000000000a20435000027ea0000013d000000000001004b000027dc0000c13d000000400100043d00000b0d02000041000000000021043500000a0c0010009c00000a0c01008041000000400110021000000ad9011001c70000282f00010430000000000001042f00000a0c0030009c00000a0c03008041000000400230021000000a0c0010009c00000a0c010080410000006001100210000000000121019f0000282f00010430000000400100043d00000b0e02000041000000000021043500000004021000390000000203000029000000000032043500000a0c0010009c00000a0c01008041000000400110021000000a2b011001c70000282f00010430000000000001042f00000a0c0010009c00000a0c01008041000000400110021000000a0c0020009c00000a0c020080410000006002200210000000000112019f000000000200041400000a0c0020009c00000a0c02008041000000c002200210000000000112019f00000a13011001c70000801002000039282d28280000040f0000000100200190000028030000613d000000000101043b000000000001042d00000000010000190000282f0001043000000000050100190000000000200443000000050030008c000028130000413d000000040100003900000000020000190000000506200210000000000664001900000005066002700000000006060031000000000161043a0000000102200039000000000031004b0000280b0000413d00000a0c0030009c00000a0c030080410000006001300210000000000200041400000a0c0020009c00000a0c02008041000000c002200210000000000112019f00000b10011001c70000000002050019282d28280000040f0000000100200190000028220000613d000000000101043b000000000001042d000000000001042f00002826002104210000000102000039000000000001042d0000000002000019000000000001042d0000282b002104230000000102000039000000000001042d0000000002000019000000000001042d0000282d000004320000282e0001042e0000282f0001043000000000000000000000000000000000000000000000000000000000ffffffff00000000000000000000000000000000000000000000000000000001ffffffe000000000000000000000000000000000000000000000000000000000ffffffe0000000000000000000000000000000000000000000000000ffffffffffffffff8000000000000000000000000000000000000000000000000000000000000000000000000000000000000000ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff000000000000000000000000000000000000000002000000000000000000000000000000000000000000000000000000000000008be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e03da8a5f161a6c3ff06a60736d0ed24d7963cc6a5c4fafd2fa1dae9bb908e07a5c2575a0e9e593c00f959f8c92f12db2869c3395a3b0502d05e2516446f71f85b3da8a5f161a6c3ff06a60736d0ed24d7963cc6a5c4fafd2fa1dae9bb908e07a475ca53043ea007e5c65182cbb028f60d7179ff4b55739a3949b401801c942e658a35acfbc15ff81a39ae7d344fd709f28e8600b4aa8c65c6b64bfe7fe36bd19b75ca53043ea007e5c65182cbb028f60d7179ff4b55739a3949b401801c942e64796b89b91644bc98cd93958e4c9038275d622183e25ac5af08cc6b5d955391320200000200000000000000000000000000000004000000000000000000000000319284ad7d4265c99e51f9e0112e2425b1ad54f8c4e06d7a4191eaa263c72b14ce6d7b5282bd9a3661ae061feed1dbda4e52ab073b1f9285be6e155d9c38d4ec319284ad7d4265c99e51f9e0112e2425b1ad54f8c4e06d7a4191eaa263c72b1339a5844729cae3e308f36a5ce933956d7c6367997d26743ca06a70b77c062d58c65a7bb8d6351c1cf70c95a316cc6a92839c986682d98bc35f958f4883f9d2a839a5844729cae3e308f36a5ce933956d7c6367997d26743ca06a70b77c062d57fe8a4859c7bd88fc0f24184464406785daae8e84cb1860cc4a4eff72e05fe2470175b7a638427703f0dbe7bb9bbf987a2551717b34e79f33b5b1008d1fa01db9fe8a4859c7bd88fc0f24184464406785daae8e84cb1860cc4a4eff72e05fe2460000000200000000000000000000000000000180000001000000000000000000496e76616c696420636f6d6d697373696f6e2070657263656e7461676500000008c379a00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000640000000000000000000000001e4fbdf700000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000002400000000000000000000000000000000000000000000000000000000000000000000000000000000848769dd00000000000000000000000000000000000000000000000000000000c87b56dc00000000000000000000000000000000000000000000000000000000e985e9c400000000000000000000000000000000000000000000000000000000f2c4ce1d00000000000000000000000000000000000000000000000000000000f4f3b1ff00000000000000000000000000000000000000000000000000000000f4f3b20000000000000000000000000000000000000000000000000000000000f799bed400000000000000000000000000000000000000000000000000000000f9f92be400000000000000000000000000000000000000000000000000000000f2c4ce1e00000000000000000000000000000000000000000000000000000000f2fde38b00000000000000000000000000000000000000000000000000000000f45a2cd700000000000000000000000000000000000000000000000000000000eb7c571c00000000000000000000000000000000000000000000000000000000eb7c571d00000000000000000000000000000000000000000000000000000000ee5c2e6600000000000000000000000000000000000000000000000000000000f0bc91c000000000000000000000000000000000000000000000000000000000e985e9c500000000000000000000000000000000000000000000000000000000eae483c500000000000000000000000000000000000000000000000000000000d8cde1c500000000000000000000000000000000000000000000000000000000e26127e700000000000000000000000000000000000000000000000000000000e26127e800000000000000000000000000000000000000000000000000000000e58306f900000000000000000000000000000000000000000000000000000000e8a3d48500000000000000000000000000000000000000000000000000000000d8cde1c600000000000000000000000000000000000000000000000000000000db2e21bc00000000000000000000000000000000000000000000000000000000dfef66f500000000000000000000000000000000000000000000000000000000cce54e4300000000000000000000000000000000000000000000000000000000cce54e4400000000000000000000000000000000000000000000000000000000cd05de6d00000000000000000000000000000000000000000000000000000000d5abeb0100000000000000000000000000000000000000000000000000000000c87b56dd00000000000000000000000000000000000000000000000000000000cc1acb9d00000000000000000000000000000000000000000000000000000000a945bf7f00000000000000000000000000000000000000000000000000000000bfac4d0e00000000000000000000000000000000000000000000000000000000c4be5b5800000000000000000000000000000000000000000000000000000000c4be5b5900000000000000000000000000000000000000000000000000000000c627525500000000000000000000000000000000000000000000000000000000c7dd451700000000000000000000000000000000000000000000000000000000bfac4d0f00000000000000000000000000000000000000000000000000000000c01b59db00000000000000000000000000000000000000000000000000000000c2e5ec0400000000000000000000000000000000000000000000000000000000acaa27fb00000000000000000000000000000000000000000000000000000000acaa27fc00000000000000000000000000000000000000000000000000000000ad9e5e3e00000000000000000000000000000000000000000000000000000000b88d4fde00000000000000000000000000000000000000000000000000000000a945bf8000000000000000000000000000000000000000000000000000000000ac3adb1800000000000000000000000000000000000000000000000000000000925f78950000000000000000000000000000000000000000000000000000000095d89b400000000000000000000000000000000000000000000000000000000095d89b4100000000000000000000000000000000000000000000000000000000a0712d6800000000000000000000000000000000000000000000000000000000a22cb46500000000000000000000000000000000000000000000000000000000925f789600000000000000000000000000000000000000000000000000000000938e3d7b00000000000000000000000000000000000000000000000000000000944964cf000000000000000000000000000000000000000000000000000000008add9f05000000000000000000000000000000000000000000000000000000008add9f06000000000000000000000000000000000000000000000000000000008d56ef1d000000000000000000000000000000000000000000000000000000008da5cb5b00000000000000000000000000000000000000000000000000000000848769de00000000000000000000000000000000000000000000000000000000849355dc0000000000000000000000000000000000000000000000000000000043d3e4680000000000000000000000000000000000000000000000000000000062b974e1000000000000000000000000000000000000000000000000000000006f8b44af000000000000000000000000000000000000000000000000000000007225037f0000000000000000000000000000000000000000000000000000000072250380000000000000000000000000000000000000000000000000000000008034ed0600000000000000000000000000000000000000000000000000000000838aed02000000000000000000000000000000000000000000000000000000006f8b44b00000000000000000000000000000000000000000000000000000000070a0823100000000000000000000000000000000000000000000000000000000715018a6000000000000000000000000000000000000000000000000000000006bac9e8f000000000000000000000000000000000000000000000000000000006bac9e90000000000000000000000000000000000000000000000000000000006bb7b1d9000000000000000000000000000000000000000000000000000000006d5d40c60000000000000000000000000000000000000000000000000000000062b974e2000000000000000000000000000000000000000000000000000000006352211e00000000000000000000000000000000000000000000000000000000535ba42e0000000000000000000000000000000000000000000000000000000055f804b20000000000000000000000000000000000000000000000000000000055f804b3000000000000000000000000000000000000000000000000000000005b8ad42900000000000000000000000000000000000000000000000000000000600dd5ea00000000000000000000000000000000000000000000000000000000535ba42f00000000000000000000000000000000000000000000000000000000537df3b60000000000000000000000000000000000000000000000000000000054214f69000000000000000000000000000000000000000000000000000000004831302800000000000000000000000000000000000000000000000000000000483130290000000000000000000000000000000000000000000000000000000049b05fab000000000000000000000000000000000000000000000000000000004ada218b0000000000000000000000000000000000000000000000000000000043d3e4690000000000000000000000000000000000000000000000000000000044337ea10000000000000000000000000000000000000000000000000000000011131f1e000000000000000000000000000000000000000000000000000000002ccf7144000000000000000000000000000000000000000000000000000000003e5fe94e000000000000000000000000000000000000000000000000000000003e5fe94f0000000000000000000000000000000000000000000000000000000040c442de0000000000000000000000000000000000000000000000000000000042842e0e000000000000000000000000000000000000000000000000000000002ccf7145000000000000000000000000000000000000000000000000000000002f285709000000000000000000000000000000000000000000000000000000003ccfd60b000000000000000000000000000000000000000000000000000000001aef99b6000000000000000000000000000000000000000000000000000000001aef99b70000000000000000000000000000000000000000000000000000000023b872dd000000000000000000000000000000000000000000000000000000002a55205a0000000000000000000000000000000000000000000000000000000011131f1f0000000000000000000000000000000000000000000000000000000018160ddd00000000000000000000000000000000000000000000000000000000083d40c9000000000000000000000000000000000000000000000000000000000b0bfbe8000000000000000000000000000000000000000000000000000000000b0bfbe9000000000000000000000000000000000000000000000000000000000c894cfe000000000000000000000000000000000000000000000000000000000ee196f500000000000000000000000000000000000000000000000000000000083d40ca00000000000000000000000000000000000000000000000000000000095ea7b300000000000000000000000000000000000000000000000000000000039af9ea00000000000000000000000000000000000000000000000000000000039af9eb0000000000000000000000000000000000000000000000000000000006fdde0300000000000000000000000000000000000000000000000000000000081812fc00000000000000000000000000000000000000000000000000000000011013c60000000000000000000000000000000000000000000000000000000001ffc9a7000000000000000000000000000000000000002000000080000000000000000000000000000000000000000000000000ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff00000000000000000000000000000000310ab089e4439a4c15d089f94afb7896ff553aecb10793d0ab882de59d99a32e02000002000000000000000000000000000000440000000000000000000000006e742063616e20776974686472617700000000000000000000000000000000004f6e6c79206f776e6572206f7220636f6d6d697373696f6e2072656369706965000000000000000000000000000000000000008400000000000000000000000070a0823100000000000000000000000000000000000000000000000000000000a9059cbb00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000ffffffffffffff7f9cc7f708afc65944829bd487b90b72536b1951864fbfc14e125fc972a6507f3902000002000000000000000000000000000000240000000000000000000000005472616e73666572206661696c65640000000000000000000000000000000000118cdaa7000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000024000000800000000000000000313ce56700000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000400000080000000000000000000000000000000ff000000000000000000000000000000000000000000000000ffffffff0000000000000000ffffffffffffffffffffffffffffffffffffffff0000000000000000ffffffff000000000000000000000000000000000000000000000000000000000000000000000000000000200000000000000000000000000000000000000000000000000000000000000080000000800000000000000000fffffffffffffffffffffffffffffffffffffffffffffffffffffffffefd9a00fffffffffffffffffffffffffffffffffffffffffffffffffffffffffefd99ff546f6f206561726c7920746f20656d657267656e6379207769746864726177003a6357012c1a3ae0a17d304c9920310382d968ebcc4b1771f41c6b304205b57002000000000000000000000000000000000000400000000000000000000000000000000100000000000000000000000000000000000000000000000000000000546f6b656e20646f6573206e6f742065786973740000000000000000000000000000000000184f03e93ff9f4daa797ed6e38ed64bf6a1f01000000000000000000000000000000000000000000000000000004ee2d6d415b85acef810000000000000000000000000000000000000000000004ee2d6d415b85acef80ffffffff000000000000000000000000000000000000000000000000002386f26fc100000000000000000000000000000000000000000000000000000000000005f5e10000ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff30313233343536373839616263646566000000000000000000000000000000007f000000000000000000000000000000000000000000000000000000000000002e6a736f6e000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000ffffffffffffffdf6e6578697374656e7420746f6b656e00000000000000000000000000000000004552433732314d657461646174613a2055524920717565727920666f72206e6f4e487b7100000000000000000000000000000000000000000000000000000000546f74616c20737570706c792065786365656465640000000000000000000000000000000000000000000000000000000000000000000000ffffffffffffffbf7fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe0496e73756666696369656e742066756e647320666f72206d696e740000000000546965722073616c65206e6f74207374617274656400000000000000000000000200000000000000000000000000000000000020000000000000000000000000c59ca8fed3e5c51f5e82cfb366dfcefc7d26971433b4e88e0be394cfbdfa4a90496e76616c696420646973636f756e742070657263656e7461676500000000000000000000000000000000000000000000000064000000800000000000000000feaf968c0000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000ffffffffffffffffffff7fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff0000000000000000000000000000000000000004000000000000000000000000496e76616c69642070726963652066726f6d20436861696e6c696e6b00000000000000000000000000000000000000000000006000000000000000000000000017307eab39ab6107e8899845ad3d59bd9653f200f220920489ca2b5937696c31656e740000000000000000000000000000000000000000000000000000000000436f737420697320686967686572207468616e2074686520616d6f756e742073496e76616c696420616d6f756e7420746f206265206d696e74656400000000008f4eb604000000000000000000000000000000000000000000000000000000005075626c69632073616c65206e6f742061637469766500000000000000000000a5d4097edda6d87cb9329af83fb3712ef77eeb13738ffe43cc35a4ce305ad96250726963652070657220746f6b656e206e6f742073657400000000000000000072000000000000000000000000000000000000000000000000000000000000004e6f7420696e2070726573616c65206c69737420666f722074686973207469655072696365206665656420616464726573736573206e6f7420736574000000005061796d656e7420746f6b656e2061646472657373206e6f742073657400000045786365656473206d617820737570706c790000000000000000000000000000457863656564732074696572206d6178206d696e7420616d6f756e74000000005469657220646f6573206e6f74206578697374000000000000000000000000000000000000000000000000000000000000000040000000000000000000000000cfb3b942000000000000000000000000000000000000000000000000000000008c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b925df2d9b4200000000000000000000000000000000000000000000000000000000cf4700e40000000000000000000000000000000000000000000000000000000000000000ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff000000000000000000000000000000000000000000000000000000002a55205a0000000000000000000000000000000000000000000000000000000080ac58cd0000000000000000000000000000000000000000000000000000000001ffc9a7000000000000000000000000000000000000000000000000000000005b5e139f00000000000000000000000000000000000000000000000000000000ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff00ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe0ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff000000000000000000000000000000000000000000000000ffffffffffffffe00000000000000000000000000000000000000000000000010000000000000000a11481000000000000000000000000000000000000000000000000000000000059c896be000000000000000000000000000000000000000000000000000000007574652074726164657300000000000000000000000000000000000000000000426c61636b6c697374656420656e7469746965732063616e6e6f74206578656354726164696e672069732064697361626c6564000000000000000000000000000000000200000000000000000000000000000000000000000000000000000000ddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3efea553b34000000000000000000000000000000000000000000000000000000001806aa1896bbf26568e884a7374b41e002500962caba6a15023a8d90e8508b83150b7a020000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000003ffffffe0d1a57ed600000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000ffffffffffffff8023b872dd00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000ffffffffffffff6000000000000000000000000000000000000000000000000100000000000000012e07630000000000000000000000000000000000000000000000000000000000b562e8dd000000000000000000000000000000000000000000000000000000001425ea42000000000000000000000000000000000000000000000000000000009996b315000000000000000000000000000000000000000000000000000000005274afe70000000000000000000000000000000000000000000000000000000002000002000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000188192d3649382320be4f19d4eb479bb904f5f1359a990eebb8d32767691f71a
Constructor Arguments (ABI-Encoded and is the last bytes of the Contract Creation Code above)
00000000000000000000000000000000000000000000000000000000000001a000000000000000000000000000000000000000000000000000000000000001e00000000000000000000000000000000000000000000000000000000000000220000000000000000000000000000000000000000000000000000000000000270f000000000000000000000000000000000000000000000000008e1bc9bf040000000000000000000000000000000000000000000000000000000000000000024000000000000000000000000000000000000000000000000000000000000002a0000000000000000000000000ff383ed09ae2d216bd37b03797dd9a3a0f75c77a0000000000000000000000008f995e8961d2ff09d444ab4ec72d67f36aa2c8cc00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000064000000000000000000000000aa4306c1b90782ce2710d9512becd03168eaf7a200000000000000000000000000000000000000000000000000000000000001f400000000000000000000000000000000000000000000000000000000000000055a65656b7300000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000055a45454b5300000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000036697066733a2f2f516d61443766374c3152505679324d726b61483662796653424141554232373172724175357a525a4e6b6d547a372f000000000000000000000000000000000000000000000000000000000000000000000000000000000000
-----Decoded View---------------
Arg [0] : _name (string): Zeeks
Arg [1] : _symbol (string): ZEEKS
Arg [2] : _contractURI (string):
Arg [3] : _maxSupply (uint256): 9999
Arg [4] : _publicPrice (uint256): 40000000000000000
Arg [5] : _defaultBaseURI (string): ipfs://QmaD7f7L1RPVy2MrkaH6byfSBAAUB271rrAu5zRZNkmTz7/
Arg [6] : _notRevealedURI (string):
Arg [7] : _withdrawalRecipientAddress (address): 0xFF383ED09aE2D216BD37B03797DD9a3A0f75c77a
Arg [8] : _commissionRecipientAddress (address): 0x8F995E8961D2FF09d444aB4eC72d67f36aa2c8CC
Arg [9] : _fixedCommisionThreshold (uint256): 0
Arg [10] : _commissionPercentageIn10000 (uint256): 100
Arg [11] : _defaultRoyaltyRecipient (address): 0xAA4306c1b90782Ce2710D9512beCD03168eaF7A2
Arg [12] : _defaultRoyaltyPercentageIn10000 (uint256): 500
-----Encoded View---------------
22 Constructor Arguments found :
Arg [0] : 00000000000000000000000000000000000000000000000000000000000001a0
Arg [1] : 00000000000000000000000000000000000000000000000000000000000001e0
Arg [2] : 0000000000000000000000000000000000000000000000000000000000000220
Arg [3] : 000000000000000000000000000000000000000000000000000000000000270f
Arg [4] : 000000000000000000000000000000000000000000000000008e1bc9bf040000
Arg [5] : 0000000000000000000000000000000000000000000000000000000000000240
Arg [6] : 00000000000000000000000000000000000000000000000000000000000002a0
Arg [7] : 000000000000000000000000ff383ed09ae2d216bd37b03797dd9a3a0f75c77a
Arg [8] : 0000000000000000000000008f995e8961d2ff09d444ab4ec72d67f36aa2c8cc
Arg [9] : 0000000000000000000000000000000000000000000000000000000000000000
Arg [10] : 0000000000000000000000000000000000000000000000000000000000000064
Arg [11] : 000000000000000000000000aa4306c1b90782ce2710d9512becd03168eaf7a2
Arg [12] : 00000000000000000000000000000000000000000000000000000000000001f4
Arg [13] : 0000000000000000000000000000000000000000000000000000000000000005
Arg [14] : 5a65656b73000000000000000000000000000000000000000000000000000000
Arg [15] : 0000000000000000000000000000000000000000000000000000000000000005
Arg [16] : 5a45454b53000000000000000000000000000000000000000000000000000000
Arg [17] : 0000000000000000000000000000000000000000000000000000000000000000
Arg [18] : 0000000000000000000000000000000000000000000000000000000000000036
Arg [19] : 697066733a2f2f516d61443766374c3152505679324d726b6148366279665342
Arg [20] : 4141554232373172724175357a525a4e6b6d547a372f00000000000000000000
Arg [21] : 0000000000000000000000000000000000000000000000000000000000000000
Loading...
Loading
Loading...
Loading
[ Download: CSV Export ]
[ Download: CSV Export ]
A contract address hosts a smart contract, which is a set of code stored on the blockchain that runs when predetermined conditions are met. Learn more about addresses in our Knowledge Base.