Saturday, March 31, 2018

ERC721SmartToken - ERC721 token with ERC20 adapter

ERC721SmartToken contract implements non-fungible tokens based on ERC721 standard that also supports ERC20interface. The main idea is to create a token that can be both non-fungible and has transferable value.
Inside ERC721 contract each not fungible token represented as a NFT structure where there is a member element called value.
struct NFT {
        uint256 value;          // NFT value 
        string  metadata;       // ... metadata: IPFS path
        uint256 kind;           // ... type
        uint256 level;          // ... activities level
        uint256 state;          // ... state
    }
This element holds ERC20 token inside ERC721 token.
To work with ERC721 token as normal ERC20 token we need to create a ERC20Adapter that implements standard ERC20 methods, as for example balanceOf method.
function balanceOf(address _owner) public view returns (uint256)
But in our case due to ERC721 structure for each _owner there can be many non-fungible tokens belong to this particular address. To select particular token for ERC20 compatible operations like transfer token value to another address we need to implement new methods that will accept address and token ID as parameters.
function approveFrom(uint256 _fromId, address _spender, uint256 _toId, uint256 _value) public returns (bool success)
This method calls ERC20Controller method implemented by ERC721SmartToken contract that will use the following mappings to store information abount NFT tokens:
mapping (address => mapping (address => uint256)) public allowanceIds; // transfer allowance
mapping (address => mapping (address => uint256)) public allowanceAmt; // transfer allowance
Now we can use ERC20 standard methods transfer and transferFrom to transfer values between two NFT tokens.
ERC20Controller implements defaultId methods that provides a default NFT id for the _owner address.
function defaultId(address _owner) internal returns (uint256 id);
So, we can use ERC20 standard method approve to approve transfer between two NFT default tokens for msg.senderand _spender.