编译器
0.8.19+commit.7dd6d404
文件 1 的 20:Address.sol
pragma solidity ^0.8.1;
library Address {
function isContract(address account) internal view returns (bool) {
return account.code.length > 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 functionCallWithValue(target, data, 0, "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");
(bool success, bytes memory returndata) = target.call{value: value}(data);
return verifyCallResultFromTarget(target, 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) {
(bool success, bytes memory returndata) = target.staticcall(data);
return verifyCallResultFromTarget(target, 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) {
(bool success, bytes memory returndata) = target.delegatecall(data);
return verifyCallResultFromTarget(target, success, returndata, errorMessage);
}
function verifyCallResultFromTarget(
address target,
bool success,
bytes memory returndata,
string memory errorMessage
) internal view returns (bytes memory) {
if (success) {
if (returndata.length == 0) {
require(isContract(target), "Address: call to non-contract");
}
return returndata;
} else {
_revert(returndata, errorMessage);
}
}
function verifyCallResult(
bool success,
bytes memory returndata,
string memory errorMessage
) internal pure returns (bytes memory) {
if (success) {
return returndata;
} else {
_revert(returndata, errorMessage);
}
}
function _revert(bytes memory returndata, string memory errorMessage) private pure {
if (returndata.length > 0) {
assembly {
let returndata_size := mload(returndata)
revert(add(32, returndata), returndata_size)
}
} else {
revert(errorMessage);
}
}
}
文件 2 的 20: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;
}
}
文件 3 的 20:Counters.sol
pragma solidity ^0.8.0;
library Counters {
struct Counter {
uint256 _value;
}
function current(Counter storage counter) internal view returns (uint256) {
return counter._value;
}
function increment(Counter storage counter) internal {
unchecked {
counter._value += 1;
}
}
function decrement(Counter storage counter) internal {
uint256 value = counter._value;
require(value > 0, "Counter: decrement overflow");
unchecked {
counter._value = value - 1;
}
}
function reset(Counter storage counter) internal {
counter._value = 0;
}
}
文件 4 的 20: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 的 20: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: address zero is not a valid owner");
return _balances[owner];
}
function ownerOf(uint256 tokenId) public view virtual override returns (address) {
address owner = _ownerOf(tokenId);
require(owner != address(0), "ERC721: invalid token ID");
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) {
_requireMinted(tokenId);
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 token owner or approved for all"
);
_approve(to, tokenId);
}
function getApproved(uint256 tokenId) public view virtual override returns (address) {
_requireMinted(tokenId);
return _tokenApprovals[tokenId];
}
function setApprovalForAll(address operator, bool approved) public virtual override {
_setApprovalForAll(_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: caller is not token owner or 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: caller is not token owner or 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 _ownerOf(uint256 tokenId) internal view virtual returns (address) {
return _owners[tokenId];
}
function _exists(uint256 tokenId) internal view virtual returns (bool) {
return _ownerOf(tokenId) != address(0);
}
function _isApprovedOrOwner(address spender, uint256 tokenId) internal view virtual returns (bool) {
address owner = ERC721.ownerOf(tokenId);
return (spender == owner || isApprovedForAll(owner, spender) || getApproved(tokenId) == 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, 1);
require(!_exists(tokenId), "ERC721: token already minted");
unchecked {
_balances[to] += 1;
}
_owners[tokenId] = to;
emit Transfer(address(0), to, tokenId);
_afterTokenTransfer(address(0), to, tokenId, 1);
}
function _burn(uint256 tokenId) internal virtual {
address owner = ERC721.ownerOf(tokenId);
_beforeTokenTransfer(owner, address(0), tokenId, 1);
owner = ERC721.ownerOf(tokenId);
delete _tokenApprovals[tokenId];
unchecked {
_balances[owner] -= 1;
}
delete _owners[tokenId];
emit Transfer(owner, address(0), tokenId);
_afterTokenTransfer(owner, address(0), tokenId, 1);
}
function _transfer(
address from,
address to,
uint256 tokenId
) internal virtual {
require(ERC721.ownerOf(tokenId) == from, "ERC721: transfer from incorrect owner");
require(to != address(0), "ERC721: transfer to the zero address");
_beforeTokenTransfer(from, to, tokenId, 1);
require(ERC721.ownerOf(tokenId) == from, "ERC721: transfer from incorrect owner");
delete _tokenApprovals[tokenId];
unchecked {
_balances[from] -= 1;
_balances[to] += 1;
}
_owners[tokenId] = to;
emit Transfer(from, to, tokenId);
_afterTokenTransfer(from, to, tokenId, 1);
}
function _approve(address to, uint256 tokenId) internal virtual {
_tokenApprovals[tokenId] = to;
emit Approval(ERC721.ownerOf(tokenId), to, tokenId);
}
function _setApprovalForAll(
address owner,
address operator,
bool approved
) internal virtual {
require(owner != operator, "ERC721: approve to caller");
_operatorApprovals[owner][operator] = approved;
emit ApprovalForAll(owner, operator, approved);
}
function _requireMinted(uint256 tokenId) internal view virtual {
require(_exists(tokenId), "ERC721: invalid token ID");
}
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 firstTokenId,
uint256 batchSize
) internal virtual {}
function _afterTokenTransfer(
address from,
address to,
uint256 firstTokenId,
uint256 batchSize
) internal virtual {}
function __unsafe_increaseBalance(address account, uint256 amount) internal {
_balances[account] += amount;
}
}
文件 6 的 20: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 firstTokenId,
uint256 batchSize
) internal virtual override {
super._beforeTokenTransfer(from, to, firstTokenId, batchSize);
if (batchSize > 1) {
revert("ERC721Enumerable: consecutive transfers not supported");
}
uint256 tokenId = firstTokenId;
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 的 20:EnumerableSet.sol
pragma solidity ^0.8.0;
library EnumerableSet {
struct Set {
bytes32[] _values;
mapping(bytes32 => uint256) _indexes;
}
function _add(Set storage set, bytes32 value) private returns (bool) {
if (!_contains(set, value)) {
set._values.push(value);
set._indexes[value] = set._values.length;
return true;
} else {
return false;
}
}
function _remove(Set storage set, bytes32 value) private returns (bool) {
uint256 valueIndex = set._indexes[value];
if (valueIndex != 0) {
uint256 toDeleteIndex = valueIndex - 1;
uint256 lastIndex = set._values.length - 1;
if (lastIndex != toDeleteIndex) {
bytes32 lastValue = set._values[lastIndex];
set._values[toDeleteIndex] = lastValue;
set._indexes[lastValue] = valueIndex;
}
set._values.pop();
delete set._indexes[value];
return true;
} else {
return false;
}
}
function _contains(Set storage set, bytes32 value) private view returns (bool) {
return set._indexes[value] != 0;
}
function _length(Set storage set) private view returns (uint256) {
return set._values.length;
}
function _at(Set storage set, uint256 index) private view returns (bytes32) {
return set._values[index];
}
function _values(Set storage set) private view returns (bytes32[] memory) {
return set._values;
}
struct Bytes32Set {
Set _inner;
}
function add(Bytes32Set storage set, bytes32 value) internal returns (bool) {
return _add(set._inner, value);
}
function remove(Bytes32Set storage set, bytes32 value) internal returns (bool) {
return _remove(set._inner, value);
}
function contains(Bytes32Set storage set, bytes32 value) internal view returns (bool) {
return _contains(set._inner, value);
}
function length(Bytes32Set storage set) internal view returns (uint256) {
return _length(set._inner);
}
function at(Bytes32Set storage set, uint256 index) internal view returns (bytes32) {
return _at(set._inner, index);
}
function values(Bytes32Set storage set) internal view returns (bytes32[] memory) {
bytes32[] memory store = _values(set._inner);
bytes32[] memory result;
assembly {
result := store
}
return result;
}
struct AddressSet {
Set _inner;
}
function add(AddressSet storage set, address value) internal returns (bool) {
return _add(set._inner, bytes32(uint256(uint160(value))));
}
function remove(AddressSet storage set, address value) internal returns (bool) {
return _remove(set._inner, bytes32(uint256(uint160(value))));
}
function contains(AddressSet storage set, address value) internal view returns (bool) {
return _contains(set._inner, bytes32(uint256(uint160(value))));
}
function length(AddressSet storage set) internal view returns (uint256) {
return _length(set._inner);
}
function at(AddressSet storage set, uint256 index) internal view returns (address) {
return address(uint160(uint256(_at(set._inner, index))));
}
function values(AddressSet storage set) internal view returns (address[] memory) {
bytes32[] memory store = _values(set._inner);
address[] memory result;
assembly {
result := store
}
return result;
}
struct UintSet {
Set _inner;
}
function add(UintSet storage set, uint256 value) internal returns (bool) {
return _add(set._inner, bytes32(value));
}
function remove(UintSet storage set, uint256 value) internal returns (bool) {
return _remove(set._inner, bytes32(value));
}
function contains(UintSet storage set, uint256 value) internal view returns (bool) {
return _contains(set._inner, bytes32(value));
}
function length(UintSet storage set) internal view returns (uint256) {
return _length(set._inner);
}
function at(UintSet storage set, uint256 index) internal view returns (uint256) {
return uint256(_at(set._inner, index));
}
function values(UintSet storage set) internal view returns (uint256[] memory) {
bytes32[] memory store = _values(set._inner);
uint256[] memory result;
assembly {
result := store
}
return result;
}
}
文件 8 的 20:FoxifyAffiliation.sol
pragma solidity 0.8.19;
import "@openzeppelin/contracts/token/ERC721/extensions/ERC721Enumerable.sol";
import "@openzeppelin/contracts/access/Ownable.sol";
import "@openzeppelin/contracts/security/ReentrancyGuard.sol";
import "@openzeppelin/contracts/utils/structs/EnumerableSet.sol";
import "@openzeppelin/contracts/utils/Counters.sol";
import "@openzeppelin/contracts/utils/cryptography/MerkleProof.sol";
import "./interfaces/IFoxifyAffiliation.sol";
contract FoxifyAffiliation is IFoxifyAffiliation, ERC721Enumerable, Ownable, ReentrancyGuard {
using Counters for Counters.Counter;
using EnumerableSet for EnumerableSet.AddressSet;
using EnumerableSet for EnumerableSet.UintSet;
uint256 public constant TOTAL_SHARE = 1000;
Counters.Counter private _tokensCount;
string private _baseTokenURI;
mapping(uint256 => EnumerableSet.AddressSet) private _teamUsers;
mapping(address => EnumerableSet.UintSet) private _usersIDs;
IRandomizer public immutable randomizer;
uint256 public callbackGasLimit;
MergeLevelRates public mergeLevelRates;
MergeLevelPermissions public mergeLevelPermissions;
uint256 public teamsCount;
Wave[] public waves;
mapping(uint256 => mapping(address => bool)) public claimed;
mapping(uint256 => NFTData) public data;
mapping(uint256 => MintRequest) public requests;
mapping(address => uint256) public usersActiveID;
mapping(address => uint256) public usersTeam;
function currentWave() public view returns (uint256 id, Wave memory output) {
if (waves.length > 0) {
for (uint256 i = waves.length; i > 0; i--) {
Wave memory wave = waves[i - 1];
if (wave.start <= block.timestamp && wave.end >= block.timestamp) {
id = i - 1;
output = wave;
break;
}
}
}
}
function dataList(uint256 offset, uint256 limit) external view returns (NFTData[] memory output) {
uint256 tokensLength = _tokensCount.current();
if (offset >= tokensLength) return new NFTData[](0);
uint256 to = offset + limit;
if (tokensLength < to) to = tokensLength;
output = new NFTData[](to - offset);
for (uint256 i = 0; i < output.length; i++) output[i] = data[offset + i];
}
function exists(uint256 tokenId) external view returns (bool) {
return _exists(tokenId);
}
function supportsInterface(bytes4 interfaceId) public view override returns (bool) {
return super.supportsInterface(interfaceId);
}
function teamUsers(uint256 team, uint256 index) external view returns (address) {
return _teamUsers[team].at(index);
}
function teamUsersContains(uint256 team, address user) external view returns (bool) {
return _teamUsers[team].contains(user);
}
function teamUsersLength(uint256 team) external view returns (uint256) {
return _teamUsers[team].length();
}
function teamUsersList(
uint256 team,
uint256 offset,
uint256 limit
) external view returns (address[] memory output) {
uint256 usersLength = _teamUsers[team].length();
if (offset >= usersLength) return new address[](0);
uint256 to = offset + limit;
if (usersLength < to) to = usersLength;
output = new address[](to - offset);
for (uint256 i = 0; i < output.length; i++) output[i] = _teamUsers[team].at(offset + i);
}
function tokensCount() external view returns (uint256) {
return _tokensCount.current();
}
function usersIDs(address user, uint256 index) external view returns (uint256) {
return _usersIDs[user].at(index);
}
function usersIDsContains(address user, uint256 id) external view returns (bool) {
return _usersIDs[user].contains(id);
}
function usersIDsLength(address user) external view returns (uint256) {
return _usersIDs[user].length();
}
function usersIDsList(address user, uint256 offset, uint256 limit) external view returns (uint256[] memory output) {
uint256 idsLength = _usersIDs[user].length();
if (offset >= idsLength) return new uint256[](0);
uint256 to = offset + limit;
if (idsLength < to) to = idsLength;
output = new uint256[](to - offset);
for (uint256 i = 0; i < output.length; i++) output[i] = _usersIDs[user].at(offset + i);
}
function usersTeamList(address[] memory users) external view returns (uint256[] memory output) {
output = new uint256[](users.length);
for (uint256 i = 0; i < users.length; i++) output[i] = usersTeam[users[i]];
}
function wavesLength() external view returns (uint256) {
return waves.length;
}
function wavesList(uint256 offset, uint256 limit) external view returns (Wave[] memory output) {
uint256 wavesLength_ = waves.length;
if (offset >= wavesLength_) return new Wave[](0);
uint256 to = offset + limit;
if (wavesLength_ < to) to = wavesLength_;
output = new Wave[](to - offset);
for (uint256 i = 0; i < output.length; i++) output[i] = waves[offset + i];
}
constructor(
string memory name_,
string memory symbol_,
string memory baseTokenURI_,
uint256 teamsCount_,
address randomizer_,
uint256 callbackGasLimit_,
MergeLevelRates memory mergeLevelRates_,
MergeLevelPermissions memory mergeLevelPermissions_
) ERC721(name_, symbol_) {
require(teamsCount_ > 0, "FoxifyAffiliation: TeamsCount is not positive");
require(randomizer_ != address(0), "FoxifyAffiliation: Randomizer is zero address");
require(callbackGasLimit_ > 0, "FoxifyAffiliation: GasLimit is not positive");
_baseTokenURI = baseTokenURI_;
teamsCount = teamsCount_;
randomizer = IRandomizer(randomizer_);
callbackGasLimit = callbackGasLimit_;
_updateMergeLevelRates(mergeLevelRates_);
mergeLevelPermissions = mergeLevelPermissions_;
}
function batchTransferFrom(BatchParams[] memory params) external returns (bool) {
for (uint256 i = 0; i < params.length; i++) {
BatchParams memory param = params[i];
transferFrom(param.from, param.to, param.id);
}
return true;
}
function merge(uint256[] memory ids, Level from) external nonReentrant returns (bool) {
bool fromIsBronze = from == Level.BRONZE;
require(fromIsBronze || from == Level.SILVER, "FoxifyAffiliation: Invalid from level");
uint256 mergeRate = fromIsBronze ? mergeLevelRates.bronzeToSilver : mergeLevelRates.silverToGold;
MergeLevelPermissions memory permissions = mergeLevelPermissions;
Level to;
if (fromIsBronze) {
require(permissions.bronzeToSilver, "FoxifyAffiliation: BronzeToSilver permission denied");
to = Level.SILVER;
} else {
require(permissions.silverToGold, "FoxifyAffiliation: SilverToGold permission denied");
to = Level.GOLD;
}
require(mergeRate == ids.length, "FoxifyAffiliation: Invalid params length");
uint256 tokenId = _safeMint(msg.sender);
NFTData memory data_ = NFTData(to, keccak256(abi.encode(blockhash(block.number - 1))), block.timestamp);
data[tokenId] = data_;
for (uint256 i = 0; i < ids.length; i++) {
uint256 id = ids[i];
require(data[id].level == from, "FoxifyAffiliation: Level is not match");
require(ownerOf(id) == msg.sender, "FoxifyAffiliation: Invalid id");
_burn(id);
}
emit Merged(tokenId, ids, from, to);
emit Minted(msg.sender, tokenId, data_);
return true;
}
function mintRequest(bytes32[] calldata merkleProof, uint256 team) external returns (bool) {
(uint256 waveIndex, Wave memory wave) = currentWave();
require(block.timestamp >= wave.start && block.timestamp <= wave.end, "FoxifyAffiliation: All waves expired");
require(merkleProof.length > 0, "FoxifyAffiliation: Invalid proofs length");
require(
MerkleProof.verify(merkleProof, wave.root, keccak256(bytes.concat(keccak256(abi.encode(msg.sender))))) ==
true,
"FoxifyAffiliation: Invalid proofs"
);
require(!claimed[waveIndex][msg.sender], "FoxifyAffiliation: Already claimed");
uint256 requestID = randomizer.request(callbackGasLimit);
MintRequest memory mintRequestData = MintRequest(msg.sender, team, waveIndex);
requests[requestID] = mintRequestData;
claimed[waveIndex][msg.sender] = true;
emit MintRequested(requestID, mintRequestData);
return true;
}
function preMint(LevelsDistribution memory shares) external onlyOwner returns (bool) {
require(waves.length == 0, "FoxifyAffiliation: Waves exists");
uint256 tokensCount_ = shares.bronze + shares.silver + shares.gold;
require(tokensCount_ > 0, "FoxifyAffiliation: Tokens count not positive");
bytes32 pseudoRandom = keccak256(abi.encode(blockhash(block.number - 1)));
for (uint256 i = 1; i <= tokensCount_; i++) {
pseudoRandom = keccak256(abi.encode(pseudoRandom));
Level level;
if (i <= shares.bronze) {
level = Level.BRONZE;
} else if (i <= shares.bronze + shares.silver) {
level = Level.SILVER;
} else {
level = Level.GOLD;
}
uint256 tokenId = _safeMint(msg.sender);
NFTData memory data_ = NFTData(level, pseudoRandom, block.timestamp);
data[tokenId] = data_;
emit Minted(msg.sender, tokenId, data_);
}
return true;
}
function randomizerCallback(uint256 _id, bytes32 _value) external {
require(msg.sender == address(randomizer), "FoxifyAffiliation: Caller is not Randomizer");
MintRequest memory mintRequestData = requests[_id];
uint256 randomValue = uint256(_value) % TOTAL_SHARE;
Wave memory wave = waves[mintRequestData.waveIndex];
Level level;
if (randomValue < wave.distribution.bronze) {
level = Level.BRONZE;
} else if (randomValue < wave.distribution.bronze + wave.distribution.silver) {
level = Level.SILVER;
} else {
level = Level.GOLD;
}
uint256 tokenId = _safeMint(mintRequestData.user);
NFTData memory data_ = NFTData(level, _value, block.timestamp);
data[tokenId] = data_;
emit Minted(mintRequestData.user, tokenId, data_);
_switchTeam(mintRequestData.user, mintRequestData.team);
}
function randomizerWithdraw(uint256 amount) external onlyOwner returns (bool) {
randomizer.clientWithdrawTo(msg.sender, amount);
emit RandomizerWithdrawn(msg.sender, amount);
return true;
}
function scheduleWave(Wave memory wave) external onlyOwner returns (bool) {
require(wave.root != bytes32(0), "FoxifyAffiliation: Root is zero bytes");
require(wave.start >= block.timestamp, "FoxifyAffiliation: Current lt start");
require(wave.end > wave.start, "FoxifyAffiliation: End lte start");
require(
wave.distribution.bronze + wave.distribution.silver + wave.distribution.gold == TOTAL_SHARE,
"FoxifyAffiliation: Distribution ne TOTAL_SHARE"
);
if (waves.length > 0) {
Wave memory lastWave = waves[waves.length - 1];
require(wave.start > lastWave.end, "FoxifyAffiliation: New wave start lt last wave end");
}
waves.push(wave);
emit WaveScheduled(waves.length - 1, wave);
return true;
}
function switchTeam(uint256 team) external returns (bool) {
_switchTeam(msg.sender, team);
return true;
}
function unscheduleWave(uint256 index) external onlyOwner returns (bool) {
require(index < waves.length, "FoxifyAffiliation: Wave not exist");
Wave memory targetWave = waves[index];
require(targetWave.start > block.timestamp, "FoxifyAffiliation: Wave already started");
for (uint256 i = index + 1; i < waves.length; i++) waves[i - 1] = waves[i];
waves.pop();
emit WaveUnscheduled(targetWave);
return true;
}
function updateBaseURI(string memory uri) external onlyOwner returns (bool) {
_baseTokenURI = uri;
emit BaseURIUpdated(uri);
return true;
}
function updateCallbackGasLimit(uint256 gasLimit) external onlyOwner returns (bool) {
require(gasLimit > 0, "FoxifyAffiliation: GasLimit is not positive");
callbackGasLimit = gasLimit;
emit CallbackGasLimitUpdated(gasLimit);
return true;
}
function updateMergeLevelRates(MergeLevelRates memory rates) external onlyOwner returns (bool) {
_updateMergeLevelRates(rates);
return true;
}
function updateMergeLevelPermissions(MergeLevelPermissions memory permissions) external onlyOwner returns (bool) {
mergeLevelPermissions = permissions;
emit MergeLevelPermissionsUpdated(permissions);
return true;
}
function updateTeamsCount(uint256 count) external onlyOwner returns (bool) {
require(count > teamsCount, "FoxifyAffiliation: Teams count lte current");
teamsCount = count;
emit TeamsCountUpdated(count);
return true;
}
function updateUserActiveID(uint256 tokenId) external returns (bool) {
if (tokenId > 0) require(ownerOf(tokenId) == msg.sender, "FoxifyAffiliation: Incorrect owner");
usersActiveID[msg.sender] = tokenId;
emit UserActiveIDUpdated(msg.sender, tokenId);
return true;
}
function _baseURI() internal view override returns (string memory) {
return _baseTokenURI;
}
function _afterTokenTransfer(address from, address to, uint256 tokenId, uint256) internal override {
_usersIDs[from].remove(tokenId);
_usersIDs[to].add(tokenId);
if (from != address(0) && balanceOf(from) == 0) {
_teamUsers[usersTeam[from]].remove(from);
delete usersTeam[from];
}
if (to != address(0) && balanceOf(to) == 1 && to != owner()) {
usersActiveID[to] = tokenId;
}
}
function _beforeTokenTransfer(
address from,
address to,
uint256 tokenId,
uint256 batchSize
) internal override(ERC721Enumerable) {
super._beforeTokenTransfer(from, to, tokenId, batchSize);
require(usersActiveID[from] != tokenId, "FoxifyAffiliation: TokenId is active");
}
function _safeMint(address to) private returns (uint256 tokenId) {
_tokensCount.increment();
tokenId = _tokensCount.current();
_safeMint(to, tokenId);
}
function _switchTeam(address user, uint256 team) private {
uint256 currentTeam = usersTeam[user];
require(team <= teamsCount, "FoxifyAffiliation: Team not found");
if (currentTeam != team) {
_teamUsers[currentTeam].remove(user);
_teamUsers[team].add(user);
usersTeam[user] = team;
}
emit TeamSwitched(user, team);
}
function _updateMergeLevelRates(MergeLevelRates memory rates) private {
require(rates.bronzeToSilver > 0 && rates.silverToGold > 0, "FoxifyAffiliation: Rate is not positive");
mergeLevelRates = rates;
emit MergeLevelRatesUpdated(rates);
}
}
文件 9 的 20:IERC165.sol
pragma solidity ^0.8.0;
interface IERC165 {
function supportsInterface(bytes4 interfaceId) external view returns (bool);
}
文件 10 的 20: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,
bytes calldata data
) external;
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 setApprovalForAll(address operator, bool _approved) external;
function getApproved(uint256 tokenId) external view returns (address operator);
function isApprovedForAll(address owner, address operator) external view returns (bool);
}
文件 11 的 20: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);
function tokenByIndex(uint256 index) external view returns (uint256);
}
文件 12 的 20: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);
}
文件 13 的 20:IERC721Receiver.sol
pragma solidity ^0.8.0;
interface IERC721Receiver {
function onERC721Received(
address operator,
address from,
uint256 tokenId,
bytes calldata data
) external returns (bytes4);
}
文件 14 的 20:IFoxifyAffiliation.sol
pragma solidity 0.8.19;
import "./IRandomizer.sol";
interface IFoxifyAffiliation {
enum Level {
UNKNOWN,
BRONZE,
SILVER,
GOLD
}
struct BatchParams {
address from;
address to;
uint256 id;
}
struct LevelsDistribution {
uint256 bronze;
uint256 silver;
uint256 gold;
}
struct MergeLevelRates {
uint256 bronzeToSilver;
uint256 silverToGold;
}
struct MergeLevelPermissions {
bool bronzeToSilver;
bool silverToGold;
}
struct MintRequest {
address user;
uint256 team;
uint256 waveIndex;
}
struct NFTData {
Level level;
bytes32 randomValue;
uint256 timestamp;
}
struct Wave {
bytes32 root;
uint256 start;
uint256 end;
LevelsDistribution distribution;
}
event BaseURIUpdated(string uri);
event CallbackGasLimitUpdated(uint256 amount);
event Merged(uint256 indexed tokenId, uint256[] ids, Level from, Level to);
event MergeLevelRatesUpdated(MergeLevelRates rates);
event MergeLevelPermissionsUpdated(MergeLevelPermissions permissions);
event Minted(address indexed recipient, uint256 tokenId, NFTData data);
event MintRequested(uint256 id, MintRequest request);
event RandomizerWithdrawn(address indexed caller, uint256 amount);
event TeamsCountUpdated(uint256 count);
event TeamSwitched(address indexed user, uint256 teamId);
event UserActiveIDUpdated(address indexed user, uint256 indexed tokenId);
event WaveScheduled(uint256 index, Wave wave);
event WaveUnscheduled(Wave wave);
function randomizer() external view returns (IRandomizer);
function TOTAL_SHARE() external view returns (uint256);
function callbackGasLimit() external view returns (uint256);
function teamsCount() external view returns (uint256);
function claimed(uint256, address) external view returns (bool);
function usersActiveID(address) external view returns (uint256);
function usersTeam(address) external view returns (uint256);
function currentWave() external view returns (uint256 id, Wave memory output);
function dataList(uint256 offset, uint256 limit) external view returns (NFTData[] memory output);
function exists(uint256 tokenId) external view returns (bool);
function teamUsers(uint256 team, uint256 index) external view returns (address);
function teamUsersContains(uint256 team, address user) external view returns (bool);
function teamUsersLength(uint256 team) external view returns (uint256);
function teamUsersList(uint256 offset, uint256 limit, uint256 team) external view returns (address[] memory output);
function tokensCount() external view returns (uint256);
function usersIDs(address user, uint256 index) external view returns (uint256);
function usersIDsContains(address user, uint256 id) external view returns (bool);
function usersIDsLength(address user) external view returns (uint256);
function usersIDsList(address user, uint256 offset, uint256 limit) external view returns (uint256[] memory output);
function usersTeamList(address[] memory users) external view returns (uint256[] memory output);
function wavesList(uint256 offset, uint256 limit) external view returns (Wave[] memory output);
function batchTransferFrom(BatchParams[] memory params) external returns (bool);
function merge(uint256[] memory ids, Level from) external returns (bool);
function mintRequest(bytes32[] calldata merkleProof, uint256 team) external returns (bool);
function preMint(LevelsDistribution memory shares) external returns (bool);
function randomizerCallback(uint256 _id, bytes32 _value) external;
function randomizerWithdraw(uint256 amount) external returns (bool);
function scheduleWave(Wave memory wave) external returns (bool);
function switchTeam(uint256 team) external returns (bool);
function unscheduleWave(uint256 index) external returns (bool);
function updateBaseURI(string memory uri) external returns (bool);
function updateCallbackGasLimit(uint256 gasLimit) external returns (bool);
function updateMergeLevelRates(MergeLevelRates memory rates) external returns (bool);
function updateMergeLevelPermissions(MergeLevelPermissions memory permissions) external returns (bool);
function updateTeamsCount(uint256 count) external returns (bool);
function updateUserActiveID(uint256 tokenId) external returns (bool);
}
文件 15 的 20:IRandomizer.sol
pragma solidity 0.8.19;
interface IRandomizer {
function request(uint256 callbackGasLimit) external returns (uint256);
function request(uint256 callbackGasLimit, uint256 confirmations) external returns (uint256);
function clientWithdrawTo(address _to, uint256 _amount) external;
}
文件 16 的 20:Math.sol
pragma solidity ^0.8.0;
library Math {
enum Rounding {
Down,
Up,
Zero
}
function max(uint256 a, uint256 b) internal pure returns (uint256) {
return a > b ? a : b;
}
function min(uint256 a, uint256 b) internal pure returns (uint256) {
return a < b ? a : b;
}
function average(uint256 a, uint256 b) internal pure returns (uint256) {
return (a & b) + (a ^ b) / 2;
}
function ceilDiv(uint256 a, uint256 b) internal pure returns (uint256) {
return a == 0 ? 0 : (a - 1) / b + 1;
}
function mulDiv(
uint256 x,
uint256 y,
uint256 denominator
) internal pure returns (uint256 result) {
unchecked {
uint256 prod0;
uint256 prod1;
assembly {
let mm := mulmod(x, y, not(0))
prod0 := mul(x, y)
prod1 := sub(sub(mm, prod0), lt(mm, prod0))
}
if (prod1 == 0) {
return prod0 / denominator;
}
require(denominator > prod1);
uint256 remainder;
assembly {
remainder := mulmod(x, y, denominator)
prod1 := sub(prod1, gt(remainder, prod0))
prod0 := sub(prod0, remainder)
}
uint256 twos = denominator & (~denominator + 1);
assembly {
denominator := div(denominator, twos)
prod0 := div(prod0, twos)
twos := add(div(sub(0, twos), twos), 1)
}
prod0 |= prod1 * twos;
uint256 inverse = (3 * denominator) ^ 2;
inverse *= 2 - denominator * inverse;
inverse *= 2 - denominator * inverse;
inverse *= 2 - denominator * inverse;
inverse *= 2 - denominator * inverse;
inverse *= 2 - denominator * inverse;
inverse *= 2 - denominator * inverse;
result = prod0 * inverse;
return result;
}
}
function mulDiv(
uint256 x,
uint256 y,
uint256 denominator,
Rounding rounding
) internal pure returns (uint256) {
uint256 result = mulDiv(x, y, denominator);
if (rounding == Rounding.Up && mulmod(x, y, denominator) > 0) {
result += 1;
}
return result;
}
function sqrt(uint256 a) internal pure returns (uint256) {
if (a == 0) {
return 0;
}
uint256 result = 1 << (log2(a) >> 1);
unchecked {
result = (result + a / result) >> 1;
result = (result + a / result) >> 1;
result = (result + a / result) >> 1;
result = (result + a / result) >> 1;
result = (result + a / result) >> 1;
result = (result + a / result) >> 1;
result = (result + a / result) >> 1;
return min(result, a / result);
}
}
function sqrt(uint256 a, Rounding rounding) internal pure returns (uint256) {
unchecked {
uint256 result = sqrt(a);
return result + (rounding == Rounding.Up && result * result < a ? 1 : 0);
}
}
function log2(uint256 value) internal pure returns (uint256) {
uint256 result = 0;
unchecked {
if (value >> 128 > 0) {
value >>= 128;
result += 128;
}
if (value >> 64 > 0) {
value >>= 64;
result += 64;
}
if (value >> 32 > 0) {
value >>= 32;
result += 32;
}
if (value >> 16 > 0) {
value >>= 16;
result += 16;
}
if (value >> 8 > 0) {
value >>= 8;
result += 8;
}
if (value >> 4 > 0) {
value >>= 4;
result += 4;
}
if (value >> 2 > 0) {
value >>= 2;
result += 2;
}
if (value >> 1 > 0) {
result += 1;
}
}
return result;
}
function log2(uint256 value, Rounding rounding) internal pure returns (uint256) {
unchecked {
uint256 result = log2(value);
return result + (rounding == Rounding.Up && 1 << result < value ? 1 : 0);
}
}
function log10(uint256 value) internal pure returns (uint256) {
uint256 result = 0;
unchecked {
if (value >= 10**64) {
value /= 10**64;
result += 64;
}
if (value >= 10**32) {
value /= 10**32;
result += 32;
}
if (value >= 10**16) {
value /= 10**16;
result += 16;
}
if (value >= 10**8) {
value /= 10**8;
result += 8;
}
if (value >= 10**4) {
value /= 10**4;
result += 4;
}
if (value >= 10**2) {
value /= 10**2;
result += 2;
}
if (value >= 10**1) {
result += 1;
}
}
return result;
}
function log10(uint256 value, Rounding rounding) internal pure returns (uint256) {
unchecked {
uint256 result = log10(value);
return result + (rounding == Rounding.Up && 10**result < value ? 1 : 0);
}
}
function log256(uint256 value) internal pure returns (uint256) {
uint256 result = 0;
unchecked {
if (value >> 128 > 0) {
value >>= 128;
result += 16;
}
if (value >> 64 > 0) {
value >>= 64;
result += 8;
}
if (value >> 32 > 0) {
value >>= 32;
result += 4;
}
if (value >> 16 > 0) {
value >>= 16;
result += 2;
}
if (value >> 8 > 0) {
result += 1;
}
}
return result;
}
function log256(uint256 value, Rounding rounding) internal pure returns (uint256) {
unchecked {
uint256 result = log256(value);
return result + (rounding == Rounding.Up && 1 << (result * 8) < value ? 1 : 0);
}
}
}
文件 17 的 20:MerkleProof.sol
pragma solidity ^0.8.0;
library MerkleProof {
function verify(
bytes32[] memory proof,
bytes32 root,
bytes32 leaf
) internal pure returns (bool) {
return processProof(proof, leaf) == root;
}
function verifyCalldata(
bytes32[] calldata proof,
bytes32 root,
bytes32 leaf
) internal pure returns (bool) {
return processProofCalldata(proof, leaf) == root;
}
function processProof(bytes32[] memory proof, bytes32 leaf) internal pure returns (bytes32) {
bytes32 computedHash = leaf;
for (uint256 i = 0; i < proof.length; i++) {
computedHash = _hashPair(computedHash, proof[i]);
}
return computedHash;
}
function processProofCalldata(bytes32[] calldata proof, bytes32 leaf) internal pure returns (bytes32) {
bytes32 computedHash = leaf;
for (uint256 i = 0; i < proof.length; i++) {
computedHash = _hashPair(computedHash, proof[i]);
}
return computedHash;
}
function multiProofVerify(
bytes32[] memory proof,
bool[] memory proofFlags,
bytes32 root,
bytes32[] memory leaves
) internal pure returns (bool) {
return processMultiProof(proof, proofFlags, leaves) == root;
}
function multiProofVerifyCalldata(
bytes32[] calldata proof,
bool[] calldata proofFlags,
bytes32 root,
bytes32[] memory leaves
) internal pure returns (bool) {
return processMultiProofCalldata(proof, proofFlags, leaves) == root;
}
function processMultiProof(
bytes32[] memory proof,
bool[] memory proofFlags,
bytes32[] memory leaves
) internal pure returns (bytes32 merkleRoot) {
uint256 leavesLen = leaves.length;
uint256 totalHashes = proofFlags.length;
require(leavesLen + proof.length - 1 == totalHashes, "MerkleProof: invalid multiproof");
bytes32[] memory hashes = new bytes32[](totalHashes);
uint256 leafPos = 0;
uint256 hashPos = 0;
uint256 proofPos = 0;
for (uint256 i = 0; i < totalHashes; i++) {
bytes32 a = leafPos < leavesLen ? leaves[leafPos++] : hashes[hashPos++];
bytes32 b = proofFlags[i] ? leafPos < leavesLen ? leaves[leafPos++] : hashes[hashPos++] : proof[proofPos++];
hashes[i] = _hashPair(a, b);
}
if (totalHashes > 0) {
return hashes[totalHashes - 1];
} else if (leavesLen > 0) {
return leaves[0];
} else {
return proof[0];
}
}
function processMultiProofCalldata(
bytes32[] calldata proof,
bool[] calldata proofFlags,
bytes32[] memory leaves
) internal pure returns (bytes32 merkleRoot) {
uint256 leavesLen = leaves.length;
uint256 totalHashes = proofFlags.length;
require(leavesLen + proof.length - 1 == totalHashes, "MerkleProof: invalid multiproof");
bytes32[] memory hashes = new bytes32[](totalHashes);
uint256 leafPos = 0;
uint256 hashPos = 0;
uint256 proofPos = 0;
for (uint256 i = 0; i < totalHashes; i++) {
bytes32 a = leafPos < leavesLen ? leaves[leafPos++] : hashes[hashPos++];
bytes32 b = proofFlags[i] ? leafPos < leavesLen ? leaves[leafPos++] : hashes[hashPos++] : proof[proofPos++];
hashes[i] = _hashPair(a, b);
}
if (totalHashes > 0) {
return hashes[totalHashes - 1];
} else if (leavesLen > 0) {
return leaves[0];
} else {
return proof[0];
}
}
function _hashPair(bytes32 a, bytes32 b) private pure returns (bytes32) {
return a < b ? _efficientHash(a, b) : _efficientHash(b, a);
}
function _efficientHash(bytes32 a, bytes32 b) private pure returns (bytes32 value) {
assembly {
mstore(0x00, a)
mstore(0x20, b)
value := keccak256(0x00, 0x40)
}
}
}
文件 18 的 20: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() {
_transferOwnership(_msgSender());
}
modifier onlyOwner() {
_checkOwner();
_;
}
function owner() public view virtual returns (address) {
return _owner;
}
function _checkOwner() internal view virtual {
require(owner() == _msgSender(), "Ownable: caller is not the owner");
}
function renounceOwnership() public virtual onlyOwner {
_transferOwnership(address(0));
}
function transferOwnership(address newOwner) public virtual onlyOwner {
require(newOwner != address(0), "Ownable: new owner is the zero address");
_transferOwnership(newOwner);
}
function _transferOwnership(address newOwner) internal virtual {
address oldOwner = _owner;
_owner = newOwner;
emit OwnershipTransferred(oldOwner, newOwner);
}
}
文件 19 的 20:ReentrancyGuard.sol
pragma solidity ^0.8.0;
abstract contract ReentrancyGuard {
uint256 private constant _NOT_ENTERED = 1;
uint256 private constant _ENTERED = 2;
uint256 private _status;
constructor() {
_status = _NOT_ENTERED;
}
modifier nonReentrant() {
_nonReentrantBefore();
_;
_nonReentrantAfter();
}
function _nonReentrantBefore() private {
require(_status != _ENTERED, "ReentrancyGuard: reentrant call");
_status = _ENTERED;
}
function _nonReentrantAfter() private {
_status = _NOT_ENTERED;
}
}
文件 20 的 20:Strings.sol
pragma solidity ^0.8.0;
import "./math/Math.sol";
library Strings {
bytes16 private constant _SYMBOLS = "0123456789abcdef";
uint8 private constant _ADDRESS_LENGTH = 20;
function toString(uint256 value) internal pure returns (string memory) {
unchecked {
uint256 length = Math.log10(value) + 1;
string memory buffer = new string(length);
uint256 ptr;
assembly {
ptr := add(buffer, add(32, length))
}
while (true) {
ptr--;
assembly {
mstore8(ptr, byte(mod(value, 10), _SYMBOLS))
}
value /= 10;
if (value == 0) break;
}
return buffer;
}
}
function toHexString(uint256 value) internal pure returns (string memory) {
unchecked {
return toHexString(value, Math.log256(value) + 1);
}
}
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] = _SYMBOLS[value & 0xf];
value >>= 4;
}
require(value == 0, "Strings: hex length insufficient");
return string(buffer);
}
function toHexString(address addr) internal pure returns (string memory) {
return toHexString(uint256(uint160(addr)), _ADDRESS_LENGTH);
}
}
{
"compilationTarget": {
"contracts/FoxifyAffiliation.sol": "FoxifyAffiliation"
},
"evmVersion": "paris",
"libraries": {},
"metadata": {
"bytecodeHash": "ipfs",
"useLiteralContent": true
},
"optimizer": {
"enabled": true,
"runs": 200
},
"remappings": []
}
[{"inputs":[{"internalType":"string","name":"name_","type":"string"},{"internalType":"string","name":"symbol_","type":"string"},{"internalType":"string","name":"baseTokenURI_","type":"string"},{"internalType":"uint256","name":"teamsCount_","type":"uint256"},{"internalType":"address","name":"randomizer_","type":"address"},{"internalType":"uint256","name":"callbackGasLimit_","type":"uint256"},{"components":[{"internalType":"uint256","name":"bronzeToSilver","type":"uint256"},{"internalType":"uint256","name":"silverToGold","type":"uint256"}],"internalType":"struct IFoxifyAffiliation.MergeLevelRates","name":"mergeLevelRates_","type":"tuple"},{"components":[{"internalType":"bool","name":"bronzeToSilver","type":"bool"},{"internalType":"bool","name":"silverToGold","type":"bool"}],"internalType":"struct IFoxifyAffiliation.MergeLevelPermissions","name":"mergeLevelPermissions_","type":"tuple"}],"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":false,"internalType":"string","name":"uri","type":"string"}],"name":"BaseURIUpdated","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"CallbackGasLimitUpdated","type":"event"},{"anonymous":false,"inputs":[{"components":[{"internalType":"bool","name":"bronzeToSilver","type":"bool"},{"internalType":"bool","name":"silverToGold","type":"bool"}],"indexed":false,"internalType":"struct IFoxifyAffiliation.MergeLevelPermissions","name":"permissions","type":"tuple"}],"name":"MergeLevelPermissionsUpdated","type":"event"},{"anonymous":false,"inputs":[{"components":[{"internalType":"uint256","name":"bronzeToSilver","type":"uint256"},{"internalType":"uint256","name":"silverToGold","type":"uint256"}],"indexed":false,"internalType":"struct IFoxifyAffiliation.MergeLevelRates","name":"rates","type":"tuple"}],"name":"MergeLevelRatesUpdated","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"uint256","name":"tokenId","type":"uint256"},{"indexed":false,"internalType":"uint256[]","name":"ids","type":"uint256[]"},{"indexed":false,"internalType":"enum IFoxifyAffiliation.Level","name":"from","type":"uint8"},{"indexed":false,"internalType":"enum IFoxifyAffiliation.Level","name":"to","type":"uint8"}],"name":"Merged","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"id","type":"uint256"},{"components":[{"internalType":"address","name":"user","type":"address"},{"internalType":"uint256","name":"team","type":"uint256"},{"internalType":"uint256","name":"waveIndex","type":"uint256"}],"indexed":false,"internalType":"struct IFoxifyAffiliation.MintRequest","name":"request","type":"tuple"}],"name":"MintRequested","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"recipient","type":"address"},{"indexed":false,"internalType":"uint256","name":"tokenId","type":"uint256"},{"components":[{"internalType":"enum IFoxifyAffiliation.Level","name":"level","type":"uint8"},{"internalType":"bytes32","name":"randomValue","type":"bytes32"},{"internalType":"uint256","name":"timestamp","type":"uint256"}],"indexed":false,"internalType":"struct IFoxifyAffiliation.NFTData","name":"data","type":"tuple"}],"name":"Minted","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":"caller","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"RandomizerWithdrawn","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"user","type":"address"},{"indexed":false,"internalType":"uint256","name":"teamId","type":"uint256"}],"name":"TeamSwitched","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"count","type":"uint256"}],"name":"TeamsCountUpdated","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"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"user","type":"address"},{"indexed":true,"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"UserActiveIDUpdated","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"index","type":"uint256"},{"components":[{"internalType":"bytes32","name":"root","type":"bytes32"},{"internalType":"uint256","name":"start","type":"uint256"},{"internalType":"uint256","name":"end","type":"uint256"},{"components":[{"internalType":"uint256","name":"bronze","type":"uint256"},{"internalType":"uint256","name":"silver","type":"uint256"},{"internalType":"uint256","name":"gold","type":"uint256"}],"internalType":"struct IFoxifyAffiliation.LevelsDistribution","name":"distribution","type":"tuple"}],"indexed":false,"internalType":"struct IFoxifyAffiliation.Wave","name":"wave","type":"tuple"}],"name":"WaveScheduled","type":"event"},{"anonymous":false,"inputs":[{"components":[{"internalType":"bytes32","name":"root","type":"bytes32"},{"internalType":"uint256","name":"start","type":"uint256"},{"internalType":"uint256","name":"end","type":"uint256"},{"components":[{"internalType":"uint256","name":"bronze","type":"uint256"},{"internalType":"uint256","name":"silver","type":"uint256"},{"internalType":"uint256","name":"gold","type":"uint256"}],"internalType":"struct IFoxifyAffiliation.LevelsDistribution","name":"distribution","type":"tuple"}],"indexed":false,"internalType":"struct IFoxifyAffiliation.Wave","name":"wave","type":"tuple"}],"name":"WaveUnscheduled","type":"event"},{"inputs":[],"name":"TOTAL_SHARE","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"approve","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"}],"name":"balanceOf","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"components":[{"internalType":"address","name":"from","type":"address"},{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"id","type":"uint256"}],"internalType":"struct IFoxifyAffiliation.BatchParams[]","name":"params","type":"tuple[]"}],"name":"batchTransferFrom","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"callbackGasLimit","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"address","name":"","type":"address"}],"name":"claimed","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"currentWave","outputs":[{"internalType":"uint256","name":"id","type":"uint256"},{"components":[{"internalType":"bytes32","name":"root","type":"bytes32"},{"internalType":"uint256","name":"start","type":"uint256"},{"internalType":"uint256","name":"end","type":"uint256"},{"components":[{"internalType":"uint256","name":"bronze","type":"uint256"},{"internalType":"uint256","name":"silver","type":"uint256"},{"internalType":"uint256","name":"gold","type":"uint256"}],"internalType":"struct IFoxifyAffiliation.LevelsDistribution","name":"distribution","type":"tuple"}],"internalType":"struct IFoxifyAffiliation.Wave","name":"output","type":"tuple"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"data","outputs":[{"internalType":"enum IFoxifyAffiliation.Level","name":"level","type":"uint8"},{"internalType":"bytes32","name":"randomValue","type":"bytes32"},{"internalType":"uint256","name":"timestamp","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"offset","type":"uint256"},{"internalType":"uint256","name":"limit","type":"uint256"}],"name":"dataList","outputs":[{"components":[{"internalType":"enum IFoxifyAffiliation.Level","name":"level","type":"uint8"},{"internalType":"bytes32","name":"randomValue","type":"bytes32"},{"internalType":"uint256","name":"timestamp","type":"uint256"}],"internalType":"struct IFoxifyAffiliation.NFTData[]","name":"output","type":"tuple[]"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"exists","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"getApproved","outputs":[{"internalType":"address","name":"","type":"address"}],"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":"uint256[]","name":"ids","type":"uint256[]"},{"internalType":"enum IFoxifyAffiliation.Level","name":"from","type":"uint8"}],"name":"merge","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"mergeLevelPermissions","outputs":[{"internalType":"bool","name":"bronzeToSilver","type":"bool"},{"internalType":"bool","name":"silverToGold","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"mergeLevelRates","outputs":[{"internalType":"uint256","name":"bronzeToSilver","type":"uint256"},{"internalType":"uint256","name":"silverToGold","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"bytes32[]","name":"merkleProof","type":"bytes32[]"},{"internalType":"uint256","name":"team","type":"uint256"}],"name":"mintRequest","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"name","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","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":[{"components":[{"internalType":"uint256","name":"bronze","type":"uint256"},{"internalType":"uint256","name":"silver","type":"uint256"},{"internalType":"uint256","name":"gold","type":"uint256"}],"internalType":"struct IFoxifyAffiliation.LevelsDistribution","name":"shares","type":"tuple"}],"name":"preMint","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"randomizer","outputs":[{"internalType":"contract IRandomizer","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_id","type":"uint256"},{"internalType":"bytes32","name":"_value","type":"bytes32"}],"name":"randomizerCallback","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"randomizerWithdraw","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"requests","outputs":[{"internalType":"address","name":"user","type":"address"},{"internalType":"uint256","name":"team","type":"uint256"},{"internalType":"uint256","name":"waveIndex","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":"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":[{"components":[{"internalType":"bytes32","name":"root","type":"bytes32"},{"internalType":"uint256","name":"start","type":"uint256"},{"internalType":"uint256","name":"end","type":"uint256"},{"components":[{"internalType":"uint256","name":"bronze","type":"uint256"},{"internalType":"uint256","name":"silver","type":"uint256"},{"internalType":"uint256","name":"gold","type":"uint256"}],"internalType":"struct IFoxifyAffiliation.LevelsDistribution","name":"distribution","type":"tuple"}],"internalType":"struct IFoxifyAffiliation.Wave","name":"wave","type":"tuple"}],"name":"scheduleWave","outputs":[{"internalType":"bool","name":"","type":"bool"}],"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":[{"internalType":"uint256","name":"team","type":"uint256"}],"name":"switchTeam","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"symbol","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"team","type":"uint256"},{"internalType":"uint256","name":"index","type":"uint256"}],"name":"teamUsers","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"team","type":"uint256"},{"internalType":"address","name":"user","type":"address"}],"name":"teamUsersContains","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"team","type":"uint256"}],"name":"teamUsersLength","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"team","type":"uint256"},{"internalType":"uint256","name":"offset","type":"uint256"},{"internalType":"uint256","name":"limit","type":"uint256"}],"name":"teamUsersList","outputs":[{"internalType":"address[]","name":"output","type":"address[]"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"teamsCount","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"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":"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":"tokenId","type":"uint256"}],"name":"tokenURI","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"tokensCount","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"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":[{"internalType":"uint256","name":"index","type":"uint256"}],"name":"unscheduleWave","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"string","name":"uri","type":"string"}],"name":"updateBaseURI","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"gasLimit","type":"uint256"}],"name":"updateCallbackGasLimit","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"components":[{"internalType":"bool","name":"bronzeToSilver","type":"bool"},{"internalType":"bool","name":"silverToGold","type":"bool"}],"internalType":"struct IFoxifyAffiliation.MergeLevelPermissions","name":"permissions","type":"tuple"}],"name":"updateMergeLevelPermissions","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"components":[{"internalType":"uint256","name":"bronzeToSilver","type":"uint256"},{"internalType":"uint256","name":"silverToGold","type":"uint256"}],"internalType":"struct IFoxifyAffiliation.MergeLevelRates","name":"rates","type":"tuple"}],"name":"updateMergeLevelRates","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"count","type":"uint256"}],"name":"updateTeamsCount","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"updateUserActiveID","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"usersActiveID","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"user","type":"address"},{"internalType":"uint256","name":"index","type":"uint256"}],"name":"usersIDs","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"user","type":"address"},{"internalType":"uint256","name":"id","type":"uint256"}],"name":"usersIDsContains","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"user","type":"address"}],"name":"usersIDsLength","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"user","type":"address"},{"internalType":"uint256","name":"offset","type":"uint256"},{"internalType":"uint256","name":"limit","type":"uint256"}],"name":"usersIDsList","outputs":[{"internalType":"uint256[]","name":"output","type":"uint256[]"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"usersTeam","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address[]","name":"users","type":"address[]"}],"name":"usersTeamList","outputs":[{"internalType":"uint256[]","name":"output","type":"uint256[]"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"waves","outputs":[{"internalType":"bytes32","name":"root","type":"bytes32"},{"internalType":"uint256","name":"start","type":"uint256"},{"internalType":"uint256","name":"end","type":"uint256"},{"components":[{"internalType":"uint256","name":"bronze","type":"uint256"},{"internalType":"uint256","name":"silver","type":"uint256"},{"internalType":"uint256","name":"gold","type":"uint256"}],"internalType":"struct IFoxifyAffiliation.LevelsDistribution","name":"distribution","type":"tuple"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"wavesLength","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"offset","type":"uint256"},{"internalType":"uint256","name":"limit","type":"uint256"}],"name":"wavesList","outputs":[{"components":[{"internalType":"bytes32","name":"root","type":"bytes32"},{"internalType":"uint256","name":"start","type":"uint256"},{"internalType":"uint256","name":"end","type":"uint256"},{"components":[{"internalType":"uint256","name":"bronze","type":"uint256"},{"internalType":"uint256","name":"silver","type":"uint256"},{"internalType":"uint256","name":"gold","type":"uint256"}],"internalType":"struct IFoxifyAffiliation.LevelsDistribution","name":"distribution","type":"tuple"}],"internalType":"struct IFoxifyAffiliation.Wave[]","name":"output","type":"tuple[]"}],"stateMutability":"view","type":"function"}]