文件 1 的 41:Address.sol
pragma solidity ^0.8.1;
library Address {
function isContract(address account) internal view returns (bool) {
return account.code.length > 0;
}
function sendValue(address payable recipient, uint256 amount) internal {
require(address(this).balance >= amount, "Address: insufficient balance");
(bool success, ) = recipient.call{value: amount}("");
require(success, "Address: unable to send value, recipient may have reverted");
}
function functionCall(address target, bytes memory data) internal returns (bytes memory) {
return functionCall(target, data, "Address: low-level call failed");
}
function functionCall(
address target,
bytes memory data,
string memory errorMessage
) internal returns (bytes memory) {
return functionCallWithValue(target, data, 0, errorMessage);
}
function functionCallWithValue(
address target,
bytes memory data,
uint256 value
) internal returns (bytes memory) {
return functionCallWithValue(target, data, value, "Address: low-level call with value failed");
}
function functionCallWithValue(
address target,
bytes memory data,
uint256 value,
string memory errorMessage
) internal returns (bytes memory) {
require(address(this).balance >= value, "Address: insufficient balance for call");
require(isContract(target), "Address: call to non-contract");
(bool success, bytes memory returndata) = target.call{value: value}(data);
return verifyCallResult(success, returndata, errorMessage);
}
function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) {
return functionStaticCall(target, data, "Address: low-level static call failed");
}
function functionStaticCall(
address target,
bytes memory data,
string memory errorMessage
) internal view returns (bytes memory) {
require(isContract(target), "Address: static call to non-contract");
(bool success, bytes memory returndata) = target.staticcall(data);
return verifyCallResult(success, returndata, errorMessage);
}
function functionDelegateCall(address target, bytes memory data) internal returns (bytes memory) {
return functionDelegateCall(target, data, "Address: low-level delegate call failed");
}
function functionDelegateCall(
address target,
bytes memory data,
string memory errorMessage
) internal returns (bytes memory) {
require(isContract(target), "Address: delegate call to non-contract");
(bool success, bytes memory returndata) = target.delegatecall(data);
return verifyCallResult(success, returndata, errorMessage);
}
function verifyCallResult(
bool success,
bytes memory returndata,
string memory errorMessage
) internal pure returns (bytes memory) {
if (success) {
return returndata;
} else {
if (returndata.length > 0) {
assembly {
let returndata_size := mload(returndata)
revert(add(32, returndata), returndata_size)
}
} else {
revert(errorMessage);
}
}
}
}
文件 2 的 41:AdventureERC721.sol
pragma solidity ^0.8.4;
import "./IAdventurous.sol";
import "./AdventureWhitelist.sol";
import "../initializable/IAdventureERC721Initializer.sol";
import "../utils/tokens/InitializableERC721.sol";
error AdventureApprovalToCaller();
error AlreadyInitializedAdventureERC721();
error AlreadyOnQuest();
error AnActiveQuestIsPreventingTransfers();
error CallerNotApprovedForAdventure();
error CallerNotTokenOwner();
error MaxSimultaneousQuestsCannotBeZero();
error MaxSimultaneousQuestsExceeded();
error NotOnQuest();
error QuestIdOutOfRange();
error TooManyActiveQuests();
abstract contract AdventureERC721 is InitializableERC721, AdventureWhitelist, IAdventurous, IAdventureERC721Initializer {
uint256 private constant MAX_CONCURRENT_QUESTS = 100;
uint256 internal constant TRANSFERRING_VIA_ERC721 = 1;
uint256 internal constant TRANSFERRING_VIA_ADVENTURE = 2;
bool private initializedAdventureERC721;
uint256 internal transferType;
uint256 private _maxSimultaneousQuests;
mapping (uint256 => uint256) internal blockingQuestCounts;
mapping (address => mapping (address => bool)) private operatorAdventureApprovals;
mapping (uint256 => mapping (address => uint32[])) public activeQuestList;
mapping (uint256 => mapping (address => mapping (uint32 => Quest))) public activeQuestLookup;
function initializeAdventureERC721(uint256 maxSimultaneousQuests_) public override onlyOwner {
if(initializedAdventureERC721) {
revert AlreadyInitializedAdventureERC721();
}
_validateMaxSimultaneousQuests(maxSimultaneousQuests_);
_maxSimultaneousQuests = maxSimultaneousQuests_;
initializedAdventureERC721 = true;
transferType = TRANSFERRING_VIA_ERC721;
}
function supportsInterface(bytes4 interfaceId) public view virtual override (InitializableERC721, IERC165) returns (bool) {
return
interfaceId == type(IAdventurous).interfaceId ||
interfaceId == type(IAdventureERC721Initializer).interfaceId ||
super.supportsInterface(interfaceId);
}
function adventureTransferFrom(address from, address to, uint256 tokenId) external override {
_requireCallerIsWhitelistedAdventure();
_requireCallerApprovedForAdventure(tokenId);
transferType = TRANSFERRING_VIA_ADVENTURE;
_transfer(from, to, tokenId);
transferType = TRANSFERRING_VIA_ERC721;
}
function adventureSafeTransferFrom(address from, address to, uint256 tokenId) external override {
_requireCallerIsWhitelistedAdventure();
_requireCallerApprovedForAdventure(tokenId);
transferType = TRANSFERRING_VIA_ADVENTURE;
_safeTransfer(from, to, tokenId, "");
transferType = TRANSFERRING_VIA_ERC721;
}
function adventureBurn(uint256 tokenId) external override {
_requireCallerIsWhitelistedAdventure();
_requireCallerApprovedForAdventure(tokenId);
transferType = TRANSFERRING_VIA_ADVENTURE;
_burn(tokenId);
transferType = TRANSFERRING_VIA_ERC721;
}
function enterQuest(uint256 tokenId, uint256 questId) external override {
_requireCallerIsWhitelistedAdventure();
_requireCallerApprovedForAdventure(tokenId);
_enterQuest(tokenId, _msgSender(), questId);
}
function exitQuest(uint256 tokenId, uint256 questId) external override {
_requireCallerIsWhitelistedAdventure();
_requireCallerApprovedForAdventure(tokenId);
_exitQuest(tokenId, _msgSender(), questId);
}
function bootFromAllQuests(uint256 tokenId, address adventure) external onlyOwner {
_requireAdventureRemovedFromWhitelist(adventure);
_exitAllQuests(tokenId, adventure, true);
}
function userExitQuest(uint256 tokenId, address adventure, uint256 questId) external {
_requireAdventureRemovedFromWhitelist(adventure);
_requireCallerOwnsToken(tokenId);
_exitQuest(tokenId, adventure, questId);
}
function userExitAllQuests(uint256 tokenId, address adventure) external {
_requireAdventureRemovedFromWhitelist(adventure);
_requireCallerOwnsToken(tokenId);
_exitAllQuests(tokenId, adventure, false);
}
function setAdventuresApprovedForAll(address operator, bool approved) external {
address tokenOwner = _msgSender();
if(tokenOwner == operator) {
revert AdventureApprovalToCaller();
}
operatorAdventureApprovals[tokenOwner][operator] = approved;
emit AdventureApprovalForAll(tokenOwner, operator, approved);
}
function areAdventuresApprovedForAll(address owner, address operator) public view returns (bool) {
return operatorAdventureApprovals[owner][operator];
}
function getQuestCount(uint256 tokenId, address adventure) public override view returns (uint256) {
return activeQuestList[tokenId][adventure].length;
}
function getTimeOnQuest(uint256 tokenId, address adventure, uint256 questId) public override view returns (uint256) {
(bool participatingInQuest, uint256 startTimestamp,) = isParticipatingInQuest(tokenId, adventure, questId);
return participatingInQuest ? (block.timestamp - startTimestamp) : 0;
}
function isParticipatingInQuest(uint256 tokenId, address adventure, uint256 questId) public override view returns (bool participatingInQuest, uint256 startTimestamp, uint256 index) {
if(questId > type(uint32).max) {
revert QuestIdOutOfRange();
}
Quest storage quest = activeQuestLookup[tokenId][adventure][uint32(questId)];
participatingInQuest = quest.isActive;
startTimestamp = quest.startTimestamp;
index = quest.arrayIndex;
return (participatingInQuest, startTimestamp, index);
}
function getActiveQuests(uint256 tokenId, address adventure) public override view returns (Quest[] memory activeQuests) {
uint256 questCount = getQuestCount(tokenId, adventure);
activeQuests = new Quest[](questCount);
uint32[] memory activeQuestIdList = activeQuestList[tokenId][adventure];
for(uint256 i = 0; i < questCount; ++i) {
activeQuests[i] = activeQuestLookup[tokenId][adventure][activeQuestIdList[i]];
}
return activeQuests;
}
function maxSimultaneousQuests() public view returns (uint256) {
return _maxSimultaneousQuests;
}
function _enterQuest(uint256 tokenId, address adventure, uint256 questId) internal {
(bool participatingInQuest,,) = isParticipatingInQuest(tokenId, adventure, questId);
if(participatingInQuest) {
revert AlreadyOnQuest();
}
uint256 currentQuestCount = getQuestCount(tokenId, adventure);
if(currentQuestCount >= _maxSimultaneousQuests) {
revert TooManyActiveQuests();
}
uint32 castedQuestId = uint32(questId);
activeQuestList[tokenId][adventure].push(castedQuestId);
activeQuestLookup[tokenId][adventure][castedQuestId].isActive = true;
activeQuestLookup[tokenId][adventure][castedQuestId].startTimestamp = uint64(block.timestamp);
activeQuestLookup[tokenId][adventure][castedQuestId].questId = castedQuestId;
activeQuestLookup[tokenId][adventure][castedQuestId].arrayIndex = uint32(currentQuestCount);
address ownerOfToken = ownerOf(tokenId);
emit QuestUpdated(tokenId, ownerOfToken, adventure, questId, true, false);
if(IAdventure(adventure).questsLockTokens()) {
unchecked {
++blockingQuestCounts[tokenId];
}
}
IAdventure(adventure).onQuestEntered(ownerOfToken, tokenId, questId);
}
function _exitQuest(uint256 tokenId, address adventure, uint256 questId) internal {
(bool participatingInQuest, uint256 startTimestamp, uint256 index) = isParticipatingInQuest(tokenId, adventure, questId);
if(!participatingInQuest) {
revert NotOnQuest();
}
uint32 castedQuestId = uint32(questId);
uint256 lastArrayIndex = getQuestCount(tokenId, adventure) - 1;
if(index != lastArrayIndex) {
activeQuestList[tokenId][adventure][index] = activeQuestList[tokenId][adventure][lastArrayIndex];
activeQuestLookup[tokenId][adventure][activeQuestList[tokenId][adventure][lastArrayIndex]].arrayIndex = uint32(index);
}
activeQuestList[tokenId][adventure].pop();
delete activeQuestLookup[tokenId][adventure][castedQuestId];
address ownerOfToken = ownerOf(tokenId);
emit QuestUpdated(tokenId, ownerOfToken, adventure, questId, false, false);
if(IAdventure(adventure).questsLockTokens()) {
--blockingQuestCounts[tokenId];
}
IAdventure(adventure).onQuestExited(ownerOfToken, tokenId, questId, startTimestamp);
}
function _exitAllQuests(uint256 tokenId, address adventure, bool booted) internal {
address tokenOwner = ownerOf(tokenId);
uint256 questCount = getQuestCount(tokenId, adventure);
if(IAdventure(adventure).questsLockTokens()) {
blockingQuestCounts[tokenId] -= questCount;
}
for(uint256 i = 0; i < questCount;) {
uint32 questId = activeQuestList[tokenId][adventure][i];
Quest memory quest = activeQuestLookup[tokenId][adventure][questId];
uint256 startTimestamp = quest.startTimestamp;
emit QuestUpdated(tokenId, tokenOwner, adventure, questId, false, booted);
delete activeQuestLookup[tokenId][adventure][questId];
IAdventure(adventure).onQuestExited(tokenOwner, tokenId, questId, startTimestamp);
unchecked {
++i;
}
}
delete activeQuestList[tokenId][adventure];
}
function _beforeTokenTransfer(address , address , uint256 tokenId) internal virtual override {
if(blockingQuestCounts[tokenId] > 0) {
revert AnActiveQuestIsPreventingTransfers();
}
}
function _requireCallerApprovedForAdventure(uint256 tokenId) internal view {
if(!areAdventuresApprovedForAll(ownerOf(tokenId), _msgSender())) {
revert CallerNotApprovedForAdventure();
}
}
function _requireCallerOwnsToken(uint256 tokenId) internal view {
if(ownerOf(tokenId) != _msgSender()) {
revert CallerNotTokenOwner();
}
}
function _validateMaxSimultaneousQuests(uint256 maxSimultaneousQuests_) internal pure {
if(maxSimultaneousQuests_ == 0) {
revert MaxSimultaneousQuestsCannotBeZero();
}
if(maxSimultaneousQuests_ > MAX_CONCURRENT_QUESTS) {
revert MaxSimultaneousQuestsExceeded();
}
}
}
文件 3 的 41:AdventureNFT.sol
pragma solidity ^0.8.4;
import "./AdventureERC721.sol";
import "../initializable/IRoyaltiesInitializer.sol";
import "../initializable/IURIInitializer.sol";
import "@openzeppelin/contracts/token/common/ERC2981.sol";
error AlreadyInitializedRoyalties();
error AlreadyInitializedURI();
error ExceedsMaxRoyaltyFee();
error NonexistentToken();
abstract contract AdventureNFT is AdventureERC721, ERC2981, IRoyaltiesInitializer, IURIInitializer {
using Strings for uint256;
uint96 private constant MAX_ROYALTY_FEE_NUMERATOR = 10000;
bool private initializedRoyalties;
bool private initializedURI;
string public baseTokenURI;
string public suffixURI = ".json";
event BaseURISet(string baseTokenURI);
event SuffixURISet(string suffixURI);
event RoyaltySet(address receiver, uint96 feeNumerator);
function initializeRoyalties(address receiver, uint96 feeNumerator) public override onlyOwner {
if(initializedRoyalties) {
revert AlreadyInitializedRoyalties();
}
setRoyaltyInfo(receiver, feeNumerator);
initializedRoyalties = true;
}
function initializeURI(string memory baseURI_, string memory suffixURI_) public override onlyOwner {
if(initializedURI) {
revert AlreadyInitializedURI();
}
setBaseURI(baseURI_);
setSuffixURI(suffixURI_);
initializedURI = true;
}
function _baseURI() internal view virtual override returns (string memory) {
return baseTokenURI;
}
function setBaseURI(string memory baseTokenURI_) public onlyOwner {
baseTokenURI = baseTokenURI_;
emit BaseURISet(baseTokenURI_);
}
function setSuffixURI(string memory suffixURI_) public onlyOwner {
suffixURI = suffixURI_;
emit SuffixURISet(suffixURI_);
}
function setRoyaltyInfo(address receiver, uint96 feeNumerator) public onlyOwner {
if(feeNumerator > MAX_ROYALTY_FEE_NUMERATOR) {
revert ExceedsMaxRoyaltyFee();
}
_setDefaultRoyalty(receiver, feeNumerator);
emit RoyaltySet(receiver, feeNumerator);
}
function tokenURI(uint256 tokenId) public view virtual override returns (string memory) {
if(!_exists(tokenId)) {
revert NonexistentToken();
}
string memory baseURI = _baseURI();
return bytes(baseURI).length > 0
? string(abi.encodePacked(baseURI, tokenId.toString(), suffixURI))
: "";
}
function supportsInterface(bytes4 interfaceId) public view virtual override (AdventureERC721, ERC2981, IERC165) returns (bool) {
return
interfaceId == type(IRoyaltiesInitializer).interfaceId ||
interfaceId == type(IURIInitializer).interfaceId ||
super.supportsInterface(interfaceId);
}
}
文件 4 的 41:AdventureWhitelist.sol
pragma solidity ^0.8.4;
import "./IAdventure.sol";
import "../utils/access/InitializableOwnable.sol";
import "@openzeppelin/contracts/token/ERC721/IERC721.sol";
error AdventureIsStillWhitelisted();
error AlreadyWhitelisted();
error ArrayIndexOverflowsUint128();
error CallerNotAWhitelistedAdventure();
error InvalidAdventureContract();
error NotWhitelisted();
abstract contract AdventureWhitelist is InitializableOwnable {
struct AdventureDetails {
bool isWhitelisted;
uint128 arrayIndex;
}
event AdventureWhitelistUpdated(address indexed adventure, bool whitelisted);
address[] public whitelistedAdventureList;
mapping (address => AdventureDetails) public whitelistedAdventures;
function isAdventureWhitelisted(address account) public view returns (bool) {
return whitelistedAdventures[account].isWhitelisted;
}
function whitelistAdventure(address adventure) external onlyOwner {
if(isAdventureWhitelisted(adventure)) {
revert AlreadyWhitelisted();
}
if(!IERC165(adventure).supportsInterface(type(IAdventure).interfaceId)) {
revert InvalidAdventureContract();
}
uint256 arrayIndex = whitelistedAdventureList.length;
if(arrayIndex > type(uint128).max) {
revert ArrayIndexOverflowsUint128();
}
whitelistedAdventures[adventure].isWhitelisted = true;
whitelistedAdventures[adventure].arrayIndex = uint128(arrayIndex);
whitelistedAdventureList.push(adventure);
emit AdventureWhitelistUpdated(adventure, true);
}
function unwhitelistAdventure(address adventure) external onlyOwner {
if(!isAdventureWhitelisted(adventure)) {
revert NotWhitelisted();
}
uint128 itemPositionToDelete = whitelistedAdventures[adventure].arrayIndex;
uint256 arrayEndIndex = whitelistedAdventureList.length - 1;
if(itemPositionToDelete != arrayEndIndex) {
whitelistedAdventureList[itemPositionToDelete] = whitelistedAdventureList[arrayEndIndex];
whitelistedAdventures[whitelistedAdventureList[itemPositionToDelete]].arrayIndex = itemPositionToDelete;
}
whitelistedAdventureList.pop();
delete whitelistedAdventures[adventure];
emit AdventureWhitelistUpdated(adventure, false);
}
function _requireCallerIsWhitelistedAdventure() internal view {
if(!isAdventureWhitelisted(_msgSender())) {
revert CallerNotAWhitelistedAdventure();
}
}
function _requireAdventureRemovedFromWhitelist(address adventure) internal view {
if(isAdventureWhitelisted(adventure)) {
revert AdventureIsStillWhitelisted();
}
}
}
文件 5 的 41:BlacklistedTransferAdventureNFT.sol
pragma solidity ^0.8.4;
import "../adventures/AdventureNFT.sol";
import "../opensea/operator-filter-registry/InitializableDefaultOperatorFilterer.sol";
abstract contract BlacklistedTransferAdventureNFT is AdventureNFT, InitializableDefaultOperatorFilterer {
function setApprovalForAll(address operator, bool approved) public virtual override onlyAllowedOperatorApproval(operator) {
super.setApprovalForAll(operator, approved);
}
function approve(address operator, uint256 tokenId) public virtual override onlyAllowedOperatorApproval(operator) {
super.approve(operator, tokenId);
}
function transferFrom(address from, address to, uint256 tokenId) public virtual override onlyAllowedOperator(from) {
super.transferFrom(from, to, tokenId);
}
function safeTransferFrom(address from, address to, uint256 tokenId) public virtual override onlyAllowedOperator(from) {
super.safeTransferFrom(from, to, tokenId);
}
function safeTransferFrom(address from, address to, uint256 tokenId, bytes memory data) public virtual override onlyAllowedOperator(from) {
super.safeTransferFrom(from, to, tokenId, data);
}
}
文件 6 的 41:ClaimPeriodBase.sol
pragma solidity ^0.8.4;
import "../access/InitializableOwnable.sol";
error ClaimsMustBeClosedToReopen();
error ClaimPeriodAlreadyInitialized();
error ClaimPeriodIsNotOpen();
error ClaimPeriodMustBeClosedInTheFuture();
error ClaimPeriodMustBeInitialized();
abstract contract ClaimPeriodBase is InitializableOwnable {
bool private claimPeriodInitialized;
uint256 private claimPeriodClosingTimestamp;
event ClaimPeriodClosing(uint256 claimPeriodClosingTimestamp);
event ClaimPeriodOpened(uint256 claimPeriodClosingTimestamp);
function openClaims(uint256 claimPeriodClosingTimestamp_) external onlyOwner {
if(claimPeriodClosingTimestamp_ <= block.timestamp) {
revert ClaimPeriodMustBeClosedInTheFuture();
}
_onClaimPeriodOpening();
if(claimPeriodInitialized) {
if(block.timestamp < claimPeriodClosingTimestamp) {
revert ClaimsMustBeClosedToReopen();
}
} else {
claimPeriodInitialized = true;
}
claimPeriodClosingTimestamp = claimPeriodClosingTimestamp_;
emit ClaimPeriodOpened(claimPeriodClosingTimestamp_);
}
function closeClaims(uint256 claimPeriodClosingTimestamp_) external onlyOwner {
_requireClaimsOpen();
if(claimPeriodClosingTimestamp_ <= block.timestamp) {
revert ClaimPeriodMustBeClosedInTheFuture();
}
claimPeriodClosingTimestamp = claimPeriodClosingTimestamp_;
emit ClaimPeriodClosing(claimPeriodClosingTimestamp_);
}
function getClaimPeriodClosingTimestamp() external view returns (uint256) {
return claimPeriodClosingTimestamp;
}
function isClaimPeriodOpen() external view returns (bool) {
return _isClaimPeriodOpen();
}
function _isClaimPeriodOpen() internal view returns (bool) {
return claimPeriodInitialized && block.timestamp < claimPeriodClosingTimestamp;
}
function _requireClaimsOpen() internal view {
if(!_isClaimPeriodOpen()) {
revert ClaimPeriodIsNotOpen();
}
}
function _onClaimPeriodOpening() internal virtual {}
}
文件 7 的 41:ClaimableHolderMint.sol
pragma solidity ^0.8.4;
import "./ClaimPeriodBase.sol";
import "./MintTokenBase.sol";
import "./SequentialMintBase.sol";
import "../access/InitializableOwnable.sol";
import "../../initializable/IRootCollectionInitializer.sol";
import "@openzeppelin/contracts/token/ERC721/IERC721.sol";
import "@openzeppelin/contracts/utils/introspection/ERC165.sol";
error CallerDoesNotOwnRootTokenId();
error CannotInitializeIneligibleTokensAfterClaimsOpen();
error CollectionAddressIsNotAnERC721Token();
error IneligibleTokensAlreadyInitialized();
error IneligibleTokenArrayMustBeInAscendingOrder();
error IneligibleTokensFinalized();
error IneligibleTokensHaveNotBeenFinalized();
error IneligibleTokenMustBeWithinMaxSupply();
error InputArrayLengthMismatch();
error InvalidRootCollectionAddress();
error InvalidRootCollectionTokenId();
error MaxSupplyOfRootTokenCannotBeZero();
error MustSpecifyAtLeastOneIneligibleToken();
error MustSpecifyAtLeastOneRootCollection();
error RootCollectionsAlreadyInitialized();
error RootCollectionsHaveNotBeenInitialized();
error TokenIdAlreadyClaimed();
error TokensPerClaimMustBeBetweenOneAndTen();
error MaxNumberOfRootCollectionsExceeded();
error BatchSizeMustBeGreaterThanZero();
error BatchSizeGreaterThanMaximum();
abstract contract ClaimableHolderMint is InitializableOwnable, ClaimPeriodBase, MintTokenBase, SequentialMintBase, IRootCollectionInitializer {
struct ClaimableRootCollection {
bool isRootCollection;
IERC721 rootCollection;
uint256 maxSupply;
uint256 tokensPerClaim;
uint256[] claimedTokenTracker;
mapping(uint256 => uint256) ineligibleTokenBitmaps;
}
uint256 private constant MAX_MINTS_PER_TRANSACTION = 300;
uint256 private constant MAX_ROOT_COLLECTIONS = 25;
bool private initializedRootCollections;
bool private finalizedIneligibleTokens;
mapping (address => ClaimableRootCollection) private rootCollectionLookup;
event ClaimMinted(address indexed rootCollection, uint256 indexed rootCollectionTokenId, uint256 startTokenId, uint256 endTokenId);
event RootCollectionInitialized(address indexed rootCollection, uint256 maxSupply, uint256 tokensPerClaim);
event IneligibleTokensInitialized(address indexed rootCollectionAddress, uint256[] ineligibleTokenSlots, uint256[] ineligibleTokenBitmaps);
function initializeRootCollections(address[] memory rootCollections_, uint256[] memory rootCollectionMaxSupplies_, uint256[] memory tokensPerClaimArray_) public override onlyOwner {
if(initializedRootCollections) {
revert RootCollectionsAlreadyInitialized();
}
uint256 rootCollectionsArrayLength = rootCollections_.length;
_requireInputArrayLengthsMatch(rootCollectionsArrayLength, rootCollectionMaxSupplies_.length);
_requireInputArrayLengthsMatch(rootCollectionsArrayLength, tokensPerClaimArray_.length);
if(rootCollectionsArrayLength == 0) {
revert MustSpecifyAtLeastOneRootCollection();
}
if(rootCollectionsArrayLength > MAX_ROOT_COLLECTIONS) {
revert MaxNumberOfRootCollectionsExceeded();
}
for(uint256 i = 0; i < rootCollectionsArrayLength;) {
address rootCollection_ = rootCollections_[i];
uint256 rootCollectionMaxSupply_ = rootCollectionMaxSupplies_[i];
uint256 tokensPerClaim_ = tokensPerClaimArray_[i];
emit RootCollectionInitialized(address(rootCollection_), rootCollectionMaxSupply_, tokensPerClaim_);
if(!IERC165(rootCollection_).supportsInterface(type(IERC721).interfaceId)) {
revert CollectionAddressIsNotAnERC721Token();
}
if(tokensPerClaim_ == 0 || tokensPerClaim_ > 10) {
revert TokensPerClaimMustBeBetweenOneAndTen();
}
if(rootCollectionMaxSupply_ == 0) {
revert MaxSupplyOfRootTokenCannotBeZero();
}
rootCollectionLookup[rootCollection_].isRootCollection = true;
rootCollectionLookup[rootCollection_].rootCollection = IERC721(rootCollection_);
rootCollectionLookup[rootCollection_].maxSupply = rootCollectionMaxSupply_;
rootCollectionLookup[rootCollection_].tokensPerClaim = tokensPerClaim_;
unchecked {
uint256 numberOfTokenTrackerSlots = _getNumberOfTokenTrackerSlots(rootCollectionMaxSupply_);
for(uint256 j = 0; j < numberOfTokenTrackerSlots; ++j) {
rootCollectionLookup[rootCollection_].claimedTokenTracker.push(type(uint256).max);
}
++i;
}
}
_initializeNextTokenIdCounter();
initializedRootCollections = true;
}
function initializeIneligibleTokens(bool finalize, address rootCollectionAddress, uint256[] memory ineligibleTokenSlots, uint256[] memory ineligibleTokenBitmaps) external {
if(!initializedRootCollections) {
revert RootCollectionsHaveNotBeenInitialized();
}
if(finalizedIneligibleTokens) {
revert IneligibleTokensFinalized();
}
if(ineligibleTokenSlots.length != ineligibleTokenBitmaps.length) {
revert InputArrayLengthMismatch();
}
ClaimableRootCollection storage rootCollectionDetails = _getRootCollectionDetailsSafe(rootCollectionAddress);
if(finalize) {
finalizedIneligibleTokens = true;
}
unchecked {
for (uint256 i = 0; i < ineligibleTokenSlots.length; ++i) {
rootCollectionDetails.claimedTokenTracker[ineligibleTokenSlots[i]] = ~ineligibleTokenBitmaps[i];
rootCollectionDetails.ineligibleTokenBitmaps[ineligibleTokenSlots[i]] = ineligibleTokenBitmaps[i];
}
}
emit IneligibleTokensInitialized(rootCollectionAddress, ineligibleTokenSlots, ineligibleTokenBitmaps);
}
function claimBatch(address rootCollectionAddress, uint256[] calldata rootCollectionTokenIds) external {
_requireClaimsOpen();
if (rootCollectionTokenIds.length == 0) {
revert BatchSizeMustBeGreaterThanZero();
}
ClaimableRootCollection storage rootCollectionDetails = _getRootCollectionDetailsSafe(rootCollectionAddress);
uint256 maxBatchSize = MAX_MINTS_PER_TRANSACTION / rootCollectionDetails.tokensPerClaim;
if (rootCollectionTokenIds.length > maxBatchSize) {
revert BatchSizeGreaterThanMaximum();
}
for(uint256 i = 0; i < rootCollectionTokenIds.length;) {
_claim(rootCollectionDetails, rootCollectionTokenIds[i]);
unchecked {
++i;
}
}
}
function _claim(ClaimableRootCollection storage rootCollectionDetails, uint256 rootCollectionTokenId) internal {
if(rootCollectionDetails.rootCollection.ownerOf(rootCollectionTokenId) != _msgSender()) {
revert CallerDoesNotOwnRootTokenId();
}
(bool claimed, uint256 slot, uint256 offset, uint256 slotValue) = _isClaimed(rootCollectionDetails, rootCollectionTokenId);
if(claimed) {
revert TokenIdAlreadyClaimed();
}
uint256 claimedTokenId = getNextTokenId();
uint256 tokensPerClaim_ = rootCollectionDetails.tokensPerClaim;
emit ClaimMinted(address(rootCollectionDetails.rootCollection), rootCollectionTokenId, claimedTokenId, claimedTokenId + tokensPerClaim_ - 1);
rootCollectionDetails.claimedTokenTracker[slot] = slotValue & ~(uint256(1) << offset);
unchecked {
_advanceNextTokenIdCounter(tokensPerClaim_);
for(uint256 i = 0; i < tokensPerClaim_; ++i) {
_mintToken(_msgSender(), claimedTokenId + i);
}
}
}
function computeIneligibleTokensBitmap(uint256[] calldata ineligibleTokenIds) external pure returns (uint256[] memory, uint256[] memory) {
if (ineligibleTokenIds.length == 0) {
revert MustSpecifyAtLeastOneIneligibleToken();
}
uint256 lastTokenId = ineligibleTokenIds[ineligibleTokenIds.length - 1];
uint256 lastSeenId = 0;
uint256 numberOfTokenTrackerSlots = _getNumberOfTokenTrackerSlots(lastTokenId);
uint256[] memory tempBitmapArray = new uint256[](numberOfTokenTrackerSlots);
unchecked {
for (uint256 i = 0; i < ineligibleTokenIds.length; ++i) {
uint256 tokenId = ineligibleTokenIds[i];
if(i > 0 && tokenId <= lastSeenId) {
revert IneligibleTokenArrayMustBeInAscendingOrder();
}
lastSeenId = tokenId;
uint256 slot = tokenId / 256;
uint256 offset = tokenId % 256;
uint256 bitmap = tempBitmapArray[slot];
tempBitmapArray[slot] = bitmap | (uint256(1) << offset);
}
uint256 count;
for (uint256 i = 0; i < numberOfTokenTrackerSlots; ++i) {
if(tempBitmapArray[i] > 0) {
++count;
}
}
uint256[] memory bitmapArray = new uint256[](count);
uint256[] memory slotArray = new uint256[](count);
uint256 index;
for (uint256 i = 0; i < numberOfTokenTrackerSlots; ++i) {
if(tempBitmapArray[i] > 0) {
bitmapArray[index] = tempBitmapArray[i];
slotArray[index] = i;
++index;
}
}
return (slotArray, bitmapArray);
}
}
function getTokensPerClaim(address rootCollectionAddress) public view returns (uint256) {
ClaimableRootCollection storage rootCollectionDetails = _getRootCollectionDetailsSafe(rootCollectionAddress);
return rootCollectionDetails.tokensPerClaim;
}
function isEligible(address rootCollectionAddress, uint256 tokenId) public view returns (bool) {
ClaimableRootCollection storage rootCollectionDetails = _getRootCollectionDetailsSafe(rootCollectionAddress);
if(tokenId > rootCollectionDetails.maxSupply) {
revert InvalidRootCollectionTokenId();
}
uint256 slot = tokenId / 256;
uint256 offset = tokenId % 256;
uint256 slotValue = rootCollectionDetails.ineligibleTokenBitmaps[slot];
bool ineligible = ((slotValue >> offset) & uint256(1)) == 1;
return !ineligible;
}
function isClaimed(address rootCollectionAddress, uint256 tokenId) public view returns (bool) {
ClaimableRootCollection storage rootCollectionDetails = _getRootCollectionDetailsSafe(rootCollectionAddress);
if(tokenId > rootCollectionDetails.maxSupply) {
revert InvalidRootCollectionTokenId();
}
(bool claimed,,,) = _isClaimed(rootCollectionDetails, tokenId);
return claimed;
}
function _isClaimed(ClaimableRootCollection storage rootCollectionDetails, uint256 tokenId) internal view returns (bool claimed, uint256 slot, uint256 offset, uint256 slotValue) {
unchecked {
slot = tokenId / 256;
offset = tokenId % 256;
slotValue = rootCollectionDetails.claimedTokenTracker[slot];
claimed = ((slotValue >> offset) & uint256(1)) == 0;
}
return (claimed, slot, offset, slotValue);
}
function _getNumberOfTokenTrackerSlots(uint256 maxSupply_) internal pure returns (uint256 tokenTrackerSlotsRequired) {
unchecked {
uint256 maxSupplyPlusOne = 1 + maxSupply_;
tokenTrackerSlotsRequired = maxSupplyPlusOne / 256;
if(maxSupplyPlusOne % 256 > 0) {
++tokenTrackerSlotsRequired;
}
}
return tokenTrackerSlotsRequired;
}
function _requireInputArrayLengthsMatch(uint256 inputArray1Length, uint256 inputArray2Length) internal pure {
if(inputArray1Length != inputArray2Length) {
revert InputArrayLengthMismatch();
}
}
function _getRootCollectionDetailsSafe(address rootCollectionAddress) private view returns (ClaimableRootCollection storage) {
ClaimableRootCollection storage rootCollectionDetails = rootCollectionLookup[rootCollectionAddress];
if(!rootCollectionDetails.isRootCollection) {
revert InvalidRootCollectionAddress();
}
return rootCollectionDetails;
}
function _onClaimPeriodOpening() internal virtual override {
if(!initializedRootCollections) {
revert RootCollectionsHaveNotBeenInitialized();
}
if(!finalizedIneligibleTokens) {
revert IneligibleTokensHaveNotBeenFinalized();
}
}
}
文件 8 的 41:Context.sol
pragma solidity ^0.8.0;
abstract contract Context {
function _msgSender() internal view virtual returns (address) {
return msg.sender;
}
function _msgData() internal view virtual returns (bytes calldata) {
return msg.data;
}
}
文件 9 的 41:DigiDaigakuDragonEssence.sol
pragma solidity 0.8.9;
import "@limit-break/presets/BlacklistedTransferAdventureNFT.sol";
import "@limit-break/utils/tokens/ClaimableHolderMint.sol";
import "@limit-break/utils/tokens/SignedApprovalMint.sol";
contract DigiDaigakuDragonEssence is BlacklistedTransferAdventureNFT, ClaimableHolderMint, SignedApprovalMint {
constructor(address royaltyReceiver_, uint96 royaltyFeeNumerator_)
ERC721("", "")
EIP712("DigiDaigakuDragonEssence", "1")
{
initializeERC721("DigiDaigakuDragonEssence", "DIDE");
initializeURI("https://digidaigaku.com/dragon-essences/metadata/", ".json");
initializeAdventureERC721(100);
initializeRoyalties(royaltyReceiver_, royaltyFeeNumerator_);
initializeOperatorFilterer(address(0x3cc6CddA760b79bAfa08dF41ECFA224f810dCeB6), true);
}
function supportsInterface(bytes4 interfaceId) public view virtual override(AdventureNFT, IERC165) returns (bool) {
return interfaceId == type(ISignedApprovalInitializer).interfaceId
|| interfaceId == type(IRootCollectionInitializer).interfaceId || super.supportsInterface(interfaceId);
}
function _safeMintToken(address to, uint256 tokenId) internal virtual override {
_safeMint(to, tokenId);
}
function _mintToken(address to, uint256 tokenId) internal virtual override {
_mint(to, tokenId);
}
}
文件 10 的 41:ECDSA.sol
pragma solidity ^0.8.0;
import "../Strings.sol";
library ECDSA {
enum RecoverError {
NoError,
InvalidSignature,
InvalidSignatureLength,
InvalidSignatureS,
InvalidSignatureV
}
function _throwError(RecoverError error) private pure {
if (error == RecoverError.NoError) {
return;
} else if (error == RecoverError.InvalidSignature) {
revert("ECDSA: invalid signature");
} else if (error == RecoverError.InvalidSignatureLength) {
revert("ECDSA: invalid signature length");
} else if (error == RecoverError.InvalidSignatureS) {
revert("ECDSA: invalid signature 's' value");
} else if (error == RecoverError.InvalidSignatureV) {
revert("ECDSA: invalid signature 'v' value");
}
}
function tryRecover(bytes32 hash, bytes memory signature) internal pure returns (address, RecoverError) {
if (signature.length == 65) {
bytes32 r;
bytes32 s;
uint8 v;
assembly {
r := mload(add(signature, 0x20))
s := mload(add(signature, 0x40))
v := byte(0, mload(add(signature, 0x60)))
}
return tryRecover(hash, v, r, s);
} else if (signature.length == 64) {
bytes32 r;
bytes32 vs;
assembly {
r := mload(add(signature, 0x20))
vs := mload(add(signature, 0x40))
}
return tryRecover(hash, r, vs);
} else {
return (address(0), RecoverError.InvalidSignatureLength);
}
}
function recover(bytes32 hash, bytes memory signature) internal pure returns (address) {
(address recovered, RecoverError error) = tryRecover(hash, signature);
_throwError(error);
return recovered;
}
function tryRecover(
bytes32 hash,
bytes32 r,
bytes32 vs
) internal pure returns (address, RecoverError) {
bytes32 s = vs & bytes32(0x7fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff);
uint8 v = uint8((uint256(vs) >> 255) + 27);
return tryRecover(hash, v, r, s);
}
function recover(
bytes32 hash,
bytes32 r,
bytes32 vs
) internal pure returns (address) {
(address recovered, RecoverError error) = tryRecover(hash, r, vs);
_throwError(error);
return recovered;
}
function tryRecover(
bytes32 hash,
uint8 v,
bytes32 r,
bytes32 s
) internal pure returns (address, RecoverError) {
if (uint256(s) > 0x7FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF5D576E7357A4501DDFE92F46681B20A0) {
return (address(0), RecoverError.InvalidSignatureS);
}
if (v != 27 && v != 28) {
return (address(0), RecoverError.InvalidSignatureV);
}
address signer = ecrecover(hash, v, r, s);
if (signer == address(0)) {
return (address(0), RecoverError.InvalidSignature);
}
return (signer, RecoverError.NoError);
}
function recover(
bytes32 hash,
uint8 v,
bytes32 r,
bytes32 s
) internal pure returns (address) {
(address recovered, RecoverError error) = tryRecover(hash, v, r, s);
_throwError(error);
return recovered;
}
function toEthSignedMessageHash(bytes32 hash) internal pure returns (bytes32) {
return keccak256(abi.encodePacked("\x19Ethereum Signed Message:\n32", hash));
}
function toEthSignedMessageHash(bytes memory s) internal pure returns (bytes32) {
return keccak256(abi.encodePacked("\x19Ethereum Signed Message:\n", Strings.toString(s.length), s));
}
function toTypedDataHash(bytes32 domainSeparator, bytes32 structHash) internal pure returns (bytes32) {
return keccak256(abi.encodePacked("\x19\x01", domainSeparator, structHash));
}
}
文件 11 的 41:ERC165.sol
pragma solidity ^0.8.0;
import "./IERC165.sol";
abstract contract ERC165 is IERC165 {
function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) {
return interfaceId == type(IERC165).interfaceId;
}
}
文件 12 的 41:ERC2981.sol
pragma solidity ^0.8.0;
import "../../interfaces/IERC2981.sol";
import "../../utils/introspection/ERC165.sol";
abstract contract ERC2981 is IERC2981, ERC165 {
struct RoyaltyInfo {
address receiver;
uint96 royaltyFraction;
}
RoyaltyInfo private _defaultRoyaltyInfo;
mapping(uint256 => RoyaltyInfo) private _tokenRoyaltyInfo;
function supportsInterface(bytes4 interfaceId) public view virtual override(IERC165, ERC165) returns (bool) {
return interfaceId == type(IERC2981).interfaceId || super.supportsInterface(interfaceId);
}
function royaltyInfo(uint256 _tokenId, uint256 _salePrice) public view virtual override returns (address, uint256) {
RoyaltyInfo memory royalty = _tokenRoyaltyInfo[_tokenId];
if (royalty.receiver == address(0)) {
royalty = _defaultRoyaltyInfo;
}
uint256 royaltyAmount = (_salePrice * royalty.royaltyFraction) / _feeDenominator();
return (royalty.receiver, royaltyAmount);
}
function _feeDenominator() internal pure virtual returns (uint96) {
return 10000;
}
function _setDefaultRoyalty(address receiver, uint96 feeNumerator) internal virtual {
require(feeNumerator <= _feeDenominator(), "ERC2981: royalty fee will exceed salePrice");
require(receiver != address(0), "ERC2981: invalid receiver");
_defaultRoyaltyInfo = RoyaltyInfo(receiver, feeNumerator);
}
function _deleteDefaultRoyalty() internal virtual {
delete _defaultRoyaltyInfo;
}
function _setTokenRoyalty(
uint256 tokenId,
address receiver,
uint96 feeNumerator
) internal virtual {
require(feeNumerator <= _feeDenominator(), "ERC2981: royalty fee will exceed salePrice");
require(receiver != address(0), "ERC2981: Invalid parameters");
_tokenRoyaltyInfo[tokenId] = RoyaltyInfo(receiver, feeNumerator);
}
function _resetTokenRoyalty(uint256 tokenId) internal virtual {
delete _tokenRoyaltyInfo[tokenId];
}
}
文件 13 的 41:ERC721.sol
pragma solidity ^0.8.0;
import "./IERC721.sol";
import "./IERC721Receiver.sol";
import "./extensions/IERC721Metadata.sol";
import "../../utils/Address.sol";
import "../../utils/Context.sol";
import "../../utils/Strings.sol";
import "../../utils/introspection/ERC165.sol";
contract ERC721 is Context, ERC165, IERC721, IERC721Metadata {
using Address for address;
using Strings for uint256;
string private _name;
string private _symbol;
mapping(uint256 => address) private _owners;
mapping(address => uint256) private _balances;
mapping(uint256 => address) private _tokenApprovals;
mapping(address => mapping(address => bool)) private _operatorApprovals;
constructor(string memory name_, string memory symbol_) {
_name = name_;
_symbol = symbol_;
}
function supportsInterface(bytes4 interfaceId) public view virtual override(ERC165, IERC165) returns (bool) {
return
interfaceId == type(IERC721).interfaceId ||
interfaceId == type(IERC721Metadata).interfaceId ||
super.supportsInterface(interfaceId);
}
function balanceOf(address owner) public view virtual override returns (uint256) {
require(owner != address(0), "ERC721: address zero is not a valid owner");
return _balances[owner];
}
function ownerOf(uint256 tokenId) public view virtual override returns (address) {
address owner = _owners[tokenId];
require(owner != address(0), "ERC721: invalid token ID");
return owner;
}
function name() public view virtual override returns (string memory) {
return _name;
}
function symbol() public view virtual override returns (string memory) {
return _symbol;
}
function tokenURI(uint256 tokenId) public view virtual override returns (string memory) {
_requireMinted(tokenId);
string memory baseURI = _baseURI();
return bytes(baseURI).length > 0 ? string(abi.encodePacked(baseURI, tokenId.toString())) : "";
}
function _baseURI() internal view virtual returns (string memory) {
return "";
}
function approve(address to, uint256 tokenId) public virtual override {
address owner = ERC721.ownerOf(tokenId);
require(to != owner, "ERC721: approval to current owner");
require(
_msgSender() == owner || isApprovedForAll(owner, _msgSender()),
"ERC721: approve caller is not token owner nor approved for all"
);
_approve(to, tokenId);
}
function getApproved(uint256 tokenId) public view virtual override returns (address) {
_requireMinted(tokenId);
return _tokenApprovals[tokenId];
}
function setApprovalForAll(address operator, bool approved) public virtual override {
_setApprovalForAll(_msgSender(), operator, approved);
}
function isApprovedForAll(address owner, address operator) public view virtual override returns (bool) {
return _operatorApprovals[owner][operator];
}
function transferFrom(
address from,
address to,
uint256 tokenId
) public virtual override {
require(_isApprovedOrOwner(_msgSender(), tokenId), "ERC721: caller is not token owner nor approved");
_transfer(from, to, tokenId);
}
function safeTransferFrom(
address from,
address to,
uint256 tokenId
) public virtual override {
safeTransferFrom(from, to, tokenId, "");
}
function safeTransferFrom(
address from,
address to,
uint256 tokenId,
bytes memory data
) public virtual override {
require(_isApprovedOrOwner(_msgSender(), tokenId), "ERC721: caller is not token owner nor approved");
_safeTransfer(from, to, tokenId, data);
}
function _safeTransfer(
address from,
address to,
uint256 tokenId,
bytes memory data
) internal virtual {
_transfer(from, to, tokenId);
require(_checkOnERC721Received(from, to, tokenId, data), "ERC721: transfer to non ERC721Receiver implementer");
}
function _exists(uint256 tokenId) internal view virtual returns (bool) {
return _owners[tokenId] != address(0);
}
function _isApprovedOrOwner(address spender, uint256 tokenId) internal view virtual returns (bool) {
address owner = ERC721.ownerOf(tokenId);
return (spender == owner || isApprovedForAll(owner, spender) || getApproved(tokenId) == spender);
}
function _safeMint(address to, uint256 tokenId) internal virtual {
_safeMint(to, tokenId, "");
}
function _safeMint(
address to,
uint256 tokenId,
bytes memory data
) internal virtual {
_mint(to, tokenId);
require(
_checkOnERC721Received(address(0), to, tokenId, data),
"ERC721: transfer to non ERC721Receiver implementer"
);
}
function _mint(address to, uint256 tokenId) internal virtual {
require(to != address(0), "ERC721: mint to the zero address");
require(!_exists(tokenId), "ERC721: token already minted");
_beforeTokenTransfer(address(0), to, tokenId);
_balances[to] += 1;
_owners[tokenId] = to;
emit Transfer(address(0), to, tokenId);
_afterTokenTransfer(address(0), to, tokenId);
}
function _burn(uint256 tokenId) internal virtual {
address owner = ERC721.ownerOf(tokenId);
_beforeTokenTransfer(owner, address(0), tokenId);
_approve(address(0), tokenId);
_balances[owner] -= 1;
delete _owners[tokenId];
emit Transfer(owner, address(0), tokenId);
_afterTokenTransfer(owner, address(0), tokenId);
}
function _transfer(
address from,
address to,
uint256 tokenId
) internal virtual {
require(ERC721.ownerOf(tokenId) == from, "ERC721: transfer from incorrect owner");
require(to != address(0), "ERC721: transfer to the zero address");
_beforeTokenTransfer(from, to, tokenId);
_approve(address(0), tokenId);
_balances[from] -= 1;
_balances[to] += 1;
_owners[tokenId] = to;
emit Transfer(from, to, tokenId);
_afterTokenTransfer(from, to, tokenId);
}
function _approve(address to, uint256 tokenId) internal virtual {
_tokenApprovals[tokenId] = to;
emit Approval(ERC721.ownerOf(tokenId), to, tokenId);
}
function _setApprovalForAll(
address owner,
address operator,
bool approved
) internal virtual {
require(owner != operator, "ERC721: approve to caller");
_operatorApprovals[owner][operator] = approved;
emit ApprovalForAll(owner, operator, approved);
}
function _requireMinted(uint256 tokenId) internal view virtual {
require(_exists(tokenId), "ERC721: invalid token ID");
}
function _checkOnERC721Received(
address from,
address to,
uint256 tokenId,
bytes memory data
) private returns (bool) {
if (to.isContract()) {
try IERC721Receiver(to).onERC721Received(_msgSender(), from, tokenId, data) returns (bytes4 retval) {
return retval == IERC721Receiver.onERC721Received.selector;
} catch (bytes memory reason) {
if (reason.length == 0) {
revert("ERC721: transfer to non ERC721Receiver implementer");
} else {
assembly {
revert(add(32, reason), mload(reason))
}
}
}
} else {
return true;
}
}
function _beforeTokenTransfer(
address from,
address to,
uint256 tokenId
) internal virtual {}
function _afterTokenTransfer(
address from,
address to,
uint256 tokenId
) internal virtual {}
}
文件 14 的 41:IAdventure.sol
pragma solidity ^0.8.4;
import "@openzeppelin/contracts/utils/introspection/IERC165.sol";
interface IAdventure is IERC165 {
function questsLockTokens() external view returns (bool);
function onQuestEntered(address adventurer, uint256 tokenId, uint256 questId) external;
function onQuestExited(address adventurer, uint256 tokenId, uint256 questId, uint256 questStartTimestamp) external;
}
文件 15 的 41:IAdventureERC721Initializer.sol
pragma solidity ^0.8.4;
import "@openzeppelin/contracts/utils/introspection/IERC165.sol";
interface IAdventureERC721Initializer is IERC165 {
function initializeAdventureERC721(uint256 maxSimultaneousQuests_) external;
}
文件 16 的 41:IAdventurous.sol
pragma solidity ^0.8.4;
import "./Quest.sol";
import "@openzeppelin/contracts/utils/introspection/IERC165.sol";
interface IAdventurous is IERC165 {
event AdventureApprovalForAll(address indexed tokenOwner, address indexed operator, bool approved);
event QuestUpdated(uint256 indexed tokenId, address indexed tokenOwner, address indexed adventure, uint256 questId, bool active, bool booted);
function adventureTransferFrom(address from, address to, uint256 tokenId) external;
function adventureSafeTransferFrom(address from, address to, uint256 tokenId) external;
function adventureBurn(uint256 tokenId) external;
function enterQuest(uint256 tokenId, uint256 questId) external;
function exitQuest(uint256 tokenId, uint256 questId) external;
function getQuestCount(uint256 tokenId, address adventure) external view returns (uint256);
function getTimeOnQuest(uint256 tokenId, address adventure, uint256 questId) external view returns (uint256);
function isParticipatingInQuest(uint256 tokenId, address adventure, uint256 questId) external view returns (bool participatingInQuest, uint256 startTimestamp, uint256 index);
function getActiveQuests(uint256 tokenId, address adventure) external view returns (Quest[] memory activeQuests);
}
文件 17 的 41:IERC165.sol
pragma solidity ^0.8.0;
interface IERC165 {
function supportsInterface(bytes4 interfaceId) external view returns (bool);
}
文件 18 的 41:IERC2981.sol
pragma solidity ^0.8.0;
import "../utils/introspection/IERC165.sol";
interface IERC2981 is IERC165 {
function royaltyInfo(uint256 tokenId, uint256 salePrice)
external
view
returns (address receiver, uint256 royaltyAmount);
}
文件 19 的 41:IERC721.sol
pragma solidity ^0.8.0;
import "../../utils/introspection/IERC165.sol";
interface IERC721 is IERC165 {
event Transfer(address indexed from, address indexed to, uint256 indexed tokenId);
event Approval(address indexed owner, address indexed approved, uint256 indexed tokenId);
event ApprovalForAll(address indexed owner, address indexed operator, bool approved);
function balanceOf(address owner) external view returns (uint256 balance);
function ownerOf(uint256 tokenId) external view returns (address owner);
function safeTransferFrom(
address from,
address to,
uint256 tokenId,
bytes calldata data
) external;
function safeTransferFrom(
address from,
address to,
uint256 tokenId
) external;
function transferFrom(
address from,
address to,
uint256 tokenId
) external;
function approve(address to, uint256 tokenId) external;
function setApprovalForAll(address operator, bool _approved) external;
function getApproved(uint256 tokenId) external view returns (address operator);
function isApprovedForAll(address owner, address operator) external view returns (bool);
}
文件 20 的 41:IERC721Initializer.sol
pragma solidity ^0.8.4;
import "@openzeppelin/contracts/token/ERC721/IERC721.sol";
interface IERC721Initializer is IERC721 {
function initializeERC721(string memory name_, string memory symbol_) external;
}
文件 21 的 41:IERC721Metadata.sol
pragma solidity ^0.8.0;
import "../IERC721.sol";
interface IERC721Metadata is IERC721 {
function name() external view returns (string memory);
function symbol() external view returns (string memory);
function tokenURI(uint256 tokenId) external view returns (string memory);
}
文件 22 的 41:IERC721Receiver.sol
pragma solidity ^0.8.0;
interface IERC721Receiver {
function onERC721Received(
address operator,
address from,
uint256 tokenId,
bytes calldata data
) external returns (bytes4);
}
文件 23 的 41:IOperatorFilterRegistry.sol
pragma solidity ^0.8.4;
interface IOperatorFilterRegistry {
function isOperatorAllowed(address registrant, address operator) external view returns (bool);
function register(address registrant) external;
function registerAndSubscribe(address registrant, address subscription) external;
function registerAndCopyEntries(address registrant, address registrantToCopy) external;
function unregister(address addr) external;
function updateOperator(address registrant, address operator, bool filtered) external;
function updateOperators(address registrant, address[] calldata operators, bool filtered) external;
function updateCodeHash(address registrant, bytes32 codehash, bool filtered) external;
function updateCodeHashes(address registrant, bytes32[] calldata codeHashes, bool filtered) external;
function subscribe(address registrant, address registrantToSubscribe) external;
function unsubscribe(address registrant, bool copyExistingEntries) external;
function subscriptionOf(address addr) external returns (address registrant);
function subscribers(address registrant) external returns (address[] memory);
function subscriberAt(address registrant, uint256 index) external returns (address);
function copyEntriesOf(address registrant, address registrantToCopy) external;
function isOperatorFiltered(address registrant, address operator) external returns (bool);
function isCodeHashOfFiltered(address registrant, address operatorWithCode) external returns (bool);
function isCodeHashFiltered(address registrant, bytes32 codeHash) external returns (bool);
function filteredOperators(address addr) external returns (address[] memory);
function filteredCodeHashes(address addr) external returns (bytes32[] memory);
function filteredOperatorAt(address registrant, uint256 index) external returns (address);
function filteredCodeHashAt(address registrant, uint256 index) external returns (bytes32);
function isRegistered(address addr) external returns (bool);
function codeHashOf(address addr) external returns (bytes32);
}
文件 24 的 41:IOperatorFiltererInitializer.sol
pragma solidity ^0.8.4;
import "@openzeppelin/contracts/token/ERC721/IERC721.sol";
interface IOperatorFiltererInitializer {
function initializeOperatorFilterer(address subscriptionOrRegistrantToCopy, bool subscribe) external;
}
文件 25 的 41:IOwnableInitializer.sol
pragma solidity ^0.8.4;
import "@openzeppelin/contracts/utils/introspection/IERC165.sol";
interface IOwnableInitializer is IERC165 {
function initializeOwner(address owner_) external;
function transferOwnership(address newOwner) external;
}
文件 26 的 41:IRootCollectionInitializer.sol
pragma solidity ^0.8.4;
import "@openzeppelin/contracts/utils/introspection/IERC165.sol";
interface IRootCollectionInitializer is IERC165 {
function initializeRootCollections(address[] memory rootCollection_, uint256[] memory rootCollectionMaxSupply_, uint256[] memory tokensPerClaim_) external;
}
文件 27 的 41:IRoyaltiesInitializer.sol
pragma solidity ^0.8.4;
import "@openzeppelin/contracts/utils/introspection/IERC165.sol";
interface IRoyaltiesInitializer is IERC165 {
function initializeRoyalties(address receiver, uint96 feeNumerator) external;
}
文件 28 的 41:ISignedApprovalInitializer.sol
pragma solidity ^0.8.4;
import "@openzeppelin/contracts/utils/introspection/IERC165.sol";
interface ISignedApprovalInitializer is IERC165 {
function initializeSigner(address signer, uint256 maxQuantity) external;
}
文件 29 的 41:IURIInitializer.sol
pragma solidity ^0.8.4;
import "@openzeppelin/contracts/utils/introspection/IERC165.sol";
interface IURIInitializer is IERC165 {
function initializeURI(string memory baseURI_, string memory suffixURI_) external;
}
文件 30 的 41:InitializableDefaultOperatorFilterer.sol
pragma solidity ^0.8.4;
import {InitializableOperatorFilterer} from "./InitializableOperatorFilterer.sol";
abstract contract InitializableDefaultOperatorFilterer is InitializableOperatorFilterer {
address internal constant DEFAULT_SUBSCRIPTION = address(0x3cc6CddA760b79bAfa08dF41ECFA224f810dCeB6);
function initializeOperatorFilterer(address , bool ) public virtual override {
super.initializeOperatorFilterer(DEFAULT_SUBSCRIPTION, true);
}
}
文件 31 的 41:InitializableERC721.sol
pragma solidity ^0.8.4;
import "../access/InitializableOwnable.sol";
import "../../initializable/IERC721Initializer.sol";
import "@openzeppelin/contracts/token/ERC721/ERC721.sol";
error AlreadyInitializedERC721();
abstract contract InitializableERC721 is InitializableOwnable, ERC721, IERC721Initializer {
bool private initializedERC721;
string internal _contractName;
string internal _contractSymbol;
function initializeERC721(string memory name_, string memory symbol_) public override onlyOwner {
if(initializedERC721) {
revert AlreadyInitializedERC721();
}
_contractName = name_;
_contractSymbol = symbol_;
initializedERC721 = true;
}
function supportsInterface(bytes4 interfaceId) public view virtual override(ERC721, IERC165) returns (bool) {
return
interfaceId == type(IERC721Initializer).interfaceId ||
super.supportsInterface(interfaceId);
}
function name() public view virtual override returns (string memory) {
return _contractName;
}
function symbol() public view virtual override returns (string memory) {
return _contractSymbol;
}
}
文件 32 的 41:InitializableOperatorFilterer.sol
pragma solidity ^0.8.4;
import {IOperatorFilterRegistry} from "./IOperatorFilterRegistry.sol";
import {IOperatorFiltererInitializer} from "../../initializable/IOperatorFiltererInitializer.sol";
abstract contract InitializableOperatorFilterer is IOperatorFiltererInitializer {
error OperatorNotAllowed(address operator);
IOperatorFilterRegistry public constant OPERATOR_FILTER_REGISTRY = IOperatorFilterRegistry(0x000000000000AAeB6D7670E522A718067333cd4E);
modifier onlyAllowedOperator(address from) virtual {
if (from != msg.sender) {
_checkFilterOperator(msg.sender);
}
_;
}
modifier onlyAllowedOperatorApproval(address operator) virtual {
_checkFilterOperator(operator);
_;
}
function initializeOperatorFilterer(address subscriptionOrRegistrantToCopy, bool subscribe) public virtual override {
if (address(OPERATOR_FILTER_REGISTRY).code.length > 0) {
if (subscribe) {
OPERATOR_FILTER_REGISTRY.registerAndSubscribe(address(this), subscriptionOrRegistrantToCopy);
} else {
if (subscriptionOrRegistrantToCopy != address(0)) {
OPERATOR_FILTER_REGISTRY.registerAndCopyEntries(address(this), subscriptionOrRegistrantToCopy);
} else {
OPERATOR_FILTER_REGISTRY.register(address(this));
}
}
}
}
function _checkFilterOperator(address operator) internal view virtual {
if (address(OPERATOR_FILTER_REGISTRY).code.length > 0) {
if (!OPERATOR_FILTER_REGISTRY.isOperatorAllowed(address(this), operator)) {
revert OperatorNotAllowed(operator);
}
}
}
}
文件 33 的 41:InitializableOwnable.sol
pragma solidity ^0.8.4;
import "../../initializable/IOwnableInitializer.sol";
import "@openzeppelin/contracts/utils/Context.sol";
error CallerIsNotTheContractOwner();
error NewOwnerIsTheZeroAddress();
error OwnerAlreadyInitialized();
abstract contract InitializableOwnable is Context, IOwnableInitializer {
address private _owner;
event OwnershipTransferred(address indexed previousOwner, address indexed newOwner);
constructor() {
_transferOwnership(_msgSender());
}
modifier onlyOwner() {
_checkOwner();
_;
}
function initializeOwner(address owner_) public override {
if(_owner != address(0)) {
revert OwnerAlreadyInitialized();
}
_transferOwnership(owner_);
}
function owner() public view virtual returns (address) {
return _owner;
}
function _checkOwner() internal view virtual {
if(owner() != _msgSender()) {
revert CallerIsNotTheContractOwner();
}
}
function transferOwnership(address newOwner) public virtual override onlyOwner {
if(newOwner == address(0)) {
revert NewOwnerIsTheZeroAddress();
}
_transferOwnership(newOwner);
}
function _transferOwnership(address newOwner) internal virtual {
address oldOwner = _owner;
_owner = newOwner;
emit OwnershipTransferred(oldOwner, newOwner);
}
}
文件 34 的 41:MintTokenBase.sol
pragma solidity ^0.8.4;
abstract contract MintTokenBase {
function _mintToken(address to, uint256 tokenId) internal virtual;
}
文件 35 的 41:Quest.sol
pragma solidity ^0.8.4;
struct Quest {
bool isActive;
uint32 questId;
uint64 startTimestamp;
uint32 arrayIndex;
}
文件 36 的 41:ReentrancyGuard.sol
pragma solidity ^0.8.0;
abstract contract ReentrancyGuard {
uint256 private constant _NOT_ENTERED = 1;
uint256 private constant _ENTERED = 2;
uint256 private _status;
constructor() {
_status = _NOT_ENTERED;
}
modifier nonReentrant() {
require(_status != _ENTERED, "ReentrancyGuard: reentrant call");
_status = _ENTERED;
_;
_status = _NOT_ENTERED;
}
}
文件 37 的 41:SafeMintTokenBase.sol
pragma solidity ^0.8.4;
abstract contract SafeMintTokenBase {
function _safeMintToken(address to, uint256 tokenId) internal virtual;
}
文件 38 的 41:SequentialMintBase.sol
pragma solidity ^0.8.4;
import "../access/InitializableOwnable.sol";
import "../../initializable/IRootCollectionInitializer.sol";
import "@openzeppelin/contracts/security/ReentrancyGuard.sol";
import "@openzeppelin/contracts/token/ERC721/IERC721.sol";
import "@openzeppelin/contracts/utils/introspection/ERC165.sol";
abstract contract SequentialMintBase {
uint256 private nextTokenIdCounter;
function _initializeNextTokenIdCounter() internal {
if(nextTokenIdCounter == 0) {
nextTokenIdCounter = 1;
}
}
function _advanceNextTokenIdCounter(uint256 amount) internal {
nextTokenIdCounter += amount;
}
function getNextTokenId() public view returns (uint256) {
return nextTokenIdCounter;
}
}
文件 39 的 41:SignedApprovalMint.sol
pragma solidity ^0.8.4;
import "./SafeMintTokenBase.sol";
import "./SequentialMintBase.sol";
import "../access/InitializableOwnable.sol";
import "../../initializable/ISignedApprovalInitializer.sol";
import "@openzeppelin/contracts/token/ERC721/IERC721.sol";
import "@openzeppelin/contracts/utils/introspection/ERC165.sol";
import "@openzeppelin/contracts/utils/cryptography/draft-EIP712.sol";
import "@openzeppelin/contracts/utils/cryptography/ECDSA.sol";
error AddressAlreadyMinted();
error InvalidSignature();
error MaxQuantityMustBeGreaterThanZero();
error MintExceedsMaximumAmountBySignedApproval();
error SignedClaimsAreDecommissioned();
error SignerAlreadyInitialized();
error SignerCannotBeInitializedAsAddressZero();
error SignerIsAddressZero();
abstract contract SignedApprovalMint is InitializableOwnable, SafeMintTokenBase, SequentialMintBase, EIP712, ISignedApprovalInitializer {
bool private _signedClaimsDecommissioned;
address private _approvalSigner;
uint256 private _maxQuantityMintable;
uint256 private _mintedAmount;
mapping(address => bool) private addressMinted;
event SignedClaimsDecommissioned();
event SignedMintClaimed(address indexed minter, uint256 startTokenId, uint256 endTokenId);
event SignerUpdated(address oldSigner, address newSigner);
function claimSignedMint(bytes calldata signature, uint256 quantity) external {
if (addressMinted[_msgSender()]) {
revert AddressAlreadyMinted();
}
if (_approvalSigner == address(0)) {
revert SignerIsAddressZero();
}
_requireSignedClaimsActive();
uint256 newTotal = _mintedAmount + quantity;
if (newTotal > _maxQuantityMintable) {
revert MintExceedsMaximumAmountBySignedApproval();
}
_mintedAmount = newTotal;
bytes32 hash = _hashTypedDataV4(
keccak256(
abi.encode(
keccak256("Approved(address wallet,uint256 quantity)"),
_msgSender(),
quantity
)
)
);
if (_approvalSigner != ECDSA.recover(hash, signature)) {
revert InvalidSignature();
}
addressMinted[_msgSender()] = true;
uint256 tokenIdToMint = getNextTokenId();
emit SignedMintClaimed(_msgSender(), tokenIdToMint, tokenIdToMint + quantity - 1);
unchecked {
_advanceNextTokenIdCounter(quantity);
for(uint256 i = 0; i < quantity; ++i) {
_safeMintToken(_msgSender(), tokenIdToMint + i);
}
}
}
function decommissionSignedApprovals() external onlyOwner {
_requireSignedClaimsActive();
_signedClaimsDecommissioned = true;
emit SignedClaimsDecommissioned();
}
function initializeSigner(address signer, uint256 maxQuantity) public override onlyOwner {
if(_approvalSigner != address(0)) {
revert SignerAlreadyInitialized();
}
if(signer == address(0)) {
revert SignerCannotBeInitializedAsAddressZero();
}
if(maxQuantity == 0) {
revert MaxQuantityMustBeGreaterThanZero();
}
_initializeNextTokenIdCounter();
_approvalSigner = signer;
_maxQuantityMintable = maxQuantity;
}
function setSigner(address newSigner) public onlyOwner {
if(_signedClaimsDecommissioned) {
revert SignedClaimsAreDecommissioned();
}
emit SignerUpdated(_approvalSigner, newSigner);
_approvalSigner = newSigner;
}
function hasMintedBySignedApproval(address account) public view returns (bool) {
return addressMinted[account];
}
function approvalSigner() public view returns (address) {
return _approvalSigner;
}
function maxQuantityMintable() public view returns (uint256) {
return _maxQuantityMintable;
}
function mintedAmount() public view returns (uint256) {
return _mintedAmount;
}
function signedClaimsDecommissioned() public view returns (bool) {
return _signedClaimsDecommissioned;
}
function _requireSignedClaimsActive() internal view {
if(_signedClaimsDecommissioned) {
revert SignedClaimsAreDecommissioned();
}
}
}
文件 40 的 41:Strings.sol
pragma solidity ^0.8.0;
library Strings {
bytes16 private constant _HEX_SYMBOLS = "0123456789abcdef";
uint8 private constant _ADDRESS_LENGTH = 20;
function toString(uint256 value) internal pure returns (string memory) {
if (value == 0) {
return "0";
}
uint256 temp = value;
uint256 digits;
while (temp != 0) {
digits++;
temp /= 10;
}
bytes memory buffer = new bytes(digits);
while (value != 0) {
digits -= 1;
buffer[digits] = bytes1(uint8(48 + uint256(value % 10)));
value /= 10;
}
return string(buffer);
}
function toHexString(uint256 value) internal pure returns (string memory) {
if (value == 0) {
return "0x00";
}
uint256 temp = value;
uint256 length = 0;
while (temp != 0) {
length++;
temp >>= 8;
}
return toHexString(value, length);
}
function toHexString(uint256 value, uint256 length) internal pure returns (string memory) {
bytes memory buffer = new bytes(2 * length + 2);
buffer[0] = "0";
buffer[1] = "x";
for (uint256 i = 2 * length + 1; i > 1; --i) {
buffer[i] = _HEX_SYMBOLS[value & 0xf];
value >>= 4;
}
require(value == 0, "Strings: hex length insufficient");
return string(buffer);
}
function toHexString(address addr) internal pure returns (string memory) {
return toHexString(uint256(uint160(addr)), _ADDRESS_LENGTH);
}
}
文件 41 的 41:draft-EIP712.sol
pragma solidity ^0.8.0;
import "./ECDSA.sol";
abstract contract EIP712 {
bytes32 private immutable _CACHED_DOMAIN_SEPARATOR;
uint256 private immutable _CACHED_CHAIN_ID;
address private immutable _CACHED_THIS;
bytes32 private immutable _HASHED_NAME;
bytes32 private immutable _HASHED_VERSION;
bytes32 private immutable _TYPE_HASH;
constructor(string memory name, string memory version) {
bytes32 hashedName = keccak256(bytes(name));
bytes32 hashedVersion = keccak256(bytes(version));
bytes32 typeHash = keccak256(
"EIP712Domain(string name,string version,uint256 chainId,address verifyingContract)"
);
_HASHED_NAME = hashedName;
_HASHED_VERSION = hashedVersion;
_CACHED_CHAIN_ID = block.chainid;
_CACHED_DOMAIN_SEPARATOR = _buildDomainSeparator(typeHash, hashedName, hashedVersion);
_CACHED_THIS = address(this);
_TYPE_HASH = typeHash;
}
function _domainSeparatorV4() internal view returns (bytes32) {
if (address(this) == _CACHED_THIS && block.chainid == _CACHED_CHAIN_ID) {
return _CACHED_DOMAIN_SEPARATOR;
} else {
return _buildDomainSeparator(_TYPE_HASH, _HASHED_NAME, _HASHED_VERSION);
}
}
function _buildDomainSeparator(
bytes32 typeHash,
bytes32 nameHash,
bytes32 versionHash
) private view returns (bytes32) {
return keccak256(abi.encode(typeHash, nameHash, versionHash, block.chainid, address(this)));
}
function _hashTypedDataV4(bytes32 structHash) internal view virtual returns (bytes32) {
return ECDSA.toTypedDataHash(_domainSeparatorV4(), structHash);
}
}
{
"compilationTarget": {
"src/DigiDaigakuDragonEssence.sol": "DigiDaigakuDragonEssence"
},
"evmVersion": "london",
"libraries": {},
"metadata": {
"bytecodeHash": "ipfs"
},
"optimizer": {
"enabled": true,
"runs": 200
},
"remappings": [
":@creator-token-contracts/=lib/creator-token-contracts/",
":@hypermint/=lib/hypermint-contracts/",
":@limit-break/=lib/limit-break-contracts/contracts/",
":@openzeppelin/=lib/openzeppelin-contracts/",
":ERC721A/=lib/ERC721A/contracts/",
":creator-token-contracts/=lib/creator-token-contracts/contracts/",
":ds-test/=lib/forge-std/lib/ds-test/src/",
":erc721a/=lib/ERC721A/",
":forge-std/=lib/forge-std/src/",
":hypermint-contracts/=lib/hypermint-contracts/contracts/",
":limit-break-contracts/=lib/limit-break-contracts/contracts/",
":openzeppelin-contracts/=lib/openzeppelin-contracts/"
]
}
[{"inputs":[{"internalType":"address","name":"royaltyReceiver_","type":"address"},{"internalType":"uint96","name":"royaltyFeeNumerator_","type":"uint96"}],"stateMutability":"nonpayable","type":"constructor"},{"inputs":[],"name":"AddressAlreadyMinted","type":"error"},{"inputs":[],"name":"AdventureApprovalToCaller","type":"error"},{"inputs":[],"name":"AdventureIsStillWhitelisted","type":"error"},{"inputs":[],"name":"AlreadyInitializedAdventureERC721","type":"error"},{"inputs":[],"name":"AlreadyInitializedERC721","type":"error"},{"inputs":[],"name":"AlreadyInitializedRoyalties","type":"error"},{"inputs":[],"name":"AlreadyInitializedURI","type":"error"},{"inputs":[],"name":"AlreadyOnQuest","type":"error"},{"inputs":[],"name":"AlreadyWhitelisted","type":"error"},{"inputs":[],"name":"AnActiveQuestIsPreventingTransfers","type":"error"},{"inputs":[],"name":"ArrayIndexOverflowsUint128","type":"error"},{"inputs":[],"name":"BatchSizeGreaterThanMaximum","type":"error"},{"inputs":[],"name":"BatchSizeMustBeGreaterThanZero","type":"error"},{"inputs":[],"name":"CallerDoesNotOwnRootTokenId","type":"error"},{"inputs":[],"name":"CallerIsNotTheContractOwner","type":"error"},{"inputs":[],"name":"CallerNotAWhitelistedAdventure","type":"error"},{"inputs":[],"name":"CallerNotApprovedForAdventure","type":"error"},{"inputs":[],"name":"CallerNotTokenOwner","type":"error"},{"inputs":[],"name":"ClaimPeriodIsNotOpen","type":"error"},{"inputs":[],"name":"ClaimPeriodMustBeClosedInTheFuture","type":"error"},{"inputs":[],"name":"ClaimsMustBeClosedToReopen","type":"error"},{"inputs":[],"name":"CollectionAddressIsNotAnERC721Token","type":"error"},{"inputs":[],"name":"ExceedsMaxRoyaltyFee","type":"error"},{"inputs":[],"name":"IneligibleTokenArrayMustBeInAscendingOrder","type":"error"},{"inputs":[],"name":"IneligibleTokensFinalized","type":"error"},{"inputs":[],"name":"IneligibleTokensHaveNotBeenFinalized","type":"error"},{"inputs":[],"name":"InputArrayLengthMismatch","type":"error"},{"inputs":[],"name":"InvalidAdventureContract","type":"error"},{"inputs":[],"name":"InvalidRootCollectionAddress","type":"error"},{"inputs":[],"name":"InvalidRootCollectionTokenId","type":"error"},{"inputs":[],"name":"InvalidSignature","type":"error"},{"inputs":[],"name":"MaxNumberOfRootCollectionsExceeded","type":"error"},{"inputs":[],"name":"MaxQuantityMustBeGreaterThanZero","type":"error"},{"inputs":[],"name":"MaxSimultaneousQuestsCannotBeZero","type":"error"},{"inputs":[],"name":"MaxSimultaneousQuestsExceeded","type":"error"},{"inputs":[],"name":"MaxSupplyOfRootTokenCannotBeZero","type":"error"},{"inputs":[],"name":"MintExceedsMaximumAmountBySignedApproval","type":"error"},{"inputs":[],"name":"MustSpecifyAtLeastOneIneligibleToken","type":"error"},{"inputs":[],"name":"MustSpecifyAtLeastOneRootCollection","type":"error"},{"inputs":[],"name":"NewOwnerIsTheZeroAddress","type":"error"},{"inputs":[],"name":"NonexistentToken","type":"error"},{"inputs":[],"name":"NotOnQuest","type":"error"},{"inputs":[],"name":"NotWhitelisted","type":"error"},{"inputs":[{"internalType":"address","name":"operator","type":"address"}],"name":"OperatorNotAllowed","type":"error"},{"inputs":[],"name":"OwnerAlreadyInitialized","type":"error"},{"inputs":[],"name":"QuestIdOutOfRange","type":"error"},{"inputs":[],"name":"RootCollectionsAlreadyInitialized","type":"error"},{"inputs":[],"name":"RootCollectionsHaveNotBeenInitialized","type":"error"},{"inputs":[],"name":"SignedClaimsAreDecommissioned","type":"error"},{"inputs":[],"name":"SignerAlreadyInitialized","type":"error"},{"inputs":[],"name":"SignerCannotBeInitializedAsAddressZero","type":"error"},{"inputs":[],"name":"SignerIsAddressZero","type":"error"},{"inputs":[],"name":"TokenIdAlreadyClaimed","type":"error"},{"inputs":[],"name":"TokensPerClaimMustBeBetweenOneAndTen","type":"error"},{"inputs":[],"name":"TooManyActiveQuests","type":"error"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"tokenOwner","type":"address"},{"indexed":true,"internalType":"address","name":"operator","type":"address"},{"indexed":false,"internalType":"bool","name":"approved","type":"bool"}],"name":"AdventureApprovalForAll","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"adventure","type":"address"},{"indexed":false,"internalType":"bool","name":"whitelisted","type":"bool"}],"name":"AdventureWhitelistUpdated","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"owner","type":"address"},{"indexed":true,"internalType":"address","name":"approved","type":"address"},{"indexed":true,"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"Approval","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"owner","type":"address"},{"indexed":true,"internalType":"address","name":"operator","type":"address"},{"indexed":false,"internalType":"bool","name":"approved","type":"bool"}],"name":"ApprovalForAll","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"string","name":"baseTokenURI","type":"string"}],"name":"BaseURISet","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"rootCollection","type":"address"},{"indexed":true,"internalType":"uint256","name":"rootCollectionTokenId","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"startTokenId","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"endTokenId","type":"uint256"}],"name":"ClaimMinted","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"claimPeriodClosingTimestamp","type":"uint256"}],"name":"ClaimPeriodClosing","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"claimPeriodClosingTimestamp","type":"uint256"}],"name":"ClaimPeriodOpened","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"rootCollectionAddress","type":"address"},{"indexed":false,"internalType":"uint256[]","name":"ineligibleTokenSlots","type":"uint256[]"},{"indexed":false,"internalType":"uint256[]","name":"ineligibleTokenBitmaps","type":"uint256[]"}],"name":"IneligibleTokensInitialized","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":"uint256","name":"tokenId","type":"uint256"},{"indexed":true,"internalType":"address","name":"tokenOwner","type":"address"},{"indexed":true,"internalType":"address","name":"adventure","type":"address"},{"indexed":false,"internalType":"uint256","name":"questId","type":"uint256"},{"indexed":false,"internalType":"bool","name":"active","type":"bool"},{"indexed":false,"internalType":"bool","name":"booted","type":"bool"}],"name":"QuestUpdated","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"rootCollection","type":"address"},{"indexed":false,"internalType":"uint256","name":"maxSupply","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"tokensPerClaim","type":"uint256"}],"name":"RootCollectionInitialized","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"receiver","type":"address"},{"indexed":false,"internalType":"uint96","name":"feeNumerator","type":"uint96"}],"name":"RoyaltySet","type":"event"},{"anonymous":false,"inputs":[],"name":"SignedClaimsDecommissioned","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"minter","type":"address"},{"indexed":false,"internalType":"uint256","name":"startTokenId","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"endTokenId","type":"uint256"}],"name":"SignedMintClaimed","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"oldSigner","type":"address"},{"indexed":false,"internalType":"address","name":"newSigner","type":"address"}],"name":"SignerUpdated","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"string","name":"suffixURI","type":"string"}],"name":"SuffixURISet","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":"OPERATOR_FILTER_REGISTRY","outputs":[{"internalType":"contract IOperatorFilterRegistry","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"address","name":"","type":"address"},{"internalType":"uint256","name":"","type":"uint256"}],"name":"activeQuestList","outputs":[{"internalType":"uint32","name":"","type":"uint32"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"address","name":"","type":"address"},{"internalType":"uint32","name":"","type":"uint32"}],"name":"activeQuestLookup","outputs":[{"internalType":"bool","name":"isActive","type":"bool"},{"internalType":"uint32","name":"questId","type":"uint32"},{"internalType":"uint64","name":"startTimestamp","type":"uint64"},{"internalType":"uint32","name":"arrayIndex","type":"uint32"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"adventureBurn","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"from","type":"address"},{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"adventureSafeTransferFrom","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"from","type":"address"},{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"adventureTransferFrom","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"approvalSigner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"operator","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"approve","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"},{"internalType":"address","name":"operator","type":"address"}],"name":"areAdventuresApprovedForAll","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"}],"name":"balanceOf","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"baseTokenURI","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"},{"internalType":"address","name":"adventure","type":"address"}],"name":"bootFromAllQuests","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"rootCollectionAddress","type":"address"},{"internalType":"uint256[]","name":"rootCollectionTokenIds","type":"uint256[]"}],"name":"claimBatch","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bytes","name":"signature","type":"bytes"},{"internalType":"uint256","name":"quantity","type":"uint256"}],"name":"claimSignedMint","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"claimPeriodClosingTimestamp_","type":"uint256"}],"name":"closeClaims","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256[]","name":"ineligibleTokenIds","type":"uint256[]"}],"name":"computeIneligibleTokensBitmap","outputs":[{"internalType":"uint256[]","name":"","type":"uint256[]"},{"internalType":"uint256[]","name":"","type":"uint256[]"}],"stateMutability":"pure","type":"function"},{"inputs":[],"name":"decommissionSignedApprovals","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"},{"internalType":"uint256","name":"questId","type":"uint256"}],"name":"enterQuest","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"},{"internalType":"uint256","name":"questId","type":"uint256"}],"name":"exitQuest","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"},{"internalType":"address","name":"adventure","type":"address"}],"name":"getActiveQuests","outputs":[{"components":[{"internalType":"bool","name":"isActive","type":"bool"},{"internalType":"uint32","name":"questId","type":"uint32"},{"internalType":"uint64","name":"startTimestamp","type":"uint64"},{"internalType":"uint32","name":"arrayIndex","type":"uint32"}],"internalType":"struct Quest[]","name":"activeQuests","type":"tuple[]"}],"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":"getClaimPeriodClosingTimestamp","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getNextTokenId","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"},{"internalType":"address","name":"adventure","type":"address"}],"name":"getQuestCount","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"},{"internalType":"address","name":"adventure","type":"address"},{"internalType":"uint256","name":"questId","type":"uint256"}],"name":"getTimeOnQuest","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"rootCollectionAddress","type":"address"}],"name":"getTokensPerClaim","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"account","type":"address"}],"name":"hasMintedBySignedApproval","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"maxSimultaneousQuests_","type":"uint256"}],"name":"initializeAdventureERC721","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"string","name":"name_","type":"string"},{"internalType":"string","name":"symbol_","type":"string"}],"name":"initializeERC721","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bool","name":"finalize","type":"bool"},{"internalType":"address","name":"rootCollectionAddress","type":"address"},{"internalType":"uint256[]","name":"ineligibleTokenSlots","type":"uint256[]"},{"internalType":"uint256[]","name":"ineligibleTokenBitmaps","type":"uint256[]"}],"name":"initializeIneligibleTokens","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"},{"internalType":"bool","name":"","type":"bool"}],"name":"initializeOperatorFilterer","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"owner_","type":"address"}],"name":"initializeOwner","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address[]","name":"rootCollections_","type":"address[]"},{"internalType":"uint256[]","name":"rootCollectionMaxSupplies_","type":"uint256[]"},{"internalType":"uint256[]","name":"tokensPerClaimArray_","type":"uint256[]"}],"name":"initializeRootCollections","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"receiver","type":"address"},{"internalType":"uint96","name":"feeNumerator","type":"uint96"}],"name":"initializeRoyalties","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"signer","type":"address"},{"internalType":"uint256","name":"maxQuantity","type":"uint256"}],"name":"initializeSigner","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"string","name":"baseURI_","type":"string"},{"internalType":"string","name":"suffixURI_","type":"string"}],"name":"initializeURI","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"account","type":"address"}],"name":"isAdventureWhitelisted","outputs":[{"internalType":"bool","name":"","type":"bool"}],"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":"isClaimPeriodOpen","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"rootCollectionAddress","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"isClaimed","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"rootCollectionAddress","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"isEligible","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"},{"internalType":"address","name":"adventure","type":"address"},{"internalType":"uint256","name":"questId","type":"uint256"}],"name":"isParticipatingInQuest","outputs":[{"internalType":"bool","name":"participatingInQuest","type":"bool"},{"internalType":"uint256","name":"startTimestamp","type":"uint256"},{"internalType":"uint256","name":"index","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"maxQuantityMintable","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"maxSimultaneousQuests","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"mintedAmount","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"name","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"claimPeriodClosingTimestamp_","type":"uint256"}],"name":"openClaims","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"ownerOf","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"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":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"from","type":"address"},{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"},{"internalType":"bytes","name":"data","type":"bytes"}],"name":"safeTransferFrom","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"operator","type":"address"},{"internalType":"bool","name":"approved","type":"bool"}],"name":"setAdventuresApprovedForAll","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"operator","type":"address"},{"internalType":"bool","name":"approved","type":"bool"}],"name":"setApprovalForAll","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"string","name":"baseTokenURI_","type":"string"}],"name":"setBaseURI","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"receiver","type":"address"},{"internalType":"uint96","name":"feeNumerator","type":"uint96"}],"name":"setRoyaltyInfo","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"newSigner","type":"address"}],"name":"setSigner","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"string","name":"suffixURI_","type":"string"}],"name":"setSuffixURI","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"signedClaimsDecommissioned","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"suffixURI","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"bytes4","name":"interfaceId","type":"bytes4"}],"name":"supportsInterface","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"symbol","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"tokenURI","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"from","type":"address"},{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"transferFrom","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"adventure","type":"address"}],"name":"unwhitelistAdventure","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"},{"internalType":"address","name":"adventure","type":"address"}],"name":"userExitAllQuests","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"},{"internalType":"address","name":"adventure","type":"address"},{"internalType":"uint256","name":"questId","type":"uint256"}],"name":"userExitQuest","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"adventure","type":"address"}],"name":"whitelistAdventure","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"whitelistedAdventureList","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"whitelistedAdventures","outputs":[{"internalType":"bool","name":"isWhitelisted","type":"bool"},{"internalType":"uint128","name":"arrayIndex","type":"uint128"}],"stateMutability":"view","type":"function"}]