-
Notifications
You must be signed in to change notification settings - Fork 522
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
add queryable extension to ERC721Base (#560)
* add queryable to erc721base * clean/lint * uncomment natspec * use erc721Avirtualapprove --------- Signed-off-by: WhiteOakKong <92236155+WhiteOakKong@users.noreply.github.com> Co-authored-by: nkrishang <62195808+nkrishang@users.noreply.github.com>
- Loading branch information
1 parent
c7a4554
commit 8da4090
Showing
3 changed files
with
244 additions
and
3 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,168 @@ | ||
// SPDX-License-Identifier: MIT | ||
// ERC721A Contracts v3.3.0 | ||
// Creator: Chiru Labs | ||
|
||
pragma solidity ^0.8.4; | ||
|
||
import "./IERC721AQueryable.sol"; | ||
import "../ERC721AVirtualApprove.sol"; | ||
|
||
/** | ||
* @title ERC721A Queryable | ||
* @dev ERC721A subclass with convenience query functions. | ||
*/ | ||
abstract contract ERC721AQueryable is ERC721A, IERC721AQueryable { | ||
/** | ||
* @dev Returns the `TokenOwnership` struct at `tokenId` without reverting. | ||
* | ||
* If the `tokenId` is out of bounds: | ||
* - `addr` = `address(0)` | ||
* - `startTimestamp` = `0` | ||
* - `burned` = `false` | ||
* | ||
* If the `tokenId` is burned: | ||
* - `addr` = `<Address of owner before token was burned>` | ||
* - `startTimestamp` = `<Timestamp when token was burned>` | ||
* - `burned = `true` | ||
* | ||
* Otherwise: | ||
* - `addr` = `<Address of owner>` | ||
* - `startTimestamp` = `<Timestamp of start of ownership>` | ||
* - `burned = `false` | ||
*/ | ||
function explicitOwnershipOf(uint256 tokenId) public view override returns (TokenOwnership memory) { | ||
TokenOwnership memory ownership; | ||
if (tokenId < _startTokenId() || tokenId >= _currentIndex) { | ||
return ownership; | ||
} | ||
ownership = _ownerships[tokenId]; | ||
if (ownership.burned) { | ||
return ownership; | ||
} | ||
return _ownershipOf(tokenId); | ||
} | ||
|
||
/** | ||
* @dev Returns an array of `TokenOwnership` structs at `tokenIds` in order. | ||
* See {ERC721AQueryable-explicitOwnershipOf} | ||
*/ | ||
function explicitOwnershipsOf(uint256[] memory tokenIds) external view override returns (TokenOwnership[] memory) { | ||
unchecked { | ||
uint256 tokenIdsLength = tokenIds.length; | ||
TokenOwnership[] memory ownerships = new TokenOwnership[](tokenIdsLength); | ||
for (uint256 i; i != tokenIdsLength; ++i) { | ||
ownerships[i] = explicitOwnershipOf(tokenIds[i]); | ||
} | ||
return ownerships; | ||
} | ||
} | ||
|
||
/** | ||
* @dev Returns an array of token IDs owned by `owner`, | ||
* in the range [`start`, `stop`) | ||
* (i.e. `start <= tokenId < stop`). | ||
* | ||
* This function allows for tokens to be queried if the collection | ||
* grows too big for a single call of {ERC721AQueryable-tokensOfOwner}. | ||
* | ||
* Requirements: | ||
* | ||
* - `start` < `stop` | ||
*/ | ||
/* solhint-disable*/ | ||
function tokensOfOwnerIn( | ||
address owner, | ||
uint256 start, | ||
uint256 stop | ||
) external view override returns (uint256[] memory) { | ||
unchecked { | ||
if (start >= stop) revert InvalidQueryRange(); | ||
uint256 tokenIdsIdx; | ||
uint256 stopLimit = _currentIndex; | ||
// Set `start = max(start, _startTokenId())`. | ||
if (start < _startTokenId()) { | ||
start = _startTokenId(); | ||
} | ||
// Set `stop = min(stop, _currentIndex)`. | ||
if (stop > stopLimit) { | ||
stop = stopLimit; | ||
} | ||
uint256 tokenIdsMaxLength = balanceOf(owner); | ||
// Set `tokenIdsMaxLength = min(balanceOf(owner), stop - start)`, | ||
// to cater for cases where `balanceOf(owner)` is too big. | ||
if (start < stop) { | ||
uint256 rangeLength = stop - start; | ||
if (rangeLength < tokenIdsMaxLength) { | ||
tokenIdsMaxLength = rangeLength; | ||
} | ||
} else { | ||
tokenIdsMaxLength = 0; | ||
} | ||
uint256[] memory tokenIds = new uint256[](tokenIdsMaxLength); | ||
if (tokenIdsMaxLength == 0) { | ||
return tokenIds; | ||
} | ||
// We need to call `explicitOwnershipOf(start)`, | ||
// because the slot at `start` may not be initialized. | ||
TokenOwnership memory ownership = explicitOwnershipOf(start); | ||
address currOwnershipAddr; | ||
// If the starting slot exists (i.e. not burned), initialize `currOwnershipAddr`. | ||
// `ownership.address` will not be zero, as `start` is clamped to the valid token ID range. | ||
if (!ownership.burned) { | ||
currOwnershipAddr = ownership.addr; | ||
} | ||
for (uint256 i = start; i != stop && tokenIdsIdx != tokenIdsMaxLength; ++i) { | ||
ownership = _ownerships[i]; | ||
if (ownership.burned) { | ||
continue; | ||
} | ||
if (ownership.addr != address(0)) { | ||
currOwnershipAddr = ownership.addr; | ||
} | ||
if (currOwnershipAddr == owner) { | ||
tokenIds[tokenIdsIdx++] = i; | ||
} | ||
} | ||
// Downsize the array to fit. | ||
assembly { | ||
mstore(tokenIds, tokenIdsIdx) | ||
} | ||
return tokenIds; | ||
} | ||
} | ||
|
||
/* solhint-enable */ | ||
|
||
/** | ||
* @dev Returns an array of token IDs owned by `owner`. | ||
* | ||
* This function scans the ownership mapping and is O(totalSupply) in complexity. | ||
* It is meant to be called off-chain. | ||
* | ||
* See {ERC721AQueryable-tokensOfOwnerIn} for splitting the scan into | ||
* multiple smaller scans if the collection is large enough to cause | ||
* an out-of-gas error (10K pfp collections should be fine). | ||
*/ | ||
function tokensOfOwner(address owner) external view override returns (uint256[] memory) { | ||
unchecked { | ||
uint256 tokenIdsIdx; | ||
address currOwnershipAddr; | ||
uint256 tokenIdsLength = balanceOf(owner); | ||
uint256[] memory tokenIds = new uint256[](tokenIdsLength); | ||
TokenOwnership memory ownership; | ||
for (uint256 i = _startTokenId(); tokenIdsIdx != tokenIdsLength; ++i) { | ||
ownership = _ownerships[i]; | ||
if (ownership.burned) { | ||
continue; | ||
} | ||
if (ownership.addr != address(0)) { | ||
currOwnershipAddr = ownership.addr; | ||
} | ||
if (currOwnershipAddr == owner) { | ||
tokenIds[tokenIdsIdx++] = i; | ||
} | ||
} | ||
return tokenIds; | ||
} | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,73 @@ | ||
// SPDX-License-Identifier: MIT | ||
// ERC721A Contracts v3.3.0 | ||
// Creator: Chiru Labs | ||
|
||
pragma solidity ^0.8.4; | ||
|
||
import "../interface/IERC721A.sol"; | ||
|
||
/** | ||
* @dev Interface of an ERC721AQueryable compliant contract. | ||
*/ | ||
interface IERC721AQueryable is IERC721A { | ||
/** | ||
* Invalid query range (`start` >= `stop`). | ||
*/ | ||
error InvalidQueryRange(); | ||
|
||
/** | ||
* @dev Returns the `TokenOwnership` struct at `tokenId` without reverting. | ||
* | ||
* If the `tokenId` is out of bounds: | ||
* - `addr` = `address(0)` | ||
* - `startTimestamp` = `0` | ||
* - `burned` = `false` | ||
* | ||
* If the `tokenId` is burned: | ||
* - `addr` = `<Address of owner before token was burned>` | ||
* - `startTimestamp` = `<Timestamp when token was burned>` | ||
* - `burned = `true` | ||
* | ||
* Otherwise: | ||
* - `addr` = `<Address of owner>` | ||
* - `startTimestamp` = `<Timestamp of start of ownership>` | ||
* - `burned = `false` | ||
*/ | ||
function explicitOwnershipOf(uint256 tokenId) external view returns (TokenOwnership memory); | ||
|
||
/** | ||
* @dev Returns an array of `TokenOwnership` structs at `tokenIds` in order. | ||
* See {ERC721AQueryable-explicitOwnershipOf} | ||
*/ | ||
function explicitOwnershipsOf(uint256[] memory tokenIds) external view returns (TokenOwnership[] memory); | ||
|
||
/** | ||
* @dev Returns an array of token IDs owned by `owner`, | ||
* in the range [`start`, `stop`) | ||
* (i.e. `start <= tokenId < stop`). | ||
* | ||
* This function allows for tokens to be queried if the collection | ||
* grows too big for a single call of {ERC721AQueryable-tokensOfOwner}. | ||
* | ||
* Requirements: | ||
* | ||
* - `start` < `stop` | ||
*/ | ||
function tokensOfOwnerIn( | ||
address owner, | ||
uint256 start, | ||
uint256 stop | ||
) external view returns (uint256[] memory); | ||
|
||
/** | ||
* @dev Returns an array of token IDs owned by `owner`. | ||
* | ||
* This function scans the ownership mapping and is O(totalSupply) in complexity. | ||
* It is meant to be called off-chain. | ||
* | ||
* See {ERC721AQueryable-tokensOfOwnerIn} for splitting the scan into | ||
* multiple smaller scans if the collection is large enough to cause | ||
* an out-of-gas error (10K pfp collections should be fine). | ||
*/ | ||
function tokensOfOwner(address owner) external view returns (uint256[] memory); | ||
} |