Week3- Token (ERC20) & NFT (ERC721)
===
Outline
---
* Ethereum Improvement Proposals (EIP)
* ERC20
* ERC721
Ethereum Improvement Proposals (EIP)
---
* 提案種類
![未命名绘图.drawio (4)](https://hackmd.io/_uploads/ryYPIypGA.png)
* 提案流程
![Screenshot 2024-05-11 at 8.05.17 PM](https://hackmd.io/_uploads/SyOMy16MR.png)
https://github.com/ethereum/EIPs/blob/master/EIPS/eip-1.md
Openzeppelin
---
https://docs.openzeppelin.com/contracts/5.x/
ERC20
---
* 以太坊上的「同質化代幣」標準。
* https://github.com/OpenZeppelin/openzeppelin-contracts/blob/master/contracts/token/ERC20/ERC20.sol
* 狀態變數:
* _balances: 持幣紀錄。(address => uint256)
* _allowances: 授權關係。 (address => (address => uint256))
* _totalSupply:代幣供給量。
* _name: 代幣名稱。
* _symbol: 代幣簡稱。
* Function:
* Get:
* name():取得代幣名稱。
* symbol():取得代幣簡稱。
* decimals():取得精度。
* totalSupply():取得總發行量。
* balanceOf(address account):取得「某地址」的「持幣量」。
* Write:
* transfer(address to, uint256 value):轉移代幣。
* approve(address spender, uint256 value):授權「某地址」操作「某數量」代幣。
* transferFrom(address from, address to, uint256 value):轉移代幣,通常「被授權者」會呼叫這個。
* internal:
* _mint(address account, uint256 value):鑄造代幣。
* _burn(address account, uint256 value):燒掉代幣。
* Demo
```solidity=
// SPDX-License-Identifier: GPL-3.0
pragma solidity >=0.8.20;
import "@openzeppelin/contracts/token/ERC20/ERC20.sol";
contract ERC20Example is ERC20 {
// 定義最大供給量
uint256 public maxSupply;
//建構子初始化ERC20必要參數(name與symbol),並多加設定題目要求的maxSupply
constructor(
string memory _name,
string memory _symbol,
uint256 _maxSupply
) ERC20(_name, _symbol){
//執行時填入10000000000(100億) * 1000000000000000000(單位)
maxSupply = _maxSupply;
}
// 實作mint function,主要用來demo確認用
function mint (uint256 amount) external {
//判斷這筆交易若完成,是否會超出最大供給量,如果會就回傳錯誤字串"over max supply."
require(amount + totalSupply() <= maxSupply, "over max supply.");
_mint(msg.sender, amount);
}
}
```
ERC721
---
* 以太坊上的「非同質化代幣」標準。
* https://github.com/OpenZeppelin/openzeppelin-contracts/blob/master/contracts/token/ERC721/ERC721.sol
* 狀態變數:
* _name: 代幣名稱。
* _symbol:代幣簡稱。
* _owners:代幣id與擁有者關係。(uint256 => address)
* _balances: 持幣紀錄。(address => uint256)
* _tokenApprovals: 代幣單個id授權紀關係。(uint256 => address)
* _operatorApprovals: 全部持有id授權關係。(address ==> ( address = >bool))
* Funtion:
* Get:
* name() : 取得代幣名稱。
* symbol() :取得代幣簡稱。
* balanceOf(address owner):取得「某地址」持有NFT「數量」。
* ownerOf(uint256 tokenId):取得「某代幣Id」的擁有者地址。
* tokenURI(uint256 tokenId):取得NFT的圖片資訊。
* getApproved(uint256 tokenId): 取得「某代幣Id」授權給哪個地址。
* isApprovedForAll(address owner, address operator):取得「某地址(owner)」是否授權「所有的代幣Id」給「某地址{operator}」。
* Write:
* approve(address to, uint256 tokenId):授權持有的「某代幣Id」NFT給「某地址」。
* setApprovalForAll(address operator, bool approved):設定「所有持有的代幣Id」都授權給「某地址」。
* transferFrom(address from, address to, uint256 tokenId): 轉移NFT。
* Internal:
* _mint(address to, uint256 tokenId): 鑄造NFT。
* _burn(uint256 tokenId):燒掉NFT。
* Demo
```solidity=
// SPDX-License-Identifier: GPL-3.0
pragma solidity ^0.8.20;
import "@openzeppelin/contracts/token/ERC721/ERC721.sol";
contract ERC721Example is ERC721 {
// 定義最大供給量
uint256 public maxSupply;
uint256 public counter = 0;
modifier avaialbeMint(uint256 amount) {
//判斷這筆交易若完成,是否會超出最大供給量,如果會就回傳錯誤字串"over max supply."
require(amount + counter <= maxSupply, "over max supply.");
_;
}
//建構子初始化ERC721必要參數(name與symbol),並多加設定題目要求的maxSupply
constructor(
string memory _name,
string memory _symbol,
uint256 _maxSupply)
ERC721(_name, _symbol){
//執行時填入10
maxSupply = _maxSupply;
}
// 實作mint function,主要用來demo確認用
function mint (uint256 amount) external avaialbeMint(amount){
// 迴圈值星批量鑄造NFT
for(uint256 i=0; i < amount ; i++){
// 鑄造 NFT, counter為NFT的tokenId
_mint(msg.sender, counter);
counter ++ ;
}
}
// 讓transfer無效
function safeTransferFrom(address from, address to, uint256 tokenId, bytes memory data) public override {
revert();
}
}
```
如何讓NFT圖片Opensea呈現
---
https://docs.opensea.io/docs/metadata-standards
* IPFS
* https://www.pinata.cloud/
* Step1: 上傳圖片。
* Step2: 上傳符合Opensea標準的檔案
```solidity=
// SPDX-License-Identifier: GPL-3.0
pragma solidity >=0.8.20;
import "@openzeppelin/contracts/token/ERC721/ERC721.sol";
contract MyNFT is ERC721{
using Strings for uint256;
address owner;
uint256 public maxSupply = 10; // 最大發行量
bool private isOpened = false;//盲盒是否打開
uint256 public counter = 0;
modifier onlyOwner{
require(msg.sender == owner);
_;
}
constructor (string memory _name, string memory _symbol) ERC721(_name, _symbol){
owner = msg.sender;
}
//開盲盒
function openBlindBox() external onlyOwner{
isOpened = true;
}
//設定NFT的baseURI(盲盒)
function _baseURI() internal pure override returns (string memory) {
return "ipfs://QmXxZBg4RnGxC2dDxfUSAmxgGooHsoncPQgCLiNw8kj3Ls/";
}
//查看NFT Metadata網址
function tokenURI(uint256 tokenId) public view override returns (string memory) {
if (!isOpened){
return _baseURI();
}
return string(abi.encodePacked("ipfs://QmWQcaFFCm9ofyVN2ZwGbTGopLEbQ6QSc2Xn1C7ekKAYDF/", tokenId.toString(), ".json"));
}
// 實作mint function,主要用來demo確認用
function mint (uint256 amount) external{
require(amount + counter <= maxSupply, "over max supply.");
// 迴圈批量鑄造NFT
for(uint256 i=0; i < amount ; i++){
// 鑄造 NFT, counter為NFT的tokenId
_mint(msg.sender, counter);
counter ++ ;
}
}
}
```
### 延伸:ERC721A合約!