diff --git a/src/qt/psbtoperationsdialog.cpp b/src/qt/psbtoperationsdialog.cpp
index 5a4b4442f3d..3952c7b39a6 100644
--- a/src/qt/psbtoperationsdialog.cpp
+++ b/src/qt/psbtoperationsdialog.cpp
@@ -83,6 +83,10 @@ void PSBTOperationsDialog::signTransaction()
WalletModel::UnlockContext ctx(m_wallet_model->requestUnlock());
+ // For multisigs we indicate when a participant successfully signs the PSBT
+ // even when additional signatures are required for the PSBT to be complete.
+ // Simply check the number of partial signatures before and after signing.
+ const size_t n_partial_sigs_before{m_transaction_data.inputs.at(0).partial_sigs.size()};
const auto err{m_wallet_model->wallet().fillPSBT(SIGHASH_ALL, /*sign=*/true, /*bip32derivs=*/true, &n_signed, m_transaction_data, complete)};
if (err) {
@@ -93,13 +97,26 @@ void PSBTOperationsDialog::signTransaction()
updateTransactionDisplay();
+ const size_t n_partial_sigs_after{m_transaction_data.inputs.at(0).partial_sigs.size()};
+ // We also indicate when the participant successfully signed every tx input.
+ // To keep things simple we only check if this is not true, in which case
+ // fallback to the old (vague) behavior where it's unclear if the PSBT was
+ // successfully partially signed.
+ const bool all_inputs_n_partial_sigs_equal{std::all_of(m_transaction_data.inputs.begin(), m_transaction_data.inputs.end(),
+ [&](const PSBTInput& input) { return n_partial_sigs_after == input.partial_sigs.size(); })};
+
if (!complete && !ctx.isValid()) {
showStatus(tr("Cannot sign inputs while wallet is locked."), StatusLevel::WARN);
- } else if (!complete && n_signed < 1) {
- showStatus(tr("Could not sign any more inputs."), StatusLevel::WARN);
} else if (!complete) {
- showStatus(tr("Signed %1 inputs, but more signatures are still required.").arg(n_signed),
- StatusLevel::INFO);
+ if (all_inputs_n_partial_sigs_equal && n_partial_sigs_after - n_partial_sigs_before > 0) {
+ showStatus(tr("Added %1 partial signature to each input, but more signatures are still required.").arg(n_partial_sigs_after - n_partial_sigs_before),
+ StatusLevel::INFO);
+ } else if (n_signed < 1) {
+ showStatus(tr("Could not sign any more inputs."), StatusLevel::WARN);
+ } else {
+ showStatus(tr("Signed %1 inputs, but more signatures are still required.").arg(n_signed),
+ StatusLevel::INFO);
+ }
} else {
showStatus(tr("Signed transaction successfully. Transaction is ready to broadcast."),
StatusLevel::INFO);
@@ -217,9 +234,16 @@ QString PSBTOperationsDialog::renderTransaction(const PartiallySignedTransaction
}
size_t num_unsigned = CountPSBTUnsignedInputs(psbtx);
+ const size_t num_partial_sigs{psbtx.inputs.at(0).partial_sigs.size()};
+ const bool all_inputs_n_partial_sigs_equal{std::all_of(psbtx.inputs.begin(), psbtx.inputs.end(),
+ [&](const PSBTInput& input) { return num_partial_sigs == input.partial_sigs.size(); })};
+
if (num_unsigned > 0) {
tx_description.append("
");
tx_description.append(tr("Transaction has %1 unsigned inputs.").arg(QString::number(num_unsigned)));
+ if (all_inputs_n_partial_sigs_equal && num_partial_sigs > 0) {
+ tx_description.append(tr(" Each input has %1 partial signatures.").arg(QString::number(num_partial_sigs)));
+ }
}
return tx_description;