|
@ -2,17 +2,23 @@ package com.sparrowwallet.sparrow.transaction; |
|
|
|
|
|
|
|
|
import com.sparrowwallet.drongo.protocol.Script; |
|
|
import com.sparrowwallet.drongo.protocol.Script; |
|
|
import com.sparrowwallet.drongo.protocol.ScriptChunk; |
|
|
import com.sparrowwallet.drongo.protocol.ScriptChunk; |
|
|
|
|
|
import com.sparrowwallet.drongo.protocol.Transaction; |
|
|
import com.sparrowwallet.drongo.protocol.TransactionInput; |
|
|
import com.sparrowwallet.drongo.protocol.TransactionInput; |
|
|
import com.sparrowwallet.drongo.psbt.PSBTInput; |
|
|
import com.sparrowwallet.drongo.psbt.PSBTInput; |
|
|
|
|
|
import com.sparrowwallet.sparrow.EventManager; |
|
|
import javafx.fxml.FXML; |
|
|
import javafx.fxml.FXML; |
|
|
import javafx.fxml.Initializable; |
|
|
import javafx.fxml.Initializable; |
|
|
import javafx.scene.control.Button; |
|
|
import javafx.scene.control.*; |
|
|
import javafx.scene.control.TextField; |
|
|
|
|
|
import org.fxmisc.richtext.CodeArea; |
|
|
import org.fxmisc.richtext.CodeArea; |
|
|
|
|
|
import tornadofx.control.Field; |
|
|
import tornadofx.control.Fieldset; |
|
|
import tornadofx.control.Fieldset; |
|
|
import org.fxmisc.flowless.VirtualizedScrollPane; |
|
|
import org.fxmisc.flowless.VirtualizedScrollPane; |
|
|
|
|
|
|
|
|
import java.net.URL; |
|
|
import java.net.URL; |
|
|
|
|
|
import java.time.Instant; |
|
|
|
|
|
import java.time.LocalDateTime; |
|
|
|
|
|
import java.time.ZoneId; |
|
|
|
|
|
import java.time.format.DateTimeFormatter; |
|
|
import java.util.List; |
|
|
import java.util.List; |
|
|
import java.util.ResourceBundle; |
|
|
import java.util.ResourceBundle; |
|
|
|
|
|
|
|
@ -49,6 +55,42 @@ public class InputController extends TransactionFormController implements Initia |
|
|
@FXML |
|
|
@FXML |
|
|
private CodeArea witnessesArea; |
|
|
private CodeArea witnessesArea; |
|
|
|
|
|
|
|
|
|
|
|
@FXML |
|
|
|
|
|
private ToggleGroup locktimeToggleGroup; |
|
|
|
|
|
|
|
|
|
|
|
@FXML |
|
|
|
|
|
private ToggleButton locktimeNoneType; |
|
|
|
|
|
|
|
|
|
|
|
@FXML |
|
|
|
|
|
private ToggleButton locktimeAbsoluteType; |
|
|
|
|
|
|
|
|
|
|
|
@FXML |
|
|
|
|
|
private ToggleButton locktimeRelativeType; |
|
|
|
|
|
|
|
|
|
|
|
@FXML |
|
|
|
|
|
private Fieldset locktimeFieldset; |
|
|
|
|
|
|
|
|
|
|
|
@FXML |
|
|
|
|
|
private Field locktimeNoneField; |
|
|
|
|
|
|
|
|
|
|
|
@FXML |
|
|
|
|
|
private Field locktimeAbsoluteField; |
|
|
|
|
|
|
|
|
|
|
|
@FXML |
|
|
|
|
|
private Field locktimeRelativeField; |
|
|
|
|
|
|
|
|
|
|
|
@FXML |
|
|
|
|
|
private Spinner<Integer> locktimeNone; |
|
|
|
|
|
|
|
|
|
|
|
@FXML |
|
|
|
|
|
private TextField locktimeAbsolute; |
|
|
|
|
|
|
|
|
|
|
|
@FXML |
|
|
|
|
|
private Spinner<Integer> locktimeRelative; |
|
|
|
|
|
|
|
|
|
|
|
@FXML |
|
|
|
|
|
private ComboBox<String> locktimeRelativeCombo; |
|
|
|
|
|
|
|
|
@Override |
|
|
@Override |
|
|
public void initialize(URL location, ResourceBundle resources) { |
|
|
public void initialize(URL location, ResourceBundle resources) { |
|
|
|
|
|
|
|
@ -63,7 +105,12 @@ public class InputController extends TransactionFormController implements Initia |
|
|
|
|
|
|
|
|
//TODO: Enable select outpoint when wallet present
|
|
|
//TODO: Enable select outpoint when wallet present
|
|
|
outpointSelect.setDisable(true); |
|
|
outpointSelect.setDisable(true); |
|
|
|
|
|
initializeScriptFields(txInput); |
|
|
|
|
|
|
|
|
|
|
|
initializeLocktimeFields(txInput); |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
private void initializeScriptFields(TransactionInput txInput) { |
|
|
//TODO: Is this safe?
|
|
|
//TODO: Is this safe?
|
|
|
Script redeemScript = txInput.getScriptSig().getFirstNestedScript(); |
|
|
Script redeemScript = txInput.getScriptSig().getFirstNestedScript(); |
|
|
|
|
|
|
|
@ -95,6 +142,75 @@ public class InputController extends TransactionFormController implements Initia |
|
|
} |
|
|
} |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
private void initializeLocktimeFields(TransactionInput txInput) { |
|
|
|
|
|
Transaction transaction = inputForm.getTransaction(); |
|
|
|
|
|
locktimeToggleGroup.selectedToggleProperty().addListener((ov, old_toggle, new_toggle) -> { |
|
|
|
|
|
if(locktimeToggleGroup.getSelectedToggle() != null) { |
|
|
|
|
|
String selection = locktimeToggleGroup.getSelectedToggle().getUserData().toString(); |
|
|
|
|
|
if(selection.equals("none")) { |
|
|
|
|
|
locktimeFieldset.getChildren().removeAll(locktimeRelativeField, locktimeAbsoluteField, locktimeNoneField); |
|
|
|
|
|
locktimeFieldset.getChildren().add(locktimeNoneField); |
|
|
|
|
|
txInput.setSequenceNumber(TransactionInput.SEQUENCE_LOCKTIME_DISABLED); |
|
|
|
|
|
EventManager.get().notify(transaction); |
|
|
|
|
|
} else if(selection.equals("absolute")) { |
|
|
|
|
|
locktimeFieldset.getChildren().removeAll(locktimeRelativeField, locktimeAbsoluteField, locktimeNoneField); |
|
|
|
|
|
locktimeFieldset.getChildren().add(locktimeAbsoluteField); |
|
|
|
|
|
long locktime = transaction.getLocktime(); |
|
|
|
|
|
if(locktime < Transaction.MAX_BLOCK_LOCKTIME) { |
|
|
|
|
|
locktimeAbsoluteField.setText("Block:"); |
|
|
|
|
|
locktimeAbsolute.setText(Long.toString(locktime)); |
|
|
|
|
|
} else { |
|
|
|
|
|
locktimeAbsoluteField.setText("Date:"); |
|
|
|
|
|
LocalDateTime localDateTime = Instant.ofEpochSecond(locktime).atZone(ZoneId.systemDefault()).toLocalDateTime(); |
|
|
|
|
|
locktimeAbsolute.setText(DateTimeFormatter.ofPattern(HeadersController.LOCKTIME_DATE_FORMAT).format(localDateTime)); |
|
|
|
|
|
} |
|
|
|
|
|
//TODO: Check RBF field and set appropriately
|
|
|
|
|
|
txInput.setSequenceNumber(TransactionInput.SEQUENCE_LOCKTIME_DISABLED - 1); |
|
|
|
|
|
EventManager.get().notify(transaction); |
|
|
|
|
|
} else { |
|
|
|
|
|
locktimeFieldset.getChildren().removeAll(locktimeRelativeField, locktimeAbsoluteField, locktimeNoneField); |
|
|
|
|
|
locktimeFieldset.getChildren().add(locktimeRelativeField); |
|
|
|
|
|
setRelativeLocktime(txInput, transaction, locktimeRelative.getValue()); |
|
|
|
|
|
} |
|
|
|
|
|
} |
|
|
|
|
|
}); |
|
|
|
|
|
|
|
|
|
|
|
locktimeNone.setValueFactory(new SpinnerValueFactory.IntegerSpinnerValueFactory(0, (int)Transaction.MAX_BLOCK_LOCKTIME-1, (int)transaction.getLocktime())); |
|
|
|
|
|
locktimeRelativeCombo.getSelectionModel().select(0); |
|
|
|
|
|
locktimeRelative.setValueFactory(new SpinnerValueFactory.IntegerSpinnerValueFactory(0, (int)TransactionInput.MAX_RELATIVE_TIMELOCK_IN_BLOCKS, 0)); |
|
|
|
|
|
if(txInput.isAbsoluteTimeLockDisabled()) { |
|
|
|
|
|
locktimeToggleGroup.selectToggle(locktimeNoneType); |
|
|
|
|
|
} else if(txInput.isAbsoluteTimeLocked()) { |
|
|
|
|
|
locktimeToggleGroup.selectToggle(locktimeAbsoluteType); |
|
|
|
|
|
} else { |
|
|
|
|
|
locktimeRelative.valueFactoryProperty().get().setValue((int)txInput.getRelativeLocktime()); |
|
|
|
|
|
if(txInput.isRelativeTimeLockedInBlocks()) { |
|
|
|
|
|
locktimeRelativeCombo.getSelectionModel().select(0); |
|
|
|
|
|
} else { |
|
|
|
|
|
locktimeRelativeCombo.getSelectionModel().select(1); |
|
|
|
|
|
} |
|
|
|
|
|
locktimeToggleGroup.selectToggle(locktimeRelativeType); |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
locktimeRelative.valueProperty().addListener((obs, oldValue, newValue) -> { |
|
|
|
|
|
setRelativeLocktime(txInput, transaction, newValue); |
|
|
|
|
|
}); |
|
|
|
|
|
|
|
|
|
|
|
locktimeRelativeCombo.getSelectionModel().selectedItemProperty().addListener((ov, old_toggle, new_toggle) -> { |
|
|
|
|
|
setRelativeLocktime(txInput, transaction, locktimeRelative.getValue()); |
|
|
|
|
|
}); |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
private void setRelativeLocktime(TransactionInput txInput, Transaction transaction, Integer value) { |
|
|
|
|
|
String relativeSelection = locktimeRelativeCombo.getValue(); |
|
|
|
|
|
if (relativeSelection.equals("blocks")) { |
|
|
|
|
|
txInput.setSequenceNumber(value & 0xFFFF); |
|
|
|
|
|
} else { |
|
|
|
|
|
txInput.setSequenceNumber((value & 0xFFFF) | 0x400000); |
|
|
|
|
|
} |
|
|
|
|
|
EventManager.get().notify(transaction); |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
public void setModel(InputForm form) { |
|
|
public void setModel(InputForm form) { |
|
|
this.inputForm = form; |
|
|
this.inputForm = form; |
|
|
initializeView(); |
|
|
initializeView(); |
|
|