diff --git a/core/src/main/java/haveno/core/locale/CurrencyUtil.java b/core/src/main/java/haveno/core/locale/CurrencyUtil.java index d94fb3a70c..99c65812c2 100644 --- a/core/src/main/java/haveno/core/locale/CurrencyUtil.java +++ b/core/src/main/java/haveno/core/locale/CurrencyUtil.java @@ -81,6 +81,12 @@ public class CurrencyUtil { public static List<TradeCurrency> getAllFiatCurrencies() { return getAllTraditionalCurrencies().stream() + .filter(currency -> CurrencyUtil.isFiatCurrency(currency.getCode())) + .collect(Collectors.toList()); + } + + public static List<TradeCurrency> getAllSortedFiatCurrencies() { + return getAllSortedTraditionalCurrencies().stream() .filter(currency -> CurrencyUtil.isFiatCurrency(currency.getCode())) .collect(Collectors.toList()); // sorted by currency name } diff --git a/core/src/main/java/haveno/core/offer/CreateOfferService.java b/core/src/main/java/haveno/core/offer/CreateOfferService.java index ac8abb0900..2b60736099 100644 --- a/core/src/main/java/haveno/core/offer/CreateOfferService.java +++ b/core/src/main/java/haveno/core/offer/CreateOfferService.java @@ -25,6 +25,7 @@ import haveno.core.locale.Res; import haveno.core.monetary.Price; import haveno.core.payment.PaymentAccount; import haveno.core.payment.PaymentAccountUtil; +import haveno.core.payment.payload.PaymentMethod; import haveno.core.provider.price.MarketPrice; import haveno.core.provider.price.PriceFeedService; import haveno.core.support.dispute.arbitration.arbitrator.ArbitratorManager; @@ -47,8 +48,6 @@ import java.util.List; import java.util.Map; import java.util.UUID; -import static haveno.core.payment.payload.PaymentMethod.HAL_CASH_ID; - @Slf4j @Singleton public class CreateOfferService { @@ -137,7 +136,7 @@ public class CreateOfferService { boolean useMarketBasedPriceValue = price == null && useMarketBasedPrice && isMarketPriceAvailable(currencyCode) && - !paymentAccount.hasPaymentMethodWithId(HAL_CASH_ID); + !PaymentMethod.isFixedPriceOnly(paymentAccount.getPaymentMethod().getId()); // verify price if (price == null && !useMarketBasedPriceValue) { diff --git a/core/src/main/java/haveno/core/offer/Offer.java b/core/src/main/java/haveno/core/offer/Offer.java index e62fa51aed..bfcc06a61b 100644 --- a/core/src/main/java/haveno/core/offer/Offer.java +++ b/core/src/main/java/haveno/core/offer/Offer.java @@ -246,10 +246,7 @@ public class Offer implements NetworkPayload, PersistablePayload { return null; } Volume volumeByAmount = price.getVolumeByAmount(amount); - if (offerPayload.getPaymentMethodId().equals(PaymentMethod.HAL_CASH_ID)) - volumeByAmount = VolumeUtil.getAdjustedVolumeForHalCash(volumeByAmount); - else if (isFiatOffer()) - volumeByAmount = VolumeUtil.getRoundedFiatVolume(volumeByAmount); + volumeByAmount = VolumeUtil.getAdjustedVolume(volumeByAmount, getPaymentMethod().getId()); return volumeByAmount; } diff --git a/core/src/main/java/haveno/core/offer/takeoffer/TakeOfferModel.java b/core/src/main/java/haveno/core/offer/takeoffer/TakeOfferModel.java index 15f8b11e21..f97839c944 100644 --- a/core/src/main/java/haveno/core/offer/takeoffer/TakeOfferModel.java +++ b/core/src/main/java/haveno/core/offer/takeoffer/TakeOfferModel.java @@ -19,15 +19,14 @@ package haveno.core.offer.takeoffer; import haveno.common.taskrunner.Model; import haveno.core.account.witness.AccountAgeWitnessService; -import haveno.core.locale.CurrencyUtil; import haveno.core.monetary.Price; import haveno.core.monetary.Volume; import haveno.core.offer.Offer; import haveno.core.offer.OfferUtil; import haveno.core.payment.PaymentAccount; -import haveno.core.payment.payload.PaymentMethod; import haveno.core.provider.price.PriceFeedService; import haveno.core.trade.HavenoUtils; +import haveno.core.util.VolumeUtil; import haveno.core.xmr.model.XmrAddressEntry; import haveno.core.xmr.wallet.XmrWalletService; import lombok.Getter; @@ -41,8 +40,6 @@ import java.util.Objects; import static com.google.common.base.Preconditions.checkArgument; import static com.google.common.base.Preconditions.checkNotNull; import static haveno.core.offer.OfferDirection.SELL; -import static haveno.core.util.VolumeUtil.getAdjustedVolumeForHalCash; -import static haveno.core.util.VolumeUtil.getRoundedFiatVolume; import static haveno.core.xmr.model.XmrAddressEntry.Context.OFFER_FUNDING; @Slf4j @@ -136,11 +133,7 @@ public class TakeOfferModel implements Model { private void calculateVolume() { Price tradePrice = offer.getPrice(); Volume volumeByAmount = Objects.requireNonNull(tradePrice).getVolumeByAmount(amount); - - if (offer.getPaymentMethod().getId().equals(PaymentMethod.HAL_CASH_ID)) - volumeByAmount = getAdjustedVolumeForHalCash(volumeByAmount); - else if (CurrencyUtil.isFiatCurrency(offer.getCurrencyCode())) - volumeByAmount = getRoundedFiatVolume(volumeByAmount); + volumeByAmount = VolumeUtil.getAdjustedVolume(volumeByAmount, offer.getPaymentMethod().getId()); volume = volumeByAmount; diff --git a/core/src/main/java/haveno/core/payment/CashByAtmAccount.java b/core/src/main/java/haveno/core/payment/CashByAtmAccount.java new file mode 100644 index 0000000000..b91bc78df5 --- /dev/null +++ b/core/src/main/java/haveno/core/payment/CashByAtmAccount.java @@ -0,0 +1,64 @@ +/* + * This file is part of Haveno. + * + * Haveno 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. + * + * Haveno 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 Haveno. If not, see <http://www.gnu.org/licenses/>. + */ + +package haveno.core.payment; + +import haveno.core.api.model.PaymentAccountFormField; +import haveno.core.locale.CurrencyUtil; +import haveno.core.locale.TradeCurrency; +import haveno.core.payment.payload.CashByAtmAccountPayload; +import haveno.core.payment.payload.PaymentAccountPayload; +import haveno.core.payment.payload.PaymentMethod; +import lombok.NonNull; + +import java.util.List; + +public final class CashByAtmAccount extends PaymentAccount { + + public static final List<TradeCurrency> SUPPORTED_CURRENCIES = CurrencyUtil.getAllFiatCurrencies(); + + private static final List<PaymentAccountFormField.FieldId> INPUT_FIELD_IDS = List.of( + PaymentAccountFormField.FieldId.EXTRA_INFO + ); + + public CashByAtmAccount() { + super(PaymentMethod.CASH_BY_ATM); + } + + @Override + protected PaymentAccountPayload createPayload() { + return new CashByAtmAccountPayload(paymentMethod.getId(), id); + } + + @Override + public @NonNull List<TradeCurrency> getSupportedCurrencies() { + return SUPPORTED_CURRENCIES; + } + + @Override + public @NonNull List<PaymentAccountFormField.FieldId> getInputFieldIds() { + return INPUT_FIELD_IDS; + } + + public void setExtraInfo(String extraInfo) { + ((CashByAtmAccountPayload) paymentAccountPayload).setExtraInfo(extraInfo); + } + + public String getExtraInfo() { + return ((CashByAtmAccountPayload) paymentAccountPayload).getExtraInfo(); + } +} diff --git a/core/src/main/java/haveno/core/payment/PaymentAccountFactory.java b/core/src/main/java/haveno/core/payment/PaymentAccountFactory.java index be4a74d772..3be7449d2b 100644 --- a/core/src/main/java/haveno/core/payment/PaymentAccountFactory.java +++ b/core/src/main/java/haveno/core/payment/PaymentAccountFactory.java @@ -76,6 +76,8 @@ public class PaymentAccountFactory { return new F2FAccount(); case PaymentMethod.PAY_BY_MAIL_ID: return new PayByMailAccount(); + case PaymentMethod.CASH_BY_ATM_ID: + return new CashByAtmAccount(); case PaymentMethod.PROMPT_PAY_ID: return new PromptPayAccount(); case PaymentMethod.ADVANCED_CASH_ID: diff --git a/core/src/main/java/haveno/core/payment/payload/CashByAtmAccountPayload.java b/core/src/main/java/haveno/core/payment/payload/CashByAtmAccountPayload.java new file mode 100644 index 0000000000..00ebf24942 --- /dev/null +++ b/core/src/main/java/haveno/core/payment/payload/CashByAtmAccountPayload.java @@ -0,0 +1,96 @@ +/* + * This file is part of Haveno. + * + * Haveno 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. + * + * Haveno 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 Haveno. If not, see <http://www.gnu.org/licenses/>. + */ + +package haveno.core.payment.payload; + +import com.google.protobuf.Message; +import haveno.core.locale.Res; +import lombok.EqualsAndHashCode; +import lombok.Getter; +import lombok.Setter; +import lombok.ToString; +import lombok.extern.slf4j.Slf4j; +import org.apache.commons.lang3.ArrayUtils; + +import java.nio.charset.StandardCharsets; +import java.util.HashMap; +import java.util.Map; + +@EqualsAndHashCode(callSuper = true) +@ToString +@Setter +@Getter +@Slf4j +public final class CashByAtmAccountPayload extends PaymentAccountPayload { + private String extraInfo = ""; + + public CashByAtmAccountPayload(String paymentMethod, String id) { + super(paymentMethod, id); + } + + + /////////////////////////////////////////////////////////////////////////////////////////// + // PROTO BUFFER + /////////////////////////////////////////////////////////////////////////////////////////// + + private CashByAtmAccountPayload(String paymentMethod, String id, + String extraInfo, + long maxTradePeriod, + Map<String, String> excludeFromJsonDataMap) { + super(paymentMethod, + id, + maxTradePeriod, + excludeFromJsonDataMap); + this.extraInfo = extraInfo; + } + + @Override + public Message toProtoMessage() { + return getPaymentAccountPayloadBuilder() + .setCashByAtmAccountPayload(protobuf.CashByAtmAccountPayload.newBuilder() + .setExtraInfo(extraInfo)) + .build(); + } + + public static CashByAtmAccountPayload fromProto(protobuf.PaymentAccountPayload proto) { + return new CashByAtmAccountPayload(proto.getPaymentMethodId(), + proto.getId(), + proto.getCashByAtmAccountPayload().getExtraInfo(), + proto.getMaxTradePeriod(), + new HashMap<>(proto.getExcludeFromJsonDataMap())); + } + + + /////////////////////////////////////////////////////////////////////////////////////////// + // API + /////////////////////////////////////////////////////////////////////////////////////////// + + @Override + public String getPaymentDetails() { + return Res.getWithCol("payment.shared.extraInfo") + " " + extraInfo; + } + + @Override + public String getPaymentDetailsForTradePopup() { + return Res.getWithCol("payment.shared.extraInfo") + " " + extraInfo; + } + + @Override + public byte[] getAgeWitnessInputData() { + return super.getAgeWitnessInputData(ArrayUtils.addAll(id.getBytes(StandardCharsets.UTF_8))); + } +} diff --git a/core/src/main/java/haveno/core/payment/payload/PaymentMethod.java b/core/src/main/java/haveno/core/payment/payload/PaymentMethod.java index e9fcc5d293..151c7c7e66 100644 --- a/core/src/main/java/haveno/core/payment/payload/PaymentMethod.java +++ b/core/src/main/java/haveno/core/payment/payload/PaymentMethod.java @@ -31,6 +31,7 @@ import haveno.core.payment.AmazonGiftCardAccount; import haveno.core.payment.AustraliaPayidAccount; import haveno.core.payment.BizumAccount; import haveno.core.payment.CapitualAccount; +import haveno.core.payment.CashByAtmAccount; import haveno.core.payment.PayByMailAccount; import haveno.core.payment.CashDepositAccount; import haveno.core.payment.CelPayAccount; @@ -162,6 +163,7 @@ public final class PaymentMethod implements PersistablePayload, Comparable<Payme public static final String AMAZON_GIFT_CARD_ID = "AMAZON_GIFT_CARD"; public static final String BLOCK_CHAINS_INSTANT_ID = "BLOCK_CHAINS_INSTANT"; public static final String PAY_BY_MAIL_ID = "PAY_BY_MAIL"; + public static final String CASH_BY_ATM_ID = "CASH_BY_ATM"; public static final String CAPITUAL_ID = "CAPITUAL"; public static final String CELPAY_ID = "CELPAY"; public static final String MONESE_ID = "MONESE"; @@ -224,6 +226,7 @@ public final class PaymentMethod implements PersistablePayload, Comparable<Payme public static PaymentMethod AMAZON_GIFT_CARD; public static PaymentMethod BLOCK_CHAINS_INSTANT; public static PaymentMethod PAY_BY_MAIL; + public static PaymentMethod CASH_BY_ATM; public static PaymentMethod CAPITUAL; public static PaymentMethod CELPAY; public static PaymentMethod MONESE; @@ -271,6 +274,7 @@ public final class PaymentMethod implements PersistablePayload, Comparable<Payme // Global CASH_DEPOSIT = new PaymentMethod(CASH_DEPOSIT_ID, 4 * DAY, DEFAULT_TRADE_LIMIT_HIGH_RISK, getAssetCodes(CashDepositAccount.SUPPORTED_CURRENCIES)), PAY_BY_MAIL = new PaymentMethod(PAY_BY_MAIL_ID, 8 * DAY, DEFAULT_TRADE_LIMIT_HIGH_RISK, getAssetCodes(PayByMailAccount.SUPPORTED_CURRENCIES)), + CASH_BY_ATM = new PaymentMethod(CASH_BY_ATM_ID, 4 * DAY, DEFAULT_TRADE_LIMIT_HIGH_RISK, getAssetCodes(CashByAtmAccount.SUPPORTED_CURRENCIES)), MONEY_GRAM = new PaymentMethod(MONEY_GRAM_ID, 4 * DAY, DEFAULT_TRADE_LIMIT_MID_RISK, getAssetCodes(MoneyGramAccount.SUPPORTED_CURRENCIES)), WESTERN_UNION = new PaymentMethod(WESTERN_UNION_ID, 4 * DAY, DEFAULT_TRADE_LIMIT_MID_RISK, getAssetCodes(WesternUnionAccount.SUPPORTED_CURRENCIES)), NATIONAL_BANK = new PaymentMethod(NATIONAL_BANK_ID, 4 * DAY, DEFAULT_TRADE_LIMIT_HIGH_RISK, getAssetCodes(NationalBankAccount.SUPPORTED_CURRENCIES)), @@ -560,4 +564,14 @@ public final class PaymentMethod implements PersistablePayload, Comparable<Payme id.equals(PaymentMethod.MONEY_BEAM_ID) || id.equals(PaymentMethod.UPHOLD_ID); } + + public static boolean isRoundedForAtmCash(String id) { + return id.equals(PaymentMethod.CASH_BY_ATM_ID) || + id.equals(PaymentMethod.HAL_CASH_ID); + } + + public static boolean isFixedPriceOnly(String id) { + return id.equals(PaymentMethod.CASH_BY_ATM_ID) || + id.equals(PaymentMethod.HAL_CASH_ID); + } } diff --git a/core/src/main/java/haveno/core/proto/CoreProtoResolver.java b/core/src/main/java/haveno/core/proto/CoreProtoResolver.java index 1d3732d75d..c60f3901ea 100644 --- a/core/src/main/java/haveno/core/proto/CoreProtoResolver.java +++ b/core/src/main/java/haveno/core/proto/CoreProtoResolver.java @@ -30,6 +30,7 @@ import haveno.core.payment.payload.AustraliaPayidAccountPayload; import haveno.core.payment.payload.BizumAccountPayload; import haveno.core.payment.payload.CapitualAccountPayload; import haveno.core.payment.payload.CashAppAccountPayload; +import haveno.core.payment.payload.CashByAtmAccountPayload; import haveno.core.payment.payload.PayByMailAccountPayload; import haveno.core.payment.payload.CashDepositAccountPayload; import haveno.core.payment.payload.CelPayAccountPayload; @@ -201,6 +202,8 @@ public class CoreProtoResolver implements ProtoResolver { return USPostalMoneyOrderAccountPayload.fromProto(proto); case PAY_BY_MAIL_ACCOUNT_PAYLOAD: return PayByMailAccountPayload.fromProto(proto); + case CASH_BY_ATM_ACCOUNT_PAYLOAD: + return CashByAtmAccountPayload.fromProto(proto); case PROMPT_PAY_ACCOUNT_PAYLOAD: return PromptPayAccountPayload.fromProto(proto); case ADVANCED_CASH_ACCOUNT_PAYLOAD: diff --git a/core/src/main/java/haveno/core/trade/Contract.java b/core/src/main/java/haveno/core/trade/Contract.java index df89410aca..71a360f910 100644 --- a/core/src/main/java/haveno/core/trade/Contract.java +++ b/core/src/main/java/haveno/core/trade/Contract.java @@ -22,7 +22,6 @@ import haveno.common.crypto.PubKeyRing; import haveno.common.proto.network.NetworkPayload; import haveno.common.util.JsonExclude; import haveno.common.util.Utilities; -import haveno.core.locale.CurrencyUtil; import haveno.core.monetary.Price; import haveno.core.monetary.Volume; import haveno.core.offer.OfferPayload; @@ -204,12 +203,7 @@ public final class Contract implements NetworkPayload { public Volume getTradeVolume() { Volume volumeByAmount = getPrice().getVolumeByAmount(getTradeAmount()); - - if (getPaymentMethodId().equals(PaymentMethod.HAL_CASH_ID)) - volumeByAmount = VolumeUtil.getAdjustedVolumeForHalCash(volumeByAmount); - else if (CurrencyUtil.isFiatCurrency(getOfferPayload().getCurrencyCode())) - volumeByAmount = VolumeUtil.getRoundedFiatVolume(volumeByAmount); - + volumeByAmount = VolumeUtil.getAdjustedVolume(volumeByAmount, getPaymentMethodId()); return volumeByAmount; } diff --git a/core/src/main/java/haveno/core/trade/Trade.java b/core/src/main/java/haveno/core/trade/Trade.java index 5bab5c8331..9d2358720d 100644 --- a/core/src/main/java/haveno/core/trade/Trade.java +++ b/core/src/main/java/haveno/core/trade/Trade.java @@ -28,13 +28,11 @@ import haveno.common.proto.ProtoUtil; import haveno.common.taskrunner.Model; import haveno.common.util.Utilities; import haveno.core.api.CoreMoneroConnectionsService; -import haveno.core.locale.CurrencyUtil; import haveno.core.monetary.Price; import haveno.core.monetary.Volume; import haveno.core.offer.Offer; import haveno.core.offer.OfferDirection; import haveno.core.payment.payload.PaymentAccountPayload; -import haveno.core.payment.payload.PaymentMethod; import haveno.core.proto.CoreProtoResolver; import haveno.core.proto.network.CoreNetworkProtoResolver; import haveno.core.support.dispute.Dispute; @@ -1438,14 +1436,7 @@ public abstract class Trade implements Tradable, Model { try { if (getAmount() != null && getPrice() != null) { Volume volumeByAmount = getPrice().getVolumeByAmount(getAmount()); - if (offer != null) { - if (offer.getPaymentMethod().getId().equals(PaymentMethod.HAL_CASH_ID)) - volumeByAmount = VolumeUtil.getAdjustedVolumeForHalCash(volumeByAmount); - else if (CurrencyUtil.isFiatCurrency(offer.getCurrencyCode())) - volumeByAmount = VolumeUtil.getRoundedFiatVolume(volumeByAmount); - else if (CurrencyUtil.isTraditionalCurrency(offer.getCurrencyCode())) - volumeByAmount = VolumeUtil.getRoundedTraditionalVolume(volumeByAmount); - } + if (offer != null) volumeByAmount = VolumeUtil.getAdjustedVolume(volumeByAmount, offer.getPaymentMethod().getId()); return volumeByAmount; } else { return null; diff --git a/core/src/main/java/haveno/core/util/VolumeUtil.java b/core/src/main/java/haveno/core/util/VolumeUtil.java index 9103b79032..d607f1711b 100644 --- a/core/src/main/java/haveno/core/util/VolumeUtil.java +++ b/core/src/main/java/haveno/core/util/VolumeUtil.java @@ -26,6 +26,7 @@ import haveno.core.monetary.TraditionalMoney; import haveno.core.monetary.TraditionalExchangeRate; import haveno.core.monetary.Volume; import haveno.core.offer.Offer; +import haveno.core.payment.payload.PaymentMethod; import haveno.core.trade.HavenoUtils; import org.bitcoinj.core.Monetary; import org.bitcoinj.utils.MonetaryFormat; @@ -42,23 +43,33 @@ public class VolumeUtil { private static double EXPONENT = Math.pow(10, TraditionalMoney.SMALLEST_UNIT_EXPONENT); // 1000000000000 with precision 8 + public static Volume getAdjustedVolume(Volume volumeByAmount, String paymentMethodId) { + if (PaymentMethod.isRoundedForAtmCash(paymentMethodId)) + return VolumeUtil.getRoundedAtmCashVolume(volumeByAmount); + else if (CurrencyUtil.isFiatCurrency(volumeByAmount.getCurrencyCode())) + return VolumeUtil.getRoundedFiatVolume(volumeByAmount); + else if (CurrencyUtil.isTraditionalCurrency(volumeByAmount.getCurrencyCode())) + return VolumeUtil.getRoundedTraditionalVolume(volumeByAmount); + return volumeByAmount; + } + public static Volume getRoundedFiatVolume(Volume volumeByAmount) { // We want to get rounded to 1 unit of the fiat currency, e.g. 1 EUR. return getAdjustedFiatVolume(volumeByAmount, 1); } + private static Volume getRoundedAtmCashVolume(Volume volumeByAmount) { + // EUR has precision TraditionalMoney.SMALLEST_UNIT_EXPONENT and we want multiple of 10 so we divide by EXPONENT then + // round and multiply with 10 + return getAdjustedFiatVolume(volumeByAmount, 10); + } + public static Volume getRoundedTraditionalVolume(Volume volumeByAmount) { DecimalFormat decimalFormat = new DecimalFormat("#.####"); double roundedVolume = Double.parseDouble(decimalFormat.format(Double.parseDouble(volumeByAmount.toString()))); return Volume.parse(String.valueOf(roundedVolume), volumeByAmount.getCurrencyCode()); } - public static Volume getAdjustedVolumeForHalCash(Volume volumeByAmount) { - // EUR has precision TraditionalMoney.SMALLEST_UNIT_EXPONENT and we want multiple of 10 so we divide by EXPONENT then - // round and multiply with 10 - return getAdjustedFiatVolume(volumeByAmount, 10); - } - /** * * @param volumeByAmount The volume generated from an amount diff --git a/core/src/main/java/haveno/core/util/coin/CoinUtil.java b/core/src/main/java/haveno/core/util/coin/CoinUtil.java index 80caf4adbb..4de0603009 100644 --- a/core/src/main/java/haveno/core/util/coin/CoinUtil.java +++ b/core/src/main/java/haveno/core/util/coin/CoinUtil.java @@ -19,14 +19,17 @@ package haveno.core.util.coin; import com.google.common.annotations.VisibleForTesting; import haveno.common.util.MathUtils; +import haveno.core.locale.CurrencyUtil; import haveno.core.monetary.Price; import haveno.core.monetary.Volume; +import haveno.core.payment.payload.PaymentMethod; import haveno.core.trade.HavenoUtils; import haveno.core.xmr.wallet.Restrictions; import org.bitcoinj.core.Coin; import java.math.BigDecimal; import java.math.BigInteger; +import java.text.DecimalFormat; import static com.google.common.base.Preconditions.checkArgument; import static haveno.core.util.VolumeUtil.getAdjustedFiatVolume; @@ -76,6 +79,21 @@ public class CoinUtil { return BigDecimal.valueOf(percent).multiply(new BigDecimal(amount)).toBigInteger(); } + public static BigInteger getRoundedAmount(BigInteger amount, Price price, long maxTradeLimit, String currencyCode, String paymentMethodId) { + if (PaymentMethod.isRoundedForAtmCash(paymentMethodId)) { + return getRoundedAtmCashAmount(amount, price, maxTradeLimit); + } else if (CurrencyUtil.isFiatCurrency(currencyCode)) { + return getRoundedFiatAmount(amount, price, maxTradeLimit); + } else if (CurrencyUtil.isTraditionalCurrency(currencyCode)) { + return getRoundedTraditionalAmount(amount, price, maxTradeLimit); + } + return amount; + } + + public static BigInteger getRoundedAtmCashAmount(BigInteger amount, Price price, long maxTradeLimit) { + return getAdjustedAmount(amount, price, maxTradeLimit, 10); + } + /** * Calculate the possibly adjusted amount for {@code amount}, taking into account the * {@code price} and {@code maxTradeLimit} and {@code factor}. @@ -88,9 +106,11 @@ public class CoinUtil { public static BigInteger getRoundedFiatAmount(BigInteger amount, Price price, long maxTradeLimit) { return getAdjustedAmount(amount, price, maxTradeLimit, 1); } - - public static BigInteger getAdjustedAmountForHalCash(BigInteger amount, Price price, long maxTradeLimit) { - return getAdjustedAmount(amount, price, maxTradeLimit, 10); + + public static BigInteger getRoundedTraditionalAmount(BigInteger amount, Price price, long maxTradeLimit) { + DecimalFormat decimalFormat = new DecimalFormat("#.####"); + double roundedXmrAmount = Double.parseDouble(decimalFormat.format(HavenoUtils.atomicUnitsToXmr(amount))); + return HavenoUtils.xmrToAtomicUnits(roundedXmrAmount); } /** diff --git a/core/src/main/resources/i18n/displayStrings.properties b/core/src/main/resources/i18n/displayStrings.properties index a1465f65ca..920925b5fd 100644 --- a/core/src/main/resources/i18n/displayStrings.properties +++ b/core/src/main/resources/i18n/displayStrings.properties @@ -2943,13 +2943,6 @@ payment.payByMail.info=Trading using pay-by-mail (PBM) on Haveno requires that y payment.payByMail.contact=Contact info payment.payByMail.contact.prompt=Name or nym envelope should be addressed to -payment.f2f.contact=Contact info -payment.f2f.contact.prompt=How would you like to be contacted by the trading peer? (email address, phone number,...) -payment.f2f.city=City for 'Face to face' meeting -payment.f2f.city.prompt=The city will be displayed with the offer -payment.shared.optionalExtra=Optional additional information -payment.shared.extraInfo=Additional information -payment.shared.extraInfo.prompt=Define any special terms, conditions, or details you would like to be displayed with your offers for this payment account (users will see this info before accepting offers). payment.payByMail.extraInfo.prompt=Please state on your offers: \n\n\ Country you are located (eg France); \n\ Countries / regions you would accept trades from (eg France, EU, or any European country); \n\ @@ -2957,6 +2950,23 @@ Any special terms/conditions; \n\ Any other details. payment.payByMail.tradingRestrictions=Please review the maker's terms and conditions.\n\ If you do not meet the requirements do not take this trade. +payment.cashByAtm.info=Cash at ATM: Cardless withdraw at ATM using code\n\n\ + 1. List your accepted banks, regions, or other terms.\n\n\ + 2. Chat with your peer trader to coordinate a time and share the withdraw code.\n\n\ + If you cannot complete the transaction as specified in your trade contract, you may lose some (or all) of your security deposit. +payment.cashByAtm.extraInfo.prompt=Please state on your offers: \n\n\ +Your accepted banks / locations; \n\ +Any special terms/conditions; \n\ +Any other details. +payment.payByMail.tradingRestrictions=Please review the maker's terms and conditions.\n\ + If you do not meet the requirements do not take this trade. +payment.f2f.contact=Contact info +payment.f2f.contact.prompt=How would you like to be contacted by the trading peer? (email address, phone number,...) +payment.f2f.city=City for 'Face to face' meeting +payment.f2f.city.prompt=The city will be displayed with the offer +payment.shared.optionalExtra=Optional additional information +payment.shared.extraInfo=Additional information +payment.shared.extraInfo.prompt=Define any special terms, conditions, or details you would like to be displayed with your offers for this payment account (users will see this info before accepting offers). payment.f2f.info='Face to Face' trades have different rules and come with different risks than online transactions.\n\n\ The main differences are:\n\ ● The trading peers need to exchange information about the meeting location and time by using their provided contact details.\n\ @@ -3001,6 +3011,7 @@ SPECIFIC_BANKS=Transfers with specific banks US_POSTAL_MONEY_ORDER=US Postal Money Order CASH_DEPOSIT=Cash Deposit PAY_BY_MAIL=Pay By Mail +CASH_BY_ATM=Cash by ATM MONEY_GRAM=MoneyGram WESTERN_UNION=Western Union F2F=Face to face (in person) @@ -3018,7 +3029,9 @@ US_POSTAL_MONEY_ORDER_SHORT=US Money Order # suppress inspection "UnusedProperty" CASH_DEPOSIT_SHORT=Cash Deposit # suppress inspection "UnusedProperty" -PAY_BY_MAIL_SHORT=PayByMail +PAY_BY_MAIL_SHORT=Pay By Mail +# suppress inspection "UnusedProperty" +CASH_BY_ATM_SHORT=Cash By ATM # suppress inspection "UnusedProperty" MONEY_GRAM_SHORT=MoneyGram # suppress inspection "UnusedProperty" diff --git a/desktop/src/main/java/haveno/desktop/components/paymentmethods/CashByAtmForm.java b/desktop/src/main/java/haveno/desktop/components/paymentmethods/CashByAtmForm.java new file mode 100644 index 0000000000..b4ea6b9327 --- /dev/null +++ b/desktop/src/main/java/haveno/desktop/components/paymentmethods/CashByAtmForm.java @@ -0,0 +1,116 @@ +/* + * This file is part of Haveno. + * + * Haveno 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. + * + * Haveno 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 Haveno. If not, see <http://www.gnu.org/licenses/>. + */ + +package haveno.desktop.components.paymentmethods; + +import com.jfoenix.controls.JFXTextArea; +import haveno.core.account.witness.AccountAgeWitnessService; +import haveno.core.locale.CurrencyUtil; +import haveno.core.locale.Res; +import haveno.core.locale.TradeCurrency; +import haveno.core.payment.CashByAtmAccount; +import haveno.core.payment.PaymentAccount; +import haveno.core.payment.payload.CashByAtmAccountPayload; +import haveno.core.payment.payload.PaymentAccountPayload; +import haveno.core.util.coin.CoinFormatter; +import haveno.core.util.validation.InputValidator; +import haveno.desktop.util.Layout; +import javafx.collections.FXCollections; +import javafx.scene.control.TextArea; +import javafx.scene.layout.GridPane; + +import static haveno.desktop.util.FormBuilder.addCompactTopLabelTextArea; +import static haveno.desktop.util.FormBuilder.addCompactTopLabelTextField; +import static haveno.desktop.util.FormBuilder.addTopLabelTextArea; +import static haveno.desktop.util.FormBuilder.addTopLabelTextFieldWithCopyIcon; + +public class CashByAtmForm extends PaymentMethodForm { + private final CashByAtmAccount cashByAtmAccount; + + public static int addFormForBuyer(GridPane gridPane, int gridRow, + PaymentAccountPayload paymentAccountPayload) { + CashByAtmAccountPayload cbm = (CashByAtmAccountPayload) paymentAccountPayload; + addTopLabelTextFieldWithCopyIcon(gridPane, gridRow, 1, + Res.get("payment.shared.extraInfo"), + cbm.getExtraInfo(), + Layout.COMPACT_FIRST_ROW_AND_GROUP_DISTANCE); + + TextArea textExtraInfo = addCompactTopLabelTextArea(gridPane, gridRow, 1, Res.get("payment.shared.extraInfo"), "").second; + textExtraInfo.setMinHeight(70); + textExtraInfo.setEditable(false); + textExtraInfo.setText(cbm.getExtraInfo()); + return gridRow; + } + + public CashByAtmForm(PaymentAccount paymentAccount, + AccountAgeWitnessService accountAgeWitnessService, + InputValidator inputValidator, GridPane gridPane, int gridRow, CoinFormatter formatter) { + super(paymentAccount, accountAgeWitnessService, inputValidator, gridPane, gridRow, formatter); + this.cashByAtmAccount = (CashByAtmAccount) paymentAccount; + } + + @Override + public void addFormForAddAccount() { + gridRowFrom = gridRow + 1; + + addTradeCurrencyComboBox(); + currencyComboBox.setItems(FXCollections.observableArrayList(CurrencyUtil.getAllSortedFiatCurrencies())); + + TextArea extraTextArea = addTopLabelTextArea(gridPane, ++gridRow, + Res.get("payment.shared.optionalExtra"), Res.get("payment.cashByAtm.extraInfo.prompt")).second; + extraTextArea.setMinHeight(70); + ((JFXTextArea) extraTextArea).setLabelFloat(false); + extraTextArea.textProperty().addListener((ov, oldValue, newValue) -> { + cashByAtmAccount.setExtraInfo(newValue); + updateFromInputs(); + }); + + addLimitations(false); + addAccountNameTextFieldWithAutoFillToggleButton(); + } + + @Override + protected void autoFillNameTextField() { + setAccountNameWithString(cashByAtmAccount.getExtraInfo().substring(0, Math.min(50, cashByAtmAccount.getExtraInfo().length()))); + } + + @Override + public void addFormForEditAccount() { + gridRowFrom = gridRow; + addAccountNameTextFieldWithAutoFillToggleButton(); + addCompactTopLabelTextField(gridPane, ++gridRow, Res.get("shared.paymentMethod"), + Res.get(cashByAtmAccount.getPaymentMethod().getId())); + + TradeCurrency tradeCurrency = paymentAccount.getSingleTradeCurrency(); + String nameAndCode = tradeCurrency != null ? tradeCurrency.getNameAndCode() : ""; + addCompactTopLabelTextField(gridPane, ++gridRow, Res.get("shared.currency"), nameAndCode); + + TextArea textAreaExtra = addCompactTopLabelTextArea(gridPane, ++gridRow, Res.get("payment.shared.extraInfo"), "").second; + textAreaExtra.setText(cashByAtmAccount.getExtraInfo()); + textAreaExtra.setMinHeight(70); + textAreaExtra.setEditable(false); + + addLimitations(true); + } + + @Override + public void updateAllInputsValid() { + allInputsValid.set(isAccountNameValid() + && !cashByAtmAccount.getExtraInfo().isEmpty() + && paymentAccount.getSingleTradeCurrency() != null); + } +} diff --git a/desktop/src/main/java/haveno/desktop/main/account/content/traditionalaccounts/TraditionalAccountsView.java b/desktop/src/main/java/haveno/desktop/main/account/content/traditionalaccounts/TraditionalAccountsView.java index aef3d182fe..e654bfef97 100644 --- a/desktop/src/main/java/haveno/desktop/main/account/content/traditionalaccounts/TraditionalAccountsView.java +++ b/desktop/src/main/java/haveno/desktop/main/account/content/traditionalaccounts/TraditionalAccountsView.java @@ -26,6 +26,7 @@ import haveno.core.locale.Res; import haveno.core.offer.OfferRestrictions; import haveno.core.payment.AmazonGiftCardAccount; import haveno.core.payment.AustraliaPayidAccount; +import haveno.core.payment.CashByAtmAccount; import haveno.core.payment.PayByMailAccount; import haveno.core.payment.CashDepositAccount; import haveno.core.payment.ZelleAccount; @@ -72,6 +73,7 @@ import haveno.desktop.components.paymentmethods.AmazonGiftCardForm; import haveno.desktop.components.paymentmethods.AustraliaPayidForm; import haveno.desktop.components.paymentmethods.BizumForm; import haveno.desktop.components.paymentmethods.CapitualForm; +import haveno.desktop.components.paymentmethods.CashByAtmForm; import haveno.desktop.components.paymentmethods.PayByMailForm; import haveno.desktop.components.paymentmethods.CashDepositForm; import haveno.desktop.components.paymentmethods.CelPayForm; @@ -270,6 +272,14 @@ public class TraditionalAccountsView extends PaymentAccountsView<GridPane, Tradi .actionButtonText(Res.get("shared.iUnderstand")) .onAction(() -> doSaveNewAccount(paymentAccount)) .show(); + } else if (paymentAccount instanceof CashByAtmAccount) { + // CashByAtm has no chargeback risk so we don't show the text from payment.limits.info. + new Popup().information(Res.get("payment.cashByAtm.info")) + .width(850) + .closeButtonText(Res.get("shared.cancel")) + .actionButtonText(Res.get("shared.iUnderstand")) + .onAction(() -> doSaveNewAccount(paymentAccount)) + .show(); } else if (paymentAccount instanceof HalCashAccount) { // HalCash has no chargeback risk so we don't show the text from payment.limits.info. new Popup().information(Res.get("payment.halCash.info")) @@ -559,6 +569,8 @@ public class TraditionalAccountsView extends PaymentAccountsView<GridPane, Tradi return new CashDepositForm(paymentAccount, accountAgeWitnessService, inputValidator, root, gridRow, formatter); case PaymentMethod.PAY_BY_MAIL_ID: return new PayByMailForm(paymentAccount, accountAgeWitnessService, inputValidator, root, gridRow, formatter); + case PaymentMethod.CASH_BY_ATM_ID: + return new CashByAtmForm(paymentAccount, accountAgeWitnessService, inputValidator, root, gridRow, formatter); case PaymentMethod.HAL_CASH_ID: return new HalCashForm(paymentAccount, accountAgeWitnessService, halCashValidator, inputValidator, root, gridRow, formatter); case PaymentMethod.F2F_ID: diff --git a/desktop/src/main/java/haveno/desktop/main/offer/MutableOfferDataModel.java b/desktop/src/main/java/haveno/desktop/main/offer/MutableOfferDataModel.java index 1c79a4e8a5..543053a413 100644 --- a/desktop/src/main/java/haveno/desktop/main/offer/MutableOfferDataModel.java +++ b/desktop/src/main/java/haveno/desktop/main/offer/MutableOfferDataModel.java @@ -32,6 +32,7 @@ import haveno.core.offer.OfferDirection; import haveno.core.offer.OfferUtil; import haveno.core.offer.OpenOfferManager; import haveno.core.payment.PaymentAccount; +import haveno.core.payment.payload.PaymentMethod; import haveno.core.provider.price.PriceFeedService; import haveno.core.trade.HavenoUtils; import haveno.core.trade.handlers.TransactionResultHandler; @@ -81,7 +82,6 @@ import java.util.function.Predicate; import java.util.stream.Collectors; import static com.google.common.base.Preconditions.checkNotNull; -import static haveno.core.payment.payload.PaymentMethod.HAL_CASH_ID; import static java.util.Comparator.comparing; public abstract class MutableOfferDataModel extends OfferDataModel { @@ -503,12 +503,7 @@ public abstract class MutableOfferDataModel extends OfferDataModel { private Volume calculateVolumeForAmount(ObjectProperty<BigInteger> minAmount) { Volume volumeByAmount = price.get().getVolumeByAmount(minAmount.get()); - - // For HalCash we want multiple of 10 EUR - if (isUsingHalCashAccount()) - volumeByAmount = VolumeUtil.getAdjustedVolumeForHalCash(volumeByAmount); - else if (CurrencyUtil.isFiatCurrency(tradeCurrencyCode.get())) - volumeByAmount = VolumeUtil.getRoundedFiatVolume(volumeByAmount); + volumeByAmount = VolumeUtil.getAdjustedVolume(volumeByAmount, paymentAccount.getPaymentMethod().getId()); return volumeByAmount; } @@ -516,10 +511,7 @@ public abstract class MutableOfferDataModel extends OfferDataModel { if (isNonZeroPrice.test(price) && isNonZeroVolume.test(volume) && allowAmountUpdate) { try { BigInteger value = HavenoUtils.coinToAtomicUnits(DisplayUtils.reduceTo4Decimals(HavenoUtils.atomicUnitsToCoin(price.get().getAmountByVolume(volume.get())), btcFormatter)); - if (isUsingHalCashAccount()) - value = CoinUtil.getAdjustedAmountForHalCash(value, price.get(), getMaxTradeLimit()); - else if (CurrencyUtil.isFiatCurrency(tradeCurrencyCode.get())) - value = CoinUtil.getRoundedFiatAmount(value, price.get(), getMaxTradeLimit()); + value = CoinUtil.getRoundedAmount(value, price.get(), getMaxTradeLimit(), tradeCurrencyCode.get(), paymentAccount.getPaymentMethod().getId()); calculateVolume(); @@ -680,7 +672,7 @@ public abstract class MutableOfferDataModel extends OfferDataModel { this.triggerPrice = triggerPrice; } - public boolean isUsingHalCashAccount() { - return paymentAccount.hasPaymentMethodWithId(HAL_CASH_ID); + public boolean isUsingRoundedAtmCashAccount() { + return PaymentMethod.isRoundedForAtmCash(paymentAccount.getPaymentMethod().getId()); } } diff --git a/desktop/src/main/java/haveno/desktop/main/offer/MutableOfferView.java b/desktop/src/main/java/haveno/desktop/main/offer/MutableOfferView.java index 4316cc92fd..55741c10a1 100644 --- a/desktop/src/main/java/haveno/desktop/main/offer/MutableOfferView.java +++ b/desktop/src/main/java/haveno/desktop/main/offer/MutableOfferView.java @@ -100,7 +100,6 @@ import java.util.HashMap; import java.util.List; import java.util.concurrent.TimeUnit; -import static haveno.core.payment.payload.PaymentMethod.HAL_CASH_ID; import static haveno.desktop.main.offer.OfferViewUtil.addPayInfoEntry; import static haveno.desktop.util.FormBuilder.add2ButtonsAfterGroup; import static haveno.desktop.util.FormBuilder.addAddressTextField; @@ -828,7 +827,7 @@ public abstract class MutableOfferView<M extends MutableOfferViewModel<?>> exten int marketPriceAvailableValue = model.marketPriceAvailableProperty.get(); if (marketPriceAvailableValue > -1) { boolean showPriceToggle = marketPriceAvailableValue == 1 && - !model.getDataModel().paymentAccount.hasPaymentMethodWithId(HAL_CASH_ID); + !PaymentMethod.isFixedPriceOnly(model.getDataModel().paymentAccount.getPaymentMethod().getId()); percentagePriceBox.setVisible(showPriceToggle); priceTypeToggleButton.setVisible(showPriceToggle); boolean fixedPriceSelected = !model.getDataModel().getUseMarketBasedPrice().get() || !showPriceToggle; diff --git a/desktop/src/main/java/haveno/desktop/main/offer/MutableOfferViewModel.java b/desktop/src/main/java/haveno/desktop/main/offer/MutableOfferViewModel.java index 5970818444..3b62b3c0ea 100644 --- a/desktop/src/main/java/haveno/desktop/main/offer/MutableOfferViewModel.java +++ b/desktop/src/main/java/haveno/desktop/main/offer/MutableOfferViewModel.java @@ -841,12 +841,7 @@ public abstract class MutableOfferViewModel<M extends MutableOfferDataModel> ext Volume volume = dataModel.getVolume().get(); if (volume != null) { - // For HalCash we want multiple of 10 EUR - if (dataModel.isUsingHalCashAccount()) - volume = VolumeUtil.getAdjustedVolumeForHalCash(volume); - else if (CurrencyUtil.isFiatCurrency(tradeCurrencyCode.get())) - volume = VolumeUtil.getRoundedFiatVolume(volume); - + volume = VolumeUtil.getAdjustedVolume(volume, dataModel.getPaymentAccount().getPaymentMethod().getId()); this.volume.set(VolumeUtil.formatVolume(volume)); } @@ -1082,10 +1077,7 @@ public abstract class MutableOfferViewModel<M extends MutableOfferDataModel> ext long maxTradeLimit = dataModel.getMaxTradeLimit(); Price price = dataModel.getPrice().get(); if (price != null && price.isPositive()) { - if (dataModel.isUsingHalCashAccount()) - amount = CoinUtil.getAdjustedAmountForHalCash(amount, price, maxTradeLimit); - else if (CurrencyUtil.isFiatCurrency(tradeCurrencyCode.get())) - amount = CoinUtil.getRoundedFiatAmount(amount, price, maxTradeLimit); + amount = CoinUtil.getRoundedAmount(amount, price, maxTradeLimit, tradeCurrencyCode.get(), dataModel.getPaymentAccount().getPaymentMethod().getId()); } dataModel.setAmount(amount); if (syncMinAmountWithAmount || @@ -1106,10 +1098,7 @@ public abstract class MutableOfferViewModel<M extends MutableOfferDataModel> ext Price price = dataModel.getPrice().get(); long maxTradeLimit = dataModel.getMaxTradeLimit(); if (price != null && price.isPositive()) { - if (dataModel.isUsingHalCashAccount()) - minAmount = CoinUtil.getAdjustedAmountForHalCash(minAmount, price, maxTradeLimit); - else if (CurrencyUtil.isFiatCurrency(tradeCurrencyCode.get())) - minAmount = CoinUtil.getRoundedFiatAmount(minAmount, price, maxTradeLimit); + minAmount = CoinUtil.getRoundedAmount(minAmount, price, maxTradeLimit, tradeCurrencyCode.get(), dataModel.getPaymentAccount().getPaymentMethod().getId()); } dataModel.setMinAmount(minAmount); diff --git a/desktop/src/main/java/haveno/desktop/main/offer/takeoffer/TakeOfferDataModel.java b/desktop/src/main/java/haveno/desktop/main/offer/takeoffer/TakeOfferDataModel.java index 30fa04b6cc..61745fb8cb 100644 --- a/desktop/src/main/java/haveno/desktop/main/offer/takeoffer/TakeOfferDataModel.java +++ b/desktop/src/main/java/haveno/desktop/main/offer/takeoffer/TakeOfferDataModel.java @@ -63,7 +63,6 @@ import java.util.Set; import static com.google.common.base.Preconditions.checkArgument; import static com.google.common.base.Preconditions.checkNotNull; -import static haveno.core.payment.payload.PaymentMethod.HAL_CASH_ID; /** * Domain for that UI element. @@ -376,10 +375,7 @@ class TakeOfferDataModel extends OfferDataModel { amount.get() != null && amount.get().compareTo(BigInteger.valueOf(0)) != 0) { Volume volumeByAmount = tradePrice.getVolumeByAmount(amount.get()); - if (offer.getPaymentMethod().getId().equals(PaymentMethod.HAL_CASH_ID)) - volumeByAmount = VolumeUtil.getAdjustedVolumeForHalCash(volumeByAmount); - else if (offer.isFiatOffer()) - volumeByAmount = VolumeUtil.getRoundedFiatVolume(volumeByAmount); + volumeByAmount = VolumeUtil.getAdjustedVolume(volumeByAmount, offer.getPaymentMethod().getId()); volume.set(volumeByAmount); @@ -491,7 +487,7 @@ class TakeOfferDataModel extends OfferDataModel { return offer.getSellerSecurityDeposit(); } - public boolean isUsingHalCashAccount() { - return paymentAccount.hasPaymentMethodWithId(HAL_CASH_ID); + public boolean isRoundedForAtmCash() { + return PaymentMethod.isRoundedForAtmCash(paymentAccount.getPaymentMethod().getId()); } } diff --git a/desktop/src/main/java/haveno/desktop/main/offer/takeoffer/TakeOfferViewModel.java b/desktop/src/main/java/haveno/desktop/main/offer/takeoffer/TakeOfferViewModel.java index 674c92cc90..6063d4599c 100644 --- a/desktop/src/main/java/haveno/desktop/main/offer/takeoffer/TakeOfferViewModel.java +++ b/desktop/src/main/java/haveno/desktop/main/offer/takeoffer/TakeOfferViewModel.java @@ -302,18 +302,19 @@ class TakeOfferViewModel extends ActivatableWithDataModel<TakeOfferDataModel> im Price tradePrice = dataModel.tradePrice; long maxTradeLimit = dataModel.getMaxTradeLimit(); - if (dataModel.getPaymentMethod().getId().equals(PaymentMethod.HAL_CASH_ID)) { - BigInteger adjustedAmountForHalCash = CoinUtil.getAdjustedAmountForHalCash(dataModel.getAmount().get(), + if (PaymentMethod.isRoundedForAtmCash(dataModel.getPaymentMethod().getId())) { + BigInteger adjustedAmountForHalCash = CoinUtil.getRoundedAtmCashAmount(dataModel.getAmount().get(), tradePrice, maxTradeLimit); dataModel.applyAmount(adjustedAmountForHalCash); amount.set(HavenoUtils.formatXmr(dataModel.getAmount().get())); - } else if (dataModel.getOffer().isFiatOffer()) { + } else if (dataModel.getOffer().isTraditionalOffer()) { if (!isAmountEqualMinAmount(dataModel.getAmount().get()) && (!isAmountEqualMaxAmount(dataModel.getAmount().get()))) { // We only apply the rounding if the amount is variable (minAmount is lower as amount). // Otherwise we could get an amount lower then the minAmount set by rounding - BigInteger roundedAmount = CoinUtil.getRoundedFiatAmount(dataModel.getAmount().get(), tradePrice, - maxTradeLimit); + BigInteger roundedAmount = dataModel.getOffer().isFiatOffer() ? + CoinUtil.getRoundedFiatAmount(dataModel.getAmount().get(), tradePrice, maxTradeLimit) : + CoinUtil.getRoundedTraditionalAmount(dataModel.getAmount().get(), tradePrice, maxTradeLimit); dataModel.applyAmount(roundedAmount); } amount.set(HavenoUtils.formatXmr(dataModel.getAmount().get())); @@ -585,13 +586,15 @@ class TakeOfferViewModel extends ActivatableWithDataModel<TakeOfferDataModel> im long maxTradeLimit = dataModel.getMaxTradeLimit(); Price price = dataModel.tradePrice; if (price != null) { - if (dataModel.isUsingHalCashAccount()) { - amount = CoinUtil.getAdjustedAmountForHalCash(amount, price, maxTradeLimit); - } else if (dataModel.getOffer().isFiatOffer() + if (dataModel.isRoundedForAtmCash()) { + amount = CoinUtil.getRoundedAtmCashAmount(amount, price, maxTradeLimit); + } else if (dataModel.getOffer().isTraditionalOffer() && !isAmountEqualMinAmount(amount) && !isAmountEqualMaxAmount(amount)) { // We only apply the rounding if the amount is variable (minAmount is lower as amount). // Otherwise we could get an amount lower then the minAmount set by rounding - amount = CoinUtil.getRoundedFiatAmount(amount, price, maxTradeLimit); + amount = dataModel.getOffer().isFiatOffer() ? + CoinUtil.getRoundedFiatAmount(amount, price, maxTradeLimit) : + CoinUtil.getRoundedTraditionalAmount(amount, price, maxTradeLimit); } } dataModel.applyAmount(amount); diff --git a/proto/src/main/proto/pb.proto b/proto/src/main/proto/pb.proto index cddf8be050..b5bbf530bf 100644 --- a/proto/src/main/proto/pb.proto +++ b/proto/src/main/proto/pb.proto @@ -856,6 +856,7 @@ message PaymentAccountPayload { CelPayAccountPayload cel_pay_account_payload = 37; MoneseAccountPayload monese_account_payload = 38; VerseAccountPayload verse_account_payload = 39; + CashByAtmAccountPayload cash_by_atm_account_payload = 40; } } @@ -1112,6 +1113,10 @@ message PayByMailAccountPayload { string extra_info = 3; } +message CashByAtmAccountPayload { + string extra_info = 1; +} + message PromptPayAccountPayload { string prompt_pay_id = 1; }