diff --git a/build.gradle b/build.gradle index 5fb3e52d..c63ec6cb 100644 --- a/build.gradle +++ b/build.gradle @@ -16,7 +16,7 @@ repositories { javafx { version = "14" - modules = [ 'javafx.controls', 'javafx.fxml' ] + modules = [ 'javafx.controls', 'javafx.fxml', 'javafx.swing' ] } java { @@ -46,7 +46,7 @@ dependencies { mainClassName = 'com.sparrowwallet.sparrow/com.sparrowwallet.sparrow.MainApp' run { - applicationDefaultJvmArgs = ["-Xdock:name=Sparrow", "-Xdock:icon=/Users/scy/git/sparrow/src/main/resources/sparrow.png", "--add-opens=javafx.graphics/com.sun.javafx.css=org.controlsfx.controls", "--add-opens=javafx.graphics/javafx.scene=org.controlsfx.controls"] + applicationDefaultJvmArgs = ["-Xdock:name=Sparrow", "-Xdock:icon=/Users/scy/git/sparrow/src/main/resources/sparrow.png", "--add-opens=javafx.graphics/com.sun.javafx.css=org.controlsfx.controls", "--add-opens=javafx.graphics/javafx.scene=org.controlsfx.controls", "--add-opens=javafx.controls/com.sun.javafx.scene.control.behavior=org.controlsfx.controls", "--add-opens=javafx.controls/com.sun.javafx.scene.control.inputmap=org.controlsfx.controls", "--add-opens=javafx.graphics/com.sun.javafx.scene.traversal=org.controlsfx.controls"] } jlink { @@ -61,7 +61,7 @@ jlink { options = ['--strip-debug', '--compress', '2', '--no-header-files', '--no-man-pages', '--ignore-signing-information', '--exclude-files', '**.png'] launcher { name = 'sparrow' - jvmArgs = ["--add-opens=javafx.graphics/com.sun.javafx.css=org.controlsfx.controls", "--add-opens=javafx.graphics/javafx.scene=org.controlsfx.controls"] + jvmArgs = ["--add-opens=javafx.graphics/com.sun.javafx.css=org.controlsfx.controls", "--add-opens=javafx.graphics/javafx.scene=org.controlsfx.controls", "--add-opens=javafx.controls/com.sun.javafx.scene.control.behavior=org.controlsfx.controls", "--add-opens=javafx.controls/com.sun.javafx.scene.control.inputmap=org.controlsfx.controls", "--add-opens=javafx.graphics/com.sun.javafx.scene.traversal=org.controlsfx.controls"] } addExtraDependencies("javafx") jpackage { diff --git a/drongo b/drongo index 0cebee3d..08e8df08 160000 --- a/drongo +++ b/drongo @@ -1 +1 @@ -Subproject commit 0cebee3d22740d0ab38657b1deaba67fe282f920 +Subproject commit 08e8df0807a01b2716ac80e9d6d9e2c113025848 diff --git a/src/main/java/com/sparrowwallet/sparrow/AppController.java b/src/main/java/com/sparrowwallet/sparrow/AppController.java index c8a742da..f0788ce1 100644 --- a/src/main/java/com/sparrowwallet/sparrow/AppController.java +++ b/src/main/java/com/sparrowwallet/sparrow/AppController.java @@ -7,11 +7,14 @@ import com.sparrowwallet.drongo.Utils; import com.sparrowwallet.drongo.protocol.Transaction; import com.sparrowwallet.drongo.psbt.PSBT; import com.sparrowwallet.drongo.psbt.PSBTParseException; +import com.sparrowwallet.drongo.wallet.Wallet; import com.sparrowwallet.sparrow.control.TextAreaDialog; import com.sparrowwallet.sparrow.event.TabEvent; import com.sparrowwallet.sparrow.event.TransactionTabChangedEvent; import com.sparrowwallet.sparrow.event.TransactionTabSelectedEvent; import com.sparrowwallet.sparrow.transaction.TransactionController; +import com.sparrowwallet.sparrow.wallet.WalletController; +import com.sparrowwallet.sparrow.wallet.WalletForm; import javafx.event.ActionEvent; import javafx.fxml.FXML; import javafx.fxml.FXMLLoader; @@ -88,6 +91,8 @@ public class AppController implements Initializable { showTxHex.setSelected(true); showTxHexProperty = true; + + addWalletTab(null, new Wallet()); } public void openFromFile(ActionEvent event) { @@ -192,6 +197,36 @@ public class AppController implements Initializable { showTxHexProperty = item.isSelected(); } + public void newWallet(ActionEvent event) { + Tab tab = addWalletTab(null, new Wallet()); + tabs.getSelectionModel().select(tab); + } + + public Tab addWalletTab(String name, Wallet wallet) { + try { + String tabName = name; + if(tabName == null || tabName.isEmpty()) { + tabName = "New wallet"; + } + + Tab tab = new Tab(tabName); + TabData tabData = new TabData(TabData.TabType.WALLET); + tab.setUserData(tabData); + tab.setContextMenu(getTabContextMenu(tab)); + tab.setClosable(true); + FXMLLoader walletLoader = new FXMLLoader(getClass().getResource("wallet/wallet.fxml")); + tab.setContent(walletLoader.load()); + WalletController controller = walletLoader.getController(); + WalletForm walletForm = new WalletForm(wallet); + controller.setWalletForm(walletForm); + + tabs.getTabs().add(tab); + return tab; + } catch(IOException e) { + throw new RuntimeException(e); + } + } + public void openExamples(ActionEvent event) { try { addTransactionTab("p2pkh", "01000000019c2e0f24a03e72002a96acedb12a632e72b6b74c05dc3ceab1fe78237f886c48010000006a47304402203da9d487be5302a6d69e02a861acff1da472885e43d7528ed9b1b537a8e2cac9022002d1bca03a1e9715a99971bafe3b1852b7a4f0168281cbd27a220380a01b3307012102c9950c622494c2e9ff5a003e33b690fe4832477d32c2d256c67eab8bf613b34effffffff02b6f50500000000001976a914bdf63990d6dc33d705b756e13dd135466c06b3b588ac845e0201000000001976a9145fb0e9755a3424efd2ba0587d20b1e98ee29814a88ac06241559"); diff --git a/src/main/java/com/sparrowwallet/sparrow/TabData.java b/src/main/java/com/sparrowwallet/sparrow/TabData.java index 857701c2..3844d5e1 100644 --- a/src/main/java/com/sparrowwallet/sparrow/TabData.java +++ b/src/main/java/com/sparrowwallet/sparrow/TabData.java @@ -32,6 +32,6 @@ public class TabData { } public enum TabType { - TRANSACTION + WALLET, TRANSACTION } } diff --git a/src/main/java/com/sparrowwallet/sparrow/control/EnumChoiceBox.java b/src/main/java/com/sparrowwallet/sparrow/control/EnumChoiceBox.java new file mode 100644 index 00000000..0764a28c --- /dev/null +++ b/src/main/java/com/sparrowwallet/sparrow/control/EnumChoiceBox.java @@ -0,0 +1,14 @@ +package com.sparrowwallet.sparrow.control; + +import javafx.beans.NamedArg; +import javafx.scene.control.ChoiceBox; + +public class EnumChoiceBox> extends ChoiceBox { + + public EnumChoiceBox(@NamedArg("enumType") String enumType) throws Exception { + Class enumClass = (Class) Class.forName(enumType); + getItems().setAll(enumClass.getEnumConstants()); + } +} + + diff --git a/src/main/java/com/sparrowwallet/sparrow/transaction/TransactionFormController.java b/src/main/java/com/sparrowwallet/sparrow/transaction/TransactionFormController.java index 7c1cae65..d15ab13b 100644 --- a/src/main/java/com/sparrowwallet/sparrow/transaction/TransactionFormController.java +++ b/src/main/java/com/sparrowwallet/sparrow/transaction/TransactionFormController.java @@ -17,6 +17,7 @@ import java.time.Duration; import java.util.List; import static org.fxmisc.richtext.model.TwoDimensional.Bias.Backward; +import static com.sparrowwallet.drongo.protocol.ScriptType.*; public abstract class TransactionFormController { protected void addPieData(PieChart pie, List outputs) { @@ -57,7 +58,7 @@ public abstract class TransactionFormController { } protected void appendScript(CodeArea codeArea, Script script, Script redeemScript, Script witnessScript) { - if(ScriptPattern.isP2PKH(script)) { + if(P2PKH.isScriptType(script)) { codeArea.append(script.getChunks().get(0).toString(), "script-opcode"); codeArea.append(" ", ""); codeArea.append(script.getChunks().get(1).toString(), "script-opcode"); @@ -67,17 +68,17 @@ public abstract class TransactionFormController { codeArea.append(script.getChunks().get(3).toString(), "script-opcode"); codeArea.append(" ", ""); codeArea.append(script.getChunks().get(4).toString(), "script-opcode"); - } else if(ScriptPattern.isP2SH(script)) { + } else if(P2SH.isScriptType(script)) { codeArea.append(script.getChunks().get(0).toString(), "script-opcode"); codeArea.append(" ", ""); codeArea.append("", "script-hash"); codeArea.append(" ", ""); codeArea.append(script.getChunks().get(2).toString(), "script-opcode"); - } else if(ScriptPattern.isP2WPKH(script)) { + } else if(P2WPKH.isScriptType(script)) { codeArea.append(script.getChunks().get(0).toString(), "script-opcode"); codeArea.append(" ", ""); codeArea.append("", "script-hash"); - } else if(ScriptPattern.isP2WSH(script)) { + } else if(P2WSH.isScriptType(script)) { codeArea.append(script.getChunks().get(0).toString(), "script-opcode"); codeArea.append(" ", ""); codeArea.append("", "script-hash"); diff --git a/src/main/java/com/sparrowwallet/sparrow/wallet/Function.java b/src/main/java/com/sparrowwallet/sparrow/wallet/Function.java new file mode 100644 index 00000000..f1999b82 --- /dev/null +++ b/src/main/java/com/sparrowwallet/sparrow/wallet/Function.java @@ -0,0 +1,5 @@ +package com.sparrowwallet.sparrow.wallet; + +public enum Function { + TRANSACTIONS, SEND, RECEIVE, ADDRESSES, POLICIES, SETTINGS; +} diff --git a/src/main/java/com/sparrowwallet/sparrow/wallet/SettingsController.java b/src/main/java/com/sparrowwallet/sparrow/wallet/SettingsController.java new file mode 100644 index 00000000..fa145a54 --- /dev/null +++ b/src/main/java/com/sparrowwallet/sparrow/wallet/SettingsController.java @@ -0,0 +1,69 @@ +package com.sparrowwallet.sparrow.wallet; + +import com.sparrowwallet.drongo.policy.PolicyType; +import com.sparrowwallet.drongo.protocol.ScriptType; +import com.sparrowwallet.sparrow.EventManager; +import com.sparrowwallet.sparrow.control.CopyableLabel; +import javafx.fxml.FXML; +import javafx.fxml.Initializable; +import javafx.scene.control.*; +import org.controlsfx.control.RangeSlider; +import tornadofx.control.Fieldset; + +import java.net.URL; +import java.util.ResourceBundle; + +public class SettingsController extends WalletFormController implements Initializable { + + @FXML + private ComboBox policyType; + + @FXML + private TextField policy; + + @FXML + private ComboBox scriptType; + + @FXML + private Fieldset multisigFieldset; + + @FXML + private RangeSlider multisigControl; + + @FXML + private CopyableLabel multisigLowLabel; + + @FXML + private CopyableLabel multisigHighLabel; + + @FXML + private TabPane keystoreTabs; + + @FXML ComboBox testType; + + @Override + public void initialize(URL location, ResourceBundle resources) { + EventManager.get().register(this); + } + + @Override + public void initializeView() { + policyType.getSelectionModel().selectedItemProperty().addListener((observable, oldValue, policyType) -> { + scriptType.getSelectionModel().select(policyType.getDefaultScriptType()); + multisigFieldset.setVisible(policyType.equals(PolicyType.MULTI)); + }); + + multisigLowLabel.textProperty().bind(multisigControl.lowValueProperty().asString("%.0f") ); + multisigHighLabel.textProperty().bind(multisigControl.highValueProperty().asString("%.0f")); + + multisigFieldset.managedProperty().bind(multisigFieldset.visibleProperty()); + + if(walletForm.getWallet().getPolicyType() != null) { + policyType.getSelectionModel().select(walletForm.getWallet().getPolicyType()); + } else { + policyType.getSelectionModel().select(0); + } + + + } +} diff --git a/src/main/java/com/sparrowwallet/sparrow/wallet/WalletController.java b/src/main/java/com/sparrowwallet/sparrow/wallet/WalletController.java new file mode 100644 index 00000000..9dee0f27 --- /dev/null +++ b/src/main/java/com/sparrowwallet/sparrow/wallet/WalletController.java @@ -0,0 +1,61 @@ +package com.sparrowwallet.sparrow.wallet; + +import com.sparrowwallet.drongo.wallet.Wallet; +import com.sparrowwallet.sparrow.AppController; +import com.sparrowwallet.sparrow.EventManager; +import javafx.fxml.FXML; +import javafx.fxml.FXMLLoader; +import javafx.fxml.Initializable; +import javafx.scene.Node; +import javafx.scene.control.ToggleGroup; +import javafx.scene.layout.BorderPane; +import javafx.scene.layout.StackPane; + +import java.io.IOException; +import java.net.URL; +import java.util.ResourceBundle; + +public class WalletController extends WalletFormController implements Initializable { + @FXML + private BorderPane tabContent; + + @FXML + private StackPane walletPane; + + @FXML + private ToggleGroup walletMenu; + + @Override + public void initialize(URL location, ResourceBundle resources) { + EventManager.get().register(this); + } + + public void initializeView() { + walletMenu.selectedToggleProperty().addListener((observable, oldValue, selectedToggle) -> { + Function function = (Function)selectedToggle.getUserData(); + + boolean existing = false; + for(Node walletFunction : walletPane.getChildren()) { + if(walletFunction.getUserData().equals(function)) { + existing = true; + walletFunction.setViewOrder(1); + } else { + walletFunction.setViewOrder(0); + } + } + + try { + if(!existing) { + FXMLLoader functionLoader = new FXMLLoader(AppController.class.getResource("wallet/" + function.toString().toLowerCase() + ".fxml")); + Node walletFunction = functionLoader.load(); + WalletFormController controller = functionLoader.getController(); + controller.setWalletForm(getWalletForm()); + walletFunction.setViewOrder(1); + walletPane.getChildren().add(walletFunction); + } + } catch (IOException e) { + throw new IllegalStateException("Can't find pane", e); + } + }); + } +} diff --git a/src/main/java/com/sparrowwallet/sparrow/wallet/WalletForm.java b/src/main/java/com/sparrowwallet/sparrow/wallet/WalletForm.java new file mode 100644 index 00000000..8f13706b --- /dev/null +++ b/src/main/java/com/sparrowwallet/sparrow/wallet/WalletForm.java @@ -0,0 +1,19 @@ +package com.sparrowwallet.sparrow.wallet; + +import com.sparrowwallet.drongo.wallet.Wallet; + +public class WalletForm { + private Wallet wallet; + + public WalletForm(Wallet wallet) { + this.wallet = wallet; + } + + public Wallet getWallet() { + return wallet; + } + + public void setWallet(Wallet wallet) { + this.wallet = wallet; + } +} diff --git a/src/main/java/com/sparrowwallet/sparrow/wallet/WalletFormController.java b/src/main/java/com/sparrowwallet/sparrow/wallet/WalletFormController.java new file mode 100644 index 00000000..845b531e --- /dev/null +++ b/src/main/java/com/sparrowwallet/sparrow/wallet/WalletFormController.java @@ -0,0 +1,16 @@ +package com.sparrowwallet.sparrow.wallet; + +public abstract class WalletFormController { + public WalletForm walletForm; + + public WalletForm getWalletForm() { + return walletForm; + } + + public void setWalletForm(WalletForm walletForm) { + this.walletForm = walletForm; + initializeView(); + } + + public abstract void initializeView(); +} diff --git a/src/main/java/module-info.java b/src/main/java/module-info.java index 5bd4b964..95d55dba 100644 --- a/src/main/java/module-info.java +++ b/src/main/java/module-info.java @@ -8,4 +8,5 @@ open module com.sparrowwallet.sparrow { requires com.sparrowwallet.drongo; requires com.google.common; requires flowless; + requires javafx.swing; } \ No newline at end of file diff --git a/src/main/resources/bird.png b/src/main/resources/bird.png new file mode 100755 index 00000000..ae8b43f8 Binary files /dev/null and b/src/main/resources/bird.png differ diff --git a/src/main/resources/com/sparrowwallet/sparrow/app.fxml b/src/main/resources/com/sparrowwallet/sparrow/app.fxml index 27e73ae7..6f82cc19 100644 --- a/src/main/resources/com/sparrowwallet/sparrow/app.fxml +++ b/src/main/resources/com/sparrowwallet/sparrow/app.fxml @@ -12,6 +12,7 @@ + diff --git a/src/main/resources/com/sparrowwallet/sparrow/wallet/settings.css b/src/main/resources/com/sparrowwallet/sparrow/wallet/settings.css new file mode 100644 index 00000000..50845478 --- /dev/null +++ b/src/main/resources/com/sparrowwallet/sparrow/wallet/settings.css @@ -0,0 +1,11 @@ +.form .fieldset:horizontal .field { + -fx-pref-height: 40px; +} + +.form .fieldset:horizontal .label-container { + -fx-alignment: center-left; +} + +.form .fieldset:horizontal .input-container { + -fx-alignment: center-left; +} diff --git a/src/main/resources/com/sparrowwallet/sparrow/wallet/settings.fxml b/src/main/resources/com/sparrowwallet/sparrow/wallet/settings.fxml new file mode 100644 index 00000000..dc254872 --- /dev/null +++ b/src/main/resources/com/sparrowwallet/sparrow/wallet/settings.fxml @@ -0,0 +1,82 @@ + + + + + + + + + + + + + + + + + + + + + + + + + +
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + +
+
+ +
+
+ + + + + + + + +
+
+ +
+
+ + + +
+
+ + + + +
diff --git a/src/main/resources/com/sparrowwallet/sparrow/wallet/wallet.css b/src/main/resources/com/sparrowwallet/sparrow/wallet/wallet.css new file mode 100644 index 00000000..fee37f1a --- /dev/null +++ b/src/main/resources/com/sparrowwallet/sparrow/wallet/wallet.css @@ -0,0 +1,21 @@ +.list-menu { + -fx-pref-width: 180; + -fx-background-color: #3da0e3; +} + +.list-item { + -fx-pref-width: 180; + -fx-background-color: #3da0e3; +} + +.list-item * { + -fx-fill: #fff; +} + +.list-item:hover { + -fx-background-color: #4aa7e5; +} + +.list-item:selected { + -fx-background-color: #1e88cf; +} \ No newline at end of file diff --git a/src/main/resources/com/sparrowwallet/sparrow/wallet/wallet.fxml b/src/main/resources/com/sparrowwallet/sparrow/wallet/wallet.fxml new file mode 100644 index 00000000..9b11691f --- /dev/null +++ b/src/main/resources/com/sparrowwallet/sparrow/wallet/wallet.fxml @@ -0,0 +1,72 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + +
+