167 lines
5.7 KiB
Solidity
167 lines
5.7 KiB
Solidity
/*
|
|
|
|
Copyright 2020 DODO ZOO.
|
|
SPDX-License-Identifier: Apache-2.0
|
|
|
|
*/
|
|
|
|
pragma solidity 0.6.9;
|
|
|
|
import {SafeMath} from "../../lib/SafeMath.sol";
|
|
import {InitializableOwnable} from "../../lib/InitializableOwnable.sol";
|
|
import {IERC721} from "../../intf/IERC721.sol";
|
|
import {IERC721Receiver} from "../../intf/IERC721Receiver.sol";
|
|
import {IERC1155} from "../../intf/IERC1155.sol";
|
|
import {IERC1155Receiver} from "../../intf/IERC1155Receiver.sol";
|
|
|
|
|
|
contract NFTCollateralVault is InitializableOwnable, IERC721Receiver, IERC1155Receiver {
|
|
using SafeMath for uint256;
|
|
|
|
// ============ Storage ============
|
|
string public name;
|
|
struct NftInfo {
|
|
uint256 tokenId;
|
|
uint256 amount;
|
|
address nftContract;
|
|
}
|
|
NftInfo[] public nftInfos;
|
|
|
|
constructor (string memory _name) public {
|
|
name = _name;
|
|
}
|
|
|
|
// ============ Event ============
|
|
event RemoveNftToken(address nftContract, uint256 tokenId, uint256 amount);
|
|
event AddNftToken(address nftContract, uint256 tokenId, uint256 amount);
|
|
|
|
|
|
// ============ Ownable ============
|
|
function directTransferOwnership(address newOwner) external onlyOwner {
|
|
_OWNER_ = newOwner;
|
|
emit OwnershipTransferred(_OWNER_, newOwner);
|
|
}
|
|
|
|
function createFragment(address nftProxy, bytes calldata data) external onlyOwner {
|
|
require(nftProxy != address(0), "DODONftVault: PROXY_INVALID");
|
|
_OWNER_ = nftProxy;
|
|
emit OwnershipTransferred(_OWNER_, nftProxy);
|
|
(bool success, ) = nftProxy.call(data);
|
|
require(success, "DODONftVault: TRANSFER_OWNER_FAILED");
|
|
}
|
|
|
|
function withdrawERC721(address nftContract, uint256 tokenId) external onlyOwner {
|
|
_removeNftInfo(nftContract, tokenId, 1);
|
|
IERC721(nftContract).safeTransferFrom(address(this), _OWNER_, tokenId, "");
|
|
}
|
|
|
|
function withdrawERC1155(address nftContract, uint256[] memory tokenIds, uint256[] memory amounts) external onlyOwner {
|
|
require(tokenIds.length == amounts.length, "PARAMS_NOT_MATCH");
|
|
for(uint256 i = 0; i < tokenIds.length; i++) {
|
|
_removeNftInfo(nftContract, tokenIds[i], amounts[i]);
|
|
}
|
|
IERC1155(nftContract).safeBatchTransferFrom(address(this), _OWNER_, tokenIds, amounts, "");
|
|
}
|
|
|
|
// ============ View ============
|
|
|
|
function getNftInfoById(uint256 i) external view returns (address nftContract, uint256 tokenId, uint256 amount) {
|
|
require(i < nftInfos.length, "ID_OVERFLOW");
|
|
NftInfo memory ni = nftInfos[i];
|
|
nftContract = ni.nftContract;
|
|
tokenId = ni.tokenId;
|
|
amount = ni.amount;
|
|
}
|
|
|
|
function getIdByTokenIdAndAddr(address nftContract, uint256 tokenId) external view returns(uint256) {
|
|
uint256 len = nftInfos.length;
|
|
for (uint256 i = 0; i < len; i++) {
|
|
if (nftContract == nftInfos[i].nftContract && tokenId == nftInfos[i].tokenId) {
|
|
return i;
|
|
}
|
|
}
|
|
require(false, "TOKEN_ID_NOT_FOUND");
|
|
}
|
|
|
|
function supportsInterface(bytes4 interfaceId) public override view returns (bool) {
|
|
return interfaceId == type(IERC1155Receiver).interfaceId
|
|
|| interfaceId == type(IERC721Receiver).interfaceId;
|
|
}
|
|
|
|
// ============ Callback ============
|
|
function onERC721Received(
|
|
address,
|
|
address,
|
|
uint256 tokenId,
|
|
bytes calldata
|
|
) external override returns (bytes4) {
|
|
_addNftInfo(msg.sender, tokenId, 1);
|
|
return IERC721Receiver.onERC721Received.selector;
|
|
}
|
|
|
|
function onERC1155Received(
|
|
address,
|
|
address,
|
|
uint256 id,
|
|
uint256 value,
|
|
bytes calldata
|
|
) external override returns (bytes4){
|
|
_addNftInfo(msg.sender, id, value);
|
|
return IERC1155Receiver.onERC1155Received.selector;
|
|
}
|
|
|
|
function onERC1155BatchReceived(
|
|
address,
|
|
address,
|
|
uint256[] calldata ids,
|
|
uint256[] calldata values,
|
|
bytes calldata
|
|
) external override returns (bytes4){
|
|
require(ids.length == values.length, "PARAMS_NOT_MATCH");
|
|
for(uint256 i = 0; i < ids.length; i++) {
|
|
_addNftInfo(msg.sender, ids[i], values[i]);
|
|
}
|
|
return IERC1155Receiver.onERC1155BatchReceived.selector;
|
|
}
|
|
|
|
// ========== internal ===============
|
|
function _addNftInfo(address nftContract, uint256 tokenId, uint256 addAmount) internal {
|
|
uint256 len = nftInfos.length;
|
|
for(uint256 i = 0; i < len; i++) {
|
|
NftInfo memory nftInfo = nftInfos[i];
|
|
if (nftContract == nftInfo.nftContract && tokenId == nftInfo.tokenId) {
|
|
nftInfos[i].amount = nftInfo.amount.add(addAmount);
|
|
emit AddNftToken(nftContract, tokenId, addAmount);
|
|
return;
|
|
}
|
|
}
|
|
nftInfos.push(
|
|
NftInfo({
|
|
tokenId: tokenId,
|
|
amount: addAmount,
|
|
nftContract: nftContract
|
|
})
|
|
);
|
|
emit AddNftToken(nftContract, tokenId, addAmount);
|
|
}
|
|
|
|
function _removeNftInfo(address nftContract, uint256 tokenId, uint256 removeAmount) internal {
|
|
uint256 len = nftInfos.length;
|
|
for (uint256 i = 0; i < len; i++) {
|
|
NftInfo memory nftInfo = nftInfos[i];
|
|
if (nftContract == nftInfo.nftContract && tokenId == nftInfo.tokenId) {
|
|
if(removeAmount >= nftInfo.amount) {
|
|
if(i != len - 1) {
|
|
nftInfos[i] = nftInfos[len - 1];
|
|
}
|
|
nftInfos.pop();
|
|
}else {
|
|
nftInfos[i].amount = nftInfo.amount.sub(removeAmount);
|
|
}
|
|
emit RemoveNftToken(nftContract, tokenId, removeAmount);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|