commit2897abccc9Author: Hadrien Croubois <hadrien.croubois@gmail.com> Date: Fri Jul 14 15:31:29 2023 +0200 Update ERC721.sol commite26d5c0951Author: Hadrien Croubois <hadrien.croubois@gmail.com> Date: Fri Jul 14 08:46:48 2023 +0200 Update IERC721.sol commita475ffae0aAuthor: Hadrien Croubois <hadrien.croubois@gmail.com> Date: Fri Jul 14 08:45:25 2023 +0200 Update ERC721.sol commit20bb47f439Author: Hadrien Croubois <hadrien.croubois@gmail.com> Date: Fri Jul 14 08:43:14 2023 +0200 Update contracts/token/ERC721/ERC721.sol commitf404802d55Author: Hadrien Croubois <hadrien.croubois@gmail.com> Date: Fri Jul 14 08:41:30 2023 +0200 Update ERC721.sol commitb982e2a808Author: Hadrien Croubois <hadrien.croubois@gmail.com> Date: Fri Jul 14 08:38:46 2023 +0200 Update ERC721.behavior.js commitca32b459ecAuthor: Francisco Giordano <fg@frang.io> Date: Thu Jul 13 19:14:15 2023 -0300 fix _safeTransfer docs commitcaabbf3c46Author: Francisco Giordano <fg@frang.io> Date: Thu Jul 13 19:08:36 2023 -0300 improve warnings and notes commita023cad591Author: Francisco Giordano <fg@frang.io> Date: Thu Jul 13 18:21:27 2023 -0300 wrap long line commit5ce49a45fdAuthor: Francisco Giordano <fg@frang.io> Date: Thu Jul 13 18:19:30 2023 -0300 remove unnecessary solhint annotation commitd0375301f1Author: Francisco <fg@frang.io> Date: Thu Jul 13 18:17:24 2023 -0300 Apply suggestions from code review Co-authored-by: Ernesto García <ernestognw@gmail.com> commit81aca96467Author: Francisco <fg@frang.io> Date: Thu Jul 13 18:16:42 2023 -0300 Update CHANGELOG.md Co-authored-by: Ernesto García <ernestognw@gmail.com> commit12f63b3b1bAuthor: Hadrien Croubois <hadrien.croubois@gmail.com> Date: Thu Jul 13 17:28:04 2023 +0200 add test commit08da709ba7Author: Hadrien Croubois <hadrien.croubois@gmail.com> Date: Thu Jul 13 16:45:30 2023 +0200 refactor _checkAuhtorized commit328b16bf8cAuthor: Hadrien Croubois <hadrien.croubois@gmail.com> Date: Thu Jul 13 16:29:05 2023 +0200 Authorised → Authorized commitb29e573383Author: Hadrien Croubois <hadrien.croubois@gmail.com> Date: Thu Jul 13 16:14:57 2023 +0200 rename from → previousOwner commite996ba49d8Author: Hadrien Croubois <hadrien.croubois@gmail.com> Date: Thu Jul 13 16:00:38 2023 +0200 add ERC721 specific details in the 'How to upgrade from 4.x' section of the CHANGELOG commit20048ca3b9Author: Hadrien Croubois <hadrien.croubois@gmail.com> Date: Thu Jul 13 11:00:11 2023 +0200 Changes suggested in the PR discussions commit4c25b48803Merge:d7a6aaf4fb4d9510Author: Hadrien Croubois <hadrien.croubois@gmail.com> Date: Thu Jul 13 10:00:55 2023 +0200 Merge branch 'refactor/erc721-update-fnPointer' of https://github.com/Amxx/openzeppelin-contracts into refactor/erc721-update-fnPointer commitd7a6aaf41fAuthor: Hadrien Croubois <hadrien.croubois@gmail.com> Date: Thu Jul 13 10:00:50 2023 +0200 remove _exists commitfb4d9510deAuthor: Hadrien Croubois <hadrien.croubois@gmail.com> Date: Thu Jul 13 10:00:39 2023 +0200 Apply suggestions from code review Co-authored-by: Francisco <fg@frang.io> Co-authored-by: Ernesto García <ernestognw@gmail.com> commit10815081f7Author: ernestognw <ernestognw@gmail.com> Date: Wed Jul 12 21:09:18 2023 -0600 Lint commit9ba012005fAuthor: ernestognw <ernestognw@gmail.com> Date: Wed Jul 12 20:28:50 2023 -0600 Format _increaseBalance NatSpec commit7c3f1615b0Author: Francisco <fg@frang.io> Date: Wed Jul 12 20:29:11 2023 -0300 Update .changeset/eighty-lemons-shake.md Co-authored-by: Ernesto García <ernestognw@gmail.com> commit4516803058Author: Hadrien Croubois <hadrien.croubois@gmail.com> Date: Wed Jul 12 17:15:48 2023 +0200 make the safe function without a data field non virtual commite4b0e725dfAuthor: Hadrien Croubois <hadrien.croubois@gmail.com> Date: Wed Jul 12 16:56:07 2023 +0200 use whenNotPaused in ERC721Pausable commitb973d985a4Author: Hadrien Croubois <hadrien.croubois@gmail.com> Date: Wed Jul 12 14:11:59 2023 +0200 changesets commit7121ff7c5fMerge:2558c8fade570d0dAuthor: Hadrien Croubois <hadrien.croubois@gmail.com> Date: Wed Jul 12 13:46:07 2023 +0200 Merge branch 'erc721-approve-0' into refactor/erc721-update-fnPointer commitde570d0d14Author: Hadrien Croubois <hadrien.croubois@gmail.com> Date: Wed Jul 12 13:42:46 2023 +0200 allow using approve/_approve to clean approval commit2558c8fac8Author: Hadrien Croubois <hadrien.croubois@gmail.com> Date: Wed Jul 12 10:16:55 2023 +0200 change _increaseBalance type to uint128 commit16f2f15673Author: Hadrien Croubois <hadrien.croubois@gmail.com> Date: Wed Jul 12 10:01:30 2023 +0200 remove _isApproedOrOwner in favor of _isApproved + refactor _checkOnERC721Received commit7e9d024d08Author: Hadrien Croubois <hadrien.croubois@gmail.com> Date: Wed Jul 12 09:31:18 2023 +0200 Apply suggestions from code review Co-authored-by: Ernesto García <ernestognw@gmail.com> commit1a9552009bAuthor: Hadrien Croubois <hadrien.croubois@gmail.com> Date: Tue Jul 11 21:47:23 2023 +0200 replace constraints with a simple operator check commitbd0c52e34aAuthor: Hadrien Croubois <hadrien.croubois@gmail.com> Date: Tue Jul 11 18:06:29 2023 +0200 refactor constraint into an optionalChecks bitmap commit5ab254cf95Author: Hadrien Croubois <hadrien.croubois@gmail.com> Date: Fri Jul 7 16:13:17 2023 +0200 lint commit0bb98cb8c6Merge:562ddf567ccea54dAuthor: Hadrien Croubois <hadrien.croubois@gmail.com> Date: Fri Jul 7 16:11:40 2023 +0200 Merge branch 'master' into feature/Governor-storage commit562ddf566aAuthor: Hadrien Croubois <hadrien.croubois@gmail.com> Date: Wed Jul 5 18:45:42 2023 +0200 implement hybrid _update commit54cb3ca05fMerge:c7303ec2bb644589Author: Hadrien Croubois <hadrien.croubois@gmail.com> Date: Mon Jul 3 21:09:30 2023 +0200 Merge branch 'master' into refactor/erc721-update-fnPointer commitc7303ec2aeAuthor: Hadrien Croubois <hadrien.croubois@gmail.com> Date: Mon Jul 3 09:37:53 2023 +0200 fix lint commit1cc7f54ab5Merge:78c280b506861dceAuthor: Hadrien Croubois <hadrien.croubois@gmail.com> Date: Mon Jul 3 09:35:35 2023 +0200 Merge remote-tracking branch 'upstream' into refactor/erc721-update-fnPointer commit78c280b537Merge:e2fdbacd04342118Author: Hadrien Croubois <hadrien.croubois@gmail.com> Date: Fri Jun 30 18:40:55 2023 +0200 Merge branch 'master' into refactor/erc721-update-fnPointer commite9f03bd211Author: Francisco Giordano <fg@frang.io> Date: Fri Jun 30 12:09:15 2023 -0300 Exclude address(0) in ERC721._isApprovedOrOwner commite2fdbacd63Author: Hadrien Croubois <hadrien.croubois@gmail.com> Date: Wed Jun 21 22:09:50 2023 +0200 fix lint commit7ec34355aeAuthor: Hadrien Croubois <hadrien.croubois@gmail.com> Date: Wed Jun 21 17:59:22 2023 +0200 Apply suggestions from code review commit1ed8f9ef2cAuthor: Hadrien Croubois <hadrien.croubois@gmail.com> Date: Wed Jun 21 17:56:00 2023 +0200 use __unsafe_increaseBalance to react to batch minting commita3526acdf2Author: Hadrien Croubois <hadrien.croubois@gmail.com> Date: Thu Apr 27 16:37:40 2023 +0200 Rebase ERC721._update on top of next-v5
179 lines
7.3 KiB
Solidity
179 lines
7.3 KiB
Solidity
// SPDX-License-Identifier: MIT
|
|
// OpenZeppelin Contracts (last updated v4.8.0) (token/ERC721/extensions/ERC721Enumerable.sol)
|
|
|
|
pragma solidity ^0.8.19;
|
|
|
|
import {ERC721} from "../ERC721.sol";
|
|
import {IERC721Enumerable} from "./IERC721Enumerable.sol";
|
|
import {IERC165} from "../../../utils/introspection/ERC165.sol";
|
|
|
|
/**
|
|
* @dev This implements an optional extension of {ERC721} defined in the EIP that adds enumerability
|
|
* of all the token ids in the contract as well as all token ids owned by each account.
|
|
*
|
|
* CAUTION: `ERC721` extensions that implement custom `balanceOf` logic, such as `ERC721Consecutive`,
|
|
* interfere with enumerability and should not be used together with `ERC721Enumerable`.
|
|
*/
|
|
abstract contract ERC721Enumerable is ERC721, IERC721Enumerable {
|
|
// Mapping from owner to list of owned token IDs
|
|
mapping(address => mapping(uint256 => uint256)) private _ownedTokens;
|
|
|
|
// Mapping from token ID to index of the owner tokens list
|
|
mapping(uint256 => uint256) private _ownedTokensIndex;
|
|
|
|
// Array with all token ids, used for enumeration
|
|
uint256[] private _allTokens;
|
|
|
|
// Mapping from token id to position in the allTokens array
|
|
mapping(uint256 => uint256) private _allTokensIndex;
|
|
|
|
/**
|
|
* @dev An `owner`'s token query was out of bounds for `index`.
|
|
*
|
|
* NOTE: The owner being `address(0)` indicates a global out of bounds index.
|
|
*/
|
|
error ERC721OutOfBoundsIndex(address owner, uint256 index);
|
|
|
|
/**
|
|
* @dev Batch mint is not allowed.
|
|
*/
|
|
error ERC721EnumerableForbiddenBatchMint();
|
|
|
|
/**
|
|
* @dev See {IERC165-supportsInterface}.
|
|
*/
|
|
function supportsInterface(bytes4 interfaceId) public view virtual override(IERC165, ERC721) returns (bool) {
|
|
return interfaceId == type(IERC721Enumerable).interfaceId || super.supportsInterface(interfaceId);
|
|
}
|
|
|
|
/**
|
|
* @dev See {IERC721Enumerable-tokenOfOwnerByIndex}.
|
|
*/
|
|
function tokenOfOwnerByIndex(address owner, uint256 index) public view virtual returns (uint256) {
|
|
if (index >= balanceOf(owner)) {
|
|
revert ERC721OutOfBoundsIndex(owner, index);
|
|
}
|
|
return _ownedTokens[owner][index];
|
|
}
|
|
|
|
/**
|
|
* @dev See {IERC721Enumerable-totalSupply}.
|
|
*/
|
|
function totalSupply() public view virtual returns (uint256) {
|
|
return _allTokens.length;
|
|
}
|
|
|
|
/**
|
|
* @dev See {IERC721Enumerable-tokenByIndex}.
|
|
*/
|
|
function tokenByIndex(uint256 index) public view virtual returns (uint256) {
|
|
if (index >= totalSupply()) {
|
|
revert ERC721OutOfBoundsIndex(address(0), index);
|
|
}
|
|
return _allTokens[index];
|
|
}
|
|
|
|
/**
|
|
* @dev See {ERC721-_update}.
|
|
*/
|
|
function _update(address to, uint256 tokenId, address auth) internal virtual override returns (address) {
|
|
address previousOwner = super._update(to, tokenId, auth);
|
|
|
|
if (previousOwner == address(0)) {
|
|
_addTokenToAllTokensEnumeration(tokenId);
|
|
} else if (previousOwner != to) {
|
|
_removeTokenFromOwnerEnumeration(previousOwner, tokenId);
|
|
}
|
|
if (to == address(0)) {
|
|
_removeTokenFromAllTokensEnumeration(tokenId);
|
|
} else if (previousOwner != to) {
|
|
_addTokenToOwnerEnumeration(to, tokenId);
|
|
}
|
|
|
|
return previousOwner;
|
|
}
|
|
|
|
/**
|
|
* @dev Private function to add a token to this extension's ownership-tracking data structures.
|
|
* @param to address representing the new owner of the given token ID
|
|
* @param tokenId uint256 ID of the token to be added to the tokens list of the given address
|
|
*/
|
|
function _addTokenToOwnerEnumeration(address to, uint256 tokenId) private {
|
|
uint256 length = balanceOf(to) - 1;
|
|
_ownedTokens[to][length] = tokenId;
|
|
_ownedTokensIndex[tokenId] = length;
|
|
}
|
|
|
|
/**
|
|
* @dev Private function to add a token to this extension's token tracking data structures.
|
|
* @param tokenId uint256 ID of the token to be added to the tokens list
|
|
*/
|
|
function _addTokenToAllTokensEnumeration(uint256 tokenId) private {
|
|
_allTokensIndex[tokenId] = _allTokens.length;
|
|
_allTokens.push(tokenId);
|
|
}
|
|
|
|
/**
|
|
* @dev Private function to remove a token from this extension's ownership-tracking data structures. Note that
|
|
* while the token is not assigned a new owner, the `_ownedTokensIndex` mapping is _not_ updated: this allows for
|
|
* gas optimizations e.g. when performing a transfer operation (avoiding double writes).
|
|
* This has O(1) time complexity, but alters the order of the _ownedTokens array.
|
|
* @param from address representing the previous owner of the given token ID
|
|
* @param tokenId uint256 ID of the token to be removed from the tokens list of the given address
|
|
*/
|
|
function _removeTokenFromOwnerEnumeration(address from, uint256 tokenId) private {
|
|
// To prevent a gap in from's tokens array, we store the last token in the index of the token to delete, and
|
|
// then delete the last slot (swap and pop).
|
|
|
|
uint256 lastTokenIndex = balanceOf(from);
|
|
uint256 tokenIndex = _ownedTokensIndex[tokenId];
|
|
|
|
// When the token to delete is the last token, the swap operation is unnecessary
|
|
if (tokenIndex != lastTokenIndex) {
|
|
uint256 lastTokenId = _ownedTokens[from][lastTokenIndex];
|
|
|
|
_ownedTokens[from][tokenIndex] = lastTokenId; // Move the last token to the slot of the to-delete token
|
|
_ownedTokensIndex[lastTokenId] = tokenIndex; // Update the moved token's index
|
|
}
|
|
|
|
// This also deletes the contents at the last position of the array
|
|
delete _ownedTokensIndex[tokenId];
|
|
delete _ownedTokens[from][lastTokenIndex];
|
|
}
|
|
|
|
/**
|
|
* @dev Private function to remove a token from this extension's token tracking data structures.
|
|
* This has O(1) time complexity, but alters the order of the _allTokens array.
|
|
* @param tokenId uint256 ID of the token to be removed from the tokens list
|
|
*/
|
|
function _removeTokenFromAllTokensEnumeration(uint256 tokenId) private {
|
|
// To prevent a gap in the tokens array, we store the last token in the index of the token to delete, and
|
|
// then delete the last slot (swap and pop).
|
|
|
|
uint256 lastTokenIndex = _allTokens.length - 1;
|
|
uint256 tokenIndex = _allTokensIndex[tokenId];
|
|
|
|
// When the token to delete is the last token, the swap operation is unnecessary. However, since this occurs so
|
|
// rarely (when the last minted token is burnt) that we still do the swap here to avoid the gas cost of adding
|
|
// an 'if' statement (like in _removeTokenFromOwnerEnumeration)
|
|
uint256 lastTokenId = _allTokens[lastTokenIndex];
|
|
|
|
_allTokens[tokenIndex] = lastTokenId; // Move the last token to the slot of the to-delete token
|
|
_allTokensIndex[lastTokenId] = tokenIndex; // Update the moved token's index
|
|
|
|
// This also deletes the contents at the last position of the array
|
|
delete _allTokensIndex[tokenId];
|
|
_allTokens.pop();
|
|
}
|
|
|
|
/**
|
|
* See {ERC721-_increaseBalance}. We need that to account tokens that were minted in batch
|
|
*/
|
|
function _increaseBalance(address account, uint128 amount) internal virtual override {
|
|
if (amount > 0) {
|
|
revert ERC721EnumerableForbiddenBatchMint();
|
|
}
|
|
super._increaseBalance(account, amount);
|
|
}
|
|
}
|