在 MetaMask 中显示自定义 ERC20 函数的交易值

Displaying the transaction value of a custom ERC20 function in MetaMask

我在基于 ERC20 的 Polygon 上创建了一个新合约,其中包含一些额外的 public 功能。我在向 MetaMask 传达用户正在做什么时遇到了麻烦。

当我执行正常的 ERC20 transfer() 交易时,签名弹出窗口正确显示了 myNewToken 中交易的价值加上 MATIC gas。但是,如果我使用新的 commissionTransfer() 交易,根据合同,该交易将一部分付款发送给收款人,另一部分作为佣金发送给“房子”,签名弹出窗口不会显示交易的价值,只是汽油费。

如果用户签署交易,它会顺利进行,正确数量的代币将发送到正确的地址,但我真的需要用户能够看到他们正在签署的内容。它显示我是否在交易中添加了“价值:金额”,但这会将交易变成了 MATIC 转账,而不是我的代币。

这就是我执行 commissionTransfer() 的方式。

const tx = {
   from: userAddress,
   to: contractAddress,
   data: contract.methods.commissionTransfer(payeeAddress, 
         totalTransactionValue).encodeABI()
}
const sentTx = await web3.eth.sendTransaction(tx)

所以。我的问题是,MetaMask 从哪里获得它显示的交易价值?是来自交易对象吗?是来自合同中的 Transfer 事件(我在交易中两次调用 _transfer() 是否会导致问题)?有没有办法指示 MetaMask 需要向用户显示哪个值?

如果有帮助,这里是智能合约方法。佣金值(百分之一的十分之一)是全局的,由另一种方法设置。

function commissionTransfer(address recipient, uint256 amount) public virtual returns (bool) {
    address sender = _msgSender();
    uint256 senderBalance = balanceOf(sender);
    require(senderBalance >= amount, "ERC20: transfer amount exceeds balance");
    uint256 payAmount = (amount / 1000) * (1000 - commission);
    uint256 comAmount = amount - payAmount;
    _transfer(sender, recipient, payAmount);
    _transfer(sender, theHouse, comAmount);
    return true;
}

使用的解决方案:

根据下面 NuMa 的回复,MetaMask 似乎使用 transfer() 函数的签名来确定它是否被使用。这是从名称(“transfer”)和参数类型(地址,unit256)派生而来的,因此任何更改参数的尝试都无法在 MetaMask 上显示。

解决此问题的一种方法是从我的应用程序内的金额中删除 1 wei,并让合同对交易价值收取佣金(如果(金额 % 100000)== 99999)。这非常有效,但对用户在签署交易时看到所有这些 9。

我可能会采用的方法是创建一个新合约,其唯一的工作是与我的令牌交互并从它自己的 transfer() 函数调用 commissionTransfer() 函数。这样,在我的应用程序中,我可以控制调用哪个合约是否应用佣金。佣金合约代码为:

interface IToken {
    function commissionTransfer(address, address, uint256) external returns (bool);
}

contract tokenCommission {
    address tokenAddress=0xetc;

    function transfer(address recipient, uint256 amount) public returns (bool) {
        return IToken(tokenAddress).commissionTransfer(msg.sender, recipient, amount);
    }

    function decimals() public view virtual returns (uint8) {
        return 18;
    }
}

我还必须更改我的 commissionTransfer() 函数以包含发件人地址并验证 msg.sender 是新的佣金合同。

MetaMask 似乎使用 decimals() 函数来显示令牌的正确顺序,因此这也是必需的。完成后,MetaMask 会将 transfer() 函数识别为来自 ERC20 合约。

如果能找到更优雅的解决方案那就太好了。

首先,向交易发送一个值不会改变任何东西,因为正如你所说,该值在以太币中(在本例中为 Matic)。

Metamask 通过函数名称识别函数调用。因此,如果您尝试调用 approve() 方法,您会注意到元掩码弹出窗口与普通的 transfer() 调用有很大不同。

所以如果你想让它显示你想要的,你需要把你所有的逻辑都放在transfer()函数中。这样,当您调用 transfer() 方法时,它将显示包含您想要的信息的弹出窗口。事实上,你可以让你的 comissionTransfer() 函数被调用为 transfer(),并覆盖 transfer() 函数,一切都会如你所愿。

希望此信息对您有所帮助:)