diff --git a/core/src/main/resources/i18n/displayStrings.properties b/core/src/main/resources/i18n/displayStrings.properties
index 5bc1eaee3d5..6e4bf20cf99 100644
--- a/core/src/main/resources/i18n/displayStrings.properties
+++ b/core/src/main/resources/i18n/displayStrings.properties
@@ -192,7 +192,7 @@ shared.tradeWalletBalance=Trade wallet balance
shared.makerTxFee=Maker: {0}
shared.takerTxFee=Taker: {0}
shared.iConfirm=I confirm
-shared.tradingFeeInBsqInfo=equivalent to {0} used as trading fee
+shared.tradingFeeInBsqInfo=≈ {0}
shared.openURL=Open {0}
shared.fiat=Fiat
shared.crypto=Crypto
@@ -454,14 +454,20 @@ createOffer.warning.sellBelowMarketPrice=You will always get {0}% less than the
createOffer.warning.buyAboveMarketPrice=You will always pay {0}% more than the current market price as the price of your offer will be continuously updated.
createOffer.tradeFee.descriptionBTCOnly=Trade fee
createOffer.tradeFee.descriptionBSQEnabled=Select trade fee currency
-createOffer.tradeFee.fiatAndPercent=≈ {0} / {1} of trade amount
# new entries
createOffer.placeOfferButton=Review: Place offer to {0} bitcoin
createOffer.createOfferFundWalletInfo.headline=Fund your offer
# suppress inspection "TrailingSpacesInProperty"
createOffer.createOfferFundWalletInfo.tradeAmount=- Trade amount: {0} \n
-createOffer.createOfferFundWalletInfo.msg=You need to deposit {0} to this offer.\n\nThose funds are reserved in your local wallet and will get locked into the multisig deposit address once someone takes your offer.\n\nThe amount is the sum of:\n{1}- Your security deposit: {2}\n- Trading fee: {3}\n- Mining fee: {4}\n\nYou can choose between two options when funding your trade:\n- Use your Bisq wallet (convenient, but transactions may be linkable) OR\n- Transfer from an external wallet (potentially more private)\n\nYou will see all funding options and details after closing this popup.
+createOffer.createOfferFundWalletInfo.msg=You need to deposit {0} to this offer.\n\n\
+ Those funds are reserved in your local wallet and will get locked into the multisig deposit address once someone takes your offer.\n\n\
+ The amount is the sum of:\n\
+ {1}\
+ - Your security deposit: {2}\n\
+ - Trading fee: {3}\n\
+ - Mining fee: {4}\n\n\
+ You can choose between two options when funding your trade:\n- Use your Bisq wallet (convenient, but transactions may be linkable) OR\n- Transfer from an external wallet (potentially more private)\n\nYou will see all funding options and details after closing this popup.
# only first part "An error occurred when placing the offer:" has been used before. We added now the rest (need update in existing translations!)
createOffer.amountPriceBox.error.message=An error occurred when placing the offer:\n\n{0}\n\n\
@@ -2700,6 +2706,8 @@ feeOptionWindow.info=You can choose to pay the trade fee in BSQ or in BTC. If yo
feeOptionWindow.optionsLabel=Choose currency for trade fee payment
feeOptionWindow.useBTC=Use BTC
feeOptionWindow.fee={0} (≈ {1})
+feeOptionWindow.btcFeeWithFiatAndPercentage={0} (≈ {1} / {2})
+feeOptionWindow.btcFeeWithPercentage={0} ({1})
####################################################################
diff --git a/desktop/src/main/java/bisq/desktop/main/offer/FeeUtil.java b/desktop/src/main/java/bisq/desktop/main/offer/FeeUtil.java
new file mode 100644
index 00000000000..82f2363e5ed
--- /dev/null
+++ b/desktop/src/main/java/bisq/desktop/main/offer/FeeUtil.java
@@ -0,0 +1,82 @@
+/*
+ * This file is part of Bisq.
+ *
+ * Bisq is free software: you can redistribute it and/or modify it
+ * under the terms of the GNU Affero General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or (at
+ * your option) any later version.
+ *
+ * Bisq is distributed in the hope that it will be useful, but WITHOUT
+ * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
+ * FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero General Public
+ * License for more details.
+ *
+ * You should have received a copy of the GNU Affero General Public License
+ * along with Bisq. If not, see .
+ */
+
+package bisq.desktop.main.offer;
+
+import bisq.desktop.util.DisplayUtils;
+import bisq.desktop.util.GUIUtil;
+
+import bisq.core.locale.Res;
+import bisq.core.monetary.Volume;
+import bisq.core.offer.OfferUtil;
+import bisq.core.util.coin.CoinFormatter;
+
+import bisq.common.app.DevEnv;
+
+import org.bitcoinj.core.Coin;
+
+import java.util.Optional;
+
+public class FeeUtil {
+ public static String getTradeFeeWithFiatEquivalent(OfferUtil offerUtil,
+ Coin tradeFee,
+ boolean isCurrencyForMakerFeeBtc,
+ CoinFormatter formatter) {
+ if (!isCurrencyForMakerFeeBtc && !DevEnv.isDaoActivated()) {
+ return "";
+ }
+
+ Optional optionalBtcFeeInFiat = offerUtil.getFeeInUserFiatCurrency(tradeFee,
+ isCurrencyForMakerFeeBtc,
+ formatter);
+
+ return DisplayUtils.getFeeWithFiatAmount(tradeFee, optionalBtcFeeInFiat, formatter);
+ }
+
+ public static String getTradeFeeWithFiatEquivalentAndPercentage(OfferUtil offerUtil,
+ Coin tradeFee,
+ Coin tradeAmount,
+ boolean isCurrencyForMakerFeeBtc,
+ CoinFormatter formatter,
+ Coin minTradeFee) {
+ if (isCurrencyForMakerFeeBtc) {
+ String feeAsBtc = formatter.formatCoinWithCode(tradeFee);
+ String percentage;
+ if (!tradeFee.isGreaterThan(minTradeFee)) {
+ percentage = Res.get("guiUtil.requiredMinimum")
+ .replace("(", "")
+ .replace(")", "");
+ } else {
+ percentage = GUIUtil.getPercentage(tradeFee, tradeAmount) +
+ " " + Res.get("guiUtil.ofTradeAmount");
+ }
+ return offerUtil.getFeeInUserFiatCurrency(tradeFee,
+ isCurrencyForMakerFeeBtc,
+ formatter)
+ .map(DisplayUtils::formatAverageVolumeWithCode)
+ .map(feeInFiat -> Res.get("feeOptionWindow.btcFeeWithFiatAndPercentage", feeAsBtc, feeInFiat, percentage))
+ .orElseGet(() -> Res.get("feeOptionWindow.btcFeeWithPercentage", feeAsBtc, percentage));
+ } else {
+ // For BSQ we use the fiat equivalent only. Calculating the % value would be more effort.
+ // We could calculate the BTC value if the BSQ fee and use that...
+ return FeeUtil.getTradeFeeWithFiatEquivalent(offerUtil,
+ tradeFee,
+ false,
+ formatter);
+ }
+ }
+}
diff --git a/desktop/src/main/java/bisq/desktop/main/offer/MutableOfferViewModel.java b/desktop/src/main/java/bisq/desktop/main/offer/MutableOfferViewModel.java
index 83bd3d9f741..0b0566988b1 100644
--- a/desktop/src/main/java/bisq/desktop/main/offer/MutableOfferViewModel.java
+++ b/desktop/src/main/java/bisq/desktop/main/offer/MutableOfferViewModel.java
@@ -87,11 +87,13 @@
import javafx.util.Callback;
-import java.util.Optional;
import java.util.concurrent.TimeUnit;
+import lombok.extern.slf4j.Slf4j;
+
import static javafx.beans.binding.Bindings.createStringBinding;
+@Slf4j
public abstract class MutableOfferViewModel extends ActivatableWithDataModel {
private final BtcValidator btcValidator;
private final BsqValidator bsqValidator;
@@ -489,55 +491,28 @@ private void createListeners() {
}
private void applyMakerFee() {
- Coin makerFeeAsCoin = dataModel.getMakerFee();
- if (makerFeeAsCoin != null) {
- isTradeFeeVisible.setValue(true);
-
- tradeFee.set(getFormatterForMakerFee().formatCoin(makerFeeAsCoin));
-
- Coin makerFeeInBtc = dataModel.getMakerFeeInBtc();
- Optional optionalBtcFeeInFiat = offerUtil.getFeeInUserFiatCurrency(makerFeeInBtc,
- true,
- bsqFormatter);
- String btcFeeWithFiatAmount = DisplayUtils.getFeeWithFiatAmount(makerFeeInBtc, optionalBtcFeeInFiat, btcFormatter);
- if (DevEnv.isDaoActivated()) {
- tradeFeeInBtcWithFiat.set(btcFeeWithFiatAmount);
- } else {
- tradeFeeInBtcWithFiat.set(btcFormatter.formatCoinWithCode(makerFeeAsCoin));
- }
-
- Coin makerFeeInBsq = dataModel.getMakerFeeInBsq();
- Optional optionalBsqFeeInFiat = offerUtil.getFeeInUserFiatCurrency(makerFeeInBsq,
- false,
- bsqFormatter);
- String bsqFeeWithFiatAmount = DisplayUtils.getFeeWithFiatAmount(makerFeeInBsq,
- optionalBsqFeeInFiat,
- bsqFormatter);
- if (DevEnv.isDaoActivated()) {
- tradeFeeInBsqWithFiat.set(bsqFeeWithFiatAmount);
- } else {
- // Before DAO is enabled we show fee as fiat and % in second line
- String feeInFiatAsString;
- if (optionalBtcFeeInFiat != null && optionalBtcFeeInFiat.isPresent()) {
- feeInFiatAsString = DisplayUtils.formatVolumeWithCode(optionalBtcFeeInFiat.get());
- } else {
- feeInFiatAsString = Res.get("shared.na");
- }
-
- double amountAsDouble = (double) dataModel.getAmount().get().value;
- double makerFeeInBtcAsDouble = (double) makerFeeInBtc.value;
- double percent = makerFeeInBtcAsDouble / amountAsDouble;
-
- tradeFeeInBsqWithFiat.set(Res.get("createOffer.tradeFee.fiatAndPercent",
- feeInFiatAsString,
- FormattingUtils.formatToPercentWithSymbol(percent)));
- }
- }
tradeFeeCurrencyCode.set(dataModel.isCurrencyForMakerFeeBtc() ? Res.getBaseCurrencyCode() : "BSQ");
tradeFeeDescription.set(DevEnv.isDaoActivated() ? Res.get("createOffer.tradeFee.descriptionBSQEnabled") :
Res.get("createOffer.tradeFee.descriptionBTCOnly"));
+
+ Coin makerFeeAsCoin = dataModel.getMakerFee();
+ if (makerFeeAsCoin == null) {
+ return;
+ }
+
+ isTradeFeeVisible.setValue(true);
+ tradeFee.set(getFormatterForMakerFee().formatCoin(makerFeeAsCoin));
+ tradeFeeInBtcWithFiat.set(FeeUtil.getTradeFeeWithFiatEquivalent(offerUtil,
+ dataModel.getMakerFeeInBtc(),
+ true,
+ btcFormatter));
+ tradeFeeInBsqWithFiat.set(FeeUtil.getTradeFeeWithFiatEquivalent(offerUtil,
+ dataModel.getMakerFeeInBsq(),
+ false,
+ bsqFormatter));
}
+
private void updateMarketPriceAvailable() {
marketPrice = priceFeedService.getMarketPrice(dataModel.getTradeCurrencyCode().get());
marketPriceAvailableProperty.set(marketPrice == null || !marketPrice.isExternallyProvidedPrice() ? 0 : 1);
@@ -985,15 +960,23 @@ public String getSecurityDepositWithCode() {
return btcFormatter.formatCoinWithCode(dataModel.getSecurityDeposit());
}
+
public String getTradeFee() {
- //TODO use last bisq market price to estimate BSQ val
- final Coin makerFeeAsCoin = dataModel.getMakerFee();
- final String makerFee = getFormatterForMakerFee().formatCoinWithCode(makerFeeAsCoin);
- if (dataModel.isCurrencyForMakerFeeBtc())
- return makerFee + GUIUtil.getPercentageOfTradeAmount(makerFeeAsCoin, dataModel.getAmount().get(),
+ if (dataModel.isCurrencyForMakerFeeBtc()) {
+ return FeeUtil.getTradeFeeWithFiatEquivalentAndPercentage(offerUtil,
+ dataModel.getMakerFeeInBtc(),
+ dataModel.getAmount().get(),
+ true,
+ btcFormatter,
FeeService.getMinMakerFee(dataModel.isCurrencyForMakerFeeBtc()));
- else
- return makerFee + " (" + Res.get("shared.tradingFeeInBsqInfo", btcFormatter.formatCoinWithCode(makerFeeAsCoin)) + ")";
+ } else {
+ // For BSQ we use the fiat equivalent only. Calculating the % value would require to
+ // calculate the BTC value of the BSQ fee and use that...
+ return FeeUtil.getTradeFeeWithFiatEquivalent(offerUtil,
+ dataModel.getMakerFeeInBsq(),
+ false,
+ bsqFormatter);
+ }
}
public String getMakerFeePercentage() {
@@ -1025,10 +1008,13 @@ public String getFundsStructure() {
}
public String getTxFee() {
- Coin txFeeAsCoin = dataModel.getTxFee();
- return btcFormatter.formatCoinWithCode(txFeeAsCoin) +
- GUIUtil.getPercentageOfTradeAmount(txFeeAsCoin, dataModel.getAmount().get(), Coin.ZERO);
-
+ return FeeUtil.getTradeFeeWithFiatEquivalentAndPercentage(offerUtil,
+ dataModel.getTxFee(),
+ dataModel.getAmount().get(),
+ true,
+ btcFormatter,
+ Coin.ZERO
+ );
}
public String getTxFeePercentage() {
diff --git a/desktop/src/main/java/bisq/desktop/main/offer/takeoffer/TakeOfferViewModel.java b/desktop/src/main/java/bisq/desktop/main/offer/takeoffer/TakeOfferViewModel.java
index 0bf3855eb24..e7f7b3b6cc8 100644
--- a/desktop/src/main/java/bisq/desktop/main/offer/takeoffer/TakeOfferViewModel.java
+++ b/desktop/src/main/java/bisq/desktop/main/offer/takeoffer/TakeOfferViewModel.java
@@ -23,6 +23,7 @@
import bisq.desktop.main.MainView;
import bisq.desktop.main.funds.FundsView;
import bisq.desktop.main.funds.deposit.DepositView;
+import bisq.desktop.main.offer.FeeUtil;
import bisq.desktop.main.overlays.popups.Popup;
import bisq.desktop.util.DisplayUtils;
import bisq.desktop.util.GUIUtil;
@@ -33,7 +34,6 @@
import bisq.core.locale.CurrencyUtil;
import bisq.core.locale.Res;
import bisq.core.monetary.Price;
-import bisq.core.monetary.Volume;
import bisq.core.offer.Offer;
import bisq.core.offer.OfferPayload;
import bisq.core.offer.OfferRestrictions;
@@ -76,8 +76,6 @@
import javafx.util.Callback;
-import java.util.Optional;
-
import javax.annotation.Nullable;
import static com.google.common.base.Preconditions.checkNotNull;
@@ -279,50 +277,23 @@ public void setIsCurrencyForTakerFeeBtc(boolean isCurrencyForTakerFeeBtc) {
}
private void applyTakerFee() {
- Coin takerFeeAsCoin = dataModel.getTakerFee();
- if (takerFeeAsCoin != null) {
- isTradeFeeVisible.setValue(true);
-
- tradeFee.set(getFormatterForTakerFee().formatCoin(takerFeeAsCoin));
-
- Coin makerFeeInBtc = dataModel.getTakerFeeInBtc();
- Optional optionalBtcFeeInFiat = offerUtil.getFeeInUserFiatCurrency(makerFeeInBtc,
- true,
- bsqFormatter);
- String btcFeeWithFiatAmount = DisplayUtils.getFeeWithFiatAmount(makerFeeInBtc, optionalBtcFeeInFiat, btcFormatter);
- if (DevEnv.isDaoActivated()) {
- tradeFeeInBtcWithFiat.set(btcFeeWithFiatAmount);
- } else {
- tradeFeeInBtcWithFiat.set(btcFormatter.formatCoinWithCode(takerFeeAsCoin));
- }
-
- Coin makerFeeInBsq = dataModel.getTakerFeeInBsq();
- Optional optionalBsqFeeInFiat = offerUtil.getFeeInUserFiatCurrency(makerFeeInBsq,
- false,
- bsqFormatter);
- String bsqFeeWithFiatAmount = DisplayUtils.getFeeWithFiatAmount(makerFeeInBsq, optionalBsqFeeInFiat, bsqFormatter);
- if (DevEnv.isDaoActivated()) {
- tradeFeeInBsqWithFiat.set(bsqFeeWithFiatAmount);
- } else {
- // Before DAO is enabled we show fee as fiat and % in second line
- String feeInFiatAsString;
- if (optionalBtcFeeInFiat != null && optionalBtcFeeInFiat.isPresent()) {
- feeInFiatAsString = DisplayUtils.formatVolumeWithCode(optionalBtcFeeInFiat.get());
- } else {
- feeInFiatAsString = Res.get("shared.na");
- }
-
- double amountAsLong = (double) dataModel.getAmount().get().value;
- double makerFeeInBtcAsLong = (double) makerFeeInBtc.value;
- double percent = makerFeeInBtcAsLong / amountAsLong;
-
- tradeFeeInBsqWithFiat.set(Res.get("createOffer.tradeFee.fiatAndPercent",
- feeInFiatAsString,
- FormattingUtils.formatToPercentWithSymbol(percent)));
- }
- }
tradeFeeDescription.set(DevEnv.isDaoActivated() ? Res.get("createOffer.tradeFee.descriptionBSQEnabled") :
Res.get("createOffer.tradeFee.descriptionBTCOnly"));
+ Coin takerFeeAsCoin = dataModel.getTakerFee();
+ if (takerFeeAsCoin == null) {
+ return;
+ }
+
+ isTradeFeeVisible.setValue(true);
+ tradeFee.set(getFormatterForTakerFee().formatCoin(takerFeeAsCoin));
+ tradeFeeInBtcWithFiat.set(FeeUtil.getTradeFeeWithFiatEquivalent(offerUtil,
+ dataModel.getTakerFeeInBtc(),
+ true,
+ btcFormatter));
+ tradeFeeInBsqWithFiat.set(FeeUtil.getTradeFeeWithFiatEquivalent(offerUtil,
+ dataModel.getTakerFeeInBsq(),
+ false,
+ bsqFormatter));
}
@@ -706,14 +677,21 @@ public String getSecurityDepositWithCode() {
}
public String getTradeFee() {
- //TODO use last bisq market price to estimate BSQ val
- final Coin takerFeeAsCoin = dataModel.getTakerFee();
- final String takerFee = getFormatterForTakerFee().formatCoinWithCode(takerFeeAsCoin);
- if (dataModel.isCurrencyForTakerFeeBtc())
- return takerFee + GUIUtil.getPercentageOfTradeAmount(takerFeeAsCoin, dataModel.getAmount().get(),
- FeeService.getMinTakerFee(dataModel.isCurrencyForTakerFeeBtc()));
- else
- return takerFee + " (" + Res.get("shared.tradingFeeInBsqInfo", btcFormatter.formatCoinWithCode(takerFeeAsCoin)) + ")";
+ if (dataModel.isCurrencyForTakerFeeBtc()) {
+ return FeeUtil.getTradeFeeWithFiatEquivalentAndPercentage(offerUtil,
+ dataModel.getTakerFeeInBtc(),
+ dataModel.getAmount().get(),
+ true,
+ btcFormatter,
+ FeeService.getMinMakerFee(dataModel.isCurrencyForTakerFeeBtc()));
+ } else {
+ // For BSQ we use the fiat equivalent only. Calculating the % value would require to
+ // calculate the BTC value of the BSQ fee and use that...
+ return FeeUtil.getTradeFeeWithFiatEquivalent(offerUtil,
+ dataModel.getTakerFeeInBsq(),
+ false,
+ bsqFormatter);
+ }
}
public String getTakerFeePercentage() {
@@ -733,11 +711,13 @@ public String getTotalToPayInfo() {
}
public String getTxFee() {
- Coin txFeeAsCoin = dataModel.getTotalTxFee();
- return btcFormatter.formatCoinWithCode(txFeeAsCoin) +
- GUIUtil.getPercentageOfTradeAmount(txFeeAsCoin, dataModel.getAmount().get(),
- Coin.ZERO);
-
+ return FeeUtil.getTradeFeeWithFiatEquivalentAndPercentage(offerUtil,
+ dataModel.getTotalTxFee(),
+ dataModel.getAmount().get(),
+ true,
+ btcFormatter,
+ Coin.ZERO
+ );
}
public String getTxFeePercentage() {
diff --git a/desktop/src/main/java/bisq/desktop/util/DisplayUtils.java b/desktop/src/main/java/bisq/desktop/util/DisplayUtils.java
index 83a0d157de3..f8d144e6fa9 100644
--- a/desktop/src/main/java/bisq/desktop/util/DisplayUtils.java
+++ b/desktop/src/main/java/bisq/desktop/util/DisplayUtils.java
@@ -126,7 +126,7 @@ public static String formatVolumeWithCode(Volume volume) {
return formatVolume(volume, FIAT_VOLUME_FORMAT, true);
}
- static String formatAverageVolumeWithCode(Volume volume) {
+ public static String formatAverageVolumeWithCode(Volume volume) {
return formatVolume(volume, FIAT_VOLUME_FORMAT.minDecimals(2), true);
}
@@ -257,16 +257,16 @@ public static String formatPrice(Price price, Boolean decimalAligned, int maxPla
public static String getFeeWithFiatAmount(Coin makerFeeAsCoin,
Optional optionalFeeInFiat,
CoinFormatter formatter) {
- String fee = makerFeeAsCoin != null ? formatter.formatCoinWithCode(makerFeeAsCoin) : Res.get("shared.na");
- String feeInFiatAsString;
+ String feeInBtc = makerFeeAsCoin != null ? formatter.formatCoinWithCode(makerFeeAsCoin) : Res.get("shared.na");
if (optionalFeeInFiat != null && optionalFeeInFiat.isPresent()) {
- feeInFiatAsString = formatAverageVolumeWithCode(optionalFeeInFiat.get());
+ String feeInFiat = formatAverageVolumeWithCode(optionalFeeInFiat.get());
+ return Res.get("feeOptionWindow.fee", feeInBtc, feeInFiat);
} else {
- feeInFiatAsString = Res.get("shared.na");
+ return feeInBtc;
}
- return Res.get("feeOptionWindow.fee", fee, feeInFiatAsString);
}
+
/**
* Converts to a coin with max. 4 decimal places. Last place gets rounded.
* 0.01234 -> 0.0123