Feature Tip: Add private address tag to any address under My Name Tag !
Overview
ETH Balance
0 ETH
Eth Value
$0.00More Info
Private Name Tags
ContractCreator
TokenTracker
Latest 25 from a total of 148 transactions
Transaction Hash |
Method
|
Block
|
From
|
To
|
|||||
---|---|---|---|---|---|---|---|---|---|
Set Approval For... | 21495321 | 9 days ago | IN | 0 ETH | 0.00048381 | ||||
Set Approval For... | 21487966 | 10 days ago | IN | 0 ETH | 0.00048498 | ||||
Set Approval For... | 19607734 | 272 days ago | IN | 0 ETH | 0.00034162 | ||||
Safe Transfer Fr... | 19551390 | 280 days ago | IN | 0 ETH | 0.00244402 | ||||
Set Approval For... | 18836090 | 381 days ago | IN | 0 ETH | 0.00174847 | ||||
Set Approval For... | 18724512 | 396 days ago | IN | 0 ETH | 0.00249594 | ||||
Set Approval For... | 18315650 | 453 days ago | IN | 0 ETH | 0.00030799 | ||||
Safe Transfer Fr... | 17617414 | 551 days ago | IN | 0 ETH | 0.00168429 | ||||
Set Approval For... | 17521006 | 565 days ago | IN | 0 ETH | 0.00063622 | ||||
Set Approval For... | 17354227 | 588 days ago | IN | 0 ETH | 0.00130508 | ||||
Set Approval For... | 17327086 | 592 days ago | IN | 0 ETH | 0.00143864 | ||||
Set Approval For... | 17317186 | 594 days ago | IN | 0 ETH | 0.00319867 | ||||
Set Approval For... | 17303291 | 595 days ago | IN | 0 ETH | 0.00159793 | ||||
Set Approval For... | 17294485 | 597 days ago | IN | 0 ETH | 0.00391732 | ||||
Set Approval For... | 17245923 | 604 days ago | IN | 0 ETH | 0.00349005 | ||||
Set Approval For... | 17245225 | 604 days ago | IN | 0 ETH | 0.00489235 | ||||
Set Approval For... | 17176033 | 613 days ago | IN | 0 ETH | 0.00493866 | ||||
Set Approval For... | 17131205 | 620 days ago | IN | 0 ETH | 0.00110344 | ||||
Set Approval For... | 17056527 | 630 days ago | IN | 0 ETH | 0.00105493 | ||||
Set Approval For... | 17009515 | 637 days ago | IN | 0 ETH | 0.00099343 | ||||
Safe Transfer Fr... | 16976674 | 642 days ago | IN | 0 ETH | 0.0043927 | ||||
Set Approval For... | 16934805 | 648 days ago | IN | 0 ETH | 0.00179609 | ||||
Set Approval For... | 16806431 | 666 days ago | IN | 0 ETH | 0.00205136 | ||||
Set Approval For... | 16706497 | 680 days ago | IN | 0 ETH | 0.00133563 | ||||
Delegate | 16645667 | 688 days ago | IN | 0 ETH | 0.00295599 |
Latest 1 internal transaction
Advanced mode:
Parent Transaction Hash | Block |
From
|
To
|
|||
---|---|---|---|---|---|---|
15595994 | 835 days ago | Contract Creation | 0 ETH |
Loading...
Loading
Minimal Proxy Contract for 0x26f0ecef411fb856b267b0f2fb21ee694fc056e9
Contract Name:
NounletToken
Compiler Version
v0.8.13+commit.abaa5c0e
Optimization Enabled:
Yes with 200 runs
Other Settings:
default evmVersion
Contract Source Code (Solidity Standard Json-Input format)
// SPDX-License-Identifier: MIT pragma solidity 0.8.13; import {Clone} from "clones-with-immutable-args/src/Clone.sol"; import {ERC1155B, ERC1155BCheckpointable} from "./utils/ERC1155BCheckpointable.sol"; import {PermitBase} from "./utils/PermitBase.sol"; import {Strings} from "@openzeppelin/contracts/utils/Strings.sol"; import {INounletToken as INounlet} from "./interfaces/INounletToken.sol"; import {INounsDescriptor as IDescriptor} from "./interfaces/INounsDescriptor.sol"; import {INounsSeeder} from "./interfaces/INounsSeeder.sol"; import {INounsToken as INouns} from "./interfaces/INounsToken.sol"; /// @title NounletToken /// @author Tessera /// @notice An ERC-1155B implementation for Fractional Nouns contract NounletToken is Clone, ERC1155BCheckpointable, INounlet, PermitBase { /// @dev Using strings library with uint256 types using Strings for uint256; /// @notice Percentage amount for royalties uint96 public constant ROYALTY_PERCENT = 200; /// @notice Mapping of token ID to Nounlet seed mapping(uint256 => INounsSeeder.Seed) public seeds; /// @notice Mapping of token type approvals owner => operator => tokenId => approval status mapping(address => mapping(address => mapping(uint256 => bool))) public isApproved; /// @dev Modifier for restricting function calls modifier onlyAuthorized(address _sender) { if (msg.sender != _sender) revert InvalidSender(_sender, msg.sender); _; } /// @notice Mints new fractions for an ID /// @param _to Address to mint fraction tokens to /// @param _id Token ID to mint /// @param _data Extra calldata to include in the mint function mint( address _to, uint256 _id, bytes calldata _data ) external onlyAuthorized(NOUNLET_REGISTRY()) { seeds[_id] = generateSeed(_id); _mint(_to, _id, _data); } /// @notice Burns fractions for multiple IDs /// @param _from Address to burn fraction tokens from /// @param _ids Token IDs to burn function batchBurn(address _from, uint256[] calldata _ids) external onlyAuthorized(NOUNLET_REGISTRY()) { _batchBurn(_from, _ids); } /// @notice Permit function that approves an operator for token type with a valid signature /// @param _owner Address of the owner of the token type /// @param _operator Address of the spender of the token type /// @param _id ID of the token type /// @param _approved Approval status for the token type /// @param _deadline Expiration of the signature /// @param _v The recovery ID (129th byte and chain ID) of the signature used to recover the signer /// @param _r The first 64 bytes of the signature /// @param _s Bytes 64-128 of the signature function permit( address _owner, address _operator, uint256 _id, bool _approved, uint256 _deadline, uint8 _v, bytes32 _r, bytes32 _s ) external { if (block.timestamp > _deadline) revert SignatureExpired(block.timestamp, _deadline); // cannot realistically overflow on human timescales unchecked { bytes32 structHash = _computePermitStructHash( _owner, _operator, _id, _approved, _deadline ); bytes32 digest = _computeDigest(_computeDomainSeparator(), structHash); address signer = ecrecover(digest, _v, _r, _s); if (signer == address(0) || signer != _owner) revert InvalidSignature(signer, _owner); } isApproved[_owner][_operator][_id] = _approved; emit SingleApproval(_owner, _operator, _id, _approved); } /// @notice Permit function that approves an operator for all token types with a valid signature /// @param _owner Address of the owner of the token type /// @param _operator Address of the spender of the token type /// @param _approved Approval status for the token type /// @param _deadline Expiration of the signature /// @param _v The recovery ID (129th byte and chain ID) of the signature used to recover the signer /// @param _r The first 64 bytes of the signature /// @param _s Bytes 64-128 of the signature function permitAll( address _owner, address _operator, bool _approved, uint256 _deadline, uint8 _v, bytes32 _r, bytes32 _s ) external { if (block.timestamp > _deadline) revert SignatureExpired(block.timestamp, _deadline); // cannot realistically overflow on human timescales unchecked { bytes32 structHash = _computePermitAllStructHash( _owner, _operator, _approved, _deadline ); bytes32 digest = _computeDigest(_computeDomainSeparator(), structHash); address signer = ecrecover(digest, _v, _r, _s); if (signer == address(0) || signer != _owner) revert InvalidSignature(signer, _owner); } isApprovedForAll[_owner][_operator] = _approved; emit ApprovalForAll(_owner, _operator, _approved); } /// @notice Scoped approvals allow us to eliminate some of the risks associated with setting the approval for an entire collection /// @param _operator Address of spender account /// @param _id ID of the token type /// @param _approved Approval status for operator(spender) account function setApprovalFor( address _operator, uint256 _id, bool _approved ) external { isApproved[msg.sender][_operator][_id] = _approved; emit SingleApproval(msg.sender, _operator, _id, _approved); } /// @notice Returns the royalty amount for a given token /// @param _salePrice Price of token sold on secondary market function royaltyInfo( uint256, /* _id */ uint256 _salePrice ) external pure returns (address beneficiary, uint256 royaltyAmount) { beneficiary = ROYALTY_BENEFICIARY(); royaltyAmount = (_salePrice * uint256(ROYALTY_PERCENT)) / 10000; } /// @notice Transfers multiple token types /// @param _from Source address /// @param _to Destination address /// @param _ids IDs of each token type /// @param _amounts Transfer amounts per token type /// @param _data Additional calldata function safeBatchTransferFrom( address _from, address _to, uint256[] calldata _ids, uint256[] calldata _amounts, bytes calldata _data ) public override(ERC1155BCheckpointable, INounlet) { super.safeBatchTransferFrom(_from, _to, _ids, _amounts, _data); } /// @notice Transfers a single token type /// @param _from Source address /// @param _to Destination address /// @param _id ID of the token type /// @param _amount Transfer amount /// @param _data Additional calldata function safeTransferFrom( address _from, address _to, uint256 _id, uint256 _amount, bytes calldata _data ) public override(ERC1155BCheckpointable, INounlet) { require( msg.sender == _from || isApprovedForAll[_from][msg.sender] || isApproved[_from][msg.sender][_id], "NOT_AUTHORIZED" ); super.safeTransferFrom(_from, _to, _id, _amount, _data); } function batchTransferFrom( address _from, address _to, uint256[] calldata _ids, uint256[] calldata _amounts, bytes calldata _data ) public override(ERC1155BCheckpointable, INounlet) { super.batchTransferFrom(_from, _to, _ids, _amounts, _data); } /// @notice Transfers a single token type /// @param _from Source address /// @param _to Destination address /// @param _id ID of the token type /// @param _amount Transfer amount /// @param _data Additional calldata function transferFrom( address _from, address _to, uint256 _id, uint256 _amount, bytes calldata _data ) public override(ERC1155BCheckpointable, INounlet) { require( msg.sender == _from || isApprovedForAll[_from][msg.sender] || isApproved[_from][msg.sender][_id], "NOT_AUTHORIZED" ); super.transferFrom(_from, _to, _id, _amount, _data); } /// @notice Returns the URI of a token type /// @param _id ID of the token type function uri(uint256 _id) public view override(ERC1155B, INounlet) returns (string memory) { string memory nounId = NOUNS_TOKEN_ID().toString(); string memory name = string(abi.encodePacked("Nounlet #", _id.toString())); string memory description = string( abi.encodePacked("Noun ", nounId, " is collectively owned by a 100 member DAO") ); return IDescriptor(NOUNS_DESCRIPTOR()).genericDataURI(name, description, seeds[_id]); } function contractURI() public view returns (string memory) { return string(abi.encodePacked("https://nounlets.wtf/api/noun/", NOUNS_TOKEN_ID().toString())); } function owner() public view returns (address) { return ROYALTY_BENEFICIARY(); } /// @notice Generates a random seed for a given token /// @param _id ID of the token type function generateSeed(uint256 _id) public view returns (INounsSeeder.Seed memory) { address descriptor = NOUNS_DESCRIPTOR(); INounsSeeder.Seed memory noun = INouns(NOUNS_TOKEN()).seeds(NOUNS_TOKEN_ID()); uint256 pseudorandomness = uint256( keccak256(abi.encodePacked(blockhash(block.number - 1), _id)) ); uint256 backgroundCount = IDescriptor(descriptor).backgroundCount(); uint256 bodyCount = IDescriptor(descriptor).bodyCount(); uint256 accessoryCount = IDescriptor(descriptor).accessoryCount(); uint256 glassesCount = IDescriptor(descriptor).glassesCount(); return INounsSeeder.Seed({ background: uint48((pseudorandomness) % backgroundCount), body: uint48((pseudorandomness >> 48) % bodyCount), accessory: uint48((pseudorandomness >> 96) % accessoryCount), head: uint48(noun.head), glasses: uint48((pseudorandomness >> 192) % glassesCount) }); } /// @dev Returns true if this contract implements the interface defined by the given identifier function supportsInterface(bytes4 _interfaceId) public view virtual returns (bool) { return (_interfaceId == 0x01ffc9a7 || // ERC165 Interface ID for ERC165 _interfaceId == 0xd9b67a26 || // ERC165 Interface ID for ERC1155 _interfaceId == 0x2a55205a); // ERC165 Interface ID for ERC2981 } /// @notice Address of NounletRegistry contract /// @dev ClonesWithImmutableArgs appends this value to the end of the initial bytecode at index 0 function NOUNLET_REGISTRY() public pure returns (address) { return _getArgAddress(0); } /// @notice Address of NounsDescriptor contract /// @dev ClonesWithImmutableArgs appends this value to the end of the initial bytecode at index 20 function NOUNS_DESCRIPTOR() public pure returns (address) { return _getArgAddress(20); } /// @notice ID of the NounsToken /// @dev ClonesWithImmutableArgs appends this value to the end of the initial bytecode at index 40 function NOUNS_TOKEN_ID() public pure returns (uint256) { return _getArgUint256(40); } /// @notice Address of the royalty beneficiary /// @dev ClonesWithImmutableArgs appends this value to the end of the initial bytecode at index 72 function ROYALTY_BENEFICIARY() public pure returns (address) { return _getArgAddress(72); } function NOUNS_TOKEN() public pure returns (address) { return _getArgAddress(92); } }
// SPDX-License-Identifier: BSD pragma solidity ^0.8.4; /// @title Clone /// @author zefram.eth /// @notice Provides helper functions for reading immutable args from calldata contract Clone { /// @notice Reads an immutable arg with type address /// @param argOffset The offset of the arg in the packed data /// @return arg The arg value function _getArgAddress(uint256 argOffset) internal pure returns (address arg) { uint256 offset = _getImmutableArgsOffset(); // solhint-disable-next-line no-inline-assembly assembly { arg := shr(0x60, calldataload(add(offset, argOffset))) } } /// @notice Reads an immutable arg with type uint256 /// @param argOffset The offset of the arg in the packed data /// @return arg The arg value function _getArgUint256(uint256 argOffset) internal pure returns (uint256 arg) { uint256 offset = _getImmutableArgsOffset(); // solhint-disable-next-line no-inline-assembly assembly { arg := calldataload(add(offset, argOffset)) } } /// @notice Reads an immutable arg with type bytes32 /// @param argOffset The offset of the arg in the packed data /// @return arg The arg value function _getArgBytes32(uint256 argOffset) internal pure returns (bytes32 arg) { uint256 offset = _getImmutableArgsOffset(); // solhint-disable-next-line no-inline-assembly assembly { arg := calldataload(add(offset, argOffset)) } } /// @notice Reads a uint256 array stored in the immutable args. /// @param argOffset The offset of the arg in the packed data /// @param arrLen Number of elements in the array /// @return arr The array function _getArgUint256Array(uint256 argOffset, uint64 arrLen) internal pure returns (uint256[] memory arr) { uint256 offset = _getImmutableArgsOffset(); uint256 el; arr = new uint256[](arrLen); for (uint64 i = 0; i < arrLen; i++) { assembly { // solhint-disable-next-line no-inline-assembly el := calldataload(add(add(offset, argOffset), mul(i, 32))) } arr[i] = el; } return arr; } /// @notice Reads a uint256 array stored in the immutable args. /// @param argOffset The offset of the arg in the packed data /// @param arrLen Number of elements in the array /// @return arr The array function _getArgBytes32Array(uint256 argOffset, uint64 arrLen) internal pure returns (bytes32[] memory arr) { uint256 offset = _getImmutableArgsOffset(); bytes32 el; arr = new bytes32[](arrLen); for (uint64 i = 0; i < arrLen; i++) { assembly { // solhint-disable-next-line no-inline-assembly el := calldataload(add(add(offset, argOffset), mul(i, 32))) } arr[i] = el; } return arr; } /// @notice Reads an immutable arg with type uint64 /// @param argOffset The offset of the arg in the packed data /// @return arg The arg value function _getArgUint64(uint256 argOffset) internal pure returns (uint64 arg) { uint256 offset = _getImmutableArgsOffset(); // solhint-disable-next-line no-inline-assembly assembly { arg := shr(0xc0, calldataload(add(offset, argOffset))) } } /// @notice Reads an immutable arg with type uint8 /// @param argOffset The offset of the arg in the packed data /// @return arg The arg value function _getArgUint8(uint256 argOffset) internal pure returns (uint8 arg) { uint256 offset = _getImmutableArgsOffset(); // solhint-disable-next-line no-inline-assembly assembly { arg := shr(0xf8, calldataload(add(offset, argOffset))) } } /// @return offset The offset of the packed immutable args in calldata function _getImmutableArgsOffset() internal pure returns (uint256 offset) { // solhint-disable-next-line no-inline-assembly assembly { offset := sub( calldatasize(), add(shr(240, calldataload(sub(calldatasize(), 2))), 2) ) } } }
// SPDX-License-Identifier: AGPL-3.0-only pragma solidity >=0.8.0; import "./ERC1155B.sol"; import "@rari-capital/solmate/src/utils/SafeCastLib.sol"; import "@openzeppelin/contracts/utils/cryptography/draft-EIP712.sol"; abstract contract ERC1155BCheckpointable is ERC1155B { using SafeCastLib for uint256; /// @notice Defines decimals as per ERC-20 convention to make integrations with 3rd party governance platforms easier uint8 public constant decimals = 0; uint96 public totalSupply; uint96 public constant maxSupply = type(uint96).max; /// delegator -> delegatee mapping(address => address) private _delegates; /// delegator -> #votes mapping(address => uint96) public _ballots; /// delegatee -> #checkpoints mapping(address => uint32) public numCheckpoints; /// delegatee -> checkpoint# -> checkpoint(fromBlock,#votes) mapping(address => mapping(uint32 => Checkpoint)) public checkpoints; struct Checkpoint { uint32 fromBlock; uint96 votes; } /// @dev Event log for changing of delegate by user /// @param _delegator Address of delegator /// @param _fromDelegate Address of previous delegate /// @param _toDelegate Address of new delegate event DelegateChanged( address indexed _delegator, address indexed _fromDelegate, address indexed _toDelegate ); /// @dev Event log for moving of delegates /// @param _delegate Address of the delegate /// @param _previousBalance Previous balance of votes /// @param _newBalance Updated balance of votes event DelegateVotesChanged( address indexed _delegate, uint256 _previousBalance, uint256 _newBalance ); function delegate(address delegatee) external { return _delegate(msg.sender, delegatee); } function getCurrentVotes(address account) external view returns (uint96) { uint32 pos = numCheckpoints[account]; return pos != 0 ? checkpoints[account][pos - 1].votes : 0; } function getPriorVotes(address account, uint256 blockNumber) public view virtual returns (uint256) { require(block.number > blockNumber, "UNDETERMINED"); uint32 pos = numCheckpoints[account]; if (pos == 0) return 0; if (checkpoints[account][pos - 1].fromBlock <= blockNumber) return checkpoints[account][pos - 1].votes; if (checkpoints[account][0].fromBlock > blockNumber) return 0; uint32 lower; uint32 upper = pos - 1; while (upper > lower) { uint32 center = upper - (upper - lower) / 2; Checkpoint memory cp = checkpoints[account][center]; if (cp.fromBlock == blockNumber) return cp.votes; cp.fromBlock < blockNumber ? lower = center : upper = center - 1; } return checkpoints[account][lower].votes; } function votesToDelegate(address delegator) public view returns (uint96) { return _ballots[delegator]; } /** * @notice Overrides the standard `Comp.sol` delegates mapping to return * the delegator's own address if they haven't delegated. * This avoids having to delegate to oneself. */ function delegates(address delegator) public view returns (address) { address current = _delegates[delegator]; return current == address(0) ? delegator : current; } function safeTransferFrom( address src, address dst, uint256 id, uint256, bytes calldata data ) public virtual override { _ballots[src]--; _ballots[dst]++; _moveDelegates(delegates(src), delegates(dst), 1); super.safeTransferFrom(src, dst, id, 1, data); } function safeBatchTransferFrom( address src, address dst, uint256[] calldata ids, uint256[] calldata amounts, bytes calldata data ) public virtual override { uint96 amount = ids.length.safeCastTo96(); _ballots[src] -= amount; _ballots[dst] += amount; _moveDelegates(delegates(src), delegates(dst), amount); super.safeBatchTransferFrom(src, dst, ids, amounts, data); } function transferFrom( address src, address dst, uint256 id, uint256, bytes calldata data ) public virtual override { _ballots[src]--; _ballots[dst]++; _moveDelegates(delegates(src), delegates(dst), 1); super.transferFrom(src, dst, id, 1, data); } function batchTransferFrom( address src, address dst, uint256[] calldata ids, uint256[] calldata amounts, bytes calldata data ) public virtual override { uint96 amount = ids.length.safeCastTo96(); _ballots[src] -= amount; _ballots[dst] += amount; _moveDelegates(delegates(src), delegates(dst), amount); super.batchTransferFrom(src, dst, ids, amounts, data); } function _mint( address dst, uint256 id, bytes memory data ) internal virtual override { totalSupply++; _ballots[dst]++; _moveDelegates(address(0), delegates(dst), 1); super._mint(dst, id, data); } function _batchMint( address dst, uint256[] memory ids, bytes memory data ) internal virtual override { uint96 amount = ids.length.safeCastTo96(); totalSupply += amount; _ballots[dst] += amount; _moveDelegates(address(0), delegates(dst), amount); super._batchMint(dst, ids, data); } function _burn(uint256 id) internal virtual override { totalSupply--; _ballots[msg.sender]--; _moveDelegates(delegates(msg.sender), address(0), 1); super._burn(id); } function _batchBurn(address src, uint256[] memory ids) internal virtual override { uint96 amount = ids.length.safeCastTo96(); totalSupply -= amount; _ballots[src] -= amount; _moveDelegates(delegates(src), address(0), amount); super._batchBurn(src, ids); } function _delegate(address delegator, address delegatee) internal { /// @notice differs from `_delegate()` in `Comp.sol` to use `delegates` override method to simulate auto-delegation address currentDelegate = delegates(delegator); _delegates[delegator] = delegatee; uint96 amount = votesToDelegate(delegator); _moveDelegates(currentDelegate, delegatee, amount); emit DelegateChanged(delegator, currentDelegate, delegatee); } function _moveDelegates( address src, address dst, uint96 amount ) internal { if (src != dst && amount > 0) { if (src != address(0)) { uint96 srcOld; uint32 srcPos = numCheckpoints[src]; srcOld = srcPos != 0 ? checkpoints[src][srcPos - 1].votes : 0; uint96 srcNew = srcOld - amount; _writeCheckpoint(src, srcPos, srcOld, srcNew); } if (dst != address(0)) { uint32 dstPos = numCheckpoints[dst]; uint96 dstOld; dstOld = dstPos != 0 ? checkpoints[dst][dstPos - 1].votes : 0; uint96 dstNew = dstOld + amount; _writeCheckpoint(dst, dstPos, dstOld, dstNew); } } } function _writeCheckpoint( address delegatee, uint32 pos, uint96 oldVotes, uint96 newVotes ) internal { uint32 blockNumber = block.number.safeCastTo32(); if (pos > 0 && checkpoints[delegatee][pos - 1].fromBlock == blockNumber) { checkpoints[delegatee][pos - 1].votes = newVotes; } else { checkpoints[delegatee][pos] = Checkpoint(blockNumber, newVotes); numCheckpoints[delegatee] = pos + 1; } emit DelegateVotesChanged(delegatee, oldVotes, newVotes); } }
// SPDX-License-Identifier: MIT pragma solidity 0.8.13; import "../constants/Permit.sol"; /// @title PermitBase /// @author Tessera /// @notice Utility contract for computing permit signature approvals for FERC1155 tokens abstract contract PermitBase { /// @notice Mapping of token owner to nonce value mapping(address => uint256) public nonces; /// @dev Computes hash of permit struct /// @param _owner Address of the owner of the token type /// @param _operator Address of the spender of the token type /// @param _id ID of the token type /// @param _approved Approval status for the token type /// @param _deadline Expiration of the signature function _computePermitStructHash( address _owner, address _operator, uint256 _id, bool _approved, uint256 _deadline ) internal returns (bytes32) { return keccak256( abi.encode( PERMIT_TYPEHASH, _owner, _operator, _id, _approved, nonces[_owner]++, _deadline ) ); } /// @dev Computes hash of permit all struct /// @param _owner Address of the owner of the token type /// @param _operator Address of the spender of the token type /// @param _approved Approval status for the token type /// @param _deadline Expiration of the signature function _computePermitAllStructHash( address _owner, address _operator, bool _approved, uint256 _deadline ) internal returns (bytes32) { return keccak256( abi.encode( PERMIT_ALL_TYPEHASH, _owner, _operator, _approved, nonces[_owner]++, _deadline ) ); } /// @dev Computes domain separator to prevent signature collisions /// @return Hash of the contract-specific fields function _computeDomainSeparator() internal view returns (bytes32) { return keccak256( abi.encode( DOMAIN_TYPEHASH, keccak256(bytes(NAME)), keccak256(bytes(VERSION)), block.chainid, address(this) ) ); } /// @dev Computes digest of domain separator and struct hash /// @param _domainSeparator Hash of contract-specific fields /// @param _structHash Hash of signature fields struct /// @return Hash of the signature digest function _computeDigest(bytes32 _domainSeparator, bytes32 _structHash) internal pure returns (bytes32) { return keccak256(abi.encodePacked("\x19\x01", _domainSeparator, _structHash)); } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.7.0) (utils/Strings.sol) pragma solidity ^0.8.0; /** * @dev String operations. */ library Strings { bytes16 private constant _HEX_SYMBOLS = "0123456789abcdef"; uint8 private constant _ADDRESS_LENGTH = 20; /** * @dev Converts a `uint256` to its ASCII `string` decimal representation. */ function toString(uint256 value) internal pure returns (string memory) { // Inspired by OraclizeAPI's implementation - MIT licence // https://github.com/oraclize/ethereum-api/blob/b42146b063c7d6ee1358846c198246239e9360e8/oraclizeAPI_0.4.25.sol 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); } /** * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation. */ 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); } /** * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation with fixed 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); } /** * @dev Converts an `address` with fixed length of 20 bytes to its not checksummed ASCII `string` hexadecimal representation. */ function toHexString(address addr) internal pure returns (string memory) { return toHexString(uint256(uint160(addr)), _ADDRESS_LENGTH); } }
// SPDX-License-Identifier: MIT pragma solidity 0.8.13; import {IERC165} from "@openzeppelin/contracts/utils/introspection/IERC165.sol"; import {INounsSeeder} from "./INounsSeeder.sol"; /// @dev Checkpoint information struct Checkpoint { // Timestamp of checkpoint creation uint64 fromTimestamp; // Amount of votes uint192 votes; } /// @dev Interface for NounletToken contract interface INounletToken is IERC165 { /// @dev Emitted when deadline for signature has passed error SignatureExpired(uint256 _timestamp, uint256 _deadline); /// @dev Emitted when caller is not required address error InvalidSender(address _required, address _provided); /// @dev Emitted when owner signature is invalid error InvalidSignature(address _signer, address _owner); /// @dev Event log for approving a spender of a token type /// @param _owner Address of the owner of the token type /// @param _operator Address of the spender of the token type /// @param _id ID of the token type /// @param _approved Approval status for the token type event SingleApproval( address indexed _owner, address indexed _operator, uint256 _id, bool _approved ); function NOUNLET_REGISTRY() external pure returns (address); function NOUNS_DESCRIPTOR() external pure returns (address); function NOUNS_TOKEN_ID() external pure returns (uint256); function ROYALTY_BENEFICIARY() external pure returns (address); function batchBurn(address _from, uint256[] memory _ids) external; function generateSeed(uint256 _id) external view returns (INounsSeeder.Seed memory); function isApproved( address, address, uint256 ) external view returns (bool); function mint( address _to, uint256 _id, bytes memory _data ) external; function permit( address _owner, address _operator, uint256 _id, bool _approved, uint256 _deadline, uint8 _v, bytes32 _r, bytes32 _s ) external; function permitAll( address _owner, address _operator, bool _approved, uint256 _deadline, uint8 _v, bytes32 _r, bytes32 _s ) external; function royaltyInfo(uint256 _id, uint256 _salePrice) external view returns (address receiver, uint256 royaltyAmount); function safeTransferFrom( address _from, address _to, uint256 _id, uint256 _amount, bytes memory _data ) external; function safeBatchTransferFrom( address _from, address _to, uint256[] memory _ids, uint256[] memory _amounts, bytes memory _data ) external; function transferFrom( address _from, address _to, uint256 _id, uint256 _amount, bytes memory _data ) external; function batchTransferFrom( address _from, address _to, uint256[] calldata _ids, uint256[] calldata _amounts, bytes calldata _data ) external; function setApprovalFor( address _operator, uint256 _id, bool _approved ) external; function supportsInterface(bytes4 _interfaceId) external view returns (bool); function uri(uint256 _id) external view returns (string memory); }
// SPDX-License-Identifier: GPL-3.0 /// @title Common interface for NounsDescriptor versions, as used by NounsToken and NounsSeeder. /********************************* * ░░░░░░░░░░░░░░░░░░░░░░░░░░░░░ * * ░░░░░░░░░░░░░░░░░░░░░░░░░░░░░ * * ░░░░░░█████████░░█████████░░░ * * ░░░░░░██░░░████░░██░░░████░░░ * * ░░██████░░░████████░░░████░░░ * * ░░██░░██░░░████░░██░░░████░░░ * * ░░██░░██░░░████░░██░░░████░░░ * * ░░░░░░█████████░░█████████░░░ * * ░░░░░░░░░░░░░░░░░░░░░░░░░░░░░ * * ░░░░░░░░░░░░░░░░░░░░░░░░░░░░░ * *********************************/ pragma solidity ^0.8.13; import {INounsSeeder} from "./INounsSeeder.sol"; interface INounsDescriptor { function tokenURI(uint256 tokenId, INounsSeeder.Seed memory seed) external view returns (string memory); function dataURI(uint256 tokenId, INounsSeeder.Seed memory seed) external view returns (string memory); function backgroundCount() external view returns (uint256); function bodyCount() external view returns (uint256); function accessoryCount() external view returns (uint256); function headCount() external view returns (uint256); function glassesCount() external view returns (uint256); function genericDataURI( string calldata name, string calldata description, INounsSeeder.Seed memory seed ) external view returns (string memory); }
// SPDX-License-Identifier: GPL-3.0 /// @title Interface for NounsSeeder /********************************* * ░░░░░░░░░░░░░░░░░░░░░░░░░░░░░ * * ░░░░░░░░░░░░░░░░░░░░░░░░░░░░░ * * ░░░░░░█████████░░█████████░░░ * * ░░░░░░██░░░████░░██░░░████░░░ * * ░░██████░░░████████░░░████░░░ * * ░░██░░██░░░████░░██░░░████░░░ * * ░░██░░██░░░████░░██░░░████░░░ * * ░░░░░░█████████░░█████████░░░ * * ░░░░░░░░░░░░░░░░░░░░░░░░░░░░░ * * ░░░░░░░░░░░░░░░░░░░░░░░░░░░░░ * *********************************/ pragma solidity ^0.8.13; import {INounsDescriptor} from "./INounsDescriptor.sol"; interface INounsSeeder { struct Seed { uint48 background; uint48 body; uint48 accessory; uint48 head; uint48 glasses; } function generateSeed(uint256 nounId, INounsDescriptor descriptor) external view returns (Seed memory); }
// SPDX-License-Identifier: GPL-3.0 /// @title Interface for NounsToken /********************************* * ░░░░░░░░░░░░░░░░░░░░░░░░░░░░░ * * ░░░░░░░░░░░░░░░░░░░░░░░░░░░░░ * * ░░░░░░█████████░░█████████░░░ * * ░░░░░░██░░░████░░██░░░████░░░ * * ░░██████░░░████████░░░████░░░ * * ░░██░░██░░░████░░██░░░████░░░ * * ░░██░░██░░░████░░██░░░████░░░ * * ░░░░░░█████████░░█████████░░░ * * ░░░░░░░░░░░░░░░░░░░░░░░░░░░░░ * * ░░░░░░░░░░░░░░░░░░░░░░░░░░░░░ * *********************************/ pragma solidity ^0.8.13; import {IERC721} from "@openzeppelin/contracts/token/ERC721/IERC721.sol"; import {INounsDescriptor} from "./INounsDescriptor.sol"; import {INounsSeeder} from "./INounsSeeder.sol"; interface INounsToken is IERC721 { event DescriptorLocked(); event DescriptorUpdated(INounsDescriptor descriptor); event NounBurned(uint256 indexed tokenId); event NounCreated(uint256 indexed tokenId, INounsSeeder.Seed seed); event NoundersDAOUpdated(address noundersDAO); event MinterLocked(); event MinterUpdated(address minter); event SeederLocked(); event SeederUpdated(INounsSeeder seeder); function burn(uint256 tokenId) external; function dataURI(uint256 tokenId) external returns (string memory); function delegate(address delegatee) external; function descriptor() external returns (INounsDescriptor); function lockDescriptor() external; function lockMinter() external; function lockSeeder() external; function mint() external returns (uint256); function minter() external returns (address); function seeds(uint256) external view returns (INounsSeeder.Seed memory); function setDescriptor(INounsDescriptor descriptor) external; function setMinter(address minter) external; function setNoundersDAO(address noundersDAO) external; function setSeeder(INounsSeeder seeder) external; }
// SPDX-License-Identifier: MIT pragma solidity >=0.8.0; import {ERC1155TokenReceiver} from "@rari-capital/solmate/src/tokens/ERC1155.sol"; /// @notice Minimalist and gas efficient ERC1155 implementation optimized for single supply ids. /// @author Solmate (https://github.com/Rari-Capital/solmate/blob/main/src/tokens/ERC1155B.sol) abstract contract ERC1155B { /*////////////////////////////////////////////////////////////// EVENTS //////////////////////////////////////////////////////////////*/ event TransferSingle( address indexed operator, address indexed from, address indexed to, uint256 id, uint256 amount ); event TransferBatch( address indexed operator, address indexed from, address indexed to, uint256[] ids, uint256[] amounts ); event ApprovalForAll(address indexed owner, address indexed operator, bool approved); event URI(string value, uint256 indexed id); /*////////////////////////////////////////////////////////////// ERC1155 STORAGE //////////////////////////////////////////////////////////////*/ mapping(address => mapping(address => bool)) public isApprovedForAll; /*////////////////////////////////////////////////////////////// ERC1155B STORAGE //////////////////////////////////////////////////////////////*/ mapping(uint256 => address) public ownerOf; function balanceOf(address owner, uint256 id) public view virtual returns (uint256 bal) { address idOwner = ownerOf[id]; assembly { // We avoid branching by using assembly to take // the bool output of eq() and use it as a uint. bal := eq(idOwner, owner) } } /*////////////////////////////////////////////////////////////// METADATA LOGIC //////////////////////////////////////////////////////////////*/ function uri(uint256 id) public view virtual returns (string memory); /*////////////////////////////////////////////////////////////// ERC1155 LOGIC //////////////////////////////////////////////////////////////*/ function setApprovalForAll(address operator, bool approved) public virtual { isApprovedForAll[msg.sender][operator] = approved; emit ApprovalForAll(msg.sender, operator, approved); } function safeTransferFrom( address from, address to, uint256 id, uint256 amount, bytes calldata data ) public virtual { require(msg.sender == from || isApprovedForAll[from][msg.sender], "NOT_AUTHORIZED"); require(from == ownerOf[id], "WRONG_FROM"); // Can only transfer from the owner. // Can only transfer 1 with ERC1155B. require(amount == 1, "INVALID_AMOUNT"); ownerOf[id] = to; emit TransferSingle(msg.sender, from, to, id, amount); if (to.code.length != 0) { require( ERC1155TokenReceiver(to).onERC1155Received(msg.sender, from, id, amount, data) == ERC1155TokenReceiver.onERC1155Received.selector, "UNSAFE_RECIPIENT" ); } else require(to != address(0), "INVALID_RECIPIENT"); } function safeBatchTransferFrom( address from, address to, uint256[] calldata ids, uint256[] calldata amounts, bytes calldata data ) public virtual { require(ids.length == amounts.length, "LENGTH_MISMATCH"); require(msg.sender == from || isApprovedForAll[from][msg.sender], "NOT_AUTHORIZED"); // Storing these outside the loop saves ~15 gas per iteration. uint256 id; uint256 amount; // Unchecked because the only math done is incrementing // the array index counter which cannot possibly overflow. unchecked { for (uint256 i = 0; i < ids.length; i++) { id = ids[i]; amount = amounts[i]; // Can only transfer from the owner. require(from == ownerOf[id], "WRONG_FROM"); // Can only transfer 1 with ERC1155B. require(amount == 1, "INVALID_AMOUNT"); ownerOf[id] = to; } } emit TransferBatch(msg.sender, from, to, ids, amounts); if (to.code.length != 0) { require( ERC1155TokenReceiver(to).onERC1155BatchReceived( msg.sender, from, ids, amounts, data ) == ERC1155TokenReceiver.onERC1155BatchReceived.selector, "UNSAFE_RECIPIENT" ); } else require(to != address(0), "INVALID_RECIPIENT"); } function transferFrom( address from, address to, uint256 id, uint256 amount, bytes calldata /*data*/ ) public virtual { require(msg.sender == from || isApprovedForAll[from][msg.sender], "NOT_AUTHORIZED"); require(from == ownerOf[id], "WRONG_FROM"); // Can only transfer from the owner. // Can only transfer 1 with ERC1155B. require(amount == 1, "INVALID_AMOUNT"); ownerOf[id] = to; emit TransferSingle(msg.sender, from, to, id, amount); } function batchTransferFrom( address from, address to, uint256[] calldata ids, uint256[] calldata amounts, bytes calldata /*data*/ ) public virtual { require(ids.length == amounts.length, "LENGTH_MISMATCH"); require(msg.sender == from || isApprovedForAll[from][msg.sender], "NOT_AUTHORIZED"); // Storing these outside the loop saves ~15 gas per iteration. uint256 id; uint256 amount; // Unchecked because the only math done is incrementing // the array index counter which cannot possibly overflow. unchecked { for (uint256 i = 0; i < ids.length; i++) { id = ids[i]; amount = amounts[i]; // Can only transfer from the owner. require(from == ownerOf[id], "WRONG_FROM"); // Can only transfer 1 with ERC1155B. require(amount == 1, "INVALID_AMOUNT"); ownerOf[id] = to; } } emit TransferBatch(msg.sender, from, to, ids, amounts); } function balanceOfBatch(address[] calldata owners, uint256[] calldata ids) public view virtual returns (uint256[] memory balances) { require(owners.length == ids.length, "LENGTH_MISMATCH"); balances = new uint256[](owners.length); // Unchecked because the only math done is incrementing // the array index counter which cannot possibly overflow. unchecked { for (uint256 i = 0; i < owners.length; ++i) { balances[i] = balanceOf(owners[i], ids[i]); } } } /*////////////////////////////////////////////////////////////// INTERNAL MINT/BURN LOGIC //////////////////////////////////////////////////////////////*/ function _mint( address to, uint256 id, bytes memory data ) internal virtual { // Minting twice would effectively be a force transfer. require(ownerOf[id] == address(0), "ALREADY_MINTED"); ownerOf[id] = to; emit TransferSingle(msg.sender, address(0), to, id, 1); if (to.code.length != 0) { require( ERC1155TokenReceiver(to).onERC1155Received(msg.sender, address(0), id, 1, data) == ERC1155TokenReceiver.onERC1155Received.selector, "UNSAFE_RECIPIENT" ); } else require(to != address(0), "INVALID_RECIPIENT"); } function _batchMint( address to, uint256[] memory ids, bytes memory data ) internal virtual { uint256 idsLength = ids.length; // Saves MLOADs. // Generate an amounts array locally to use in the event below. uint256[] memory amounts = new uint256[](idsLength); uint256 id; // Storing outside the loop saves ~7 gas per iteration. // Unchecked because the only math done is incrementing // the array index counter which cannot possibly overflow. unchecked { for (uint256 i = 0; i < idsLength; ++i) { id = ids[i]; // Minting twice would effectively be a force transfer. require(ownerOf[id] == address(0), "ALREADY_MINTED"); ownerOf[id] = to; amounts[i] = 1; } } emit TransferBatch(msg.sender, address(0), to, ids, amounts); if (to.code.length != 0) { require( ERC1155TokenReceiver(to).onERC1155BatchReceived( msg.sender, address(0), ids, amounts, data ) == ERC1155TokenReceiver.onERC1155BatchReceived.selector, "UNSAFE_RECIPIENT" ); } else require(to != address(0), "INVALID_RECIPIENT"); } function _batchBurn(address from, uint256[] memory ids) internal virtual { // Burning unminted tokens makes no sense. require(from != address(0), "INVALID_FROM"); uint256 idsLength = ids.length; // Saves MLOADs. // Generate an amounts array locally to use in the event below. uint256[] memory amounts = new uint256[](idsLength); uint256 id; // Storing outside the loop saves ~7 gas per iteration. // Unchecked because the only math done is incrementing // the array index counter which cannot possibly overflow. unchecked { for (uint256 i = 0; i < idsLength; ++i) { id = ids[i]; require(ownerOf[id] == from, "WRONG_FROM"); ownerOf[id] = address(0); amounts[i] = 1; } } emit TransferBatch(msg.sender, from, address(0), ids, amounts); } function _burn(uint256 id) internal virtual { address owner = ownerOf[id]; require(owner != address(0), "NOT_MINTED"); ownerOf[id] = address(0); emit TransferSingle(msg.sender, owner, address(0), id, 1); } }
// SPDX-License-Identifier: AGPL-3.0-only pragma solidity >=0.8.0; /// @notice Safe unsigned integer casting library that reverts on overflow. /// @author Solmate (https://github.com/transmissions11/solmate/blob/main/src/utils/SafeCastLib.sol) /// @author Modified from OpenZeppelin (https://github.com/OpenZeppelin/openzeppelin-contracts/blob/master/contracts/utils/math/SafeCast.sol) library SafeCastLib { function safeCastTo248(uint256 x) internal pure returns (uint248 y) { require(x < 1 << 248); y = uint248(x); } function safeCastTo224(uint256 x) internal pure returns (uint224 y) { require(x < 1 << 224); y = uint224(x); } function safeCastTo192(uint256 x) internal pure returns (uint192 y) { require(x < 1 << 192); y = uint192(x); } function safeCastTo160(uint256 x) internal pure returns (uint160 y) { require(x < 1 << 160); y = uint160(x); } function safeCastTo128(uint256 x) internal pure returns (uint128 y) { require(x < 1 << 128); y = uint128(x); } function safeCastTo96(uint256 x) internal pure returns (uint96 y) { require(x < 1 << 96); y = uint96(x); } function safeCastTo64(uint256 x) internal pure returns (uint64 y) { require(x < 1 << 64); y = uint64(x); } function safeCastTo32(uint256 x) internal pure returns (uint32 y) { require(x < 1 << 32); y = uint32(x); } function safeCastTo24(uint256 x) internal pure returns (uint24 y) { require(x < 1 << 24); y = uint24(x); } function safeCastTo8(uint256 x) internal pure returns (uint8 y) { require(x < 1 << 8); y = uint8(x); } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts v4.4.1 (utils/cryptography/draft-EIP712.sol) pragma solidity ^0.8.0; import "./ECDSA.sol"; /** * @dev https://eips.ethereum.org/EIPS/eip-712[EIP 712] is a standard for hashing and signing of typed structured data. * * The encoding specified in the EIP is very generic, and such a generic implementation in Solidity is not feasible, * thus this contract does not implement the encoding itself. Protocols need to implement the type-specific encoding * they need in their contracts using a combination of `abi.encode` and `keccak256`. * * This contract implements the EIP 712 domain separator ({_domainSeparatorV4}) that is used as part of the encoding * scheme, and the final step of the encoding to obtain the message digest that is then signed via ECDSA * ({_hashTypedDataV4}). * * The implementation of the domain separator was designed to be as efficient as possible while still properly updating * the chain id to protect against replay attacks on an eventual fork of the chain. * * NOTE: This contract implements the version of the encoding known as "v4", as implemented by the JSON RPC method * https://docs.metamask.io/guide/signing-data.html[`eth_signTypedDataV4` in MetaMask]. * * _Available since v3.4._ */ abstract contract EIP712 { /* solhint-disable var-name-mixedcase */ // Cache the domain separator as an immutable value, but also store the chain id that it corresponds to, in order to // invalidate the cached domain separator if the chain id changes. bytes32 private immutable _CACHED_DOMAIN_SEPARATOR; uint256 private immutable _CACHED_CHAIN_ID; address private immutable _CACHED_THIS; bytes32 private immutable _HASHED_NAME; bytes32 private immutable _HASHED_VERSION; bytes32 private immutable _TYPE_HASH; /* solhint-enable var-name-mixedcase */ /** * @dev Initializes the domain separator and parameter caches. * * The meaning of `name` and `version` is specified in * https://eips.ethereum.org/EIPS/eip-712#definition-of-domainseparator[EIP 712]: * * - `name`: the user readable name of the signing domain, i.e. the name of the DApp or the protocol. * - `version`: the current major version of the signing domain. * * NOTE: These parameters cannot be changed except through a xref:learn::upgrading-smart-contracts.adoc[smart * contract upgrade]. */ constructor(string memory name, string memory version) { bytes32 hashedName = keccak256(bytes(name)); bytes32 hashedVersion = keccak256(bytes(version)); bytes32 typeHash = keccak256( "EIP712Domain(string name,string version,uint256 chainId,address verifyingContract)" ); _HASHED_NAME = hashedName; _HASHED_VERSION = hashedVersion; _CACHED_CHAIN_ID = block.chainid; _CACHED_DOMAIN_SEPARATOR = _buildDomainSeparator(typeHash, hashedName, hashedVersion); _CACHED_THIS = address(this); _TYPE_HASH = typeHash; } /** * @dev Returns the domain separator for the current chain. */ function _domainSeparatorV4() internal view returns (bytes32) { if (address(this) == _CACHED_THIS && block.chainid == _CACHED_CHAIN_ID) { return _CACHED_DOMAIN_SEPARATOR; } else { return _buildDomainSeparator(_TYPE_HASH, _HASHED_NAME, _HASHED_VERSION); } } function _buildDomainSeparator( bytes32 typeHash, bytes32 nameHash, bytes32 versionHash ) private view returns (bytes32) { return keccak256(abi.encode(typeHash, nameHash, versionHash, block.chainid, address(this))); } /** * @dev Given an already https://eips.ethereum.org/EIPS/eip-712#definition-of-hashstruct[hashed struct], this * function returns the hash of the fully encoded EIP712 message for this domain. * * This hash can be used together with {ECDSA-recover} to obtain the signer of a message. For example: * * ```solidity * bytes32 digest = _hashTypedDataV4(keccak256(abi.encode( * keccak256("Mail(address to,string contents)"), * mailTo, * keccak256(bytes(mailContents)) * ))); * address signer = ECDSA.recover(digest, signature); * ``` */ function _hashTypedDataV4(bytes32 structHash) internal view virtual returns (bytes32) { return ECDSA.toTypedDataHash(_domainSeparatorV4(), structHash); } }
// SPDX-License-Identifier: MIT pragma solidity 0.8.13; /// @dev Name of the NounletToken contract string constant NAME = "NounletToken"; /// @dev Version number of the NounletToken contract string constant VERSION = "1"; /// @dev The EIP-712 typehash for the contract's domain bytes32 constant DOMAIN_TYPEHASH = keccak256( "EIP712Domain(string name,string version,uint256 chainId,address verifyingContract)" ); /// @dev The EIP-712 typehash for the permit struct used by the contract bytes32 constant PERMIT_TYPEHASH = keccak256( "Permit(address owner,address operator,uint256 tokenId,bool approved,uint256 nonce,uint256 deadline)" ); /// @dev The EIP-712 typehash for the permit all struct used by the contract bytes32 constant PERMIT_ALL_TYPEHASH = keccak256( "PermitAll(address owner,address operator,bool approved,uint256 nonce,uint256 deadline)" );
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts v4.4.1 (utils/introspection/IERC165.sol) pragma solidity ^0.8.0; /** * @dev Interface of the ERC165 standard, as defined in the * https://eips.ethereum.org/EIPS/eip-165[EIP]. * * Implementers can declare support of contract interfaces, which can then be * queried by others ({ERC165Checker}). * * For an implementation, see {ERC165}. */ interface IERC165 { /** * @dev Returns true if this contract implements the interface defined by * `interfaceId`. See the corresponding * https://eips.ethereum.org/EIPS/eip-165#how-interfaces-are-identified[EIP section] * to learn more about how these ids are created. * * This function call must use less than 30 000 gas. */ function supportsInterface(bytes4 interfaceId) external view returns (bool); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.7.0) (token/ERC721/IERC721.sol) pragma solidity ^0.8.0; import "../../utils/introspection/IERC165.sol"; /** * @dev Required interface of an ERC721 compliant contract. */ interface IERC721 is IERC165 { /** * @dev Emitted when `tokenId` token is transferred from `from` to `to`. */ event Transfer(address indexed from, address indexed to, uint256 indexed tokenId); /** * @dev Emitted when `owner` enables `approved` to manage the `tokenId` token. */ event Approval(address indexed owner, address indexed approved, uint256 indexed tokenId); /** * @dev Emitted when `owner` enables or disables (`approved`) `operator` to manage all of its assets. */ event ApprovalForAll(address indexed owner, address indexed operator, bool approved); /** * @dev Returns the number of tokens in ``owner``'s account. */ function balanceOf(address owner) external view returns (uint256 balance); /** * @dev Returns the owner of the `tokenId` token. * * Requirements: * * - `tokenId` must exist. */ function ownerOf(uint256 tokenId) external view returns (address owner); /** * @dev Safely transfers `tokenId` token from `from` to `to`. * * Requirements: * * - `from` cannot be the zero address. * - `to` cannot be the zero address. * - `tokenId` token must exist and be owned by `from`. * - If the caller is not `from`, it must be approved to move this token by either {approve} or {setApprovalForAll}. * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer. * * Emits a {Transfer} event. */ function safeTransferFrom( address from, address to, uint256 tokenId, bytes calldata data ) external; /** * @dev Safely transfers `tokenId` token from `from` to `to`, checking first that contract recipients * are aware of the ERC721 protocol to prevent tokens from being forever locked. * * Requirements: * * - `from` cannot be the zero address. * - `to` cannot be the zero address. * - `tokenId` token must exist and be owned by `from`. * - If the caller is not `from`, it must have been allowed to move this token by either {approve} or {setApprovalForAll}. * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer. * * Emits a {Transfer} event. */ function safeTransferFrom( address from, address to, uint256 tokenId ) external; /** * @dev Transfers `tokenId` token from `from` to `to`. * * WARNING: Usage of this method is discouraged, use {safeTransferFrom} whenever possible. * * Requirements: * * - `from` cannot be the zero address. * - `to` cannot be the zero address. * - `tokenId` token must be owned by `from`. * - If the caller is not `from`, it must be approved to move this token by either {approve} or {setApprovalForAll}. * * Emits a {Transfer} event. */ function transferFrom( address from, address to, uint256 tokenId ) external; /** * @dev Gives permission to `to` to transfer `tokenId` token to another account. * The approval is cleared when the token is transferred. * * Only a single account can be approved at a time, so approving the zero address clears previous approvals. * * Requirements: * * - The caller must own the token or be an approved operator. * - `tokenId` must exist. * * Emits an {Approval} event. */ function approve(address to, uint256 tokenId) external; /** * @dev Approve or remove `operator` as an operator for the caller. * Operators can call {transferFrom} or {safeTransferFrom} for any token owned by the caller. * * Requirements: * * - The `operator` cannot be the caller. * * Emits an {ApprovalForAll} event. */ function setApprovalForAll(address operator, bool _approved) external; /** * @dev Returns the account approved for `tokenId` token. * * Requirements: * * - `tokenId` must exist. */ function getApproved(uint256 tokenId) external view returns (address operator); /** * @dev Returns if the `operator` is allowed to manage all of the assets of `owner`. * * See {setApprovalForAll} */ function isApprovedForAll(address owner, address operator) external view returns (bool); }
// SPDX-License-Identifier: AGPL-3.0-only pragma solidity >=0.8.0; /// @notice Minimalist and gas efficient standard ERC1155 implementation. /// @author Solmate (https://github.com/transmissions11/solmate/blob/main/src/tokens/ERC1155.sol) abstract contract ERC1155 { /*////////////////////////////////////////////////////////////// EVENTS //////////////////////////////////////////////////////////////*/ event TransferSingle( address indexed operator, address indexed from, address indexed to, uint256 id, uint256 amount ); event TransferBatch( address indexed operator, address indexed from, address indexed to, uint256[] ids, uint256[] amounts ); event ApprovalForAll(address indexed owner, address indexed operator, bool approved); event URI(string value, uint256 indexed id); /*////////////////////////////////////////////////////////////// ERC1155 STORAGE //////////////////////////////////////////////////////////////*/ mapping(address => mapping(uint256 => uint256)) public balanceOf; mapping(address => mapping(address => bool)) public isApprovedForAll; /*////////////////////////////////////////////////////////////// METADATA LOGIC //////////////////////////////////////////////////////////////*/ function uri(uint256 id) public view virtual returns (string memory); /*////////////////////////////////////////////////////////////// ERC1155 LOGIC //////////////////////////////////////////////////////////////*/ function setApprovalForAll(address operator, bool approved) public virtual { isApprovedForAll[msg.sender][operator] = approved; emit ApprovalForAll(msg.sender, operator, approved); } function safeTransferFrom( address from, address to, uint256 id, uint256 amount, bytes calldata data ) public virtual { require(msg.sender == from || isApprovedForAll[from][msg.sender], "NOT_AUTHORIZED"); balanceOf[from][id] -= amount; balanceOf[to][id] += amount; emit TransferSingle(msg.sender, from, to, id, amount); require( to.code.length == 0 ? to != address(0) : ERC1155TokenReceiver(to).onERC1155Received(msg.sender, from, id, amount, data) == ERC1155TokenReceiver.onERC1155Received.selector, "UNSAFE_RECIPIENT" ); } function safeBatchTransferFrom( address from, address to, uint256[] calldata ids, uint256[] calldata amounts, bytes calldata data ) public virtual { require(ids.length == amounts.length, "LENGTH_MISMATCH"); require(msg.sender == from || isApprovedForAll[from][msg.sender], "NOT_AUTHORIZED"); // Storing these outside the loop saves ~15 gas per iteration. uint256 id; uint256 amount; for (uint256 i = 0; i < ids.length; ) { id = ids[i]; amount = amounts[i]; balanceOf[from][id] -= amount; balanceOf[to][id] += amount; // An array can't have a total length // larger than the max uint256 value. unchecked { ++i; } } emit TransferBatch(msg.sender, from, to, ids, amounts); require( to.code.length == 0 ? to != address(0) : ERC1155TokenReceiver(to).onERC1155BatchReceived(msg.sender, from, ids, amounts, data) == ERC1155TokenReceiver.onERC1155BatchReceived.selector, "UNSAFE_RECIPIENT" ); } function balanceOfBatch(address[] calldata owners, uint256[] calldata ids) public view virtual returns (uint256[] memory balances) { require(owners.length == ids.length, "LENGTH_MISMATCH"); balances = new uint256[](owners.length); // Unchecked because the only math done is incrementing // the array index counter which cannot possibly overflow. unchecked { for (uint256 i = 0; i < owners.length; ++i) { balances[i] = balanceOf[owners[i]][ids[i]]; } } } /*////////////////////////////////////////////////////////////// ERC165 LOGIC //////////////////////////////////////////////////////////////*/ function supportsInterface(bytes4 interfaceId) public view virtual returns (bool) { return interfaceId == 0x01ffc9a7 || // ERC165 Interface ID for ERC165 interfaceId == 0xd9b67a26 || // ERC165 Interface ID for ERC1155 interfaceId == 0x0e89341c; // ERC165 Interface ID for ERC1155MetadataURI } /*////////////////////////////////////////////////////////////// INTERNAL MINT/BURN LOGIC //////////////////////////////////////////////////////////////*/ function _mint( address to, uint256 id, uint256 amount, bytes memory data ) internal virtual { balanceOf[to][id] += amount; emit TransferSingle(msg.sender, address(0), to, id, amount); require( to.code.length == 0 ? to != address(0) : ERC1155TokenReceiver(to).onERC1155Received(msg.sender, address(0), id, amount, data) == ERC1155TokenReceiver.onERC1155Received.selector, "UNSAFE_RECIPIENT" ); } function _batchMint( address to, uint256[] memory ids, uint256[] memory amounts, bytes memory data ) internal virtual { uint256 idsLength = ids.length; // Saves MLOADs. require(idsLength == amounts.length, "LENGTH_MISMATCH"); for (uint256 i = 0; i < idsLength; ) { balanceOf[to][ids[i]] += amounts[i]; // An array can't have a total length // larger than the max uint256 value. unchecked { ++i; } } emit TransferBatch(msg.sender, address(0), to, ids, amounts); require( to.code.length == 0 ? to != address(0) : ERC1155TokenReceiver(to).onERC1155BatchReceived(msg.sender, address(0), ids, amounts, data) == ERC1155TokenReceiver.onERC1155BatchReceived.selector, "UNSAFE_RECIPIENT" ); } function _batchBurn( address from, uint256[] memory ids, uint256[] memory amounts ) internal virtual { uint256 idsLength = ids.length; // Saves MLOADs. require(idsLength == amounts.length, "LENGTH_MISMATCH"); for (uint256 i = 0; i < idsLength; ) { balanceOf[from][ids[i]] -= amounts[i]; // An array can't have a total length // larger than the max uint256 value. unchecked { ++i; } } emit TransferBatch(msg.sender, from, address(0), ids, amounts); } function _burn( address from, uint256 id, uint256 amount ) internal virtual { balanceOf[from][id] -= amount; emit TransferSingle(msg.sender, from, address(0), id, amount); } } /// @notice A generic interface for a contract which properly accepts ERC1155 tokens. /// @author Solmate (https://github.com/transmissions11/solmate/blob/main/src/tokens/ERC1155.sol) abstract contract ERC1155TokenReceiver { function onERC1155Received( address, address, uint256, uint256, bytes calldata ) external virtual returns (bytes4) { return ERC1155TokenReceiver.onERC1155Received.selector; } function onERC1155BatchReceived( address, address, uint256[] calldata, uint256[] calldata, bytes calldata ) external virtual returns (bytes4) { return ERC1155TokenReceiver.onERC1155BatchReceived.selector; } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.7.0) (utils/cryptography/ECDSA.sol) pragma solidity ^0.8.0; import "../Strings.sol"; /** * @dev Elliptic Curve Digital Signature Algorithm (ECDSA) operations. * * These functions can be used to verify that a message was signed by the holder * of the private keys of a given address. */ library ECDSA { enum RecoverError { NoError, InvalidSignature, InvalidSignatureLength, InvalidSignatureS, InvalidSignatureV } function _throwError(RecoverError error) private pure { if (error == RecoverError.NoError) { return; // no error: do nothing } else if (error == RecoverError.InvalidSignature) { revert("ECDSA: invalid signature"); } else if (error == RecoverError.InvalidSignatureLength) { revert("ECDSA: invalid signature length"); } else if (error == RecoverError.InvalidSignatureS) { revert("ECDSA: invalid signature 's' value"); } else if (error == RecoverError.InvalidSignatureV) { revert("ECDSA: invalid signature 'v' value"); } } /** * @dev Returns the address that signed a hashed message (`hash`) with * `signature` or error string. This address can then be used for verification purposes. * * The `ecrecover` EVM opcode allows for malleable (non-unique) signatures: * this function rejects them by requiring the `s` value to be in the lower * half order, and the `v` value to be either 27 or 28. * * IMPORTANT: `hash` _must_ be the result of a hash operation for the * verification to be secure: it is possible to craft signatures that * recover to arbitrary addresses for non-hashed data. A safe way to ensure * this is by receiving a hash of the original message (which may otherwise * be too long), and then calling {toEthSignedMessageHash} on it. * * Documentation for signature generation: * - with https://web3js.readthedocs.io/en/v1.3.4/web3-eth-accounts.html#sign[Web3.js] * - with https://docs.ethers.io/v5/api/signer/#Signer-signMessage[ethers] * * _Available since v4.3._ */ function tryRecover(bytes32 hash, bytes memory signature) internal pure returns (address, RecoverError) { if (signature.length == 65) { bytes32 r; bytes32 s; uint8 v; // ecrecover takes the signature parameters, and the only way to get them // currently is to use assembly. /// @solidity memory-safe-assembly assembly { r := mload(add(signature, 0x20)) s := mload(add(signature, 0x40)) v := byte(0, mload(add(signature, 0x60))) } return tryRecover(hash, v, r, s); } else { return (address(0), RecoverError.InvalidSignatureLength); } } /** * @dev Returns the address that signed a hashed message (`hash`) with * `signature`. This address can then be used for verification purposes. * * The `ecrecover` EVM opcode allows for malleable (non-unique) signatures: * this function rejects them by requiring the `s` value to be in the lower * half order, and the `v` value to be either 27 or 28. * * IMPORTANT: `hash` _must_ be the result of a hash operation for the * verification to be secure: it is possible to craft signatures that * recover to arbitrary addresses for non-hashed data. A safe way to ensure * this is by receiving a hash of the original message (which may otherwise * be too long), and then calling {toEthSignedMessageHash} on it. */ function recover(bytes32 hash, bytes memory signature) internal pure returns (address) { (address recovered, RecoverError error) = tryRecover(hash, signature); _throwError(error); return recovered; } /** * @dev Overload of {ECDSA-tryRecover} that receives the `r` and `vs` short-signature fields separately. * * See https://eips.ethereum.org/EIPS/eip-2098[EIP-2098 short signatures] * * _Available since v4.3._ */ function tryRecover( bytes32 hash, bytes32 r, bytes32 vs ) internal pure returns (address, RecoverError) { bytes32 s = vs & bytes32(0x7fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff); uint8 v = uint8((uint256(vs) >> 255) + 27); return tryRecover(hash, v, r, s); } /** * @dev Overload of {ECDSA-recover} that receives the `r and `vs` short-signature fields separately. * * _Available since v4.2._ */ function recover( bytes32 hash, bytes32 r, bytes32 vs ) internal pure returns (address) { (address recovered, RecoverError error) = tryRecover(hash, r, vs); _throwError(error); return recovered; } /** * @dev Overload of {ECDSA-tryRecover} that receives the `v`, * `r` and `s` signature fields separately. * * _Available since v4.3._ */ function tryRecover( bytes32 hash, uint8 v, bytes32 r, bytes32 s ) internal pure returns (address, RecoverError) { // EIP-2 still allows signature malleability for ecrecover(). Remove this possibility and make the signature // unique. Appendix F in the Ethereum Yellow paper (https://ethereum.github.io/yellowpaper/paper.pdf), defines // the valid range for s in (301): 0 < s < secp256k1n ÷ 2 + 1, and for v in (302): v ∈ {27, 28}. Most // signatures from current libraries generate a unique signature with an s-value in the lower half order. // // If your library generates malleable signatures, such as s-values in the upper range, calculate a new s-value // with 0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEBAAEDCE6AF48A03BBFD25E8CD0364141 - s1 and flip v from 27 to 28 or // vice versa. If your library also generates signatures with 0/1 for v instead 27/28, add 27 to v to accept // these malleable signatures as well. if (uint256(s) > 0x7FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF5D576E7357A4501DDFE92F46681B20A0) { return (address(0), RecoverError.InvalidSignatureS); } if (v != 27 && v != 28) { return (address(0), RecoverError.InvalidSignatureV); } // If the signature is valid (and not malleable), return the signer address address signer = ecrecover(hash, v, r, s); if (signer == address(0)) { return (address(0), RecoverError.InvalidSignature); } return (signer, RecoverError.NoError); } /** * @dev Overload of {ECDSA-recover} that receives the `v`, * `r` and `s` signature fields separately. */ function recover( bytes32 hash, uint8 v, bytes32 r, bytes32 s ) internal pure returns (address) { (address recovered, RecoverError error) = tryRecover(hash, v, r, s); _throwError(error); return recovered; } /** * @dev Returns an Ethereum Signed Message, created from a `hash`. This * produces hash corresponding to the one signed with the * https://eth.wiki/json-rpc/API#eth_sign[`eth_sign`] * JSON-RPC method as part of EIP-191. * * See {recover}. */ function toEthSignedMessageHash(bytes32 hash) internal pure returns (bytes32) { // 32 is the length in bytes of hash, // enforced by the type signature above return keccak256(abi.encodePacked("\x19Ethereum Signed Message:\n32", hash)); } /** * @dev Returns an Ethereum Signed Message, created from `s`. This * produces hash corresponding to the one signed with the * https://eth.wiki/json-rpc/API#eth_sign[`eth_sign`] * JSON-RPC method as part of EIP-191. * * See {recover}. */ function toEthSignedMessageHash(bytes memory s) internal pure returns (bytes32) { return keccak256(abi.encodePacked("\x19Ethereum Signed Message:\n", Strings.toString(s.length), s)); } /** * @dev Returns an Ethereum Signed Typed Data, created from a * `domainSeparator` and a `structHash`. This produces hash corresponding * to the one signed with the * https://eips.ethereum.org/EIPS/eip-712[`eth_signTypedData`] * JSON-RPC method as part of EIP-712. * * See {recover}. */ function toTypedDataHash(bytes32 domainSeparator, bytes32 structHash) internal pure returns (bytes32) { return keccak256(abi.encodePacked("\x19\x01", domainSeparator, structHash)); } }
{ "remappings": [ "@openzeppelin/contracts/=lib/openzeppelin-contracts/contracts/", "@rari-capital/solmate/src/=lib/solmate/src/", "base64-sol/=lib/base64/", "base64/=lib/base64/", "clones-with-immutable-args/=lib/clones-with-immutable-args/", "ds-test/=lib/clones-with-immutable-args/lib/ds-test/src/", "forge-std/=lib/forge-std/src/", "nouns-contracts/=lib/nouns-monorepo/packages/nouns-contracts/", "nouns-monorepo/=lib/nouns-monorepo/", "openzeppelin-contracts/=lib/openzeppelin-contracts/", "solmate/=lib/solmate/src/" ], "optimizer": { "enabled": true, "runs": 200 }, "metadata": { "bytecodeHash": "ipfs" }, "outputSelection": { "*": { "*": [ "evm.bytecode", "evm.deployedBytecode", "devdoc", "userdoc", "metadata", "abi" ] } }, "evmVersion": "london", "libraries": {} }
[{"inputs":[{"internalType":"address","name":"_required","type":"address"},{"internalType":"address","name":"_provided","type":"address"}],"name":"InvalidSender","type":"error"},{"inputs":[{"internalType":"address","name":"_signer","type":"address"},{"internalType":"address","name":"_owner","type":"address"}],"name":"InvalidSignature","type":"error"},{"inputs":[{"internalType":"uint256","name":"_timestamp","type":"uint256"},{"internalType":"uint256","name":"_deadline","type":"uint256"}],"name":"SignatureExpired","type":"error"},{"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":"_delegator","type":"address"},{"indexed":true,"internalType":"address","name":"_fromDelegate","type":"address"},{"indexed":true,"internalType":"address","name":"_toDelegate","type":"address"}],"name":"DelegateChanged","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"_delegate","type":"address"},{"indexed":false,"internalType":"uint256","name":"_previousBalance","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"_newBalance","type":"uint256"}],"name":"DelegateVotesChanged","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"_owner","type":"address"},{"indexed":true,"internalType":"address","name":"_operator","type":"address"},{"indexed":false,"internalType":"uint256","name":"_id","type":"uint256"},{"indexed":false,"internalType":"bool","name":"_approved","type":"bool"}],"name":"SingleApproval","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"operator","type":"address"},{"indexed":true,"internalType":"address","name":"from","type":"address"},{"indexed":true,"internalType":"address","name":"to","type":"address"},{"indexed":false,"internalType":"uint256[]","name":"ids","type":"uint256[]"},{"indexed":false,"internalType":"uint256[]","name":"amounts","type":"uint256[]"}],"name":"TransferBatch","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"operator","type":"address"},{"indexed":true,"internalType":"address","name":"from","type":"address"},{"indexed":true,"internalType":"address","name":"to","type":"address"},{"indexed":false,"internalType":"uint256","name":"id","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"TransferSingle","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"string","name":"value","type":"string"},{"indexed":true,"internalType":"uint256","name":"id","type":"uint256"}],"name":"URI","type":"event"},{"inputs":[],"name":"NOUNLET_REGISTRY","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"pure","type":"function"},{"inputs":[],"name":"NOUNS_DESCRIPTOR","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"pure","type":"function"},{"inputs":[],"name":"NOUNS_TOKEN","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"pure","type":"function"},{"inputs":[],"name":"NOUNS_TOKEN_ID","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"pure","type":"function"},{"inputs":[],"name":"ROYALTY_BENEFICIARY","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"pure","type":"function"},{"inputs":[],"name":"ROYALTY_PERCENT","outputs":[{"internalType":"uint96","name":"","type":"uint96"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"_ballots","outputs":[{"internalType":"uint96","name":"","type":"uint96"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"},{"internalType":"uint256","name":"id","type":"uint256"}],"name":"balanceOf","outputs":[{"internalType":"uint256","name":"bal","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address[]","name":"owners","type":"address[]"},{"internalType":"uint256[]","name":"ids","type":"uint256[]"}],"name":"balanceOfBatch","outputs":[{"internalType":"uint256[]","name":"balances","type":"uint256[]"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_from","type":"address"},{"internalType":"uint256[]","name":"_ids","type":"uint256[]"}],"name":"batchBurn","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_from","type":"address"},{"internalType":"address","name":"_to","type":"address"},{"internalType":"uint256[]","name":"_ids","type":"uint256[]"},{"internalType":"uint256[]","name":"_amounts","type":"uint256[]"},{"internalType":"bytes","name":"_data","type":"bytes"}],"name":"batchTransferFrom","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"},{"internalType":"uint32","name":"","type":"uint32"}],"name":"checkpoints","outputs":[{"internalType":"uint32","name":"fromBlock","type":"uint32"},{"internalType":"uint96","name":"votes","type":"uint96"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"contractURI","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"decimals","outputs":[{"internalType":"uint8","name":"","type":"uint8"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"delegatee","type":"address"}],"name":"delegate","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"delegator","type":"address"}],"name":"delegates","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_id","type":"uint256"}],"name":"generateSeed","outputs":[{"components":[{"internalType":"uint48","name":"background","type":"uint48"},{"internalType":"uint48","name":"body","type":"uint48"},{"internalType":"uint48","name":"accessory","type":"uint48"},{"internalType":"uint48","name":"head","type":"uint48"},{"internalType":"uint48","name":"glasses","type":"uint48"}],"internalType":"struct INounsSeeder.Seed","name":"","type":"tuple"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"account","type":"address"}],"name":"getCurrentVotes","outputs":[{"internalType":"uint96","name":"","type":"uint96"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"account","type":"address"},{"internalType":"uint256","name":"blockNumber","type":"uint256"}],"name":"getPriorVotes","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"},{"internalType":"address","name":"","type":"address"},{"internalType":"uint256","name":"","type":"uint256"}],"name":"isApproved","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"},{"internalType":"address","name":"","type":"address"}],"name":"isApprovedForAll","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"maxSupply","outputs":[{"internalType":"uint96","name":"","type":"uint96"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_to","type":"address"},{"internalType":"uint256","name":"_id","type":"uint256"},{"internalType":"bytes","name":"_data","type":"bytes"}],"name":"mint","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"nonces","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"numCheckpoints","outputs":[{"internalType":"uint32","name":"","type":"uint32"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"ownerOf","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_owner","type":"address"},{"internalType":"address","name":"_operator","type":"address"},{"internalType":"uint256","name":"_id","type":"uint256"},{"internalType":"bool","name":"_approved","type":"bool"},{"internalType":"uint256","name":"_deadline","type":"uint256"},{"internalType":"uint8","name":"_v","type":"uint8"},{"internalType":"bytes32","name":"_r","type":"bytes32"},{"internalType":"bytes32","name":"_s","type":"bytes32"}],"name":"permit","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_owner","type":"address"},{"internalType":"address","name":"_operator","type":"address"},{"internalType":"bool","name":"_approved","type":"bool"},{"internalType":"uint256","name":"_deadline","type":"uint256"},{"internalType":"uint8","name":"_v","type":"uint8"},{"internalType":"bytes32","name":"_r","type":"bytes32"},{"internalType":"bytes32","name":"_s","type":"bytes32"}],"name":"permitAll","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"_salePrice","type":"uint256"}],"name":"royaltyInfo","outputs":[{"internalType":"address","name":"beneficiary","type":"address"},{"internalType":"uint256","name":"royaltyAmount","type":"uint256"}],"stateMutability":"pure","type":"function"},{"inputs":[{"internalType":"address","name":"_from","type":"address"},{"internalType":"address","name":"_to","type":"address"},{"internalType":"uint256[]","name":"_ids","type":"uint256[]"},{"internalType":"uint256[]","name":"_amounts","type":"uint256[]"},{"internalType":"bytes","name":"_data","type":"bytes"}],"name":"safeBatchTransferFrom","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_from","type":"address"},{"internalType":"address","name":"_to","type":"address"},{"internalType":"uint256","name":"_id","type":"uint256"},{"internalType":"uint256","name":"_amount","type":"uint256"},{"internalType":"bytes","name":"_data","type":"bytes"}],"name":"safeTransferFrom","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"seeds","outputs":[{"internalType":"uint48","name":"background","type":"uint48"},{"internalType":"uint48","name":"body","type":"uint48"},{"internalType":"uint48","name":"accessory","type":"uint48"},{"internalType":"uint48","name":"head","type":"uint48"},{"internalType":"uint48","name":"glasses","type":"uint48"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_operator","type":"address"},{"internalType":"uint256","name":"_id","type":"uint256"},{"internalType":"bool","name":"_approved","type":"bool"}],"name":"setApprovalFor","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":"totalSupply","outputs":[{"internalType":"uint96","name":"","type":"uint96"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_from","type":"address"},{"internalType":"address","name":"_to","type":"address"},{"internalType":"uint256","name":"_id","type":"uint256"},{"internalType":"uint256","name":"_amount","type":"uint256"},{"internalType":"bytes","name":"_data","type":"bytes"}],"name":"transferFrom","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_id","type":"uint256"}],"name":"uri","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"delegator","type":"address"}],"name":"votesToDelegate","outputs":[{"internalType":"uint96","name":"","type":"uint96"}],"stateMutability":"view","type":"function"}]
Loading...
Loading
Loading...
Loading
Multichain Portfolio | 30 Chains
Chain | Token | Portfolio % | Price | Amount | Value |
---|
Loading...
Loading
[ Download: CSV Export ]
[ Download: CSV Export ]
A contract address hosts a smart contract, which is a set of code stored on the blockchain that runs when predetermined conditions are met. Learn more about addresses in our Knowledge Base.