From bf311e36fb3b6902b0ab4cee32c94d2b418afdec Mon Sep 17 00:00:00 2001
From: jmacxx <47253594+jmacxx@users.noreply.github.com>
Date: Fri, 12 Nov 2021 11:10:44 -0600
Subject: [PATCH] Add payment methods ACH Transfer and Domestic Wire Transfer
---
.../bisq/core/payment/AchTransferAccount.java | 63 +++++++
.../payment/DomesticWireTransferAccount.java | 63 +++++++
.../core/payment/PaymentAccountFactory.java | 4 +
.../payload/AchTransferAccountPayload.java | 115 ++++++++++++
.../DomesticWireTransferAccountPayload.java | 119 ++++++++++++
.../core/payment/payload/PaymentMethod.java | 6 +
.../bisq/core/proto/CoreProtoResolver.java | 6 +
.../trade/statistics/TradeStatistics3.java | 4 +-
.../resources/i18n/displayStrings.properties | 25 ++-
.../paymentmethods/AchTransferForm.java | 102 +++++++++++
.../DomesticWireTransferForm.java | 89 +++++++++
.../paymentmethods/GeneralUsBankForm.java | 171 ++++++++++++++++++
.../paymentmethods/TransferwiseUsdForm.java | 3 +-
.../fiataccounts/FiatAccountsView.java | 6 +
.../steps/buyer/BuyerStep2View.java | 8 +
proto/src/main/proto/pb.proto | 10 +
16 files changed, 791 insertions(+), 3 deletions(-)
create mode 100644 core/src/main/java/bisq/core/payment/AchTransferAccount.java
create mode 100644 core/src/main/java/bisq/core/payment/DomesticWireTransferAccount.java
create mode 100644 core/src/main/java/bisq/core/payment/payload/AchTransferAccountPayload.java
create mode 100644 core/src/main/java/bisq/core/payment/payload/DomesticWireTransferAccountPayload.java
create mode 100644 desktop/src/main/java/bisq/desktop/components/paymentmethods/AchTransferForm.java
create mode 100644 desktop/src/main/java/bisq/desktop/components/paymentmethods/DomesticWireTransferForm.java
create mode 100644 desktop/src/main/java/bisq/desktop/components/paymentmethods/GeneralUsBankForm.java
diff --git a/core/src/main/java/bisq/core/payment/AchTransferAccount.java b/core/src/main/java/bisq/core/payment/AchTransferAccount.java
new file mode 100644
index 00000000000..386e9f2a798
--- /dev/null
+++ b/core/src/main/java/bisq/core/payment/AchTransferAccount.java
@@ -0,0 +1,63 @@
+/*
+ * 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.core.payment;
+
+import bisq.core.payment.payload.BankAccountPayload;
+import bisq.core.payment.payload.AchTransferAccountPayload;
+import bisq.core.payment.payload.PaymentAccountPayload;
+import bisq.core.payment.payload.PaymentMethod;
+
+import lombok.EqualsAndHashCode;
+
+@EqualsAndHashCode(callSuper = true)
+public final class AchTransferAccount extends CountryBasedPaymentAccount implements SameCountryRestrictedBankAccount {
+ public AchTransferAccount() {
+ super(PaymentMethod.ACH_TRANSFER);
+ }
+
+ @Override
+ protected PaymentAccountPayload createPayload() {
+ return new AchTransferAccountPayload(paymentMethod.getId(), id);
+ }
+
+ @Override
+ public String getBankId() {
+ return ((BankAccountPayload) paymentAccountPayload).getBankId();
+ }
+
+ @Override
+ public String getCountryCode() {
+ return getCountry() != null ? getCountry().code : "";
+ }
+
+ public AchTransferAccountPayload getPayload() {
+ return (AchTransferAccountPayload) paymentAccountPayload;
+ }
+
+ public String getMessageForBuyer() {
+ return "payment.achTransfer.info.buyer";
+ }
+
+ public String getMessageForSeller() {
+ return "payment.achTransfer.info.seller";
+ }
+
+ public String getMessageForAccountCreation() {
+ return "payment.achTransfer.info.account";
+ }
+}
diff --git a/core/src/main/java/bisq/core/payment/DomesticWireTransferAccount.java b/core/src/main/java/bisq/core/payment/DomesticWireTransferAccount.java
new file mode 100644
index 00000000000..4844211c335
--- /dev/null
+++ b/core/src/main/java/bisq/core/payment/DomesticWireTransferAccount.java
@@ -0,0 +1,63 @@
+/*
+ * 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.core.payment;
+
+import bisq.core.payment.payload.BankAccountPayload;
+import bisq.core.payment.payload.DomesticWireTransferAccountPayload;
+import bisq.core.payment.payload.PaymentAccountPayload;
+import bisq.core.payment.payload.PaymentMethod;
+
+import lombok.EqualsAndHashCode;
+
+@EqualsAndHashCode(callSuper = true)
+public final class DomesticWireTransferAccount extends CountryBasedPaymentAccount implements SameCountryRestrictedBankAccount {
+ public DomesticWireTransferAccount() {
+ super(PaymentMethod.DOMESTIC_WIRE_TRANSFER);
+ }
+
+ @Override
+ protected PaymentAccountPayload createPayload() {
+ return new DomesticWireTransferAccountPayload(paymentMethod.getId(), id);
+ }
+
+ @Override
+ public String getBankId() {
+ return ((BankAccountPayload) paymentAccountPayload).getBankId();
+ }
+
+ @Override
+ public String getCountryCode() {
+ return getCountry() != null ? getCountry().code : "";
+ }
+
+ public DomesticWireTransferAccountPayload getPayload() {
+ return (DomesticWireTransferAccountPayload) paymentAccountPayload;
+ }
+
+ public String getMessageForBuyer() {
+ return "payment.domesticWire.info.buyer";
+ }
+
+ public String getMessageForSeller() {
+ return "payment.domesticWire.info.seller";
+ }
+
+ public String getMessageForAccountCreation() {
+ return "payment.domesticWire.info.account";
+ }
+}
diff --git a/core/src/main/java/bisq/core/payment/PaymentAccountFactory.java b/core/src/main/java/bisq/core/payment/PaymentAccountFactory.java
index c570d749bef..8e3c1e74a36 100644
--- a/core/src/main/java/bisq/core/payment/PaymentAccountFactory.java
+++ b/core/src/main/java/bisq/core/payment/PaymentAccountFactory.java
@@ -124,6 +124,10 @@ public static PaymentAccount getPaymentAccount(PaymentMethod paymentMethod) {
return new StrikeAccount();
case PaymentMethod.SWIFT_ID:
return new SwiftAccount();
+ case PaymentMethod.ACH_TRANSFER_ID:
+ return new AchTransferAccount();
+ case PaymentMethod.DOMESTIC_WIRE_TRANSFER_ID:
+ return new DomesticWireTransferAccount();
case PaymentMethod.BSQ_SWAP_ID:
return new BsqSwapAccount();
diff --git a/core/src/main/java/bisq/core/payment/payload/AchTransferAccountPayload.java b/core/src/main/java/bisq/core/payment/payload/AchTransferAccountPayload.java
new file mode 100644
index 00000000000..7c438bb078b
--- /dev/null
+++ b/core/src/main/java/bisq/core/payment/payload/AchTransferAccountPayload.java
@@ -0,0 +1,115 @@
+/*
+ * 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.core.payment.payload;
+
+import bisq.core.locale.Res;
+
+import com.google.protobuf.Message;
+
+import java.util.HashMap;
+import java.util.Map;
+
+import lombok.EqualsAndHashCode;
+import lombok.Getter;
+import lombok.Setter;
+import lombok.ToString;
+import lombok.extern.slf4j.Slf4j;
+
+@EqualsAndHashCode(callSuper = true)
+@ToString
+@Getter
+@Setter
+@Slf4j
+public final class AchTransferAccountPayload extends BankAccountPayload {
+ private String holderAddress = "";
+
+ public AchTransferAccountPayload(String paymentMethod, String id) {
+ super(paymentMethod, id);
+ }
+
+
+ ///////////////////////////////////////////////////////////////////////////////////////////
+ // PROTO BUFFER
+ ///////////////////////////////////////////////////////////////////////////////////////////
+
+ private AchTransferAccountPayload(String paymentMethodName,
+ String id,
+ String countryCode,
+ String holderName,
+ String bankName,
+ String branchId,
+ String accountNr,
+ String accountType,
+ String holderAddress,
+ long maxTradePeriod,
+ Map excludeFromJsonDataMap) {
+ super(paymentMethodName,
+ id,
+ countryCode,
+ holderName,
+ bankName,
+ branchId,
+ accountNr,
+ accountType,
+ null, // holderTaxId not used
+ null, // bankId not used
+ null, // nationalAccountId not used
+ maxTradePeriod,
+ excludeFromJsonDataMap);
+
+ this.holderAddress = holderAddress;
+ }
+
+ @Override
+ public Message toProtoMessage() {
+ protobuf.AchTransferAccountPayload.Builder builder = protobuf.AchTransferAccountPayload.newBuilder()
+ .setHolderAddress(holderAddress);
+ protobuf.BankAccountPayload.Builder bankAccountPayloadBuilder = getPaymentAccountPayloadBuilder()
+ .getCountryBasedPaymentAccountPayloadBuilder()
+ .getBankAccountPayloadBuilder()
+ .setAchTransferAccountPayload(builder);
+ protobuf.CountryBasedPaymentAccountPayload.Builder countryBasedPaymentAccountPayloadBuilder = getPaymentAccountPayloadBuilder()
+ .getCountryBasedPaymentAccountPayloadBuilder()
+ .setBankAccountPayload(bankAccountPayloadBuilder);
+ return getPaymentAccountPayloadBuilder()
+ .setCountryBasedPaymentAccountPayload(countryBasedPaymentAccountPayloadBuilder)
+ .build();
+ }
+
+ public static AchTransferAccountPayload fromProto(protobuf.PaymentAccountPayload proto) {
+ protobuf.CountryBasedPaymentAccountPayload countryBasedPaymentAccountPayload = proto.getCountryBasedPaymentAccountPayload();
+ protobuf.BankAccountPayload bankAccountPayloadPB = countryBasedPaymentAccountPayload.getBankAccountPayload();
+ protobuf.AchTransferAccountPayload accountPayloadPB = bankAccountPayloadPB.getAchTransferAccountPayload();
+ return new AchTransferAccountPayload(proto.getPaymentMethodId(),
+ proto.getId(),
+ countryBasedPaymentAccountPayload.getCountryCode(),
+ bankAccountPayloadPB.getHolderName(),
+ bankAccountPayloadPB.getBankName().isEmpty() ? null : bankAccountPayloadPB.getBankName(),
+ bankAccountPayloadPB.getBranchId().isEmpty() ? null : bankAccountPayloadPB.getBranchId(),
+ bankAccountPayloadPB.getAccountNr().isEmpty() ? null : bankAccountPayloadPB.getAccountNr(),
+ bankAccountPayloadPB.getAccountType().isEmpty() ? null : bankAccountPayloadPB.getAccountType(),
+ accountPayloadPB.getHolderAddress().isEmpty() ? null : accountPayloadPB.getHolderAddress(),
+ proto.getMaxTradePeriod(),
+ new HashMap<>(proto.getExcludeFromJsonDataMap()));
+ }
+
+ @Override
+ public String getPaymentDetails() {
+ return Res.get(paymentMethodId) + " - " + getPaymentDetailsForTradePopup().replace("\n", ", ");
+ }
+}
diff --git a/core/src/main/java/bisq/core/payment/payload/DomesticWireTransferAccountPayload.java b/core/src/main/java/bisq/core/payment/payload/DomesticWireTransferAccountPayload.java
new file mode 100644
index 00000000000..f3bba3223e6
--- /dev/null
+++ b/core/src/main/java/bisq/core/payment/payload/DomesticWireTransferAccountPayload.java
@@ -0,0 +1,119 @@
+/*
+ * 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.core.payment.payload;
+
+import bisq.core.locale.BankUtil;
+import bisq.core.locale.Res;
+
+import com.google.protobuf.Message;
+
+import java.util.HashMap;
+import java.util.Map;
+
+import lombok.EqualsAndHashCode;
+import lombok.Getter;
+import lombok.Setter;
+import lombok.ToString;
+import lombok.extern.slf4j.Slf4j;
+
+@EqualsAndHashCode(callSuper = true)
+@ToString
+@Getter
+@Setter
+@Slf4j
+public final class DomesticWireTransferAccountPayload extends BankAccountPayload {
+ private String holderAddress = "";
+
+ public DomesticWireTransferAccountPayload(String paymentMethod, String id) {
+ super(paymentMethod, id);
+ }
+
+
+ ///////////////////////////////////////////////////////////////////////////////////////////
+ // PROTO BUFFER
+ ///////////////////////////////////////////////////////////////////////////////////////////
+
+ private DomesticWireTransferAccountPayload(String paymentMethodName,
+ String id,
+ String countryCode,
+ String holderName,
+ String bankName,
+ String branchId,
+ String accountNr,
+ String holderAddress,
+ long maxTradePeriod,
+ Map excludeFromJsonDataMap) {
+ super(paymentMethodName,
+ id,
+ countryCode,
+ holderName,
+ bankName,
+ branchId,
+ accountNr,
+ null,
+ null, // holderTaxId not used
+ null, // bankId not used
+ null, // nationalAccountId not used
+ maxTradePeriod,
+ excludeFromJsonDataMap);
+
+ this.holderAddress = holderAddress;
+ }
+
+ @Override
+ public Message toProtoMessage() {
+ protobuf.DomesticWireTransferAccountPayload.Builder builder = protobuf.DomesticWireTransferAccountPayload.newBuilder()
+ .setHolderAddress(holderAddress);
+ protobuf.BankAccountPayload.Builder bankAccountPayloadBuilder = getPaymentAccountPayloadBuilder()
+ .getCountryBasedPaymentAccountPayloadBuilder()
+ .getBankAccountPayloadBuilder()
+ .setDomesticWireTransferAccountPayload(builder);
+ protobuf.CountryBasedPaymentAccountPayload.Builder countryBasedPaymentAccountPayloadBuilder = getPaymentAccountPayloadBuilder()
+ .getCountryBasedPaymentAccountPayloadBuilder()
+ .setBankAccountPayload(bankAccountPayloadBuilder);
+ return getPaymentAccountPayloadBuilder()
+ .setCountryBasedPaymentAccountPayload(countryBasedPaymentAccountPayloadBuilder)
+ .build();
+ }
+
+ public static DomesticWireTransferAccountPayload fromProto(protobuf.PaymentAccountPayload proto) {
+ protobuf.CountryBasedPaymentAccountPayload countryBasedPaymentAccountPayload = proto.getCountryBasedPaymentAccountPayload();
+ protobuf.BankAccountPayload bankAccountPayloadPB = countryBasedPaymentAccountPayload.getBankAccountPayload();
+ protobuf.DomesticWireTransferAccountPayload accountPayloadPB = bankAccountPayloadPB.getDomesticWireTransferAccountPayload();
+ return new DomesticWireTransferAccountPayload(proto.getPaymentMethodId(),
+ proto.getId(),
+ countryBasedPaymentAccountPayload.getCountryCode(),
+ bankAccountPayloadPB.getHolderName(),
+ bankAccountPayloadPB.getBankName().isEmpty() ? null : bankAccountPayloadPB.getBankName(),
+ bankAccountPayloadPB.getBranchId().isEmpty() ? null : bankAccountPayloadPB.getBranchId(),
+ bankAccountPayloadPB.getAccountNr().isEmpty() ? null : bankAccountPayloadPB.getAccountNr(),
+ accountPayloadPB.getHolderAddress().isEmpty() ? null : accountPayloadPB.getHolderAddress(),
+ proto.getMaxTradePeriod(),
+ new HashMap<>(proto.getExcludeFromJsonDataMap()));
+ }
+
+ @Override
+ public String getPaymentDetails() {
+ String paymentDetails = (Res.get(paymentMethodId) + " - " +
+ Res.getWithCol("payment.account.owner") + " " + holderName + ", " +
+ BankUtil.getBankNameLabel(countryCode) + ": " + this.bankName + ", " +
+ BankUtil.getBranchIdLabel(countryCode) + ": " + this.branchId + ", " +
+ BankUtil.getAccountNrLabel(countryCode) + ": " + this.accountNr);
+ return paymentDetails;
+ }
+}
diff --git a/core/src/main/java/bisq/core/payment/payload/PaymentMethod.java b/core/src/main/java/bisq/core/payment/payload/PaymentMethod.java
index 59f76e04a82..9850aac2e9c 100644
--- a/core/src/main/java/bisq/core/payment/payload/PaymentMethod.java
+++ b/core/src/main/java/bisq/core/payment/payload/PaymentMethod.java
@@ -118,6 +118,8 @@ public final class PaymentMethod implements PersistablePayload, Comparable.
+ */
+
+package bisq.desktop.components.paymentmethods;
+
+import bisq.core.account.witness.AccountAgeWitnessService;
+import bisq.core.locale.BankUtil;
+import bisq.core.locale.Country;
+import bisq.core.locale.Res;
+import bisq.core.payment.AchTransferAccount;
+import bisq.core.payment.CountryBasedPaymentAccount;
+import bisq.core.payment.PaymentAccount;
+import bisq.core.payment.payload.AchTransferAccountPayload;
+import bisq.core.payment.payload.BankAccountPayload;
+import bisq.core.payment.payload.PaymentAccountPayload;
+import bisq.core.util.coin.CoinFormatter;
+import bisq.core.util.validation.InputValidator;
+
+import javafx.scene.control.ComboBox;
+import javafx.scene.layout.GridPane;
+
+import javafx.collections.FXCollections;
+
+import static bisq.desktop.util.FormBuilder.*;
+
+public class AchTransferForm extends GeneralUsBankForm {
+
+ public static int addFormForBuyer(GridPane gridPane, int gridRow, PaymentAccountPayload paymentAccountPayload) {
+ AchTransferAccountPayload achTransferAccountPayload = (AchTransferAccountPayload) paymentAccountPayload;
+ return addFormForBuyer(gridPane, gridRow, paymentAccountPayload, achTransferAccountPayload.getAccountType(), achTransferAccountPayload.getHolderAddress());
+ }
+
+ private final AchTransferAccount achTransferAccount;
+
+ public AchTransferForm(PaymentAccount paymentAccount, AccountAgeWitnessService accountAgeWitnessService, InputValidator inputValidator,
+ GridPane gridPane, int gridRow, CoinFormatter formatter) {
+ super(paymentAccount, accountAgeWitnessService, inputValidator, gridPane, gridRow, formatter);
+ this.achTransferAccount = (AchTransferAccount) paymentAccount;
+ }
+
+ @Override
+ public void addFormForDisplayAccount() {
+ addFormForDisplayAccount(achTransferAccount.getPayload(), achTransferAccount.getPayload().getHolderAddress());
+ }
+
+ @Override
+ public void addFormForAddAccount() {
+ addFormForAddAccountInternal(achTransferAccount.getPayload(), achTransferAccount.getPayload().getHolderAddress());
+ }
+
+ @Override
+ protected void setHolderAddress(String holderAddress) {
+ achTransferAccount.getPayload().setHolderAddress(holderAddress);
+ }
+
+ @Override
+ protected void maybeAddAccountTypeCombo(BankAccountPayload bankAccountPayload, Country country) {
+ ComboBox accountTypeComboBox = addComboBox(gridPane, ++gridRow, Res.get("payment.select.account"));
+ accountTypeComboBox.setItems(FXCollections.observableArrayList(BankUtil.getAccountTypeValues(country.code)));
+ accountTypeComboBox.setOnAction(e -> {
+ if (BankUtil.isAccountTypeRequired(country.code)) {
+ bankAccountPayload.setAccountType(accountTypeComboBox.getSelectionModel().getSelectedItem());
+ updateFromInputs();
+ }
+ });
+ }
+
+ @Override
+ public void updateAllInputsValid() {
+ AchTransferAccountPayload achTransferAccountPayload = achTransferAccount.getPayload();
+ boolean result = isAccountNameValid()
+ && paymentAccount.getSingleTradeCurrency() != null
+ && ((CountryBasedPaymentAccount) this.paymentAccount).getCountry() != null
+ && inputValidator.validate(achTransferAccountPayload.getHolderName()).isValid
+ && inputValidator.validate(achTransferAccountPayload.getHolderAddress()).isValid;
+
+ result = getValidationResult(result,
+ achTransferAccountPayload.getCountryCode(),
+ achTransferAccountPayload.getBankName(),
+ achTransferAccountPayload.getBankId(),
+ achTransferAccountPayload.getBranchId(),
+ achTransferAccountPayload.getAccountNr(),
+ achTransferAccountPayload.getAccountType(),
+ achTransferAccountPayload.getHolderTaxId(),
+ achTransferAccountPayload.getNationalAccountId());
+ allInputsValid.set(result);
+ }
+}
diff --git a/desktop/src/main/java/bisq/desktop/components/paymentmethods/DomesticWireTransferForm.java b/desktop/src/main/java/bisq/desktop/components/paymentmethods/DomesticWireTransferForm.java
new file mode 100644
index 00000000000..a4c3f920340
--- /dev/null
+++ b/desktop/src/main/java/bisq/desktop/components/paymentmethods/DomesticWireTransferForm.java
@@ -0,0 +1,89 @@
+/*
+ * 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.components.paymentmethods;
+
+import bisq.core.account.witness.AccountAgeWitnessService;
+import bisq.core.locale.Country;
+import bisq.core.payment.DomesticWireTransferAccount;
+import bisq.core.payment.CountryBasedPaymentAccount;
+import bisq.core.payment.PaymentAccount;
+import bisq.core.payment.payload.BankAccountPayload;
+import bisq.core.payment.payload.DomesticWireTransferAccountPayload;
+import bisq.core.payment.payload.PaymentAccountPayload;
+import bisq.core.util.coin.CoinFormatter;
+import bisq.core.util.validation.InputValidator;
+
+import javafx.scene.layout.GridPane;
+
+public class DomesticWireTransferForm extends GeneralUsBankForm {
+
+ public static int addFormForBuyer(GridPane gridPane, int gridRow, PaymentAccountPayload paymentAccountPayload) {
+ DomesticWireTransferAccountPayload domesticWireTransferAccountPayload = (DomesticWireTransferAccountPayload) paymentAccountPayload;
+ return addFormForBuyer(gridPane, gridRow, paymentAccountPayload, null,
+ domesticWireTransferAccountPayload.getHolderAddress());
+ }
+
+ private final DomesticWireTransferAccount domesticWireTransferAccount;
+
+ public DomesticWireTransferForm(PaymentAccount paymentAccount, AccountAgeWitnessService accountAgeWitnessService, InputValidator inputValidator,
+ GridPane gridPane, int gridRow, CoinFormatter formatter) {
+ super(paymentAccount, accountAgeWitnessService, inputValidator, gridPane, gridRow, formatter);
+ this.domesticWireTransferAccount = (DomesticWireTransferAccount) paymentAccount;
+ }
+
+ @Override
+ public void addFormForDisplayAccount() {
+ addFormForDisplayAccount(domesticWireTransferAccount.getPayload(), domesticWireTransferAccount.getPayload().getHolderAddress());
+ }
+
+ @Override
+ public void addFormForAddAccount() {
+ addFormForAddAccountInternal(domesticWireTransferAccount.getPayload(), domesticWireTransferAccount.getPayload().getHolderAddress());
+ }
+
+ @Override
+ protected void setHolderAddress(String holderAddress) {
+ domesticWireTransferAccount.getPayload().setHolderAddress(holderAddress);
+ }
+
+ @Override
+ protected void maybeAddAccountTypeCombo(BankAccountPayload bankAccountPayload, Country country) {
+ // DomesticWireTransfer does not use the account type combo
+ }
+
+ @Override
+ public void updateAllInputsValid() {
+ DomesticWireTransferAccountPayload domesticWireTransferAccountPayload = domesticWireTransferAccount.getPayload();
+ boolean result = isAccountNameValid()
+ && paymentAccount.getSingleTradeCurrency() != null
+ && ((CountryBasedPaymentAccount) this.paymentAccount).getCountry() != null
+ && inputValidator.validate(domesticWireTransferAccountPayload.getHolderName()).isValid
+ && inputValidator.validate(domesticWireTransferAccountPayload.getHolderAddress()).isValid;
+
+ result = getValidationResult(result,
+ domesticWireTransferAccountPayload.getCountryCode(),
+ domesticWireTransferAccountPayload.getBankName(),
+ domesticWireTransferAccountPayload.getBankId(),
+ domesticWireTransferAccountPayload.getBranchId(),
+ domesticWireTransferAccountPayload.getAccountNr(),
+ domesticWireTransferAccountPayload.getAccountNr(),
+ domesticWireTransferAccountPayload.getHolderTaxId(),
+ domesticWireTransferAccountPayload.getNationalAccountId());
+ allInputsValid.set(result);
+ }
+}
diff --git a/desktop/src/main/java/bisq/desktop/components/paymentmethods/GeneralUsBankForm.java b/desktop/src/main/java/bisq/desktop/components/paymentmethods/GeneralUsBankForm.java
new file mode 100644
index 00000000000..74d196a4a96
--- /dev/null
+++ b/desktop/src/main/java/bisq/desktop/components/paymentmethods/GeneralUsBankForm.java
@@ -0,0 +1,171 @@
+/*
+ * 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.components.paymentmethods;
+
+import bisq.desktop.components.InputTextField;
+import bisq.desktop.util.Layout;
+
+import bisq.core.account.witness.AccountAgeWitnessService;
+import bisq.core.locale.BankUtil;
+import bisq.core.locale.Country;
+import bisq.core.locale.CountryUtil;
+import bisq.core.locale.CurrencyUtil;
+import bisq.core.locale.Res;
+import bisq.core.locale.TradeCurrency;
+import bisq.core.payment.CountryBasedPaymentAccount;
+import bisq.core.payment.PaymentAccount;
+import bisq.core.payment.payload.BankAccountPayload;
+import bisq.core.payment.payload.CountryBasedPaymentAccountPayload;
+import bisq.core.payment.payload.PaymentAccountPayload;
+import bisq.core.util.coin.CoinFormatter;
+import bisq.core.util.validation.InputValidator;
+
+import javafx.scene.control.TextArea;
+import javafx.scene.layout.GridPane;
+
+import javax.annotation.Nullable;
+
+import static bisq.common.util.Utilities.cleanString;
+import static bisq.desktop.util.FormBuilder.*;
+
+public abstract class GeneralUsBankForm extends GeneralBankForm {
+
+ protected static int addFormForBuyer(GridPane gridPane, int gridRow, PaymentAccountPayload paymentAccountPayload,
+ @Nullable String accountType,
+ String holderAddress) {
+ BankAccountPayload bankAccountPayload = (BankAccountPayload) paymentAccountPayload;
+ String countryCode = bankAccountPayload.getCountryCode();
+ int colIndex = 1;
+
+ addTopLabelTextFieldWithCopyIcon(gridPane, getIndexOfColumn(colIndex) == 0 ? ++gridRow : gridRow, getIndexOfColumn(colIndex++),
+ Res.get("payment.account.owner"), bankAccountPayload.getHolderName(), Layout.COMPACT_FIRST_ROW_AND_GROUP_DISTANCE);
+
+ String branchIdLabel = BankUtil.getBranchIdLabel(countryCode);
+ String accountNrLabel = BankUtil.getAccountNrLabel(countryCode);
+ addCompactTopLabelTextFieldWithCopyIcon(gridPane, getIndexOfColumn(colIndex) == 0 ? ++gridRow : gridRow, getIndexOfColumn(colIndex++),
+ branchIdLabel + " / " + accountNrLabel,
+ bankAccountPayload.getBranchId() + " / " + bankAccountPayload.getAccountNr());
+
+ String bankNameLabel = BankUtil.getBankNameLabel(countryCode);
+ String accountTypeLabel = accountType == null ? "" : " / " + BankUtil.getAccountTypeLabel(countryCode);
+ addCompactTopLabelTextFieldWithCopyIcon(gridPane, getIndexOfColumn(colIndex) == 0 ? ++gridRow : gridRow, getIndexOfColumn(colIndex++),
+ bankNameLabel + accountTypeLabel,
+ accountType == null ? bankAccountPayload.getBankName() : bankAccountPayload.getBankName() + " / " + accountType);
+
+ if (holderAddress.length() > 0) {
+ TextArea textAddress = addCompactTopLabelTextArea(gridPane, getIndexOfColumn(colIndex) == 0 ? ++gridRow : gridRow, getIndexOfColumn(colIndex++),
+ Res.get("payment.account.address"), "").second;
+ textAddress.setMinHeight(70);
+ textAddress.setEditable(false);
+ textAddress.setText(holderAddress);
+ }
+ return gridRow;
+ }
+
+ public GeneralUsBankForm(PaymentAccount paymentAccount, AccountAgeWitnessService accountAgeWitnessService, InputValidator inputValidator,
+ GridPane gridPane, int gridRow, CoinFormatter formatter) {
+ super(paymentAccount, accountAgeWitnessService, inputValidator, gridPane, gridRow, formatter);
+ }
+
+ protected void addFormForDisplayAccount(BankAccountPayload bankAccountPayload, String holderAddress) {
+ Country country = ((CountryBasedPaymentAccount) paymentAccount).getCountry();
+ addTopLabelTextField(gridPane, gridRow, Res.get("payment.account.name"),
+ paymentAccount.getAccountName(), Layout.FIRST_ROW_AND_GROUP_DISTANCE);
+ addCompactTopLabelTextField(gridPane, ++gridRow, Res.get("shared.paymentMethod"),
+ Res.get(paymentAccount.getPaymentMethod().getId()));
+ addCompactTopLabelTextField(gridPane, ++gridRow, Res.get("payment.account.owner"), bankAccountPayload.getHolderName());
+ addCompactTopLabelTextField(gridPane, ++gridRow, Res.get("payment.account.owner.address"), cleanString(holderAddress));
+ addCompactTopLabelTextField(gridPane, ++gridRow, Res.get("payment.bank.name"), bankAccountPayload.getBankName());
+ addCompactTopLabelTextField(gridPane, ++gridRow, BankUtil.getBranchIdLabel(country.code), bankAccountPayload.getBranchId());
+ addCompactTopLabelTextField(gridPane, ++gridRow, BankUtil.getAccountNrLabel(country.code), bankAccountPayload.getAccountNr());
+ if (bankAccountPayload.getAccountType() != null) {
+ addCompactTopLabelTextField(gridPane, ++gridRow, BankUtil.getAccountTypeLabel(country.code), bankAccountPayload.getAccountType());
+ }
+ addCompactTopLabelTextField(gridPane, ++gridRow, Res.get("shared.currency"), paymentAccount.getSingleTradeCurrency().getNameAndCode());
+ addCompactTopLabelTextField(gridPane, ++gridRow, Res.get("shared.country"), country.name);
+ addLimitations(true);
+ }
+
+ protected void addFormForAddAccountInternal(BankAccountPayload bankAccountPayload, String holderAddress) {
+ // this payment method is only for United States/USD
+ CountryUtil.findCountryByCode("US").ifPresent(c -> onCountrySelected(c));
+ Country country = ((CountryBasedPaymentAccount) paymentAccount).getCountry();
+
+ InputTextField holderNameInputTextField = addInputTextField(gridPane, ++gridRow, Res.get("payment.account.owner"));
+ holderNameInputTextField.textProperty().addListener((ov, oldValue, newValue) -> {
+ bankAccountPayload.setHolderName(newValue);
+ updateFromInputs();
+ });
+ holderNameInputTextField.setValidator(inputValidator);
+
+ TextArea addressTextArea = addTopLabelTextArea(gridPane, ++gridRow,
+ Res.get("payment.account.owner.address"), Res.get("payment.account.owner.address")).second;
+ addressTextArea.setMinHeight(70);
+ addressTextArea.textProperty().addListener((ov, oldValue, newValue) -> {
+ setHolderAddress(newValue.trim());
+ updateFromInputs();
+ });
+
+ bankNameInputTextField = addInputTextField(gridPane, ++gridRow, Res.get("payment.bank.name"));
+ bankNameInputTextField.textProperty().addListener((ov, oldValue, newValue) -> {
+ bankAccountPayload.setBankName(newValue);
+ updateFromInputs();
+ });
+ bankNameInputTextField.setValidator(inputValidator);
+
+ branchIdInputTextField = addInputTextField(gridPane, ++gridRow, BankUtil.getBranchIdLabel(country.code));
+ branchIdInputTextField.textProperty().addListener((ov, oldValue, newValue) -> {
+ bankAccountPayload.setBranchId(newValue);
+ updateFromInputs();
+ });
+ branchIdInputTextField.setValidator(inputValidator);
+
+ accountNrInputTextField = addInputTextField(gridPane, ++gridRow, BankUtil.getAccountNrLabel(country.code));
+ accountNrInputTextField.textProperty().addListener((ov, oldValue, newValue) -> {
+ bankAccountPayload.setAccountNr(newValue);
+ updateFromInputs();
+ });
+ accountNrInputTextField.setValidator(inputValidator);
+
+ maybeAddAccountTypeCombo(bankAccountPayload, country);
+
+ addTopLabelTextField(gridPane, ++gridRow, Res.get("shared.currency"), paymentAccount.getSingleTradeCurrency().getNameAndCode());
+ addTopLabelTextField(gridPane, ++gridRow, Res.get("shared.country"), country.name);
+ addLimitations(false);
+ addAccountNameTextFieldWithAutoFillToggleButton();
+ this.validatorsApplied = true;
+ }
+
+ abstract protected void setHolderAddress(String holderAddress);
+
+ abstract protected void maybeAddAccountTypeCombo(BankAccountPayload bankAccountPayload, Country country);
+
+ protected void onCountrySelected(Country country) {
+ if (country != null) {
+ ((CountryBasedPaymentAccount) this.paymentAccount).setCountry(country);
+ String countryCode = country.code;
+ TradeCurrency currency = CurrencyUtil.getCurrencyByCountryCode(countryCode);
+ paymentAccount.setSingleTradeCurrency(currency);
+ }
+ }
+
+ @Override
+ protected void autoFillNameTextField() {
+ autoFillAccountTextFields((CountryBasedPaymentAccountPayload) paymentAccount.getPaymentAccountPayload());
+ }
+}
diff --git a/desktop/src/main/java/bisq/desktop/components/paymentmethods/TransferwiseUsdForm.java b/desktop/src/main/java/bisq/desktop/components/paymentmethods/TransferwiseUsdForm.java
index 69a5e9183e6..359afaff8ff 100644
--- a/desktop/src/main/java/bisq/desktop/components/paymentmethods/TransferwiseUsdForm.java
+++ b/desktop/src/main/java/bisq/desktop/components/paymentmethods/TransferwiseUsdForm.java
@@ -94,7 +94,8 @@ public void addFormForAddAccount() {
updateFromInputs();
});
- TextArea addressTextArea = addTopLabelTextArea(gridPane, ++gridRow, Res.get("payment.transferwiseUsd.address"), Res.get("payment.transferwiseUsd.address")).second;
+ String addressLabel = Res.get("payment.account.owner.address") + Res.get("payment.transferwiseUsd.address");
+ TextArea addressTextArea = addTopLabelTextArea(gridPane, ++gridRow, addressLabel, addressLabel).second;
addressTextArea.setMinHeight(70);
addressTextArea.textProperty().addListener((ov, oldValue, newValue) -> {
account.setBeneficiaryAddress(newValue.trim());
diff --git a/desktop/src/main/java/bisq/desktop/main/account/content/fiataccounts/FiatAccountsView.java b/desktop/src/main/java/bisq/desktop/main/account/content/fiataccounts/FiatAccountsView.java
index 178c63084bd..bad77df7d83 100644
--- a/desktop/src/main/java/bisq/desktop/main/account/content/fiataccounts/FiatAccountsView.java
+++ b/desktop/src/main/java/bisq/desktop/main/account/content/fiataccounts/FiatAccountsView.java
@@ -19,6 +19,7 @@
import bisq.desktop.common.view.FxmlView;
import bisq.desktop.components.TitledGroupBg;
+import bisq.desktop.components.paymentmethods.AchTransferForm;
import bisq.desktop.components.paymentmethods.AdvancedCashForm;
import bisq.desktop.components.paymentmethods.AliPayForm;
import bisq.desktop.components.paymentmethods.AmazonGiftCardForm;
@@ -30,6 +31,7 @@
import bisq.desktop.components.paymentmethods.CelPayForm;
import bisq.desktop.components.paymentmethods.ChaseQuickPayForm;
import bisq.desktop.components.paymentmethods.ClearXchangeForm;
+import bisq.desktop.components.paymentmethods.DomesticWireTransferForm;
import bisq.desktop.components.paymentmethods.F2FForm;
import bisq.desktop.components.paymentmethods.FasterPaymentsForm;
import bisq.desktop.components.paymentmethods.HalCashForm;
@@ -597,6 +599,10 @@ private PaymentMethodForm getPaymentMethodForm(PaymentMethod paymentMethod, Paym
return new StrikeForm(paymentAccount, accountAgeWitnessService, inputValidator, root, gridRow, formatter);
case PaymentMethod.SWIFT_ID:
return new SwiftForm(paymentAccount, accountAgeWitnessService, inputValidator, root, gridRow, formatter);
+ case PaymentMethod.ACH_TRANSFER_ID:
+ return new AchTransferForm(paymentAccount, accountAgeWitnessService, inputValidator, root, gridRow, formatter);
+ case PaymentMethod.DOMESTIC_WIRE_TRANSFER_ID:
+ return new DomesticWireTransferForm(paymentAccount, accountAgeWitnessService, inputValidator, root, gridRow, formatter);
default:
log.error("Not supported PaymentMethod: " + paymentMethod);
return null;
diff --git a/desktop/src/main/java/bisq/desktop/main/portfolio/pendingtrades/steps/buyer/BuyerStep2View.java b/desktop/src/main/java/bisq/desktop/main/portfolio/pendingtrades/steps/buyer/BuyerStep2View.java
index a2ee010eec7..304b2a48a29 100644
--- a/desktop/src/main/java/bisq/desktop/main/portfolio/pendingtrades/steps/buyer/BuyerStep2View.java
+++ b/desktop/src/main/java/bisq/desktop/main/portfolio/pendingtrades/steps/buyer/BuyerStep2View.java
@@ -21,6 +21,7 @@
import bisq.desktop.components.BusyAnimation;
import bisq.desktop.components.TextFieldWithCopyIcon;
import bisq.desktop.components.TitledGroupBg;
+import bisq.desktop.components.paymentmethods.AchTransferForm;
import bisq.desktop.components.paymentmethods.AdvancedCashForm;
import bisq.desktop.components.paymentmethods.AliPayForm;
import bisq.desktop.components.paymentmethods.AmazonGiftCardForm;
@@ -32,6 +33,7 @@
import bisq.desktop.components.paymentmethods.CelPayForm;
import bisq.desktop.components.paymentmethods.ChaseQuickPayForm;
import bisq.desktop.components.paymentmethods.ClearXchangeForm;
+import bisq.desktop.components.paymentmethods.DomesticWireTransferForm;
import bisq.desktop.components.paymentmethods.F2FForm;
import bisq.desktop.components.paymentmethods.FasterPaymentsForm;
import bisq.desktop.components.paymentmethods.HalCashForm;
@@ -400,6 +402,12 @@ protected void addContent() {
case PaymentMethod.SWIFT_ID:
gridRow = SwiftForm.addFormForBuyer(gridPane, gridRow, paymentAccountPayload, trade);
break;
+ case PaymentMethod.ACH_TRANSFER_ID:
+ gridRow = AchTransferForm.addFormForBuyer(gridPane, gridRow, paymentAccountPayload);
+ break;
+ case PaymentMethod.DOMESTIC_WIRE_TRANSFER_ID:
+ gridRow = DomesticWireTransferForm.addFormForBuyer(gridPane, gridRow, paymentAccountPayload);
+ break;
default:
log.error("Not supported PaymentMethod: " + paymentMethodId);
}
diff --git a/proto/src/main/proto/pb.proto b/proto/src/main/proto/pb.proto
index 255cc52c3aa..e305174798c 100644
--- a/proto/src/main/proto/pb.proto
+++ b/proto/src/main/proto/pb.proto
@@ -1151,10 +1151,20 @@ message BankAccountPayload {
NationalBankAccountPayload national_bank_account_payload = 9;
SameBankAccountPayload same_bank_accont_payload = 10;
SpecificBanksAccountPayload specific_banks_account_payload = 11;
+ AchTransferAccountPayload ach_transfer_account_payload = 13;
+ DomesticWireTransferAccountPayload domestic_wire_transfer_account_payload = 14;
}
string national_account_id = 12;
}
+message AchTransferAccountPayload {
+ string holder_address = 1;
+}
+
+message DomesticWireTransferAccountPayload {
+ string holder_address = 1;
+}
+
message NationalBankAccountPayload {
}