文件 1 的 13:Address.sol
pragma solidity ^0.8.0;
library Address {
function isContract(address account) internal view returns (bool) {
uint256 size;
assembly {
size := extcodesize(account)
}
return size > 0;
}
function sendValue(address payable recipient, uint256 amount) internal {
require(address(this).balance >= amount, "Address: insufficient balance");
(bool success, ) = recipient.call{value: amount}("");
require(success, "Address: unable to send value, recipient may have reverted");
}
function functionCall(address target, bytes memory data) internal returns (bytes memory) {
return functionCall(target, data, "Address: low-level call failed");
}
function functionCall(
address target,
bytes memory data,
string memory errorMessage
) internal returns (bytes memory) {
return functionCallWithValue(target, data, 0, errorMessage);
}
function functionCallWithValue(
address target,
bytes memory data,
uint256 value
) internal returns (bytes memory) {
return functionCallWithValue(target, data, value, "Address: low-level call with value failed");
}
function functionCallWithValue(
address target,
bytes memory data,
uint256 value,
string memory errorMessage
) internal returns (bytes memory) {
require(address(this).balance >= value, "Address: insufficient balance for call");
require(isContract(target), "Address: call to non-contract");
(bool success, bytes memory returndata) = target.call{value: value}(data);
return verifyCallResult(success, returndata, errorMessage);
}
function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) {
return functionStaticCall(target, data, "Address: low-level static call failed");
}
function functionStaticCall(
address target,
bytes memory data,
string memory errorMessage
) internal view returns (bytes memory) {
require(isContract(target), "Address: static call to non-contract");
(bool success, bytes memory returndata) = target.staticcall(data);
return verifyCallResult(success, returndata, errorMessage);
}
function functionDelegateCall(address target, bytes memory data) internal returns (bytes memory) {
return functionDelegateCall(target, data, "Address: low-level delegate call failed");
}
function functionDelegateCall(
address target,
bytes memory data,
string memory errorMessage
) internal returns (bytes memory) {
require(isContract(target), "Address: delegate call to non-contract");
(bool success, bytes memory returndata) = target.delegatecall(data);
return verifyCallResult(success, returndata, errorMessage);
}
function verifyCallResult(
bool success,
bytes memory returndata,
string memory errorMessage
) internal pure returns (bytes memory) {
if (success) {
return returndata;
} else {
if (returndata.length > 0) {
assembly {
let returndata_size := mload(returndata)
revert(add(32, returndata), returndata_size)
}
} else {
revert(errorMessage);
}
}
}
}
文件 2 的 13:ClownTownSociety.sol
pragma solidity 0.8.0;
import "@openzeppelin/contracts@v4.3/token/ERC721/extensions/ERC721Enumerable.sol";
import "@openzeppelin/contracts@v4.3/access/Ownable.sol";
import "@openzeppelin/contracts@v4.3/utils/Strings.sol";
interface CryptoPunksAssets {
function composite(bytes1, bytes1, bytes1, bytes1, bytes1) external view returns (bytes4);
function getAsset(uint8) external view returns (bytes memory);
function getAssetName(uint8) external view returns (string memory);
function getAssetType(uint8) external view returns (uint8);
function getAssetIndex(string calldata, bool) external view returns (uint8);
function getMappedAsset(uint8, bool) external view returns (uint8);
}
interface CryptoPunksData {
function punkAttributes(uint16) external view returns (string memory);
}
interface CryptoPunksMarket {
function punkIndexToAddress(uint256) external view returns (address);
}
contract ClownTownSociety is ERC721Enumerable, Ownable {
CryptoPunksAssets private cryptoPunksAssets;
CryptoPunksData private cryptoPunksData;
CryptoPunksMarket private cryptoPunksMarket;
enum Type { Kind, Face, Ear, Neck, Beard, Hair, Eyes, Mouth, Smoke, Nose }
uint16 private constant MAX_COUNT = 10000;
uint256 private BASE_PRICE_IN_WEI;
uint256 private SEED;
mapping(uint16 => bytes) private clowns;
mapping(uint16 => uint16) private clownsToPunks;
mapping(uint16 => uint16) private punksToClowns;
bool private shouldVerifyCryptoPunkOwnership;
bool private contractSealed;
modifier unsealed() {
require(!contractSealed, "Contract sealed.");
_;
}
constructor() ERC721("ClownTownSociety", "CLWN") {
cryptoPunksAssets = CryptoPunksAssets(0x2A256814597B4e3BE62ac0e599Bee9D7bED8C3cf);
cryptoPunksData = CryptoPunksData(0x16F5A35647D6F03D5D3da7b35409D65ba03aF3B2);
cryptoPunksMarket = CryptoPunksMarket(0xb47e3cd837dDF8e4c57F05d70Ab865de6e193BBB);
shouldVerifyCryptoPunkOwnership = true;
}
function sealContract() external onlyOwner unsealed {
contractSealed = true;
}
function openPublicSale(bool publicSaleOpened, uint256 basePriceInWei) external onlyOwner unsealed {
shouldVerifyCryptoPunkOwnership = !publicSaleOpened;
BASE_PRICE_IN_WEI = basePriceInWei;
}
function destroy() external onlyOwner unsealed {
selfdestruct(payable(owner()));
}
function withdraw() external onlyOwner {
payable(owner()).transfer(address(this).balance);
}
function tokenURI(uint256 index) public view override returns (string memory)
{
require(_exists(index));
uint16 punkIndex = uint16(index);
bytes memory punkAssets = getPunkAssets(uint16(punkIndex));
string memory json = base64Encode(abi.encodePacked(
'{"name": "Clown Punk #',
Strings.toString(index),
'", "description": "The clowns are in town! CryptoPunk owners can mint a randomly generated clown version of their punk for free. During the public sale, anyone can mint clowns for unclaimed punks, but half the proceeds are transferred to the CryptoPunk owners. 10% of profits donated to GiveDirectly.org (in contract). All metadata and images are fully generated and stored on-chain. Inspired by LarvaLabs (not affiliated).", "image": "data:image/svg+xml;base64,',
base64Encode(bytes(punkAssetsImageSvg(punkAssets))),
'", "attributes": [',
metadataAttributes(punkAssets, punkIndex),
']}'));
return string(abi.encodePacked('data:application/json;base64,', json));
}
function isPublicSaleOpen() external view returns (bool) {
return !shouldVerifyCryptoPunkOwnership;
}
function isClownMintedForPunkIndex(uint16 punkIndex) public view returns (bool) {
require(punkIndex < MAX_COUNT, "Invalid punk ID");
uint16 clownIndex = punksToClowns[punkIndex];
return (clownIndex != 0) || (punkIndex == clownsToPunks[0]);
}
function priceInWeiToMintClownForPunkIndex(uint16 punkIndex) external view returns (uint256) {
require(punkIndex < MAX_COUNT, "Invalid punk ID");
require(!isClownMintedForPunkIndex(punkIndex), "Already minted a clown from this punk");
return shouldVerifyCryptoPunkOwnership ? 0 :
priceInWeiToMintClownForPunkAssets(parseAssets(cryptoPunksData.punkAttributes(punkIndex)));
}
function tokenByPunkIndex(uint16 punkIndex) external view returns (uint16) {
require(isClownMintedForPunkIndex(punkIndex), "No clown minted for this punk");
return punksToClowns[punkIndex];
}
function attributesByIndex(uint16 index) external view returns (string memory text) {
require(_exists(index));
bytes memory punkAssets = getPunkAssets(index);
for (uint8 j = 0; j < 8; j++) {
uint8 asset = uint8(punkAssets[j]);
if (asset > 0) {
if (j > 0) {
text = string(abi.encodePacked(text, ", ", getAssetName(asset)));
} else {
text = getAssetName(asset);
}
} else {
break;
}
}
text = string(abi.encodePacked(text, ", CryptoPunk #", Strings.toString(clownsToPunks[index])));
}
function imageByIndex(uint16 index) external view returns (string memory svg) {
require(_exists(index));
svg = punkAssetsImageSvg(getPunkAssets(index));
}
address private constant giveDirectlyAddress = 0xc7464dbcA260A8faF033460622B23467Df5AEA42;
function mintClownFromPunk(uint16 punkIndex) external payable {
uint16 clownIndex = uint16(totalSupply());
require(clownIndex < MAX_COUNT, "Total cap reached");
require(punkIndex < MAX_COUNT, "Invalid punk ID");
require(!isClownMintedForPunkIndex(punkIndex), "Already minted a clown from this punk");
address punkOwnerAddress = cryptoPunksMarket.punkIndexToAddress(punkIndex);
bytes memory punkAssets = parseAssets(cryptoPunksData.punkAttributes(punkIndex));
if (punkOwnerAddress != msg.sender) {
require(!shouldVerifyCryptoPunkOwnership, "Minting open to CryptoPunk owners only");
uint256 mintPrice = priceInWeiToMintClownForPunkAssets(punkAssets);
require(mintPrice <= msg.value, "Insufficient Ether sent");
uint256 punkOwnerFee = mintPrice / 2;
payable(punkOwnerAddress).transfer(punkOwnerFee);
uint256 charityDonation = (mintPrice - punkOwnerFee) / 10;
payable(giveDirectlyAddress).transfer(charityDonation);
}
clowns[clownIndex] = punkToClownAssets(punkAssets);
clownsToPunks[clownIndex] = punkIndex;
punksToClowns[punkIndex] = clownIndex;
_mint(msg.sender, clownIndex);
}
function priceInWeiToMintClownForPunkAssets(bytes memory punkAssets) internal view returns (uint256) {
uint8 kind = uint8(punkAssets[0]);
if (kind < 9) {
return BASE_PRICE_IN_WEI;
} else if (kind == 9) {
return (BASE_PRICE_IN_WEI * 5) / 2;
} else if (kind == 10) {
return BASE_PRICE_IN_WEI * 5;
} else {
return BASE_PRICE_IN_WEI * 10;
}
}
function punkToClownAssets(bytes memory punkAssets) internal returns (bytes memory clownAssets) {
uint8[10] memory punkTraits;
for (uint8 j = 0; j < 8; ++j) {
uint8 asset = uint8(punkAssets[j]);
if (asset == 0) {
break;
}
punkTraits[getAssetType(asset)] = asset;
}
uint8 kind = punkTraits[uint8(Type.Kind)];
bool isMale = (kind < 5) || (kind >= 9);
uint8 assetIndex = 0;
clownAssets = new bytes(8);
for (uint8 j = 0; j < 10 && assetIndex < 8; ++j) {
uint8 asset = punkTraits[j];
if ((j == uint8(Type.Hair)) && (asset != 14) && (asset != 104)) {
if ((asset == 50) || (asset == 81)) {
clownAssets[assetIndex++] = bytes1(isMale ? 142 : 143);
} else if (asset == 28) {
clownAssets[assetIndex++] = bytes1(uint8(144));
} else if (asset == 121) {
clownAssets[assetIndex++] = bytes1(uint8(145));
} else if (nextPseudoRandom(3) == 0) {
clownAssets[assetIndex++] = bytes1(isMale ? 140 : 141);
} else if (nextPseudoRandom(3) == 0) {
clownAssets[assetIndex++] = bytes1(isMale ? 14 : 104);
} else if (nextPseudoRandom(2) == 0) {
clownAssets[assetIndex++] = bytes1(isMale ? 142 : 143);
} else {
clownAssets[assetIndex++] = bytes1(isMale ? 144 : 145);
}
} else if ((j == uint8(Type.Eyes)) && (asset != 39) && (asset != 114) && (asset != 73) && (asset != 94)) {
if (asset == 78) {
clownAssets[assetIndex++] = bytes1(uint8(94));
} else if (asset == 97) {
clownAssets[assetIndex++] = bytes1(uint8(139));
} else if (asset == 126) {
clownAssets[assetIndex++] = bytes1(uint8(114));
} else if (asset == 0) {
if (nextPseudoRandom(2) == 0) {
clownAssets[assetIndex++] = bytes1(isMale ? 138 : 139);
} else if (nextPseudoRandom(2) == 0) {
clownAssets[assetIndex++] = bytes1(isMale ? 39 : 114);
} else {
clownAssets[assetIndex++] = bytes1(isMale ? 73 : 94);
}
} else {
clownAssets[assetIndex++] = bytes1(asset);
}
} else if ((j == uint8(Type.Nose)) && (asset != 18) && (asset != 109) && (kind < 10)) {
if (nextPseudoRandom(2) == 0) {
clownAssets[assetIndex++] = bytes1(isMale ? 18 : 109);
} else if (nextPseudoRandom(3) == 0) {
clownAssets[assetIndex++] = bytes1(isMale ? 134 : 135);
} else {
clownAssets[assetIndex++] = bytes1(isMale ? 136 : 137);
}
} else if (asset != 0) {
clownAssets[assetIndex++] = bytes1(asset);
}
}
}
function parseAssets(string memory attributes) internal view returns (bytes memory punkAssets) {
punkAssets = new bytes(8);
bytes memory stringAsBytes = bytes(attributes);
bytes memory buffer = new bytes(stringAsBytes.length);
uint index = 0;
uint j = 0;
bool isMale;
for (uint i = 0; i < stringAsBytes.length; i++) {
if (i == 0) {
isMale = (stringAsBytes[i] != "F");
}
if (stringAsBytes[i] != ",") {
buffer[j++] = stringAsBytes[i];
} else {
punkAssets[index++] = bytes1(getAssetIndex(bufferToString(buffer, j), isMale));
i++;
j = 0;
}
}
if (j > 0) {
punkAssets[index++] = bytes1(getAssetIndex(bufferToString(buffer, j), isMale));
}
}
function getPunkAssets(uint16 index) internal view returns (bytes memory punkAssets) {
require(_exists(index));
punkAssets = new bytes(8);
for (uint8 j = 0; j < 8; j++) {
punkAssets[j] = clowns[index][j];
}
}
function appendAttribute(string memory prefix, string memory key, string memory value, bool asString, bool asNumber, bool append) internal pure returns (string memory text) {
string memory quote = asString ? '"' : '';
string memory displayType = asNumber ? '"display_type": "number", ' : '';
string memory attribute =
string(abi.encodePacked('{ ', displayType, '"trait_type": "', key, '", "value": ', quote, value, quote, ' }'));
if (append) {
text = string(abi.encodePacked(prefix, ', ', attribute));
} else {
text = attribute;
}
}
function metadataAttributes(bytes memory punkAssets, uint16 index) internal view returns (string memory text) {
uint8 accessoryCount = 0;
for (uint8 j = 0; j < 8; j++) {
uint8 asset = uint8(punkAssets[j]);
if (asset > 0) {
if (j > 0) {
++accessoryCount;
text = appendAttribute(text, "Accessory", getAssetName(asset), true, false, true);
} else {
text = appendAttribute(text, "Type", getAssetName(asset), true, false, false);
}
} else {
break;
}
}
text = appendAttribute(text, "# Traits", Strings.toString(accessoryCount), false, false, true);
text = appendAttribute(text, "CryptoPunk #", Strings.toString(clownsToPunks[index]), false, true, true);
}
function punkAssetsImage(bytes memory punkAssets) internal view returns (bytes memory) {
bytes memory pixels = new bytes(2304);
for (uint8 j = 0; j < 8; j++) {
uint8 asset = uint8(punkAssets[j]);
if (asset > 0) {
bytes memory a = getAsset(asset);
uint n = a.length / 3;
for (uint i = 0; i < n; i++) {
uint[4] memory v = [
uint(uint8(a[i * 3]) & 0xF0) >> 4,
uint(uint8(a[i * 3]) & 0xF),
uint(uint8(a[i * 3 + 2]) & 0xF0) >> 4,
uint(uint8(a[i * 3 + 2]) & 0xF)
];
for (uint dx = 0; dx < 2; dx++) {
for (uint dy = 0; dy < 2; dy++) {
uint p = ((2 * v[1] + dy) * 24 + (2 * v[0] + dx)) * 4;
if (v[2] & (1 << (dx * 2 + dy)) != 0) {
bytes4 c = composite(a[i * 3 + 1], pixels[p], pixels[p + 1], pixels[p + 2], pixels[p + 3]);
pixels[p] = c[0];
pixels[p+1] = c[1];
pixels[p+2] = c[2];
pixels[p+3] = c[3];
} else if (v[3] & (1 << (dx * 2 + dy)) != 0) {
pixels[p] = 0;
pixels[p+1] = 0;
pixels[p+2] = 0;
pixels[p+3] = 0xFF;
}
}
}
}
}
}
return pixels;
}
string private constant SVG_HEADER = '<svg id="crisp" xmlns="http://www.w3.org/2000/svg" preserveAspectRatio="xMidYMax meet" viewBox="0 0 360 360">';
string private constant SVG_FOOTER = '<style>rect{width:15px;height:15px;} #crisp{shape-rendering: crispEdges;}</style></svg>';
function punkAssetsImageSvg(bytes memory punkAssets) internal view returns (string memory svg) {
bytes memory pixels = punkAssetsImage(punkAssets);
svg = string(abi.encodePacked(SVG_HEADER));
for (uint y = 0; y < 24; y++) {
for (uint x = 0; x < 24; x++) {
uint p = (y * 24 + x) * 4;
if (uint8(pixels[p + 3]) > 0) {
bytes4 color = bytes4(
(uint32(uint8(pixels[p])) << 24) |
(uint32(uint8(pixels[p+1])) << 16) |
(uint32(uint8(pixels[p+2])) << 8) |
(uint32(uint8(pixels[p+3]))));
svg = string(abi.encodePacked(svg, rectSvg(15 * x, 15 * y, color)));
}
}
}
svg = string(abi.encodePacked(svg, SVG_FOOTER));
}
function composite(bytes1 index, bytes1 yr, bytes1 yg, bytes1 yb, bytes1 ya) internal view returns (bytes4) {
return cryptoPunksAssets.composite(index, yr, yg, yb, ya);
}
function getAsset(uint8 index) internal view returns (bytes memory) {
return cryptoPunksAssets.getAsset(index);
}
function getAssetName(uint8 index) internal view returns (string memory) {
return cryptoPunksAssets.getAssetName(index);
}
function getAssetType(uint8 index) internal view returns (uint8) {
return cryptoPunksAssets.getAssetType(index);
}
function getAssetIndex(string memory text, bool isMale) internal view returns (uint8) {
return cryptoPunksAssets.getAssetIndex(text, isMale);
}
function getMappedAsset(uint8 index, bool toMale) internal view returns (uint8) {
return cryptoPunksAssets.getMappedAsset(index, toMale);
}
function nextPseudoRandom(uint256 max) internal returns (uint) {
SEED = uint(keccak256(abi.encodePacked(block.difficulty, block.timestamp, SEED)));
return SEED % max;
}
bytes16 private constant HEX_SYMBOLS = "0123456789ABCDEF";
function rectSvg(uint x, uint y, bytes4 color) internal pure returns (string memory) {
bytes memory opaqueBuffer = new bytes(6);
bytes memory buffer = new bytes(8);
bool isOpaque = false;
for (uint i = 0; i < 4; i++) {
uint8 value = uint8(color[i]);
buffer[i * 2 + 1] = HEX_SYMBOLS[value & 0xf];
buffer[i * 2] = HEX_SYMBOLS[(value >> 4) & 0xf];
if (i < 3) {
opaqueBuffer[i * 2] = buffer[i * 2];
opaqueBuffer[i * 2 + 1] = buffer[i * 2 + 1];
} else if (value == 255) {
isOpaque = true;
} else if (value == 0) {
return '';
}
}
return string(abi.encodePacked(
'<rect x="', Strings.toString(x), '" y="', Strings.toString(y),
'" fill="#', isOpaque ? string(opaqueBuffer) : string(buffer), '"/>'));
}
function bufferToString(bytes memory buffer, uint length) internal pure returns (string memory text) {
bytes memory stringBuffer = new bytes(length);
for (uint i = 0; i < length; ++i) {
stringBuffer[i] = buffer[i];
}
text = string(stringBuffer);
}
bytes internal constant TABLE = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";
function base64Encode(bytes memory data) internal pure returns (string memory) {
uint256 len = data.length;
if (len == 0) return "";
uint256 encodedLen = 4 * ((len + 2) / 3);
bytes memory result = new bytes(encodedLen + 32);
bytes memory table = TABLE;
assembly {
let tablePtr := add(table, 1)
let resultPtr := add(result, 32)
for {
let i := 0
} lt(i, len) {
} {
i := add(i, 3)
let input := and(mload(add(data, i)), 0xffffff)
let out := mload(add(tablePtr, and(shr(18, input), 0x3F)))
out := shl(8, out)
out := add(out, and(mload(add(tablePtr, and(shr(12, input), 0x3F))), 0xFF))
out := shl(8, out)
out := add(out, and(mload(add(tablePtr, and(shr(6, input), 0x3F))), 0xFF))
out := shl(8, out)
out := add(out, and(mload(add(tablePtr, and(input, 0x3F))), 0xFF))
out := shl(224, out)
mstore(resultPtr, out)
resultPtr := add(resultPtr, 4)
}
switch mod(len, 3)
case 1 {
mstore(sub(resultPtr, 2), shl(240, 0x3d3d))
}
case 2 {
mstore(sub(resultPtr, 1), shl(248, 0x3d))
}
mstore(result, encodedLen)
}
return string(result);
}
}
文件 3 的 13:Context.sol
pragma solidity ^0.8.0;
abstract contract Context {
function _msgSender() internal view virtual returns (address) {
return msg.sender;
}
function _msgData() internal view virtual returns (bytes calldata) {
return msg.data;
}
}
文件 4 的 13:ERC165.sol
pragma solidity ^0.8.0;
import "./IERC165.sol";
abstract contract ERC165 is IERC165 {
function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) {
return interfaceId == type(IERC165).interfaceId;
}
}
文件 5 的 13:ERC721.sol
pragma solidity ^0.8.0;
import "./IERC721.sol";
import "./IERC721Receiver.sol";
import "./extensions/IERC721Metadata.sol";
import "../../utils/Address.sol";
import "../../utils/Context.sol";
import "../../utils/Strings.sol";
import "../../utils/introspection/ERC165.sol";
contract ERC721 is Context, ERC165, IERC721, IERC721Metadata {
using Address for address;
using Strings for uint256;
string private _name;
string private _symbol;
mapping(uint256 => address) private _owners;
mapping(address => uint256) private _balances;
mapping(uint256 => address) private _tokenApprovals;
mapping(address => mapping(address => bool)) private _operatorApprovals;
constructor(string memory name_, string memory symbol_) {
_name = name_;
_symbol = symbol_;
}
function supportsInterface(bytes4 interfaceId) public view virtual override(ERC165, IERC165) returns (bool) {
return
interfaceId == type(IERC721).interfaceId ||
interfaceId == type(IERC721Metadata).interfaceId ||
super.supportsInterface(interfaceId);
}
function balanceOf(address owner) public view virtual override returns (uint256) {
require(owner != address(0), "ERC721: balance query for the zero address");
return _balances[owner];
}
function ownerOf(uint256 tokenId) public view virtual override returns (address) {
address owner = _owners[tokenId];
require(owner != address(0), "ERC721: owner query for nonexistent token");
return owner;
}
function name() public view virtual override returns (string memory) {
return _name;
}
function symbol() public view virtual override returns (string memory) {
return _symbol;
}
function tokenURI(uint256 tokenId) public view virtual override returns (string memory) {
require(_exists(tokenId), "ERC721Metadata: URI query for nonexistent token");
string memory baseURI = _baseURI();
return bytes(baseURI).length > 0 ? string(abi.encodePacked(baseURI, tokenId.toString())) : "";
}
function _baseURI() internal view virtual returns (string memory) {
return "";
}
function approve(address to, uint256 tokenId) public virtual override {
address owner = ERC721.ownerOf(tokenId);
require(to != owner, "ERC721: approval to current owner");
require(
_msgSender() == owner || isApprovedForAll(owner, _msgSender()),
"ERC721: approve caller is not owner nor approved for all"
);
_approve(to, tokenId);
}
function getApproved(uint256 tokenId) public view virtual override returns (address) {
require(_exists(tokenId), "ERC721: approved query for nonexistent token");
return _tokenApprovals[tokenId];
}
function setApprovalForAll(address operator, bool approved) public virtual override {
require(operator != _msgSender(), "ERC721: approve to caller");
_operatorApprovals[_msgSender()][operator] = approved;
emit ApprovalForAll(_msgSender(), operator, approved);
}
function isApprovedForAll(address owner, address operator) public view virtual override returns (bool) {
return _operatorApprovals[owner][operator];
}
function transferFrom(
address from,
address to,
uint256 tokenId
) public virtual override {
require(_isApprovedOrOwner(_msgSender(), tokenId), "ERC721: transfer caller is not owner nor approved");
_transfer(from, to, tokenId);
}
function safeTransferFrom(
address from,
address to,
uint256 tokenId
) public virtual override {
safeTransferFrom(from, to, tokenId, "");
}
function safeTransferFrom(
address from,
address to,
uint256 tokenId,
bytes memory _data
) public virtual override {
require(_isApprovedOrOwner(_msgSender(), tokenId), "ERC721: transfer caller is not owner nor approved");
_safeTransfer(from, to, tokenId, _data);
}
function _safeTransfer(
address from,
address to,
uint256 tokenId,
bytes memory _data
) internal virtual {
_transfer(from, to, tokenId);
require(_checkOnERC721Received(from, to, tokenId, _data), "ERC721: transfer to non ERC721Receiver implementer");
}
function _exists(uint256 tokenId) internal view virtual returns (bool) {
return _owners[tokenId] != address(0);
}
function _isApprovedOrOwner(address spender, uint256 tokenId) internal view virtual returns (bool) {
require(_exists(tokenId), "ERC721: operator query for nonexistent token");
address owner = ERC721.ownerOf(tokenId);
return (spender == owner || getApproved(tokenId) == spender || isApprovedForAll(owner, spender));
}
function _safeMint(address to, uint256 tokenId) internal virtual {
_safeMint(to, tokenId, "");
}
function _safeMint(
address to,
uint256 tokenId,
bytes memory _data
) internal virtual {
_mint(to, tokenId);
require(
_checkOnERC721Received(address(0), to, tokenId, _data),
"ERC721: transfer to non ERC721Receiver implementer"
);
}
function _mint(address to, uint256 tokenId) internal virtual {
require(to != address(0), "ERC721: mint to the zero address");
require(!_exists(tokenId), "ERC721: token already minted");
_beforeTokenTransfer(address(0), to, tokenId);
_balances[to] += 1;
_owners[tokenId] = to;
emit Transfer(address(0), to, tokenId);
}
function _burn(uint256 tokenId) internal virtual {
address owner = ERC721.ownerOf(tokenId);
_beforeTokenTransfer(owner, address(0), tokenId);
_approve(address(0), tokenId);
_balances[owner] -= 1;
delete _owners[tokenId];
emit Transfer(owner, address(0), tokenId);
}
function _transfer(
address from,
address to,
uint256 tokenId
) internal virtual {
require(ERC721.ownerOf(tokenId) == from, "ERC721: transfer of token that is not own");
require(to != address(0), "ERC721: transfer to the zero address");
_beforeTokenTransfer(from, to, tokenId);
_approve(address(0), tokenId);
_balances[from] -= 1;
_balances[to] += 1;
_owners[tokenId] = to;
emit Transfer(from, to, tokenId);
}
function _approve(address to, uint256 tokenId) internal virtual {
_tokenApprovals[tokenId] = to;
emit Approval(ERC721.ownerOf(tokenId), to, tokenId);
}
function _checkOnERC721Received(
address from,
address to,
uint256 tokenId,
bytes memory _data
) private returns (bool) {
if (to.isContract()) {
try IERC721Receiver(to).onERC721Received(_msgSender(), from, tokenId, _data) returns (bytes4 retval) {
return retval == IERC721Receiver.onERC721Received.selector;
} catch (bytes memory reason) {
if (reason.length == 0) {
revert("ERC721: transfer to non ERC721Receiver implementer");
} else {
assembly {
revert(add(32, reason), mload(reason))
}
}
}
} else {
return true;
}
}
function _beforeTokenTransfer(
address from,
address to,
uint256 tokenId
) internal virtual {}
}
文件 6 的 13:ERC721Enumerable.sol
pragma solidity ^0.8.0;
import "../ERC721.sol";
import "./IERC721Enumerable.sol";
abstract contract ERC721Enumerable is ERC721, IERC721Enumerable {
mapping(address => mapping(uint256 => uint256)) private _ownedTokens;
mapping(uint256 => uint256) private _ownedTokensIndex;
uint256[] private _allTokens;
mapping(uint256 => uint256) private _allTokensIndex;
function supportsInterface(bytes4 interfaceId) public view virtual override(IERC165, ERC721) returns (bool) {
return interfaceId == type(IERC721Enumerable).interfaceId || super.supportsInterface(interfaceId);
}
function tokenOfOwnerByIndex(address owner, uint256 index) public view virtual override returns (uint256) {
require(index < ERC721.balanceOf(owner), "ERC721Enumerable: owner index out of bounds");
return _ownedTokens[owner][index];
}
function totalSupply() public view virtual override returns (uint256) {
return _allTokens.length;
}
function tokenByIndex(uint256 index) public view virtual override returns (uint256) {
require(index < ERC721Enumerable.totalSupply(), "ERC721Enumerable: global index out of bounds");
return _allTokens[index];
}
function _beforeTokenTransfer(
address from,
address to,
uint256 tokenId
) internal virtual override {
super._beforeTokenTransfer(from, to, tokenId);
if (from == address(0)) {
_addTokenToAllTokensEnumeration(tokenId);
} else if (from != to) {
_removeTokenFromOwnerEnumeration(from, tokenId);
}
if (to == address(0)) {
_removeTokenFromAllTokensEnumeration(tokenId);
} else if (to != from) {
_addTokenToOwnerEnumeration(to, tokenId);
}
}
function _addTokenToOwnerEnumeration(address to, uint256 tokenId) private {
uint256 length = ERC721.balanceOf(to);
_ownedTokens[to][length] = tokenId;
_ownedTokensIndex[tokenId] = length;
}
function _addTokenToAllTokensEnumeration(uint256 tokenId) private {
_allTokensIndex[tokenId] = _allTokens.length;
_allTokens.push(tokenId);
}
function _removeTokenFromOwnerEnumeration(address from, uint256 tokenId) private {
uint256 lastTokenIndex = ERC721.balanceOf(from) - 1;
uint256 tokenIndex = _ownedTokensIndex[tokenId];
if (tokenIndex != lastTokenIndex) {
uint256 lastTokenId = _ownedTokens[from][lastTokenIndex];
_ownedTokens[from][tokenIndex] = lastTokenId;
_ownedTokensIndex[lastTokenId] = tokenIndex;
}
delete _ownedTokensIndex[tokenId];
delete _ownedTokens[from][lastTokenIndex];
}
function _removeTokenFromAllTokensEnumeration(uint256 tokenId) private {
uint256 lastTokenIndex = _allTokens.length - 1;
uint256 tokenIndex = _allTokensIndex[tokenId];
uint256 lastTokenId = _allTokens[lastTokenIndex];
_allTokens[tokenIndex] = lastTokenId;
_allTokensIndex[lastTokenId] = tokenIndex;
delete _allTokensIndex[tokenId];
_allTokens.pop();
}
}
文件 7 的 13:IERC165.sol
pragma solidity ^0.8.0;
interface IERC165 {
function supportsInterface(bytes4 interfaceId) external view returns (bool);
}
文件 8 的 13:IERC721.sol
pragma solidity ^0.8.0;
import "../../utils/introspection/IERC165.sol";
interface IERC721 is IERC165 {
event Transfer(address indexed from, address indexed to, uint256 indexed tokenId);
event Approval(address indexed owner, address indexed approved, uint256 indexed tokenId);
event ApprovalForAll(address indexed owner, address indexed operator, bool approved);
function balanceOf(address owner) external view returns (uint256 balance);
function ownerOf(uint256 tokenId) external view returns (address owner);
function safeTransferFrom(
address from,
address to,
uint256 tokenId
) external;
function transferFrom(
address from,
address to,
uint256 tokenId
) external;
function approve(address to, uint256 tokenId) external;
function getApproved(uint256 tokenId) external view returns (address operator);
function setApprovalForAll(address operator, bool _approved) external;
function isApprovedForAll(address owner, address operator) external view returns (bool);
function safeTransferFrom(
address from,
address to,
uint256 tokenId,
bytes calldata data
) external;
}
文件 9 的 13:IERC721Enumerable.sol
pragma solidity ^0.8.0;
import "../IERC721.sol";
interface IERC721Enumerable is IERC721 {
function totalSupply() external view returns (uint256);
function tokenOfOwnerByIndex(address owner, uint256 index) external view returns (uint256 tokenId);
function tokenByIndex(uint256 index) external view returns (uint256);
}
文件 10 的 13:IERC721Metadata.sol
pragma solidity ^0.8.0;
import "../IERC721.sol";
interface IERC721Metadata is IERC721 {
function name() external view returns (string memory);
function symbol() external view returns (string memory);
function tokenURI(uint256 tokenId) external view returns (string memory);
}
文件 11 的 13:IERC721Receiver.sol
pragma solidity ^0.8.0;
interface IERC721Receiver {
function onERC721Received(
address operator,
address from,
uint256 tokenId,
bytes calldata data
) external returns (bytes4);
}
文件 12 的 13:Ownable.sol
pragma solidity ^0.8.0;
import "../utils/Context.sol";
abstract contract Ownable is Context {
address private _owner;
event OwnershipTransferred(address indexed previousOwner, address indexed newOwner);
constructor() {
_setOwner(_msgSender());
}
function owner() public view virtual returns (address) {
return _owner;
}
modifier onlyOwner() {
require(owner() == _msgSender(), "Ownable: caller is not the owner");
_;
}
function renounceOwnership() public virtual onlyOwner {
_setOwner(address(0));
}
function transferOwnership(address newOwner) public virtual onlyOwner {
require(newOwner != address(0), "Ownable: new owner is the zero address");
_setOwner(newOwner);
}
function _setOwner(address newOwner) private {
address oldOwner = _owner;
_owner = newOwner;
emit OwnershipTransferred(oldOwner, newOwner);
}
}
文件 13 的 13:Strings.sol
pragma solidity ^0.8.0;
library Strings {
bytes16 private constant _HEX_SYMBOLS = "0123456789abcdef";
function toString(uint256 value) internal pure returns (string memory) {
if (value == 0) {
return "0";
}
uint256 temp = value;
uint256 digits;
while (temp != 0) {
digits++;
temp /= 10;
}
bytes memory buffer = new bytes(digits);
while (value != 0) {
digits -= 1;
buffer[digits] = bytes1(uint8(48 + uint256(value % 10)));
value /= 10;
}
return string(buffer);
}
function toHexString(uint256 value) internal pure returns (string memory) {
if (value == 0) {
return "0x00";
}
uint256 temp = value;
uint256 length = 0;
while (temp != 0) {
length++;
temp >>= 8;
}
return toHexString(value, length);
}
function toHexString(uint256 value, uint256 length) internal pure returns (string memory) {
bytes memory buffer = new bytes(2 * length + 2);
buffer[0] = "0";
buffer[1] = "x";
for (uint256 i = 2 * length + 1; i > 1; --i) {
buffer[i] = _HEX_SYMBOLS[value & 0xf];
value >>= 4;
}
require(value == 0, "Strings: hex length insufficient");
return string(buffer);
}
}
{
"compilationTarget": {
"contracts/ClownTownSociety.sol": "ClownTownSociety"
},
"evmVersion": "istanbul",
"libraries": {},
"metadata": {
"bytecodeHash": "ipfs"
},
"optimizer": {
"enabled": true,
"runs": 2000
},
"remappings": []
}
[{"inputs":[],"stateMutability":"nonpayable","type":"constructor"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"owner","type":"address"},{"indexed":true,"internalType":"address","name":"approved","type":"address"},{"indexed":true,"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"Approval","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"owner","type":"address"},{"indexed":true,"internalType":"address","name":"operator","type":"address"},{"indexed":false,"internalType":"bool","name":"approved","type":"bool"}],"name":"ApprovalForAll","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"previousOwner","type":"address"},{"indexed":true,"internalType":"address","name":"newOwner","type":"address"}],"name":"OwnershipTransferred","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"from","type":"address"},{"indexed":true,"internalType":"address","name":"to","type":"address"},{"indexed":true,"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"Transfer","type":"event"},{"inputs":[{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"approve","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint16","name":"index","type":"uint16"}],"name":"attributesByIndex","outputs":[{"internalType":"string","name":"text","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"}],"name":"balanceOf","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"destroy","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"getApproved","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint16","name":"index","type":"uint16"}],"name":"imageByIndex","outputs":[{"internalType":"string","name":"svg","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"},{"internalType":"address","name":"operator","type":"address"}],"name":"isApprovedForAll","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint16","name":"punkIndex","type":"uint16"}],"name":"isClownMintedForPunkIndex","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"isPublicSaleOpen","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint16","name":"punkIndex","type":"uint16"}],"name":"mintClownFromPunk","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[],"name":"name","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"bool","name":"publicSaleOpened","type":"bool"},{"internalType":"uint256","name":"basePriceInWei","type":"uint256"}],"name":"openPublicSale","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"ownerOf","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint16","name":"punkIndex","type":"uint16"}],"name":"priceInWeiToMintClownForPunkIndex","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"from","type":"address"},{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"safeTransferFrom","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"from","type":"address"},{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"},{"internalType":"bytes","name":"_data","type":"bytes"}],"name":"safeTransferFrom","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"sealContract","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"operator","type":"address"},{"internalType":"bool","name":"approved","type":"bool"}],"name":"setApprovalForAll","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bytes4","name":"interfaceId","type":"bytes4"}],"name":"supportsInterface","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"symbol","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"index","type":"uint256"}],"name":"tokenByIndex","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint16","name":"punkIndex","type":"uint16"}],"name":"tokenByPunkIndex","outputs":[{"internalType":"uint16","name":"","type":"uint16"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"},{"internalType":"uint256","name":"index","type":"uint256"}],"name":"tokenOfOwnerByIndex","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"index","type":"uint256"}],"name":"tokenURI","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"totalSupply","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"from","type":"address"},{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"transferFrom","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"withdraw","outputs":[],"stateMutability":"nonpayable","type":"function"}]