Created
May 15, 2022 06:28
-
-
Save zubin-madon/1148e2934956e59bf7c98504f0dac6ea to your computer and use it in GitHub Desktop.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
// SPDX-License-Identifier: MIT | |
pragma solidity ^0.8.0; | |
import "@openzeppelin/contracts/token/ERC20/ERC20.sol"; | |
import "@openzeppelin/contracts/utils/Context.sol"; | |
import "@openzeppelin/contracts/token/ERC20/IERC20.sol"; | |
import "@openzeppelin/contracts/access/Ownable.sol"; | |
import "@openzeppelin/contracts/utils/math/SafeMath.sol"; | |
import "@openzeppelin/contracts/utils/Address.sol"; | |
contract ECHIGO is Context, IERC20, Ownable { | |
using SafeMath for uint256; | |
using Address for address; | |
mapping (address => uint256) private _rOwned; | |
mapping (address => uint256) private _tOwned; | |
mapping (address => mapping (address => uint256)) private _allowances; | |
mapping (address => bool) private _isExcluded; | |
address[] private _excluded; | |
uint256 private constant MAX = ~uint256(0); | |
uint256 private constant _tTotal = 10_000_000_000e18; | |
uint256 private _rTotal = (MAX - (MAX % _tTotal)); | |
uint256 private _tFeeTotal; | |
string private _name = 'ECHIGO'; | |
string private _symbol = 'ECHIGO'; | |
uint8 private _decimals = 18; | |
uint256 public tradeLimit; | |
constructor () { | |
_rOwned[_msgSender()] = _rTotal; | |
tradeLimit = 0; | |
emit Transfer(address(0), _msgSender(), _tTotal); | |
} | |
function name() public view returns (string memory) { | |
return _name; | |
} | |
function symbol() public view returns (string memory) { | |
return _symbol; | |
} | |
function decimals() public view returns (uint8) { | |
return _decimals; | |
} | |
function totalSupply() public pure override returns (uint256) { | |
return _tTotal; | |
} | |
function balanceOf(address account) public view override returns (uint256) { | |
if (_isExcluded[account]) return _tOwned[account]; | |
return tokenFromReflection(_rOwned[account]); | |
} | |
function transfer(address recipient, uint256 amount) public override returns (bool) { | |
_transfer(_msgSender(), recipient, amount); | |
return true; | |
} | |
function allowance(address owner, address spender) public view override returns (uint256) { | |
return _allowances[owner][spender]; | |
} | |
function approve(address spender, uint256 amount) public override returns (bool) { | |
_approve(_msgSender(), spender, amount); | |
return true; | |
} | |
function transferFrom(address sender, address recipient, uint256 amount) public override returns (bool) { | |
_transfer(sender, recipient, amount); | |
_approve(sender, _msgSender(), _allowances[sender][_msgSender()].sub(amount, "ECHIGO::transferFrom: transfer amount exceeds allowance")); | |
return true; | |
} | |
function increaseAllowance(address spender, uint256 addedValue) public virtual returns (bool) { | |
_approve(_msgSender(), spender, _allowances[_msgSender()][spender].add(addedValue)); | |
return true; | |
} | |
function decreaseAllowance(address spender, uint256 subtractedValue) public virtual returns (bool) { | |
_approve(_msgSender(), spender, _allowances[_msgSender()][spender].sub(subtractedValue, "ECHIGO::decreaseAllowance: decreased allowance below zero")); | |
return true; | |
} | |
function isExcluded(address account) public view returns (bool) { | |
return _isExcluded[account]; | |
} | |
function totalFees() public view returns (uint256) { | |
return _tFeeTotal; | |
} | |
function reflect(uint256 tAmount) public { | |
address sender = _msgSender(); | |
require(!_isExcluded[sender], "ECHIGO::reflect: Excluded addresses cannot call this function"); | |
(uint256 rAmount,,,,) = _getValues(tAmount); | |
_rOwned[sender] = _rOwned[sender].sub(rAmount); | |
_rTotal = _rTotal.sub(rAmount); | |
_tFeeTotal = _tFeeTotal.add(tAmount); | |
} | |
function reflectionFromToken(uint256 tAmount, bool deductTransferFee) public view returns(uint256) { | |
require(tAmount <= _tTotal, "ECHIGO::reflectionFromToken: Amount must be less than supply"); | |
if (!deductTransferFee) { | |
(uint256 rAmount,,,,) = _getValues(tAmount); | |
return rAmount; | |
} else { | |
(,uint256 rTransferAmount,,,) = _getValues(tAmount); | |
return rTransferAmount; | |
} | |
} | |
function tokenFromReflection(uint256 rAmount) public view returns(uint256) { | |
require(rAmount <= _rTotal, "ECHIGO::tokenFromReflection: Amount must be less than total reflections"); | |
uint256 currentRate = _getRate(); | |
return rAmount.div(currentRate); | |
} | |
function excludeAccount(address account) external onlyOwner() { | |
require(!_isExcluded[account], "ECHIGO::excludeAccount: Account is already excluded"); | |
if(_rOwned[account] > 0) { | |
_tOwned[account] = tokenFromReflection(_rOwned[account]); | |
} | |
_isExcluded[account] = true; | |
_excluded.push(account); | |
} | |
function includeAccount(address account) external onlyOwner() { | |
require(_isExcluded[account], "ECHIGO::includeAccount: Account is already included"); | |
for (uint256 i = 0; i < _excluded.length; i++) { | |
if (_excluded[i] == account) { | |
_excluded[i] = _excluded[_excluded.length - 1]; | |
_tOwned[account] = 0; | |
_isExcluded[account] = false; | |
_excluded.pop(); | |
break; | |
} | |
} | |
} | |
function _approve(address owner, address spender, uint256 amount) private { | |
require(owner != address(0), "ECHIGO::_approve: approve from the zero address"); | |
require(spender != address(0), "ECHIGO::_approve: approve to the zero address"); | |
_allowances[owner][spender] = amount; | |
emit Approval(owner, spender, amount); | |
} | |
function setTradeLimit(uint256 amount) external onlyOwner() { | |
require(msg.sender != address(0), "ECHIGO::setTradeLimit: set trade limit from the zero address"); | |
tradeLimit = amount; | |
} | |
function _transfer(address sender, address recipient, uint256 amount) private { | |
require(sender != address(0), "ECHIGO::transfer: transfer from the zero address"); | |
require(recipient != address(0), "ECHIGO::transfer: transfer to the zero address"); | |
require(amount > 0, "ECHIGO::transfer: Transfer amount must be greater than zero"); | |
require(tradeLimit == 0 || balanceOf(recipient).add(amount) <= tradeLimit, "ECHIGO::transfer: balance exceeds trading limit"); | |
if (_isExcluded[sender] && !_isExcluded[recipient]) { | |
_transferFromExcluded(sender, recipient, amount); | |
} else if (!_isExcluded[sender] && _isExcluded[recipient]) { | |
_transferToExcluded(sender, recipient, amount); | |
} else if (!_isExcluded[sender] && !_isExcluded[recipient]) { | |
_transferStandard(sender, recipient, amount); | |
} else if (_isExcluded[sender] && _isExcluded[recipient]) { | |
_transferBothExcluded(sender, recipient, amount); | |
} else { | |
_transferStandard(sender, recipient, amount); | |
} | |
} | |
function _transferStandard(address sender, address recipient, uint256 tAmount) private { | |
(uint256 rAmount, uint256 rTransferAmount, uint256 rFee, uint256 tTransferAmount, uint256 tFee) = _getValues(tAmount); | |
_rOwned[sender] = _rOwned[sender].sub(rAmount); | |
_rOwned[recipient] = _rOwned[recipient].add(rTransferAmount); | |
_reflectFee(rFee, tFee); | |
emit Transfer(sender, recipient, tTransferAmount); | |
} | |
function _transferToExcluded(address sender, address recipient, uint256 tAmount) private { | |
(uint256 rAmount, uint256 rTransferAmount, uint256 rFee, uint256 tTransferAmount, uint256 tFee) = _getValues(tAmount); | |
_rOwned[sender] = _rOwned[sender].sub(rAmount); | |
_tOwned[recipient] = _tOwned[recipient].add(tTransferAmount); | |
_rOwned[recipient] = _rOwned[recipient].add(rTransferAmount); | |
_reflectFee(rFee, tFee); | |
emit Transfer(sender, recipient, tTransferAmount); | |
} | |
function _transferFromExcluded(address sender, address recipient, uint256 tAmount) private { | |
(uint256 rAmount, uint256 rTransferAmount, uint256 rFee, uint256 tTransferAmount, uint256 tFee) = _getValues(tAmount); | |
_tOwned[sender] = _tOwned[sender].sub(tAmount); | |
_rOwned[sender] = _rOwned[sender].sub(rAmount); | |
_rOwned[recipient] = _rOwned[recipient].add(rTransferAmount); | |
_reflectFee(rFee, tFee); | |
emit Transfer(sender, recipient, tTransferAmount); | |
} | |
function _transferBothExcluded(address sender, address recipient, uint256 tAmount) private { | |
(uint256 rAmount, uint256 rTransferAmount, uint256 rFee, uint256 tTransferAmount, uint256 tFee) = _getValues(tAmount); | |
_tOwned[sender] = _tOwned[sender].sub(tAmount); | |
_rOwned[sender] = _rOwned[sender].sub(rAmount); | |
_tOwned[recipient] = _tOwned[recipient].add(tTransferAmount); | |
_rOwned[recipient] = _rOwned[recipient].add(rTransferAmount); | |
_reflectFee(rFee, tFee); | |
emit Transfer(sender, recipient, tTransferAmount); | |
} | |
function _reflectFee(uint256 rFee, uint256 tFee) private { | |
_rTotal = _rTotal.sub(rFee); | |
_tFeeTotal = _tFeeTotal.add(tFee); | |
} | |
function _getValues(uint256 tAmount) private view returns (uint256, uint256, uint256, uint256, uint256) { | |
(uint256 tTransferAmount, uint256 tFee) = _getTValues(tAmount); | |
uint256 currentRate = _getRate(); | |
(uint256 rAmount, uint256 rTransferAmount, uint256 rFee) = _getRValues(tAmount, tFee, currentRate); | |
return (rAmount, rTransferAmount, rFee, tTransferAmount, tFee); | |
} | |
function _getTValues(uint256 tAmount) private pure returns (uint256, uint256) { | |
uint256 tFee = tAmount.mul(10).div(100); | |
uint256 tTransferAmount = tAmount.sub(tFee); | |
return (tTransferAmount, tFee); | |
} | |
function _getRValues(uint256 tAmount, uint256 tFee, uint256 currentRate) private pure returns (uint256, uint256, uint256) { | |
uint256 rAmount = tAmount.mul(currentRate); | |
uint256 rFee = tFee.mul(currentRate); | |
uint256 rTransferAmount = rAmount.sub(rFee); | |
return (rAmount, rTransferAmount, rFee); | |
} | |
function _getRate() private view returns(uint256) { | |
(uint256 rSupply, uint256 tSupply) = _getCurrentSupply(); | |
return rSupply.div(tSupply); | |
} | |
function _getCurrentSupply() private view returns(uint256, uint256) { | |
uint256 rSupply = _rTotal; | |
uint256 tSupply = _tTotal; | |
for (uint256 i = 0; i < _excluded.length; i++) { | |
if (_rOwned[_excluded[i]] > rSupply || _tOwned[_excluded[i]] > tSupply) return (_rTotal, _tTotal); | |
rSupply = rSupply.sub(_rOwned[_excluded[i]]); | |
tSupply = tSupply.sub(_tOwned[_excluded[i]]); | |
} | |
if (rSupply < _rTotal.div(_tTotal)) return (_rTotal, _tTotal); | |
return (rSupply, tSupply); | |
} | |
} |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment