256 lines
7.8 KiB
Solidity
256 lines
7.8 KiB
Solidity
/*
|
|
Copyright 2021 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 {ICloneFactory} from "../../lib/CloneFactory.sol";
|
|
import {ReentrancyGuard} from "../../lib/ReentrancyGuard.sol";
|
|
import {IFilterModel} from "../../NFTPool/intf/IFilterModel.sol";
|
|
import {IFilterAdmin} from "../../NFTPool/intf/IFilterAdmin.sol";
|
|
import {IDODONFTApprove} from "../../intf/IDODONFTApprove.sol";
|
|
import {IERC20} from "../../intf/IERC20.sol";
|
|
import {SafeERC20} from "../../lib/SafeERC20.sol";
|
|
|
|
interface IFilterERC721V1 {
|
|
function init(
|
|
address filterAdmin,
|
|
address nftCollection,
|
|
bool[] memory switches,
|
|
uint256[] memory tokenRanges,
|
|
uint256[] memory nftAmounts,
|
|
uint256[] memory priceRules,
|
|
uint256[] memory spreadIds
|
|
) external;
|
|
}
|
|
|
|
contract DODONFTPoolProxy is ReentrancyGuard, InitializableOwnable {
|
|
using SafeMath for uint256;
|
|
using SafeERC20 for IERC20;
|
|
|
|
// ============ Storage ============
|
|
address constant _ETH_ADDRESS_ = 0xEeeeeEeeeEeEeeEeEeEeeEEEeeeeEeeeeeeeEEeE;
|
|
mapping(uint256 => address) public _FILTER_TEMPLATES_;
|
|
address public _FILTER_ADMIN_TEMPLATE_;
|
|
address public _DEFAULT_MAINTAINER_;
|
|
address public _CONTROLLER_MODEL_;
|
|
address public immutable _CLONE_FACTORY_;
|
|
address public immutable _DODO_NFT_APPROVE_;
|
|
address public immutable _DODO_APPROVE_;
|
|
|
|
|
|
// ============ Event ==============
|
|
event SetFilterTemplate(uint256 idx, address filterTemplate);
|
|
|
|
constructor(
|
|
address cloneFactory,
|
|
address filterAdminTemplate,
|
|
address controllerModel,
|
|
address defaultMaintainer,
|
|
address dodoNftApprove,
|
|
address dodoApprove
|
|
) public {
|
|
_CLONE_FACTORY_ = cloneFactory;
|
|
_FILTER_ADMIN_TEMPLATE_ = filterAdminTemplate;
|
|
_CONTROLLER_MODEL_ = controllerModel;
|
|
_DEFAULT_MAINTAINER_ = defaultMaintainer;
|
|
_DODO_NFT_APPROVE_ = dodoNftApprove;
|
|
_DODO_APPROVE_ = dodoApprove;
|
|
}
|
|
|
|
// ================ NFT In and Out ===================
|
|
function nftIn(
|
|
address filter,
|
|
address nftCollection,
|
|
uint256[] memory tokenIds,
|
|
address to,
|
|
uint256 minMintAmount
|
|
) external {
|
|
for(uint256 i = 0; i < tokenIds.length; i++) {
|
|
require(IFilterModel(filter).isNFTValid(nftCollection,tokenIds[i]), "NOT_REGISTRIED");
|
|
IDODONFTApprove(_DODO_NFT_APPROVE_).claimERC721(nftCollection, msg.sender, filter, tokenIds[i]);
|
|
}
|
|
uint256 received = IFilterModel(filter).ERC721In(tokenIds, to);
|
|
require(received >= minMintAmount, "MINT_AMOUNT_NOT_ENOUGH");
|
|
}
|
|
|
|
function nftTargetOut(
|
|
address filter,
|
|
uint256[] memory indexes,
|
|
address to,
|
|
uint256 maxBurnAmount
|
|
) external {
|
|
uint256 paid = IFilterModel(filter).ERC721TargetOut(indexes, to);
|
|
require(paid <= maxBurnAmount, "BURN_AMOUNT_EXCEED");
|
|
}
|
|
|
|
function nftRandomOut(
|
|
address filter,
|
|
uint256 amount,
|
|
address to,
|
|
uint256 maxBurnAmount
|
|
) external {
|
|
uint256 paid = IFilterModel(filter).ERC721RandomOut(amount, to);
|
|
require(paid <= maxBurnAmount, "BURN_AMOUNT_EXCEED");
|
|
}
|
|
|
|
|
|
// ================== Create NFTPool ===================
|
|
|
|
function createNewNFTPool01(
|
|
uint256 initSupply,
|
|
string memory name,
|
|
string memory symbol,
|
|
uint256 fee,
|
|
address nftCollection,
|
|
bool[] memory switches,
|
|
uint256[] memory tokenRanges,
|
|
uint256[] memory nftAmounts,
|
|
uint256[] memory priceRules,
|
|
uint256[] memory spreadIds
|
|
) external returns(address newFilterAdmin) {
|
|
newFilterAdmin = ICloneFactory(_CLONE_FACTORY_).clone(_FILTER_ADMIN_TEMPLATE_);
|
|
|
|
address filter01 = createFilterERC721V1(
|
|
newFilterAdmin,
|
|
nftCollection,
|
|
switches,
|
|
tokenRanges,
|
|
nftAmounts,
|
|
priceRules,
|
|
spreadIds
|
|
);
|
|
|
|
address[] memory filters = new address[](1);
|
|
filters[0] = filter01;
|
|
|
|
IFilterAdmin(newFilterAdmin).init(
|
|
msg.sender,
|
|
initSupply,
|
|
name,
|
|
symbol,
|
|
fee,
|
|
_CONTROLLER_MODEL_,
|
|
_DEFAULT_MAINTAINER_,
|
|
filters
|
|
);
|
|
}
|
|
|
|
|
|
// ================== Create Filter ===================
|
|
function createFilterERC721V1(
|
|
address filterAdmin,
|
|
address nftCollection,
|
|
bool[] memory switches,
|
|
uint256[] memory tokenRanges,
|
|
uint256[] memory nftAmounts,
|
|
uint256[] memory priceRules,
|
|
uint256[] memory spreadIds
|
|
) public returns(address newFilterERC721V1) {
|
|
//key = 1 => FilterERC721V1
|
|
newFilterERC721V1 = ICloneFactory(_CLONE_FACTORY_).clone(_FILTER_TEMPLATES_[1]);
|
|
IFilterERC721V1(newFilterERC721V1).init(
|
|
filterAdmin,
|
|
nftCollection,
|
|
switches,
|
|
tokenRanges,
|
|
nftAmounts,
|
|
priceRules,
|
|
spreadIds
|
|
);
|
|
}
|
|
|
|
|
|
// ================== NFT ERC20 Swap ======================
|
|
function erc721ToErc20(
|
|
address filterAdmin,
|
|
address filter,
|
|
address nftContract,
|
|
uint256 tokenId,
|
|
address toToken,
|
|
address dodoProxy,
|
|
bytes memory dodoSwapData
|
|
)
|
|
external
|
|
preventReentrant
|
|
{
|
|
IDODONFTApprove(_DODO_NFT_APPROVE_).claimERC721(nftContract, msg.sender, filter, tokenId);
|
|
|
|
uint256[] memory tokenIds = new uint256[](1);
|
|
tokenIds[0] = tokenId;
|
|
|
|
uint256 receivedFragAmount = IFilterModel(filter).ERC721In(tokenIds, address(this));
|
|
|
|
_generalApproveMax(filterAdmin, _DODO_APPROVE_, receivedFragAmount);
|
|
|
|
(bool success, ) = dodoProxy.call(dodoSwapData);
|
|
require(success, "API_SWAP_FAILED");
|
|
|
|
uint256 returnAmount = _generalBalanceOf(toToken, address(this));
|
|
|
|
_generalTransfer(toToken, msg.sender, returnAmount);
|
|
}
|
|
|
|
|
|
//====================== Ownable ========================
|
|
function changeDefaultMaintainer(address newMaintainer) external onlyOwner {
|
|
_DEFAULT_MAINTAINER_ = newMaintainer;
|
|
}
|
|
|
|
function changeFilterAdminTemplate(address newFilterAdminTemplate) external onlyOwner {
|
|
_FILTER_ADMIN_TEMPLATE_ = newFilterAdminTemplate;
|
|
}
|
|
|
|
function changeControllerModel(address newControllerModel) external onlyOwner {
|
|
_CONTROLLER_MODEL_ = newControllerModel;
|
|
}
|
|
|
|
function setFilterTemplate(uint256 idx, address newFilterTemplate) external onlyOwner {
|
|
_FILTER_TEMPLATES_[idx] = newFilterTemplate;
|
|
emit SetFilterTemplate(idx, newFilterTemplate);
|
|
}
|
|
|
|
|
|
//======================= Internal =====================
|
|
function _generalApproveMax(
|
|
address token,
|
|
address to,
|
|
uint256 amount
|
|
) internal {
|
|
uint256 allowance = IERC20(token).allowance(address(this), to);
|
|
if (allowance < amount) {
|
|
if (allowance > 0) {
|
|
IERC20(token).safeApprove(to, 0);
|
|
}
|
|
IERC20(token).safeApprove(to, uint256(-1));
|
|
}
|
|
}
|
|
|
|
function _generalBalanceOf(
|
|
address token,
|
|
address who
|
|
) internal view returns (uint256) {
|
|
if (token == _ETH_ADDRESS_) {
|
|
return who.balance;
|
|
} else {
|
|
return IERC20(token).balanceOf(who);
|
|
}
|
|
}
|
|
|
|
function _generalTransfer(
|
|
address token,
|
|
address payable to,
|
|
uint256 amount
|
|
) internal {
|
|
if (amount > 0) {
|
|
if (token == _ETH_ADDRESS_) {
|
|
to.transfer(amount);
|
|
} else {
|
|
IERC20(token).safeTransfer(to, amount);
|
|
}
|
|
}
|
|
}
|
|
} |