// SPDX-License-Identifier: agpl-3.0
pragma solidity ^0.8.19;
interface IXWallet {
function shareValue() external view returns (uint);
function transfer(address to, uint amount) external returns (bool);
function transferFrom(address from, address to, uint amount) external returns (bool);
}
contract stkWALLET {
// ERC20 stuff
// Constants
string public constant name = "Staked $WALLET";
uint8 public constant decimals = 18;
string public constant symbol = "stkWALLET";
// Immutables
IXWallet xWallet;
// Mutable variables
uint public totalSupply;
mapping(address => uint) private shares;
mapping(address => mapping(address => uint)) private allowed;
// ERC20 events
event Approval(address indexed owner, address indexed spender, uint amount);
event Transfer(address indexed from, address indexed to, uint amount);
// ERC20 methods
function balanceOf(address owner) external view returns (uint balance) {
return (shares[owner] * xWallet.shareValue()) / 1e18;
}
function transfer(address to, uint amount) external returns (bool success) {
require(to != address(this), "BAD_ADDRESS");
uint sharesAmount = (amount * 1e18) / xWallet.shareValue();
shares[msg.sender] = shares[msg.sender] - sharesAmount;
shares[to] = shares[to] + sharesAmount;
emit Transfer(msg.sender, to, amount);
return true;
}
function transferFrom(address from, address to, uint amount) external returns (bool success) {
uint sharesAmount = (amount * 1e18) / xWallet.shareValue();
shares[from] = shares[from] - sharesAmount;
allowed[from][msg.sender] = allowed[from][msg.sender] - amount;
shares[to] = shares[to] + sharesAmount;
emit Transfer(from, to, amount);
return true;
}
function approve(address spender, uint amount) external returns (bool success) {
allowed[msg.sender][spender] = amount;
emit Approval(msg.sender, spender, amount);
return true;
}
function allowance(address owner, address spender) external view returns (uint remaining) {
return allowed[owner][spender];
}
constructor(IXWallet token) {
xWallet = token;
}
// enter with $WALLLET
// no need for this atm, we can just use wrap/unwrap
// function stake(uint amount) external {}
// @TODO wrapAll?
// convert xWALLET to stkWALLET
function wrap(uint shareAmount) external {
shares[msg.sender] += shareAmount;
require(xWallet.transferFrom(msg.sender, address(this), shareAmount));
emit Transfer(address(0), msg.sender, (shareAmount * xWallet.shareValue()) / 1e18);
}
// this is used to trigger unstaking
function unwrap(uint shareAmount) external {
shares[msg.sender] -= shareAmount;
require(xWallet.transfer(msg.sender, shareAmount));
emit Transfer(msg.sender, address(0), (shareAmount * xWallet.shareValue()) / 1e18);
}
}
{
"compilationTarget": {
"stkWALLET.sol": "stkWALLET"
},
"evmVersion": "paris",
"libraries": {},
"metadata": {
"bytecodeHash": "ipfs"
},
"optimizer": {
"enabled": false,
"runs": 200
},
"remappings": []
}
[{"inputs":[{"internalType":"contract IXWallet","name":"token","type":"address"}],"stateMutability":"nonpayable","type":"constructor"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"owner","type":"address"},{"indexed":true,"internalType":"address","name":"spender","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"Approval","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"from","type":"address"},{"indexed":true,"internalType":"address","name":"to","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"Transfer","type":"event"},{"inputs":[{"internalType":"address","name":"owner","type":"address"},{"internalType":"address","name":"spender","type":"address"}],"name":"allowance","outputs":[{"internalType":"uint256","name":"remaining","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"spender","type":"address"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"approve","outputs":[{"internalType":"bool","name":"success","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"}],"name":"balanceOf","outputs":[{"internalType":"uint256","name":"balance","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"decimals","outputs":[{"internalType":"uint8","name":"","type":"uint8"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"name","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"symbol","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"totalSupply","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"transfer","outputs":[{"internalType":"bool","name":"success","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"from","type":"address"},{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"transferFrom","outputs":[{"internalType":"bool","name":"success","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"shareAmount","type":"uint256"}],"name":"unwrap","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"shareAmount","type":"uint256"}],"name":"wrap","outputs":[],"stateMutability":"nonpayable","type":"function"}]