|
@ -50,20 +50,29 @@ public class TransactionController implements Initializable { |
|
|
private PSBT psbt; |
|
|
private PSBT psbt; |
|
|
private BlockTransaction blockTransaction; |
|
|
private BlockTransaction blockTransaction; |
|
|
|
|
|
|
|
|
|
|
|
private TransactionView initialView; |
|
|
|
|
|
private Integer initialIndex; |
|
|
|
|
|
|
|
|
private int selectedInputIndex = -1; |
|
|
private int selectedInputIndex = -1; |
|
|
private int selectedOutputIndex = -1; |
|
|
private int selectedOutputIndex = -1; |
|
|
|
|
|
|
|
|
|
|
|
private int highestInputIndex; |
|
|
|
|
|
private int highestOutputIndex; |
|
|
|
|
|
|
|
|
@Override |
|
|
@Override |
|
|
public void initialize(URL location, ResourceBundle resources) { |
|
|
public void initialize(URL location, ResourceBundle resources) { |
|
|
EventManager.get().register(this); |
|
|
EventManager.get().register(this); |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
private void initializeView() { |
|
|
private void initializeView() { |
|
|
|
|
|
highestInputIndex = Math.min(transaction.getInputs().size(), PageForm.PAGE_SIZE); |
|
|
|
|
|
highestOutputIndex = Math.min(transaction.getOutputs().size(), PageForm.PAGE_SIZE); |
|
|
|
|
|
|
|
|
initializeTxTree(); |
|
|
initializeTxTree(); |
|
|
transactionMasterDetail.setShowDetailNode(AppController.showTxHexProperty); |
|
|
transactionMasterDetail.setShowDetailNode(AppController.showTxHexProperty); |
|
|
refreshTxHex(); |
|
|
refreshTxHex(); |
|
|
fetchThisAndInputBlockTransactions(); |
|
|
fetchThisAndInputBlockTransactions(0, highestInputIndex); |
|
|
fetchOutputBlockTransactions(); |
|
|
fetchOutputBlockTransactions(0, highestOutputIndex); |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
private void initializeTxTree() { |
|
|
private void initializeTxTree() { |
|
@ -74,27 +83,33 @@ public class TransactionController implements Initializable { |
|
|
InputsForm inputsForm = (psbt != null ? new InputsForm(psbt) : (blockTransaction != null ? new InputsForm(blockTransaction) : new InputsForm(transaction))); |
|
|
InputsForm inputsForm = (psbt != null ? new InputsForm(psbt) : (blockTransaction != null ? new InputsForm(blockTransaction) : new InputsForm(transaction))); |
|
|
TreeItem<TransactionForm> inputsItem = new TreeItem<>(inputsForm); |
|
|
TreeItem<TransactionForm> inputsItem = new TreeItem<>(inputsForm); |
|
|
inputsItem.setExpanded(true); |
|
|
inputsItem.setExpanded(true); |
|
|
for (TransactionInput txInput : transaction.getInputs()) { |
|
|
boolean inputPagingAdded = false; |
|
|
PSBTInput psbtInput = null; |
|
|
for(int i = 0; i < transaction.getInputs().size(); i++) { |
|
|
if (psbt != null && psbt.getPsbtInputs().size() > txInput.getIndex()) { |
|
|
if(i < PageForm.PAGE_SIZE || (TransactionView.INPUT.equals(initialView) && i == initialIndex)) { |
|
|
psbtInput = psbt.getPsbtInputs().get(txInput.getIndex()); |
|
|
TreeItem<TransactionForm> inputItem = createInputTreeItem(i); |
|
|
|
|
|
inputsItem.getChildren().add(inputItem); |
|
|
|
|
|
} else if(!inputPagingAdded) { |
|
|
|
|
|
PageForm pageForm = new PageForm(TransactionView.INPUT, i, i + PageForm.PAGE_SIZE); |
|
|
|
|
|
TreeItem<TransactionForm> pageItem = new TreeItem<>(pageForm); |
|
|
|
|
|
inputsItem.getChildren().add(pageItem); |
|
|
|
|
|
inputPagingAdded = true; |
|
|
} |
|
|
} |
|
|
InputForm inputForm = (psbt != null ? new InputForm(psbt, psbtInput) : (blockTransaction != null ? new InputForm(blockTransaction, txInput) : new InputForm(transaction, txInput))); |
|
|
|
|
|
TreeItem<TransactionForm> inputItem = new TreeItem<>(inputForm); |
|
|
|
|
|
inputsItem.getChildren().add(inputItem); |
|
|
|
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
OutputsForm outputsForm = (psbt != null ? new OutputsForm(psbt) : (blockTransaction != null ? new OutputsForm(blockTransaction) : new OutputsForm(transaction))); |
|
|
OutputsForm outputsForm = (psbt != null ? new OutputsForm(psbt) : (blockTransaction != null ? new OutputsForm(blockTransaction) : new OutputsForm(transaction))); |
|
|
TreeItem<TransactionForm> outputsItem = new TreeItem<>(outputsForm); |
|
|
TreeItem<TransactionForm> outputsItem = new TreeItem<>(outputsForm); |
|
|
outputsItem.setExpanded(true); |
|
|
outputsItem.setExpanded(true); |
|
|
for (TransactionOutput txOutput : transaction.getOutputs()) { |
|
|
boolean outputPagingAdded = false; |
|
|
PSBTOutput psbtOutput = null; |
|
|
for(int i = 0; i < transaction.getOutputs().size(); i++) { |
|
|
if (psbt != null && psbt.getPsbtOutputs().size() > txOutput.getIndex()) { |
|
|
if(i < PageForm.PAGE_SIZE || (TransactionView.OUTPUT.equals(initialView) && i == initialIndex)) { |
|
|
psbtOutput = psbt.getPsbtOutputs().get(txOutput.getIndex()); |
|
|
TreeItem<TransactionForm> outputItem = createOutputTreeItem(i); |
|
|
|
|
|
outputsItem.getChildren().add(outputItem); |
|
|
|
|
|
} else if(!outputPagingAdded) { |
|
|
|
|
|
PageForm pageForm = new PageForm(TransactionView.OUTPUT, i, i + PageForm.PAGE_SIZE); |
|
|
|
|
|
TreeItem<TransactionForm> pageItem = new TreeItem<>(pageForm); |
|
|
|
|
|
outputsItem.getChildren().add(pageItem); |
|
|
|
|
|
outputPagingAdded = true; |
|
|
} |
|
|
} |
|
|
OutputForm outputForm = (psbt != null ? new OutputForm(psbt, psbtOutput) : (blockTransaction != null ? new OutputForm(blockTransaction, txOutput) : new OutputForm(transaction, txOutput))); |
|
|
|
|
|
TreeItem<TransactionForm> outputItem = new TreeItem<>(outputForm); |
|
|
|
|
|
outputsItem.getChildren().add(outputItem); |
|
|
|
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
rootItem.getChildren().add(inputsItem); |
|
|
rootItem.getChildren().add(inputsItem); |
|
@ -113,36 +128,97 @@ public class TransactionController implements Initializable { |
|
|
} |
|
|
} |
|
|
})); |
|
|
})); |
|
|
|
|
|
|
|
|
txtree.getSelectionModel().selectedItemProperty().addListener((observable, old_val, new_val) -> { |
|
|
txtree.getSelectionModel().selectedItemProperty().addListener((observable, old_val, selectedItem) -> { |
|
|
TransactionForm transactionForm = new_val.getValue(); |
|
|
TransactionForm transactionForm = selectedItem.getValue(); |
|
|
try { |
|
|
if(transactionForm instanceof PageForm) { |
|
|
Node node = transactionForm.getContents(); |
|
|
PageForm pageForm = (PageForm)transactionForm; |
|
|
txpane.getChildren().clear(); |
|
|
Optional<TreeItem<TransactionForm>> optParentItem = txtree.getRoot().getChildren().stream() |
|
|
txpane.getChildren().add(node); |
|
|
.filter(item -> item.getValue().getView().equals(pageForm.getView().equals(TransactionView.INPUT) ? TransactionView.INPUTS : TransactionView.OUTPUTS)).findFirst(); |
|
|
|
|
|
|
|
|
if (node instanceof Parent) { |
|
|
if(optParentItem.isPresent()) { |
|
|
Parent parent = (Parent) node; |
|
|
TreeItem<TransactionForm> parentItem = optParentItem.get(); |
|
|
txhex.getStylesheets().clear(); |
|
|
parentItem.getChildren().remove(selectedItem); |
|
|
txhex.getStylesheets().addAll(parent.getStylesheets()); |
|
|
|
|
|
|
|
|
int max = pageForm.getView().equals(TransactionView.INPUT) ? transaction.getInputs().size() : transaction.getOutputs().size(); |
|
|
selectedInputIndex = -1; |
|
|
for(int i = pageForm.getPageStart(); i < max && i < pageForm.getPageEnd(); i++) { |
|
|
selectedOutputIndex = -1; |
|
|
TreeItem<TransactionForm> newItem = pageForm.getView().equals(TransactionView.INPUT) ? createInputTreeItem(i) : createOutputTreeItem(i); |
|
|
if (transactionForm instanceof InputForm) { |
|
|
parentItem.getChildren().add(newItem); |
|
|
InputForm inputForm = (InputForm) transactionForm; |
|
|
} |
|
|
selectedInputIndex = inputForm.getTransactionInput().getIndex(); |
|
|
|
|
|
} else if (transactionForm instanceof OutputForm) { |
|
|
if(pageForm.getPageEnd() < max) { |
|
|
OutputForm outputForm = (OutputForm) transactionForm; |
|
|
PageForm nextPageForm = new PageForm(pageForm.getView(), pageForm.getPageStart() + PageForm.PAGE_SIZE, pageForm.getPageEnd() + PageForm.PAGE_SIZE); |
|
|
selectedOutputIndex = outputForm.getTransactionOutput().getIndex(); |
|
|
TreeItem<TransactionForm> nextPageItem = new TreeItem<>(nextPageForm); |
|
|
|
|
|
parentItem.getChildren().add(nextPageItem); |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
Platform.runLater(this::refreshTxHex); |
|
|
if(pageForm.getView().equals(TransactionView.INPUT)) { |
|
|
|
|
|
highestInputIndex = Math.min(max, pageForm.getPageEnd()); |
|
|
|
|
|
fetchThisAndInputBlockTransactions(pageForm.getPageStart(), Math.min(max, pageForm.getPageEnd())); |
|
|
|
|
|
} else { |
|
|
|
|
|
highestOutputIndex = Math.min(max, pageForm.getPageEnd()); |
|
|
|
|
|
fetchOutputBlockTransactions(pageForm.getPageStart(), Math.min(max, pageForm.getPageEnd())); |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
setTreeSelection(pageForm.getView(), pageForm.getPageStart()); |
|
|
|
|
|
Platform.runLater(() -> { |
|
|
|
|
|
txtree.scrollTo(pageForm.getPageStart()); |
|
|
|
|
|
refreshTxHex(); |
|
|
|
|
|
}); |
|
|
|
|
|
} |
|
|
|
|
|
} else { |
|
|
|
|
|
try { |
|
|
|
|
|
Node node = transactionForm.getContents(); |
|
|
|
|
|
txpane.getChildren().clear(); |
|
|
|
|
|
txpane.getChildren().add(node); |
|
|
|
|
|
|
|
|
|
|
|
if (node instanceof Parent) { |
|
|
|
|
|
Parent parent = (Parent) node; |
|
|
|
|
|
txhex.getStylesheets().clear(); |
|
|
|
|
|
txhex.getStylesheets().addAll(parent.getStylesheets()); |
|
|
|
|
|
|
|
|
|
|
|
selectedInputIndex = -1; |
|
|
|
|
|
selectedOutputIndex = -1; |
|
|
|
|
|
if (transactionForm instanceof InputForm) { |
|
|
|
|
|
InputForm inputForm = (InputForm) transactionForm; |
|
|
|
|
|
selectedInputIndex = inputForm.getTransactionInput().getIndex(); |
|
|
|
|
|
} else if (transactionForm instanceof OutputForm) { |
|
|
|
|
|
OutputForm outputForm = (OutputForm) transactionForm; |
|
|
|
|
|
selectedOutputIndex = outputForm.getTransactionOutput().getIndex(); |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
Platform.runLater(this::refreshTxHex); |
|
|
|
|
|
} |
|
|
|
|
|
} catch (IOException e) { |
|
|
|
|
|
throw new IllegalStateException("Can't find pane", e); |
|
|
} |
|
|
} |
|
|
} catch (IOException e) { |
|
|
|
|
|
throw new IllegalStateException("Can't find pane", e); |
|
|
|
|
|
} |
|
|
} |
|
|
}); |
|
|
}); |
|
|
|
|
|
|
|
|
txtree.getSelectionModel().select(txtree.getRoot()); |
|
|
if(initialView != null) { |
|
|
|
|
|
setTreeSelection(initialView, initialIndex); |
|
|
|
|
|
} else { |
|
|
|
|
|
txtree.getSelectionModel().select(txtree.getRoot()); |
|
|
|
|
|
} |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
private TreeItem<TransactionForm> createInputTreeItem(int inputIndex) { |
|
|
|
|
|
TransactionInput txInput = transaction.getInputs().get(inputIndex); |
|
|
|
|
|
PSBTInput psbtInput = null; |
|
|
|
|
|
if (psbt != null && psbt.getPsbtInputs().size() > txInput.getIndex()) { |
|
|
|
|
|
psbtInput = psbt.getPsbtInputs().get(txInput.getIndex()); |
|
|
|
|
|
} |
|
|
|
|
|
InputForm inputForm = (psbt != null ? new InputForm(psbt, psbtInput) : (blockTransaction != null ? new InputForm(blockTransaction, txInput) : new InputForm(transaction, txInput))); |
|
|
|
|
|
return new TreeItem<>(inputForm); |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
private TreeItem<TransactionForm> createOutputTreeItem(int outputIndex) { |
|
|
|
|
|
TransactionOutput txOutput = transaction.getOutputs().get(outputIndex); |
|
|
|
|
|
PSBTOutput psbtOutput = null; |
|
|
|
|
|
if (psbt != null && psbt.getPsbtOutputs().size() > txOutput.getIndex()) { |
|
|
|
|
|
psbtOutput = psbt.getPsbtOutputs().get(txOutput.getIndex()); |
|
|
|
|
|
} |
|
|
|
|
|
OutputForm outputForm = (psbt != null ? new OutputForm(psbt, psbtOutput) : (blockTransaction != null ? new OutputForm(blockTransaction, txOutput) : new OutputForm(transaction, txOutput))); |
|
|
|
|
|
return new TreeItem<>(outputForm); |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
public void setTreeSelection(TransactionView view, Integer index) { |
|
|
public void setTreeSelection(TransactionView view, Integer index) { |
|
@ -150,9 +226,10 @@ public class TransactionController implements Initializable { |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
private void select(TreeItem<TransactionForm> treeItem, TransactionView view, Integer index) { |
|
|
private void select(TreeItem<TransactionForm> treeItem, TransactionView view, Integer index) { |
|
|
if (treeItem.getValue().getView().equals(view)) { |
|
|
if(treeItem.getValue().getView().equals(view)) { |
|
|
if (view.equals(TransactionView.INPUT) || view.equals(TransactionView.OUTPUT)) { |
|
|
if(view.equals(TransactionView.INPUT) || view.equals(TransactionView.OUTPUT)) { |
|
|
if (treeItem.getParent().getChildren().indexOf(treeItem) == index) { |
|
|
IndexedTransactionForm txForm = (IndexedTransactionForm)treeItem.getValue(); |
|
|
|
|
|
if(txForm.getIndex() == index) { |
|
|
txtree.getSelectionModel().select(treeItem); |
|
|
txtree.getSelectionModel().select(treeItem); |
|
|
return; |
|
|
return; |
|
|
} |
|
|
} |
|
@ -162,7 +239,7 @@ public class TransactionController implements Initializable { |
|
|
} |
|
|
} |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
for (TreeItem<TransactionForm> childItem : treeItem.getChildren()) { |
|
|
for(TreeItem<TransactionForm> childItem : treeItem.getChildren()) { |
|
|
select(childItem, view, index); |
|
|
select(childItem, view, index); |
|
|
} |
|
|
} |
|
|
} |
|
|
} |
|
@ -198,13 +275,18 @@ public class TransactionController implements Initializable { |
|
|
|
|
|
|
|
|
//Inputs
|
|
|
//Inputs
|
|
|
for (int i = 0; i < transaction.getInputs().size(); i++) { |
|
|
for (int i = 0; i < transaction.getInputs().size(); i++) { |
|
|
|
|
|
if(i == highestInputIndex) { |
|
|
|
|
|
txhex.append("...", ""); |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
TransactionInput input = transaction.getInputs().get(i); |
|
|
TransactionInput input = transaction.getInputs().get(i); |
|
|
cursor = addText(hex, cursor, 32 * 2, "input-" + getIndexedStyleClass(i, selectedInputIndex, "hash")); |
|
|
boolean skip = (i >= highestInputIndex); |
|
|
cursor = addText(hex, cursor, 4 * 2, "input-" + getIndexedStyleClass(i, selectedInputIndex, "index")); |
|
|
cursor = addText(hex, cursor, 32 * 2, "input-" + getIndexedStyleClass(i, selectedInputIndex, "hash"), skip); |
|
|
|
|
|
cursor = addText(hex, cursor, 4 * 2, "input-" + getIndexedStyleClass(i, selectedInputIndex, "index"), skip); |
|
|
VarInt scriptLen = new VarInt(input.getScriptBytes().length); |
|
|
VarInt scriptLen = new VarInt(input.getScriptBytes().length); |
|
|
cursor = addText(hex, cursor, scriptLen.getSizeInBytes() * 2, "input-" + getIndexedStyleClass(i, selectedInputIndex, "sigscript-length")); |
|
|
cursor = addText(hex, cursor, scriptLen.getSizeInBytes() * 2, "input-" + getIndexedStyleClass(i, selectedInputIndex, "sigscript-length"), skip); |
|
|
cursor = addText(hex, cursor, (int) scriptLen.value * 2, "input-" + getIndexedStyleClass(i, selectedInputIndex, "sigscript")); |
|
|
cursor = addText(hex, cursor, (int) scriptLen.value * 2, "input-" + getIndexedStyleClass(i, selectedInputIndex, "sigscript"), skip); |
|
|
cursor = addText(hex, cursor, 4 * 2, "input-" + getIndexedStyleClass(i, selectedInputIndex, "sequence")); |
|
|
cursor = addText(hex, cursor, 4 * 2, "input-" + getIndexedStyleClass(i, selectedInputIndex, "sequence"), skip); |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
//Number of outputs
|
|
|
//Number of outputs
|
|
@ -213,24 +295,34 @@ public class TransactionController implements Initializable { |
|
|
|
|
|
|
|
|
//Outputs
|
|
|
//Outputs
|
|
|
for (int i = 0; i < transaction.getOutputs().size(); i++) { |
|
|
for (int i = 0; i < transaction.getOutputs().size(); i++) { |
|
|
|
|
|
if(i == highestOutputIndex) { |
|
|
|
|
|
txhex.append("...", ""); |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
TransactionOutput output = transaction.getOutputs().get(i); |
|
|
TransactionOutput output = transaction.getOutputs().get(i); |
|
|
cursor = addText(hex, cursor, 8 * 2, "output-" + getIndexedStyleClass(i, selectedOutputIndex, "value")); |
|
|
boolean skip = (i >= highestOutputIndex); |
|
|
|
|
|
cursor = addText(hex, cursor, 8 * 2, "output-" + getIndexedStyleClass(i, selectedOutputIndex, "value"), skip); |
|
|
VarInt scriptLen = new VarInt(output.getScriptBytes().length); |
|
|
VarInt scriptLen = new VarInt(output.getScriptBytes().length); |
|
|
cursor = addText(hex, cursor, scriptLen.getSizeInBytes() * 2, "output-" + getIndexedStyleClass(i, selectedOutputIndex, "pubkeyscript-length")); |
|
|
cursor = addText(hex, cursor, scriptLen.getSizeInBytes() * 2, "output-" + getIndexedStyleClass(i, selectedOutputIndex, "pubkeyscript-length"), skip); |
|
|
cursor = addText(hex, cursor, (int) scriptLen.value * 2, "output-" + getIndexedStyleClass(i, selectedOutputIndex, "pubkeyscript")); |
|
|
cursor = addText(hex, cursor, (int) scriptLen.value * 2, "output-" + getIndexedStyleClass(i, selectedOutputIndex, "pubkeyscript"), skip); |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
if (transaction.hasWitnesses()) { |
|
|
if (transaction.hasWitnesses()) { |
|
|
for (int i = 0; i < transaction.getInputs().size(); i++) { |
|
|
for (int i = 0; i < transaction.getInputs().size(); i++) { |
|
|
|
|
|
if(i == highestInputIndex) { |
|
|
|
|
|
txhex.append("...", ""); |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
TransactionInput input = transaction.getInputs().get(i); |
|
|
TransactionInput input = transaction.getInputs().get(i); |
|
|
|
|
|
boolean skip = (i >= highestInputIndex); |
|
|
if (input.hasWitness()) { |
|
|
if (input.hasWitness()) { |
|
|
TransactionWitness witness = input.getWitness(); |
|
|
TransactionWitness witness = input.getWitness(); |
|
|
VarInt witnessCount = new VarInt(witness.getPushCount()); |
|
|
VarInt witnessCount = new VarInt(witness.getPushCount()); |
|
|
cursor = addText(hex, cursor, witnessCount.getSizeInBytes() * 2, "witness-" + getIndexedStyleClass(i, selectedInputIndex, "count")); |
|
|
cursor = addText(hex, cursor, witnessCount.getSizeInBytes() * 2, "witness-" + getIndexedStyleClass(i, selectedInputIndex, "count"), skip); |
|
|
for (byte[] push : witness.getPushes()) { |
|
|
for (byte[] push : witness.getPushes()) { |
|
|
VarInt witnessLen = new VarInt(push.length); |
|
|
VarInt witnessLen = new VarInt(push.length); |
|
|
cursor = addText(hex, cursor, witnessLen.getSizeInBytes() * 2, "witness-" + getIndexedStyleClass(i, selectedInputIndex, "length")); |
|
|
cursor = addText(hex, cursor, witnessLen.getSizeInBytes() * 2, "witness-" + getIndexedStyleClass(i, selectedInputIndex, "length"), skip); |
|
|
cursor = addText(hex, cursor, (int) witnessLen.value * 2, "witness-" + getIndexedStyleClass(i, selectedInputIndex, "data")); |
|
|
cursor = addText(hex, cursor, (int) witnessLen.value * 2, "witness-" + getIndexedStyleClass(i, selectedInputIndex, "data"), skip); |
|
|
} |
|
|
} |
|
|
} |
|
|
} |
|
|
} |
|
|
} |
|
@ -239,23 +331,30 @@ public class TransactionController implements Initializable { |
|
|
//Locktime
|
|
|
//Locktime
|
|
|
cursor = addText(hex, cursor, 8, "locktime"); |
|
|
cursor = addText(hex, cursor, 8, "locktime"); |
|
|
|
|
|
|
|
|
if (cursor != hex.length()) { |
|
|
if(cursor != hex.length()) { |
|
|
throw new IllegalStateException("Cursor position does not match transaction serialisation " + cursor + ": " + hex.length()); |
|
|
throw new IllegalStateException("Cursor position does not match transaction serialisation " + cursor + ": " + hex.length()); |
|
|
} |
|
|
} |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
private void fetchThisAndInputBlockTransactions() { |
|
|
private void fetchThisAndInputBlockTransactions(int indexStart, int indexEnd) { |
|
|
if (AppController.isOnline()) { |
|
|
if(AppController.isOnline() && indexStart < transaction.getInputs().size()) { |
|
|
Set<Sha256Hash> references = new HashSet<>(); |
|
|
Set<Sha256Hash> references = new HashSet<>(); |
|
|
if (psbt == null) { |
|
|
if (psbt == null) { |
|
|
references.add(transaction.getTxId()); |
|
|
references.add(transaction.getTxId()); |
|
|
} |
|
|
} |
|
|
for (TransactionInput input : transaction.getInputs()) { |
|
|
|
|
|
|
|
|
int maxIndex = Math.min(transaction.getInputs().size(), indexEnd); |
|
|
|
|
|
for(int i = indexStart; i < maxIndex; i++) { |
|
|
|
|
|
TransactionInput input = transaction.getInputs().get(i); |
|
|
if(!input.isCoinBase()) { |
|
|
if(!input.isCoinBase()) { |
|
|
references.add(input.getOutpoint().getHash()); |
|
|
references.add(input.getOutpoint().getHash()); |
|
|
} |
|
|
} |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
if(references.isEmpty()) { |
|
|
|
|
|
return; |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
ElectrumServer.TransactionReferenceService transactionReferenceService = new ElectrumServer.TransactionReferenceService(references); |
|
|
ElectrumServer.TransactionReferenceService transactionReferenceService = new ElectrumServer.TransactionReferenceService(references); |
|
|
transactionReferenceService.setOnSucceeded(successEvent -> { |
|
|
transactionReferenceService.setOnSucceeded(successEvent -> { |
|
|
Map<Sha256Hash, BlockTransaction> transactionMap = transactionReferenceService.getValue(); |
|
|
Map<Sha256Hash, BlockTransaction> transactionMap = transactionReferenceService.getValue(); |
|
@ -279,7 +378,7 @@ public class TransactionController implements Initializable { |
|
|
|
|
|
|
|
|
final BlockTransaction blockTx = thisBlockTx; |
|
|
final BlockTransaction blockTx = thisBlockTx; |
|
|
Platform.runLater(() -> { |
|
|
Platform.runLater(() -> { |
|
|
EventManager.get().post(new BlockTransactionFetchedEvent(transaction.getTxId(), blockTx, inputTransactions)); |
|
|
EventManager.get().post(new BlockTransactionFetchedEvent(transaction.getTxId(), blockTx, inputTransactions, indexStart, maxIndex)); |
|
|
}); |
|
|
}); |
|
|
}); |
|
|
}); |
|
|
transactionReferenceService.setOnFailed(failedEvent -> { |
|
|
transactionReferenceService.setOnFailed(failedEvent -> { |
|
@ -289,13 +388,14 @@ public class TransactionController implements Initializable { |
|
|
} |
|
|
} |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
private void fetchOutputBlockTransactions() { |
|
|
private void fetchOutputBlockTransactions(int indexStart, int indexEnd) { |
|
|
if (AppController.isOnline() && psbt == null) { |
|
|
if(AppController.isOnline() && psbt == null && indexStart < transaction.getOutputs().size()) { |
|
|
ElectrumServer.TransactionOutputsReferenceService transactionOutputsReferenceService = new ElectrumServer.TransactionOutputsReferenceService(transaction); |
|
|
int maxIndex = Math.min(transaction.getOutputs().size(), indexEnd); |
|
|
|
|
|
ElectrumServer.TransactionOutputsReferenceService transactionOutputsReferenceService = new ElectrumServer.TransactionOutputsReferenceService(transaction, indexStart, maxIndex); |
|
|
transactionOutputsReferenceService.setOnSucceeded(successEvent -> { |
|
|
transactionOutputsReferenceService.setOnSucceeded(successEvent -> { |
|
|
List<BlockTransaction> outputTransactions = transactionOutputsReferenceService.getValue(); |
|
|
List<BlockTransaction> outputTransactions = transactionOutputsReferenceService.getValue(); |
|
|
Platform.runLater(() -> { |
|
|
Platform.runLater(() -> { |
|
|
EventManager.get().post(new BlockTransactionOutputsFetchedEvent(transaction.getTxId(), outputTransactions)); |
|
|
EventManager.get().post(new BlockTransactionOutputsFetchedEvent(transaction.getTxId(), outputTransactions, indexStart, maxIndex)); |
|
|
}); |
|
|
}); |
|
|
}); |
|
|
}); |
|
|
transactionOutputsReferenceService.setOnFailed(failedEvent -> { |
|
|
transactionOutputsReferenceService.setOnFailed(failedEvent -> { |
|
@ -314,8 +414,15 @@ public class TransactionController implements Initializable { |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
private int addText(String hex, int cursor, int length, String styleClass) { |
|
|
private int addText(String hex, int cursor, int length, String styleClass) { |
|
|
txhex.append(hex.substring(cursor, cursor += length), styleClass); |
|
|
return addText(hex, cursor, length, styleClass, false); |
|
|
return cursor; |
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
private int addText(String hex, int cursor, int length, String styleClass, boolean skip) { |
|
|
|
|
|
if(!skip) { |
|
|
|
|
|
txhex.append(hex.substring(cursor, cursor + length), styleClass); |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
return cursor + length; |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
public void setTransaction(Transaction transaction) { |
|
|
public void setTransaction(Transaction transaction) { |
|
@ -332,6 +439,11 @@ public class TransactionController implements Initializable { |
|
|
this.blockTransaction = blockTransaction; |
|
|
this.blockTransaction = blockTransaction; |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
public void setInitialView(TransactionView initialView, Integer initialIndex) { |
|
|
|
|
|
this.initialView = initialView; |
|
|
|
|
|
this.initialIndex = initialIndex; |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
@Subscribe |
|
|
@Subscribe |
|
|
public void transactionChanged(TransactionChangedEvent event) { |
|
|
public void transactionChanged(TransactionChangedEvent event) { |
|
|
if (event.getTransaction().equals(transaction)) { |
|
|
if (event.getTransaction().equals(transaction)) { |
|
@ -360,7 +472,11 @@ public class TransactionController implements Initializable { |
|
|
private void setBlockTransaction(TreeItem<TransactionForm> treeItem, BlockTransactionFetchedEvent event) { |
|
|
private void setBlockTransaction(TreeItem<TransactionForm> treeItem, BlockTransactionFetchedEvent event) { |
|
|
TransactionForm form = treeItem.getValue(); |
|
|
TransactionForm form = treeItem.getValue(); |
|
|
form.setBlockTransaction(event.getBlockTransaction()); |
|
|
form.setBlockTransaction(event.getBlockTransaction()); |
|
|
form.setInputTransactions(event.getInputTransactions()); |
|
|
if(form.getInputTransactions() == null) { |
|
|
|
|
|
form.setInputTransactions(event.getInputTransactions()); |
|
|
|
|
|
} else { |
|
|
|
|
|
form.getInputTransactions().putAll(event.getInputTransactions()); |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
for (TreeItem<TransactionForm> childItem : treeItem.getChildren()) { |
|
|
for (TreeItem<TransactionForm> childItem : treeItem.getChildren()) { |
|
|
setBlockTransaction(childItem, event); |
|
|
setBlockTransaction(childItem, event); |
|
@ -376,7 +492,16 @@ public class TransactionController implements Initializable { |
|
|
|
|
|
|
|
|
private void setBlockTransactionOutputs(TreeItem<TransactionForm> treeItem, BlockTransactionOutputsFetchedEvent event) { |
|
|
private void setBlockTransactionOutputs(TreeItem<TransactionForm> treeItem, BlockTransactionOutputsFetchedEvent event) { |
|
|
TransactionForm form = treeItem.getValue(); |
|
|
TransactionForm form = treeItem.getValue(); |
|
|
form.setOutputTransactions(event.getOutputTransactions()); |
|
|
if(form.getOutputTransactions() == null) { |
|
|
|
|
|
form.setOutputTransactions(event.getOutputTransactions()); |
|
|
|
|
|
} else { |
|
|
|
|
|
for(int i = 0; i < event.getOutputTransactions().size(); i++) { |
|
|
|
|
|
BlockTransaction outputTransaction = event.getOutputTransactions().get(i); |
|
|
|
|
|
if(outputTransaction != null) { |
|
|
|
|
|
form.getOutputTransactions().set(i, outputTransaction); |
|
|
|
|
|
} |
|
|
|
|
|
} |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
for (TreeItem<TransactionForm> childItem : treeItem.getChildren()) { |
|
|
for (TreeItem<TransactionForm> childItem : treeItem.getChildren()) { |
|
|
setBlockTransactionOutputs(childItem, event); |
|
|
setBlockTransactionOutputs(childItem, event); |
|
|