编译器
0.8.12+commit.f00d7308
文件 1 的 10: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;
}
}
文件 2 的 10:ERC1155Holder.sol
pragma solidity ^0.8.0;
import "./ERC1155Receiver.sol";
contract ERC1155Holder is ERC1155Receiver {
function onERC1155Received(
address,
address,
uint256,
uint256,
bytes memory
) public virtual override returns (bytes4) {
return this.onERC1155Received.selector;
}
function onERC1155BatchReceived(
address,
address,
uint256[] memory,
uint256[] memory,
bytes memory
) public virtual override returns (bytes4) {
return this.onERC1155BatchReceived.selector;
}
}
文件 3 的 10:ERC1155Receiver.sol
pragma solidity ^0.8.0;
import "../IERC1155Receiver.sol";
import "../../../utils/introspection/ERC165.sol";
abstract contract ERC1155Receiver is ERC165, IERC1155Receiver {
function supportsInterface(bytes4 interfaceId) public view virtual override(ERC165, IERC165) returns (bool) {
return interfaceId == type(IERC1155Receiver).interfaceId || super.supportsInterface(interfaceId);
}
}
文件 4 的 10: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 的 10:IERC1155.sol
pragma solidity ^0.8.0;
import "../../utils/introspection/IERC165.sol";
interface IERC1155 is IERC165 {
event TransferSingle(address indexed operator, address indexed from, address indexed to, uint256 id, uint256 value);
event TransferBatch(
address indexed operator,
address indexed from,
address indexed to,
uint256[] ids,
uint256[] values
);
event ApprovalForAll(address indexed account, address indexed operator, bool approved);
event URI(string value, uint256 indexed id);
function balanceOf(address account, uint256 id) external view returns (uint256);
function balanceOfBatch(address[] calldata accounts, uint256[] calldata ids)
external
view
returns (uint256[] memory);
function setApprovalForAll(address operator, bool approved) external;
function isApprovedForAll(address account, address operator) external view returns (bool);
function safeTransferFrom(
address from,
address to,
uint256 id,
uint256 amount,
bytes calldata data
) external;
function safeBatchTransferFrom(
address from,
address to,
uint256[] calldata ids,
uint256[] calldata amounts,
bytes calldata data
) external;
}
文件 6 的 10:IERC1155Receiver.sol
pragma solidity ^0.8.0;
import "../../utils/introspection/IERC165.sol";
interface IERC1155Receiver is IERC165 {
function onERC1155Received(
address operator,
address from,
uint256 id,
uint256 value,
bytes calldata data
) external returns (bytes4);
function onERC1155BatchReceived(
address operator,
address from,
uint256[] calldata ids,
uint256[] calldata values,
bytes calldata data
) external returns (bytes4);
}
文件 7 的 10:IERC165.sol
pragma solidity ^0.8.0;
interface IERC165 {
function supportsInterface(bytes4 interfaceId) external view returns (bool);
}
文件 8 的 10:IERC721.sol
pragma solidity ^0.8.0;
import "../../utils/introspection/IERC165.sol";
interface IERC721 is IERC165 {
event Transfer(address indexed from, address indexed to, uint256 indexed tokenId);
event Approval(address indexed owner, address indexed approved, uint256 indexed tokenId);
event ApprovalForAll(address indexed owner, address indexed operator, bool approved);
function balanceOf(address owner) external view returns (uint256 balance);
function ownerOf(uint256 tokenId) external view returns (address owner);
function safeTransferFrom(
address from,
address to,
uint256 tokenId
) external;
function transferFrom(
address from,
address to,
uint256 tokenId
) external;
function approve(address to, uint256 tokenId) external;
function getApproved(uint256 tokenId) external view returns (address operator);
function setApprovalForAll(address operator, bool _approved) external;
function isApprovedForAll(address owner, address operator) external view returns (bool);
function safeTransferFrom(
address from,
address to,
uint256 tokenId,
bytes calldata data
) external;
}
文件 9 的 10: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());
}
function owner() public view virtual returns (address) {
return _owner;
}
modifier onlyOwner() {
require(owner() == _msgSender(), "Ownable: caller is not the owner");
_;
}
function renounceOwnership() public virtual onlyOwner {
_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);
}
}
文件 10 的 10:Staking.sol
pragma solidity ^0.8.12;
import "@openzeppelin/contracts/access/Ownable.sol";
import "@openzeppelin/contracts/utils/Context.sol";
import "@openzeppelin/contracts/token/ERC721/IERC721.sol";
import "@openzeppelin/contracts/token/ERC1155/IERC1155.sol";
import "@openzeppelin/contracts/token/ERC1155/utils/ERC1155Holder.sol";
contract Staking is Ownable, ERC1155Holder {
IERC721 private landContract;
IERC721 private avatarContract = IERC721(0x31eAa2E93D7AFd237F87F30c0Dbd3aDEB9934f1B);
IERC1155 private furnitureContract = IERC1155(0xb644476e44A797Db3B8a6A16f2e63e8D5a541b67);
mapping (bytes32 => bool) public proofs;
mapping (address => uint256) public tickets;
uint256 public totalTickets;
bool public stakingEnabled = true;
bool public unstakingEnabled = false;
event Stake(address indexed userAddress, uint256[] landIds, uint256[] avatarIds, uint256[] furnitureIds, uint256[] furnitureAmounts, uint256 blockNumber);
event Unstake(address indexed userAddress, uint256[] landIds, uint256[] avatarIds, uint256[] furnitureIds, uint256[] furnitureAmounts, uint256 blockNumber);
constructor() {}
function transferERC721(IERC721 contractERC721, address from, address to, uint256[] calldata ids) internal {
uint256 length = ids.length;
for(uint256 i = 0; i < length;) {
contractERC721.transferFrom(from, to, ids[i]);
unchecked {
++i;
}
}
}
function stake(uint256[] calldata landIds, uint256[] calldata avatarIds, uint256[] calldata furnitureIds, uint256[] calldata furnitureAmounts) external {
require(stakingEnabled, "Staking is disabled");
bytes32 proof = keccak256(abi.encode(_msgSender(), landIds, avatarIds, furnitureIds, furnitureAmounts, block.number));
require(!proofs[proof], "Proof already exists");
proofs[proof] = true;
transferERC721(landContract, _msgSender(), address(this), landIds);
transferERC721(avatarContract, _msgSender(), address(this), avatarIds);
uint256 earnedTickets;
unchecked {
earnedTickets = landIds.length + avatarIds.length;
}
uint256 furnitureLength = furnitureIds.length;
if(furnitureLength > 0){
furnitureContract.safeBatchTransferFrom(_msgSender(), address(this), furnitureIds, furnitureAmounts, "");
for(uint256 i = 0; i < furnitureLength;) {
require(furnitureIds[i] < 4, "Only mega-rare furnitures");
unchecked {
earnedTickets += furnitureAmounts[i++];
}
}
}
require(earnedTickets > 0, "No NFTs selected to stake");
unchecked {
tickets[_msgSender()] += earnedTickets;
totalTickets += earnedTickets;
}
emit Stake(_msgSender(), landIds, avatarIds, furnitureIds, furnitureAmounts, block.number);
}
function unstake(uint256[] calldata landIds, uint256[] calldata avatarIds, uint256[] calldata furnitureIds, uint256[] calldata furnitureAmounts, uint256 blockNumber) external {
require(unstakingEnabled, "Unstaking is disabled");
bytes32 proof = keccak256(abi.encode(_msgSender(), landIds, avatarIds, furnitureIds, furnitureAmounts, blockNumber));
require(proofs[proof], "Proof does not exist");
delete proofs[proof];
transferERC721(landContract, address(this), _msgSender(), landIds);
transferERC721(avatarContract, address(this), _msgSender(), avatarIds);
uint256 earnedTickets;
unchecked {
earnedTickets = landIds.length + avatarIds.length;
}
uint256 furnitureLength = furnitureIds.length;
if(furnitureLength > 0){
furnitureContract.safeBatchTransferFrom(address(this), _msgSender(), furnitureIds, furnitureAmounts, "");
for(uint256 i = 0; i < furnitureLength;) {
unchecked {
earnedTickets += furnitureAmounts[i++];
}
}
}
unchecked {
tickets[_msgSender()] -= earnedTickets;
totalTickets -= earnedTickets;
}
emit Unstake(_msgSender(), landIds, avatarIds, furnitureIds, furnitureAmounts, blockNumber);
}
function enableStaking() external onlyOwner {
require(!unstakingEnabled, "First disable unstaking");
stakingEnabled = true;
}
function disableStaking() external onlyOwner {
stakingEnabled = false;
}
function enableUnstaking() external onlyOwner {
require(!stakingEnabled, "First disable staking");
unstakingEnabled = true;
}
function disableUnstaking() external onlyOwner {
unstakingEnabled = false;
}
function setLandAddress(address landAddress) external onlyOwner {
landContract = IERC721(landAddress);
}
}
{
"compilationTarget": {
"Staking.sol": "Staking"
},
"evmVersion": "london",
"libraries": {},
"metadata": {
"bytecodeHash": "ipfs"
},
"optimizer": {
"enabled": true,
"runs": 200
},
"remappings": []
}
[{"inputs":[],"stateMutability":"nonpayable","type":"constructor"},{"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":"userAddress","type":"address"},{"indexed":false,"internalType":"uint256[]","name":"landIds","type":"uint256[]"},{"indexed":false,"internalType":"uint256[]","name":"avatarIds","type":"uint256[]"},{"indexed":false,"internalType":"uint256[]","name":"furnitureIds","type":"uint256[]"},{"indexed":false,"internalType":"uint256[]","name":"furnitureAmounts","type":"uint256[]"},{"indexed":false,"internalType":"uint256","name":"blockNumber","type":"uint256"}],"name":"Stake","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"userAddress","type":"address"},{"indexed":false,"internalType":"uint256[]","name":"landIds","type":"uint256[]"},{"indexed":false,"internalType":"uint256[]","name":"avatarIds","type":"uint256[]"},{"indexed":false,"internalType":"uint256[]","name":"furnitureIds","type":"uint256[]"},{"indexed":false,"internalType":"uint256[]","name":"furnitureAmounts","type":"uint256[]"},{"indexed":false,"internalType":"uint256","name":"blockNumber","type":"uint256"}],"name":"Unstake","type":"event"},{"inputs":[],"name":"disableStaking","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"disableUnstaking","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"enableStaking","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"enableUnstaking","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"},{"internalType":"address","name":"","type":"address"},{"internalType":"uint256[]","name":"","type":"uint256[]"},{"internalType":"uint256[]","name":"","type":"uint256[]"},{"internalType":"bytes","name":"","type":"bytes"}],"name":"onERC1155BatchReceived","outputs":[{"internalType":"bytes4","name":"","type":"bytes4"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"},{"internalType":"address","name":"","type":"address"},{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"bytes","name":"","type":"bytes"}],"name":"onERC1155Received","outputs":[{"internalType":"bytes4","name":"","type":"bytes4"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"bytes32","name":"","type":"bytes32"}],"name":"proofs","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"landAddress","type":"address"}],"name":"setLandAddress","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256[]","name":"landIds","type":"uint256[]"},{"internalType":"uint256[]","name":"avatarIds","type":"uint256[]"},{"internalType":"uint256[]","name":"furnitureIds","type":"uint256[]"},{"internalType":"uint256[]","name":"furnitureAmounts","type":"uint256[]"}],"name":"stake","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"stakingEnabled","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"bytes4","name":"interfaceId","type":"bytes4"}],"name":"supportsInterface","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"tickets","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"totalTickets","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256[]","name":"landIds","type":"uint256[]"},{"internalType":"uint256[]","name":"avatarIds","type":"uint256[]"},{"internalType":"uint256[]","name":"furnitureIds","type":"uint256[]"},{"internalType":"uint256[]","name":"furnitureAmounts","type":"uint256[]"},{"internalType":"uint256","name":"blockNumber","type":"uint256"}],"name":"unstake","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"unstakingEnabled","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"}]