import QtQuick 2.2 import QtQuick.Controls 1.1 import QtQuick.Layouts 1.1 import QtQuick.Dialogs 1.2 import QtQuick.Window 2.0 import QtQuick.Controls.Styles 1.3 import org.ethereum.qml.QEther 1.0 import "js/TransactionHelper.js" as TransactionHelper import "js/InputValidator.js" as InputValidator import "." Dialog { id: modalTransactionDialog modality: Qt.ApplicationModal width: 570 height: 500 visible: false title: qsTr("Edit Transaction") property int transactionIndex property int blockIndex property alias gas: gasValueEdit.gasValue; property alias gasAuto: gasAutoCheck.checked; property alias gasPrice: gasPriceField.value; property alias transactionValue: valueField.value; property string contractId: contractComboBox.currentValue(); property alias functionId: functionComboBox.currentText; property var paramValues; property var paramsModel: []; property bool useTransactionDefaultValue: false property alias stateAccounts: senderComboBox.model property bool saveStatus signal accepted; StateDialogStyle { id: transactionDialogStyle } function open(index, blockIdx, item) { rowFunction.visible = !useTransactionDefaultValue; rowValue.visible = !useTransactionDefaultValue; rowGas.visible = !useTransactionDefaultValue; rowGasPrice.visible = !useTransactionDefaultValue; transactionIndex = index blockIndex = blockIdx typeLoader.transactionIndex = index typeLoader.blockIndex = blockIdx saveStatus = item.saveStatus gasValueEdit.gasValue = item.gas; gasAutoCheck.checked = item.gasAuto ? true : false; gasPriceField.value = item.gasPrice; valueField.value = item.value; var contractId = item.contractId; var functionId = item.functionId; rowFunction.visible = true; paramValues = item.parameters !== undefined ? item.parameters : {}; if (item.sender) senderComboBox.select(item.sender); contractsModel.clear(); var contractIndex = -1; var contracts = codeModel.contracts; for (var c in contracts) { contractsModel.append({ cid: c, text: contracts[c].contract.name }); if (contracts[c].contract.name === contractId) contractIndex = contractsModel.count - 1; } if (contractIndex == -1 && contractsModel.count > 0) contractIndex = 0; //@todo suggest unused contract contractComboBox.currentIndex = contractIndex; recipients.accounts = senderComboBox.model; recipients.subType = "address"; recipients.load(); recipients.init(); recipients.select(contractId); if (item.isContractCreation) loadFunctions(contractComboBox.currentValue()); else loadFunctions(contractFromToken(recipients.currentValue())) selectFunction(functionId); trType.checked = item.isContractCreation trType.init(); paramsModel = []; if (item.isContractCreation) loadCtorParameters(); else loadParameters(); visible = true; valueField.focus = true; } function loadCtorParameters(contractId) { paramsModel = []; var contract = codeModel.contracts[contractId]; if (contract) { var params = contract.contract.constructor.parameters; for (var p = 0; p < params.length; p++) loadParameter(params[p]); } initTypeLoader(); } function loadFunctions(contractId) { functionsModel.clear(); functionsModel.append({ text: " - " }); var contract = codeModel.contracts[contractId]; if (contract) { var functions = codeModel.contracts[contractId].contract.functions; for (var f = 0; f < functions.length; f++) { functionsModel.append({ text: functions[f].name }); } } } function selectContract(contractName) { for (var k = 0; k < contractsModel.count; k++) { if (contractsModel.get(k).cid === contractName) { contractComboBox.currentIndex = k; break; } } } function selectFunction(functionId) { var functionIndex = -1; for (var f = 0; f < functionsModel.count; f++) if (functionsModel.get(f).text === functionId) functionIndex = f; if (functionIndex == -1 && functionsModel.count > 0) functionIndex = 0; //@todo suggest unused function functionComboBox.currentIndex = functionIndex; } function loadParameter(parameter) { var type = parameter.type; var pname = parameter.name; paramsModel.push({ name: pname, type: type }); } function loadParameters() { paramsModel = [] if (functionComboBox.currentIndex >= 0 && functionComboBox.currentIndex < functionsModel.count) { var contract = codeModel.contracts[contractFromToken(recipients.currentValue())]; if (contract) { var func = contract.contract.functions[functionComboBox.currentIndex - 1]; if (func) { var parameters = func.parameters; for (var p = 0; p < parameters.length; p++) loadParameter(parameters[p]); } } } initTypeLoader(); } function initTypeLoader() { typeLoader.value = {} typeLoader.members = [] typeLoader.value = paramValues; typeLoader.members = paramsModel; paramLabel.visible = paramsModel.length > 0; paramScroll.visible = paramsModel.length > 0; modalTransactionDialog.height = (paramsModel.length > 0 ? 500 : 300); } function acceptAndClose() { close(); accepted(); } function close() { visible = false; } function getItem() { var item; if (!useTransactionDefaultValue) { item = { contractId: transactionDialog.contractId, functionId: transactionDialog.functionId, gas: transactionDialog.gas, gasAuto: transactionDialog.gasAuto, gasPrice: transactionDialog.gasPrice, value: transactionDialog.transactionValue, parameters: {}, }; } else { item = TransactionHelper.defaultTransaction(); item.contractId = transactionDialog.contractId; item.functionId = transactionDialog.functionId; } item.isContractCreation = trType.checked; if (item.isContractCreation) item.functionId = item.contractId; item.isFunctionCall = item.functionId !== " - "; if (!item.isContractCreation) { item.contractId = recipients.currentText; item.label = item.contractId + " " + item.functionId; if (recipients.current().type === "address") { item.functionId = ""; item.isFunctionCall = false; } } else { item.functionId = item.contractId; item.label = qsTr("Deploy") + " " + item.contractId; } item.saveStatus = saveStatus item.sender = senderComboBox.model[senderComboBox.currentIndex].secret; item.parameters = paramValues; return item; } function contractFromToken(token) { if (token.indexOf('<') === 0) return token.replace("<", "").replace(">", "").split(" - ")[0]; return token; } contentItem: Rectangle { color: transactionDialogStyle.generic.backgroundColor ColumnLayout { anchors.fill: parent ColumnLayout { anchors.fill: parent anchors.margins: 10 ColumnLayout { id: dialogContent anchors.top: parent.top spacing: 10 RowLayout { id: rowSender Layout.fillWidth: true height: 150 DefaultLabel { Layout.preferredWidth: 75 text: qsTr("Sender") } ComboBox { function select(secret) { for (var i in model) if (model[i].secret === secret) { currentIndex = i; break; } } id: senderComboBox Layout.preferredWidth: 350 currentIndex: 0 textRole: "name" editable: false } } RowLayout { id: rowIsContract Layout.fillWidth: true height: 150 CheckBox { id: trType onCheckedChanged: { init(); } function init() { rowFunction.visible = !checked; rowContract.visible = checked; rowRecipient.visible = !checked; paramLabel.visible = checked; paramScroll.visible = checked; functionComboBox.enabled = !checked; if (checked) loadCtorParameters(contractComboBox.currentValue()); } text: qsTr("is contract creation") checked: true } } RowLayout { id: rowRecipient Layout.fillWidth: true height: 150 DefaultLabel { Layout.preferredWidth: 75 text: qsTr("Recipient") } QAddressView { id: recipients onIndexChanged: { rowFunction.visible = current().type === "contract"; paramLabel.visible = current().type === "contract"; paramScroll.visible = current().type === "contract"; if (!rowIsContract.checked) loadFunctions(contractFromToken(recipients.currentValue())) } } } RowLayout { id: rowContract Layout.fillWidth: true height: 150 DefaultLabel { Layout.preferredWidth: 75 text: qsTr("Contract") } ComboBox { id: contractComboBox function currentValue() { return (currentIndex >=0 && currentIndex < contractsModel.count) ? contractsModel.get(currentIndex).cid : ""; } Layout.preferredWidth: 350 currentIndex: -1 textRole: "text" editable: false model: ListModel { id: contractsModel } onCurrentIndexChanged: { loadCtorParameters(currentValue()); } } } RowLayout { id: rowFunction Layout.fillWidth: true height: 150 DefaultLabel { Layout.preferredWidth: 75 text: qsTr("Function") } ComboBox { id: functionComboBox Layout.preferredWidth: 350 currentIndex: -1 textRole: "text" editable: false model: ListModel { id: functionsModel } onCurrentIndexChanged: { loadParameters(); } } } CommonSeparator { Layout.fillWidth: true } RowLayout { id: rowValue Layout.fillWidth: true height: 150 DefaultLabel { Layout.preferredWidth: 75 text: qsTr("Value") } Ether { id: valueField edit: true displayFormattedValue: true } } CommonSeparator { Layout.fillWidth: true } RowLayout { id: rowGas Layout.fillWidth: true height: 150 DefaultLabel { Layout.preferredWidth: 75 text: qsTr("Gas") } DefaultTextField { property variant gasValue onGasValueChanged: text = gasValue.value(); onTextChanged: gasValue.setValue(text); implicitWidth: 200 enabled: !gasAutoCheck.checked id: gasValueEdit; } CheckBox { id: gasAutoCheck checked: true text: qsTr("Auto"); } } CommonSeparator { Layout.fillWidth: true } RowLayout { id: rowGasPrice Layout.fillWidth: true height: 150 DefaultLabel { Layout.preferredWidth: 75 text: qsTr("Gas Price") } Ether { id: gasPriceField edit: true displayFormattedValue: true } } CommonSeparator { Layout.fillWidth: true } DefaultLabel { id: paramLabel text: qsTr("Parameters:") Layout.preferredWidth: 75 } ScrollView { id: paramScroll anchors.top: paramLabel.bottom anchors.topMargin: 10 Layout.fillWidth: true Layout.fillHeight: true StructView { id: typeLoader Layout.preferredWidth: 150 members: paramsModel; accounts: senderComboBox.model context: "parameter" } } CommonSeparator { Layout.fillWidth: true visible: paramsModel.length > 0 } } } RowLayout { anchors.bottom: parent.bottom anchors.right: parent.right; Button { text: qsTr("OK"); onClicked: { var invalid = InputValidator.validate(paramsModel, paramValues); if (invalid.length === 0) { close(); accepted(); } else { errorDialog.text = qsTr("Some parameters are invalid:\n"); for (var k in invalid) errorDialog.text += invalid[k].message + "\n"; errorDialog.open(); } } } Button { text: qsTr("Cancel"); onClicked: close(); } MessageDialog { id: errorDialog standardButtons: StandardButton.Ok icon: StandardIcon.Critical } } } } }