|
|
|
import QtQuick 2.2
|
|
|
|
import QtQuick.Controls 1.1
|
|
|
|
import QtQuick.Dialogs 1.2
|
|
|
|
import QtQuick.Layouts 1.1
|
|
|
|
import QtQuick.Window 2.0
|
|
|
|
import QtQuick.Controls.Styles 1.3
|
|
|
|
import org.ethereum.qml.QEther 1.0
|
|
|
|
import "js/QEtherHelper.js" as QEtherHelper
|
|
|
|
import "js/TransactionHelper.js" as TransactionHelper
|
|
|
|
import "."
|
|
|
|
|
|
|
|
Dialog {
|
|
|
|
id: modalStateDialog
|
|
|
|
modality: Qt.ApplicationModal
|
|
|
|
|
|
|
|
width: 630
|
|
|
|
height: 500
|
|
|
|
title: qsTr("Edit State")
|
|
|
|
visible: false
|
|
|
|
|
|
|
|
property alias stateTitle: titleField.text
|
|
|
|
property alias isDefault: defaultCheckBox.checked
|
|
|
|
property alias model: transactionsModel
|
|
|
|
property alias transactionDialog: transactionDialog
|
|
|
|
property alias minerComboBox: comboMiner
|
|
|
|
property alias newAccAction: newAccountAction
|
|
|
|
property int stateIndex
|
|
|
|
property var stateTransactions: []
|
|
|
|
property var stateAccounts: []
|
|
|
|
signal accepted
|
|
|
|
|
|
|
|
StateDialogStyle {
|
|
|
|
id: stateDialogStyle
|
|
|
|
}
|
|
|
|
|
|
|
|
function open(index, item, setDefault) {
|
|
|
|
stateIndex = index;
|
|
|
|
stateTitle = item.title;
|
|
|
|
transactionsModel.clear();
|
|
|
|
|
|
|
|
stateTransactions = [];
|
|
|
|
var transactions = item.transactions;
|
|
|
|
for (var t = 0; t < transactions.length; t++) {
|
|
|
|
transactionsModel.append(item.transactions[t]);
|
|
|
|
stateTransactions.push(item.transactions[t]);
|
|
|
|
}
|
|
|
|
|
|
|
|
accountsModel.clear();
|
|
|
|
stateAccounts = [];
|
|
|
|
var miner = 0;
|
|
|
|
for (var k = 0; k < item.accounts.length; k++)
|
|
|
|
{
|
|
|
|
accountsModel.append(item.accounts[k]);
|
|
|
|
stateAccounts.push(item.accounts[k]);
|
|
|
|
if (item.miner && item.accounts[k].name === item.miner.name)
|
|
|
|
miner = k;
|
|
|
|
}
|
|
|
|
|
|
|
|
visible = true;
|
|
|
|
isDefault = setDefault;
|
|
|
|
titleField.focus = true;
|
|
|
|
defaultCheckBox.enabled = !isDefault;
|
|
|
|
comboMiner.model = stateAccounts;
|
|
|
|
comboMiner.currentIndex = miner;
|
|
|
|
forceActiveFocus();
|
|
|
|
}
|
|
|
|
|
|
|
|
function acceptAndClose() {
|
|
|
|
close();
|
|
|
|
accepted();
|
|
|
|
}
|
|
|
|
|
|
|
|
function close() {
|
|
|
|
visible = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
function getItem() {
|
|
|
|
var item = {
|
|
|
|
title: stateDialog.stateTitle,
|
|
|
|
transactions: [],
|
|
|
|
accounts: []
|
|
|
|
}
|
|
|
|
item.transactions = stateTransactions;
|
|
|
|
item.accounts = stateAccounts;
|
|
|
|
for (var k = 0; k < stateAccounts.length; k++)
|
|
|
|
{
|
|
|
|
if (stateAccounts[k].name === comboMiner.currentText)
|
|
|
|
{
|
|
|
|
item.miner = stateAccounts[k];
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return item;
|
|
|
|
}
|
|
|
|
|
|
|
|
contentItem: Rectangle {
|
|
|
|
color: stateDialogStyle.generic.backgroundColor
|
|
|
|
Rectangle {
|
|
|
|
color: stateDialogStyle.generic.backgroundColor
|
|
|
|
anchors.top: parent.top
|
|
|
|
anchors.margins: 10
|
|
|
|
anchors.fill: parent
|
|
|
|
ColumnLayout {
|
|
|
|
anchors.fill: parent
|
|
|
|
anchors.margins: 10
|
|
|
|
ColumnLayout {
|
|
|
|
id: dialogContent
|
|
|
|
anchors.top: parent.top
|
|
|
|
RowLayout
|
|
|
|
{
|
|
|
|
Layout.fillWidth: true
|
|
|
|
DefaultLabel {
|
|
|
|
Layout.preferredWidth: 85
|
|
|
|
text: qsTr("Title")
|
|
|
|
}
|
|
|
|
DefaultTextField
|
|
|
|
{
|
|
|
|
id: titleField
|
|
|
|
Layout.fillWidth: true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
CommonSeparator
|
|
|
|
{
|
|
|
|
Layout.fillWidth: true
|
|
|
|
}
|
|
|
|
|
|
|
|
RowLayout
|
|
|
|
{
|
|
|
|
Layout.fillWidth: true
|
|
|
|
|
|
|
|
Rectangle
|
|
|
|
{
|
|
|
|
Layout.preferredWidth: 85
|
|
|
|
DefaultLabel {
|
|
|
|
id: accountsLabel
|
|
|
|
Layout.preferredWidth: 85
|
|
|
|
text: qsTr("Accounts")
|
|
|
|
}
|
|
|
|
|
|
|
|
Button
|
|
|
|
{
|
|
|
|
id: newAccountButton
|
|
|
|
anchors.top: accountsLabel.bottom
|
|
|
|
anchors.topMargin: 10
|
|
|
|
iconSource: "qrc:/qml/img/plus.png"
|
|
|
|
action: newAccountAction
|
|
|
|
}
|
|
|
|
|
|
|
|
Action {
|
|
|
|
id: newAccountAction
|
|
|
|
tooltip: qsTr("Add new Account")
|
|
|
|
onTriggered:
|
|
|
|
{
|
|
|
|
add();
|
|
|
|
}
|
|
|
|
|
|
|
|
function add()
|
|
|
|
{
|
|
|
|
var account = stateListModel.newAccount("1000000", QEther.Ether);
|
|
|
|
stateAccounts.push(account);
|
|
|
|
accountsModel.append(account);
|
|
|
|
return account;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
MessageDialog
|
|
|
|
{
|
|
|
|
id: alertAlreadyUsed
|
|
|
|
text: qsTr("This account is in use. You cannot remove it. The first account is used to deploy config contract and cannot be removed.")
|
|
|
|
icon: StandardIcon.Warning
|
|
|
|
standardButtons: StandardButton.Ok
|
|
|
|
}
|
|
|
|
|
|
|
|
TableView
|
|
|
|
{
|
|
|
|
id: accountsView
|
|
|
|
Layout.fillWidth: true
|
|
|
|
model: accountsModel
|
|
|
|
headerVisible: false
|
|
|
|
TableViewColumn {
|
|
|
|
role: "name"
|
|
|
|
title: qsTr("Name")
|
|
|
|
width: 230
|
|
|
|
delegate: Item {
|
|
|
|
RowLayout
|
|
|
|
{
|
|
|
|
height: 25
|
|
|
|
width: parent.width
|
|
|
|
Button
|
|
|
|
{
|
|
|
|
iconSource: "qrc:/qml/img/delete_sign.png"
|
|
|
|
action: deleteAccountAction
|
|
|
|
}
|
|
|
|
|
|
|
|
Action {
|
|
|
|
id: deleteAccountAction
|
|
|
|
tooltip: qsTr("Delete Account")
|
|
|
|
onTriggered:
|
|
|
|
{
|
|
|
|
if (transactionsModel.isUsed(stateAccounts[styleData.row].secret))
|
|
|
|
alertAlreadyUsed.open();
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (stateAccounts[styleData.row].name === comboMiner.currentText)
|
|
|
|
comboMiner.currentIndex = 0;
|
|
|
|
stateAccounts.splice(styleData.row, 1);
|
|
|
|
accountsModel.remove(styleData.row);
|
|
|
|
comboMiner.model = stateAccounts;
|
|
|
|
comboMiner.update();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
DefaultTextField {
|
|
|
|
anchors.verticalCenter: parent.verticalCenter
|
|
|
|
onTextChanged: {
|
|
|
|
if (styleData.row > -1)
|
|
|
|
{
|
|
|
|
stateAccounts[styleData.row].name = text
|
|
|
|
var index = comboMiner.currentIndex;
|
|
|
|
comboMiner.model = stateAccounts;
|
|
|
|
comboMiner.currentIndex = index;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
text: {
|
|
|
|
return styleData.value
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TableViewColumn {
|
|
|
|
role: "balance"
|
|
|
|
title: qsTr("Balance")
|
|
|
|
width: 200
|
|
|
|
delegate: Item {
|
|
|
|
Ether {
|
|
|
|
id: balanceField
|
|
|
|
edit: true
|
|
|
|
displayFormattedValue: false
|
|
|
|
value: styleData.value
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
rowDelegate:
|
|
|
|
Rectangle {
|
|
|
|
color: styleData.alternate ? "transparent" : "#f0f0f0"
|
|
|
|
height: 30;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
CommonSeparator
|
|
|
|
{
|
|
|
|
Layout.fillWidth: true
|
|
|
|
}
|
|
|
|
|
|
|
|
RowLayout
|
|
|
|
{
|
|
|
|
Layout.fillWidth: true
|
|
|
|
DefaultLabel {
|
|
|
|
Layout.preferredWidth: 85
|
|
|
|
text: qsTr("Miner")
|
|
|
|
}
|
|
|
|
ComboBox {
|
|
|
|
id: comboMiner
|
|
|
|
textRole: "name"
|
|
|
|
Layout.fillWidth: true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
CommonSeparator
|
|
|
|
{
|
|
|
|
Layout.fillWidth: true
|
|
|
|
}
|
|
|
|
|
|
|
|
RowLayout
|
|
|
|
{
|
|
|
|
Layout.fillWidth: true
|
|
|
|
DefaultLabel {
|
|
|
|
Layout.preferredWidth: 85
|
|
|
|
text: qsTr("Default")
|
|
|
|
}
|
|
|
|
CheckBox {
|
|
|
|
id: defaultCheckBox
|
|
|
|
Layout.fillWidth: true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
CommonSeparator
|
|
|
|
{
|
|
|
|
Layout.fillWidth: true
|
|
|
|
}
|
|
|
|
|
|
|
|
RowLayout
|
|
|
|
{
|
|
|
|
Layout.fillWidth: true
|
|
|
|
|
|
|
|
Rectangle
|
|
|
|
{
|
|
|
|
Layout.preferredWidth: 85
|
|
|
|
DefaultLabel {
|
|
|
|
id: transactionsLabel
|
|
|
|
Layout.preferredWidth: 85
|
|
|
|
text: qsTr("Transactions")
|
|
|
|
}
|
|
|
|
|
|
|
|
Button
|
|
|
|
{
|
|
|
|
anchors.top: transactionsLabel.bottom
|
|
|
|
anchors.topMargin: 10
|
|
|
|
iconSource: "qrc:/qml/img/plus.png"
|
|
|
|
action: newTrAction
|
|
|
|
}
|
|
|
|
|
|
|
|
Action {
|
|
|
|
id: newTrAction
|
|
|
|
tooltip: qsTr("Create a new transaction")
|
|
|
|
onTriggered: transactionsModel.addTransaction()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TableView
|
|
|
|
{
|
|
|
|
id: transactionsView
|
|
|
|
Layout.fillWidth: true
|
|
|
|
model: transactionsModel
|
|
|
|
headerVisible: false
|
|
|
|
TableViewColumn {
|
|
|
|
role: "name"
|
|
|
|
title: qsTr("Name")
|
|
|
|
width: 150
|
|
|
|
delegate: Item {
|
|
|
|
RowLayout
|
|
|
|
{
|
|
|
|
height: 30
|
|
|
|
width: parent.width
|
|
|
|
Button
|
|
|
|
{
|
|
|
|
iconSource: "qrc:/qml/img/delete_sign.png"
|
|
|
|
action: deleteTransactionAction
|
|
|
|
}
|
|
|
|
|
|
|
|
Action {
|
|
|
|
id: deleteTransactionAction
|
|
|
|
tooltip: qsTr("Delete")
|
|
|
|
onTriggered: transactionsModel.deleteTransaction(styleData.row)
|
|
|
|
}
|
|
|
|
|
|
|
|
Button
|
|
|
|
{
|
|
|
|
iconSource: "qrc:/qml/img/edit.png"
|
|
|
|
action: editAction
|
|
|
|
visible: styleData.row >= 0 ? !transactionsModel.get(styleData.row).stdContract : false
|
|
|
|
width: 10
|
|
|
|
height: 10
|
|
|
|
Action {
|
|
|
|
id: editAction
|
|
|
|
tooltip: qsTr("Edit")
|
|
|
|
onTriggered: transactionsModel.editTransaction(styleData.row)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
DefaultLabel {
|
|
|
|
Layout.preferredWidth: 150
|
|
|
|
text: {
|
|
|
|
if (styleData.row >= 0)
|
|
|
|
{
|
|
|
|
if (transactionsModel.get(styleData.row).label !== undefined)
|
|
|
|
return transactionsModel.get(styleData.row).label;
|
|
|
|
else if (transactionsModel.get(styleData.row).functionId !== undefined)
|
|
|
|
return transactionsModel.get(styleData.row).functionId;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
rowDelegate:
|
|
|
|
Rectangle {
|
|
|
|
color: styleData.alternate ? "transparent" : "#f0f0f0"
|
|
|
|
height: 30;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
RowLayout
|
|
|
|
{
|
|
|
|
anchors.bottom: parent.bottom
|
|
|
|
anchors.right: parent.right;
|
|
|
|
|
|
|
|
Button {
|
|
|
|
text: qsTr("Delete");
|
|
|
|
enabled: !modalStateDialog.isDefault
|
|
|
|
onClicked: {
|
|
|
|
projectModel.stateListModel.deleteState(stateIndex);
|
|
|
|
close();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Button {
|
|
|
|
text: qsTr("OK");
|
|
|
|
onClicked: {
|
|
|
|
close();
|
|
|
|
accepted();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Button {
|
|
|
|
text: qsTr("Cancel");
|
|
|
|
onClicked: close();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ListModel {
|
|
|
|
id: accountsModel
|
|
|
|
|
|
|
|
function removeAccount(_i)
|
|
|
|
{
|
|
|
|
accountsModel.remove(_i);
|
|
|
|
stateAccounts.splice(_i, 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ListModel {
|
|
|
|
id: transactionsModel
|
|
|
|
|
|
|
|
function editTransaction(index) {
|
|
|
|
transactionDialog.stateAccounts = stateAccounts;
|
|
|
|
transactionDialog.open(index, transactionsModel.get(index));
|
|
|
|
}
|
|
|
|
|
|
|
|
function addTransaction() {
|
|
|
|
// Set next id here to work around Qt bug
|
|
|
|
// https://bugreports.qt-project.org/browse/QTBUG-41327
|
|
|
|
// Second call to signal handler would just edit the item that was just created, no harm done
|
|
|
|
var item = TransactionHelper.defaultTransaction();
|
|
|
|
transactionDialog.stateAccounts = stateAccounts;
|
|
|
|
transactionDialog.open(transactionsModel.count, item);
|
|
|
|
}
|
|
|
|
|
|
|
|
function deleteTransaction(index) {
|
|
|
|
stateTransactions.splice(index, 1);
|
|
|
|
transactionsModel.remove(index);
|
|
|
|
}
|
|
|
|
|
|
|
|
function isUsed(secret)
|
|
|
|
{
|
|
|
|
for (var i in stateTransactions)
|
|
|
|
{
|
|
|
|
if (stateTransactions[i].sender === secret)
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TransactionDialog
|
|
|
|
{
|
|
|
|
id: transactionDialog
|
|
|
|
onAccepted:
|
|
|
|
{
|
|
|
|
var item = transactionDialog.getItem();
|
|
|
|
if (transactionDialog.transactionIndex < transactionsModel.count) {
|
|
|
|
transactionsModel.set(transactionDialog.transactionIndex, item);
|
|
|
|
stateTransactions[transactionDialog.transactionIndex] = item;
|
|
|
|
} else {
|
|
|
|
transactionsModel.append(item);
|
|
|
|
stateTransactions.push(item);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|