This ERC is a new NFT asset designed based on the user contract wallet (including account abstraction), and is forward compatible with ERC-721. To keep NFT assets simple, this ERC removes the approve, setApprovalForAll, getApproved, isApprovedForAll and safeTransferFrom functions of ERC-721.
Motivation
ERC-721 defines Ethereum-based standard NFT that can be traded and transferred, but the essence of ERC-721 is based on the externally-owned account (EOA) wallet design. An EOA wallet has no state and code storage, and the smart contract wallet is different.
Almost all ERCs related to NFTs are add functions, but our opinion is the opposite. We think the NFT contract should be simpler, with more functions taken care of by the smart contract wallet.
Our proposal is to design a simpler NFT asset based on the smart contract wallet.
It aims to achieve the following goals:
Keep the NFT contract simple, only responsible for the transferFrom function.
approve, getApproved, setApprovalForAll and isApprovedForAll functions are not managed by the NFT contract. Instead, these permissions are managed at the user level, offering greater flexibility and control to users. This change not only enhances user autonomy but also mitigates certain risks associated with the ERC-721 contract’s implementation of these functions.
Remove the safeTransferFrom function. A better way to call the other party’s NFT assets is to access the other party’s own contract instead of directly accessing the NFT asset contract.
Forward compatibility with ERC-721 means that all NFT can be compatible with this proposal.
Specification
The key words “MUST”, “MUST NOT”, “REQUIRED”, “SHALL”, “SHALL NOT”, “SHOULD”, “SHOULD NOT”, “RECOMMENDED”, “NOT RECOMMENDED”, “MAY”, and “OPTIONAL” in this document are to be interpreted as described in RFC 2119 and RFC 8174.
Compliant contracts MUST implement the following interface:
pragmasolidity^0.8.20;/**
* @title ERC7561 Simple NFT interface
* @dev See https://ercs.ethereum.org/ERCS/erc-7561
*/interfaceIERC7561{/**
* @notice Used to notify transfer NFT.
* @param from Address of the from
* @param to Address of the receive
* @param tokenId The transaction token id
*/eventTransfer(addressindexedfrom,addressindexedto,uint256indexedtokenId);/**
* @notice Count all NFTs assigned to an owner
* @param owner Address of the owner
* @return The number of NFTs owned by `owner`, possibly zero
*/functionbalanceOf(addressowner)externalviewreturns(uint256);/**
* @notice Find the owner of an NFT
* @param tokenId The identifier for an NFT
* @return The address of the owner of the NFT
*/functionownerOf(uint256tokenId)externalviewreturns(address);/**
* @notice Transfer ownership of an NFT
* @param from Address of the from
* @param to Address of the to
* @param tokenId The NFT to transfer
*/functiontransferFrom(addressfrom,addressto,uint256tokenId)external;}
Rationale
The proposal is to simplify NFT standards by removing approve, setApprovalForAll, getApproved, isApprovedForAll and safeTransferFrom functions. This simplification aims to enhance security, reduce complexity, and improve efficiency, making the standard more suitable for smart contract wallet environments while maintaining essential functionalities.
Backwards Compatibility
As mentioned in the beginning, this ERC is forward compatible with ERC-721, ERC-721 is backward compatible with this ERC.
pragmasolidity^0.8.20;import"./IERC7561.sol";import"../../math/SafeMath.sol";/**
* @title Standard ERC7561 NFT
* @dev Note: the ERC-165 identifier for this interface is 0xc1b31357
* @dev Implementation of the basic standard NFT.
*/contractERC7561isIERC7561{// Token name
stringprivate_name;// Token symbol
stringprivate_symbol;mapping(uint256tokenId=>address)private_owners;mapping(addressowner=>uint256)private_balances;uint256private_totalSupply;functiontotalSupply()externalviewreturns(uint256){return_totalSupply;}functionbalanceOf(addressowner)publicviewreturns(uint256){require(owner!=address(0));return_balances[owner];}functionownerOf(uint256tokenId)publicviewreturns(address){return_requireOwned(tokenId);}functiontransferFrom(addressfrom,addressto,uint256tokenId)public{require(from==msg.sender);require(to!=address(0));addresspreviousOwner=_update(to,tokenId);require(previousOwner==from);}function_ownerOf(uint256tokenId)internalviewvirtualreturns(address){return_owners[tokenId];}function_requireOwned(uint256tokenId)internalviewreturns(address){addressowner=_ownerOf(tokenId);require(owner!=address(0));returnowner;}function_update(addressto,uint256tokenId)internalvirtualreturns(address){addressfrom=_ownerOf(tokenId);// Execute the update
if(from!=address(0)){unchecked{_balances[from]-=1;}}if(to!=address(0)){unchecked{_balances[to]+=1;}}_owners[tokenId]=to;emitTransfer(from,to,tokenId);returnfrom;}}
Security Considerations
It should be noted that this ERC is not backward compatible with ERC-721, so there will be incompatibility with existing dapps.