ERC1155
ERC1155
是以太坊上的一個(gè)標(biāo)準(zhǔn)協(xié)議,用于創(chuàng)建可批量鑄造、可批量傳輸?shù)臄?shù)字資產(chǎn)之景,如游戲中的物品、收藏品膏潮、證書等锻狗。每個(gè) ERC1155
合約可以鑄造多種類型的資產(chǎn),每種資產(chǎn)都有一個(gè)唯一的 ID
焕参,并且每種資產(chǎn)可以有多個(gè)實(shí)例轻纪。這個(gè)標(biāo)準(zhǔn)協(xié)議定義了一組功能和事件,使得 ERC1155
資產(chǎn)可以被安全地轉(zhuǎn)移叠纷、擁有刻帚、交易和檢索。與 ERC721
不同涩嚣,ERC1155
允許多種類型的資產(chǎn)被打包在一個(gè)交易中崇众,從而使得交易更加高效,并且減少了需要進(jìn)行的交易數(shù)量航厚。這個(gè)標(biāo)準(zhǔn)協(xié)議具有可擴(kuò)展性顷歌,并且在許多不同的場(chǎng)景下都可以使用,例如游戲阶淘、NFT 市場(chǎng)衙吩、證券發(fā)行等互妓。
Mint
可以在以下測(cè)試網(wǎng)絡(luò)自己mint溪窒,具體步驟可以參考 Github
Network | Contract Address |
---|---|
Ethereum Goerli | 0xb1ecef869cd9382ae17efb07bd813d49832ea6c1 |
開發(fā)
在 Openzeppelin 的幫助下可以很快的入手開發(fā)ERC1155
標(biāo)準(zhǔn)的NFT合約。參考之前測(cè)試用的NFT代碼冯勉,自行編譯并且發(fā)布到了更多的測(cè)試網(wǎng)絡(luò)上澈蚌,可以很方便的進(jìn)行ERC1155
NFT測(cè)試。
引入ERC1155
庫(kù)之后灼狰,自己只需要實(shí)現(xiàn)mint
等自定義方法即可宛瞄,非常簡(jiǎn)單。
合約開發(fā)使用的是VSCode和hardhat交胚,合約驗(yàn)證使用hardhat flattening份汗。
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;
import "@openzeppelin/contracts/token/ERC1155/ERC1155.sol";
import "@openzeppelin/contracts/access/Ownable.sol";
contract TestToadz is ERC1155, Ownable {
uint256 public constant maxTokens = 6969;
uint256 public numAvailableTokens = 6969;
uint256 public constant maxMintsPerTx = 5;
bool public devMintLocked = false;
uint256[10000] private _availableTokens;
constructor() ERC1155("ipfs://QmWEFSMku6yGLQ9TQr66HjSd9kay8ZDYKbBEfjNi4pLtrr/{id}") {}
function getNumAvailableTokens() public view returns (uint256) {
return numAvailableTokens;
}
//Minting
function mint(uint256 quantity, uint256 amount) public {
uint256 updatedNumAvailableTokens = numAvailableTokens;
require(
block.timestamp >= 1337133769,
"Sale starts at whatever this time is"
);
require(
quantity <= maxMintsPerTx,
"There is a limit on minting too many at a time!"
);
require(
updatedNumAvailableTokens - quantity >= 0,
"Minting this many would exceed supply!"
);
require(msg.sender == tx.origin, "No contracts!");
uint256[] memory ids = new uint256[](quantity);
uint256[] memory amounts = new uint256[](quantity);
for (uint256 i = 0; i < quantity; i++) {
uint256 tokenId = getRandomSerialToken(quantity, i);
ids[i] = tokenId;
amounts[i] = amount;
updatedNumAvailableTokens--;
}
_mintBatch(msg.sender, ids, amounts, "");
numAvailableTokens = updatedNumAvailableTokens;
}
//Dev mint special tokens
function mintSpecial(uint256[] memory specialIds, uint256[] memory amounts ) external onlyOwner {
require(!devMintLocked, "Dev Mint Permanently Locked");
_mintBatch(msg.sender, specialIds, amounts, "");
}
function getRandomSerialToken(uint256 _numToFetch, uint256 _i)
internal
returns (uint256)
{
uint256 randomNum = uint256(
keccak256(
abi.encode(
msg.sender,
tx.gasprice,
block.number,
block.timestamp,
blockhash(block.number - 1),
_numToFetch,
_i
)
)
);
uint256 randomIndex = randomNum % numAvailableTokens;
uint256 valAtIndex = _availableTokens[randomIndex];
uint256 result;
if (valAtIndex == 0) {
result = randomIndex;
} else {
result = valAtIndex;
}
uint256 lastIndex = numAvailableTokens - 1;
if (randomIndex != lastIndex) {
uint256 lastValInArray = _availableTokens[lastIndex];
if (lastValInArray == 0) {
_availableTokens[randomIndex] = lastIndex;
} else {
_availableTokens[randomIndex] = lastValInArray;
}
}
numAvailableTokens--;
return result;
}
function lockDevMint() public onlyOwner {
devMintLocked = true;
}
}