mirror of
https://github.com/monero-project/monero-gui.git
synced 2025-01-22 18:54:37 +00:00
Merge pull request #268
3788943
Addressbook: Fix size_t build error (Jaquee)28a220d
Addressbook: New tooltip implementation (old didn't work on ubuntu) (Jaquee)a0f35aa
bring tooltip to front (Jaquee)0e5329a
Addressbook: updated references to bitmonero (Jaquee)a344f17
AddressBook: basic functions (Jaquee)
This commit is contained in:
commit
575fab90c7
16 changed files with 430 additions and 47 deletions
|
@ -311,14 +311,7 @@ Rectangle {
|
||||||
color: "#505050"
|
color: "#505050"
|
||||||
height: 1
|
height: 1
|
||||||
}
|
}
|
||||||
/*
|
|
||||||
Rectangle {
|
|
||||||
anchors.left: parent.left
|
|
||||||
anchors.right: parent.right
|
|
||||||
anchors.leftMargin: 16
|
|
||||||
color: historyButton.checked || addressBookButton.checked ? "#1C1C1C" : "#505050"
|
|
||||||
height: 1
|
|
||||||
}
|
|
||||||
// ------------- AddressBook tab ---------------
|
// ------------- AddressBook tab ---------------
|
||||||
|
|
||||||
MenuButton {
|
MenuButton {
|
||||||
|
@ -339,11 +332,11 @@ Rectangle {
|
||||||
anchors.left: parent.left
|
anchors.left: parent.left
|
||||||
anchors.right: parent.right
|
anchors.right: parent.right
|
||||||
anchors.leftMargin: 16
|
anchors.leftMargin: 16
|
||||||
color: addressBookButton.checked || miningButton.checked ? "#1C1C1C" : "#505050"
|
color: "#505050"
|
||||||
height: 1
|
height: 1
|
||||||
}
|
}
|
||||||
|
|
||||||
// ------------- Mining tab ---------------
|
/* // ------------- Mining tab ---------------
|
||||||
MenuButton {
|
MenuButton {
|
||||||
id: miningButton
|
id: miningButton
|
||||||
anchors.left: parent.left
|
anchors.left: parent.left
|
||||||
|
|
|
@ -51,6 +51,7 @@ Rectangle {
|
||||||
property History historyView: History { }
|
property History historyView: History { }
|
||||||
property Sign signView: Sign { }
|
property Sign signView: Sign { }
|
||||||
property Settings settingsView: Settings { }
|
property Settings settingsView: Settings { }
|
||||||
|
property AddressBook addressBookView: AddressBook { }
|
||||||
|
|
||||||
|
|
||||||
signal paymentClicked(string address, string paymentId, string amount, int mixinCount, int priority, string description)
|
signal paymentClicked(string address, string paymentId, string amount, int mixinCount, int priority, string description)
|
||||||
|
@ -81,6 +82,12 @@ Rectangle {
|
||||||
transferView.updateStatus();
|
transferView.updateStatus();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// send from AddressBook
|
||||||
|
function sendTo(address, paymentId, description){
|
||||||
|
root.state = "Transfer";
|
||||||
|
transferView.sendTo(address, paymentId, description);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
// XXX: just for memo, to be removed
|
// XXX: just for memo, to be removed
|
||||||
// states: [
|
// states: [
|
||||||
|
@ -127,7 +134,7 @@ Rectangle {
|
||||||
PropertyChanges { target: root; currentView: txkeyView }
|
PropertyChanges { target: root; currentView: txkeyView }
|
||||||
}, State {
|
}, State {
|
||||||
name: "AddressBook"
|
name: "AddressBook"
|
||||||
PropertyChanges { /*TODO*/ }
|
PropertyChanges { target: root; currentView: addressBookView }
|
||||||
}, State {
|
}, State {
|
||||||
name: "Sign"
|
name: "Sign"
|
||||||
PropertyChanges { target: root; currentView: signView }
|
PropertyChanges { target: root; currentView: signView }
|
||||||
|
|
|
@ -77,14 +77,14 @@ ListView {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Text {
|
TextEdit {
|
||||||
id: addressText
|
id: addressText
|
||||||
|
selectByMouse: true
|
||||||
anchors.bottom: descriptionText.bottom
|
anchors.bottom: descriptionText.bottom
|
||||||
anchors.left: descriptionText.right
|
anchors.left: descriptionText.right
|
||||||
anchors.right: dropdown.left
|
anchors.right: dropdown.left
|
||||||
anchors.leftMargin: description.length > 0 ? 12 : 0
|
anchors.leftMargin: description.length > 0 ? 12 : 0
|
||||||
anchors.rightMargin: 12
|
anchors.rightMargin: 40
|
||||||
elide: Text.ElideRight
|
|
||||||
font.family: "Arial"
|
font.family: "Arial"
|
||||||
font.pixelSize: 16
|
font.pixelSize: 16
|
||||||
font.letterSpacing: -1
|
font.letterSpacing: -1
|
||||||
|
@ -103,17 +103,18 @@ ListView {
|
||||||
font.pixelSize: 12
|
font.pixelSize: 12
|
||||||
font.letterSpacing: -1
|
font.letterSpacing: -1
|
||||||
color: "#535353"
|
color: "#535353"
|
||||||
text: qsTr("Payment ID:") + + translationManager.emptyString
|
text: qsTr("Payment ID:") + translationManager.emptyString
|
||||||
}
|
}
|
||||||
|
|
||||||
Text {
|
TextEdit {
|
||||||
|
selectByMouse: true;
|
||||||
anchors.bottom: paymentLabel.bottom
|
anchors.bottom: paymentLabel.bottom
|
||||||
anchors.left: paymentLabel.right
|
anchors.left: paymentLabel.right
|
||||||
anchors.leftMargin: 12
|
anchors.leftMargin: 12
|
||||||
anchors.rightMargin: 12
|
anchors.rightMargin: 12
|
||||||
anchors.right: dropdown.left
|
anchors.right: dropdown.left
|
||||||
|
|
||||||
elide: Text.ElideRight
|
|
||||||
font.family: "Arial"
|
font.family: "Arial"
|
||||||
font.pixelSize: 13
|
font.pixelSize: 13
|
||||||
font.letterSpacing: -1
|
font.letterSpacing: -1
|
||||||
|
@ -125,7 +126,7 @@ ListView {
|
||||||
id: dropModel
|
id: dropModel
|
||||||
ListElement { name: "<b>Copy address to clipboard</b>"; icon: "../images/dropdownCopy.png" }
|
ListElement { name: "<b>Copy address to clipboard</b>"; icon: "../images/dropdownCopy.png" }
|
||||||
ListElement { name: "<b>Send to same destination</b>"; icon: "../images/dropdownSend.png" }
|
ListElement { name: "<b>Send to same destination</b>"; icon: "../images/dropdownSend.png" }
|
||||||
ListElement { name: "<b>Find similar transactions</b>"; icon: "../images/dropdownSearch.png" }
|
// ListElement { name: "<b>Find similar transactions</b>"; icon: "../images/dropdownSearch.png" }
|
||||||
ListElement { name: "<b>Remove from history</b>"; icon: "../images/dropdownDel.png" }
|
ListElement { name: "<b>Remove from history</b>"; icon: "../images/dropdownDel.png" }
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -146,6 +147,16 @@ ListView {
|
||||||
onOptionClicked: {
|
onOptionClicked: {
|
||||||
if(option === 0)
|
if(option === 0)
|
||||||
clipboard.setText(address)
|
clipboard.setText(address)
|
||||||
|
else if(option === 1){
|
||||||
|
console.log("Sending to: ", address +" "+ paymentId);
|
||||||
|
middlePanel.sendTo(address, paymentId, description);
|
||||||
|
leftPanel.selectItem(middlePanel.state)
|
||||||
|
} else if(option === 2){
|
||||||
|
console.log("Delete: ", rowId);
|
||||||
|
currentWallet.addressBookModel.deleteRow(rowId);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -120,8 +120,8 @@ Item {
|
||||||
repeat: true
|
repeat: true
|
||||||
running: false
|
running: false
|
||||||
onTriggered: {
|
onTriggered: {
|
||||||
if(((tipItem.visible && !tipItem.containsMouse) || !tipItem.visible) && !mouseArea.containsMouse) {
|
if(((appWindow.toolTip.visible && !appWindow.toolTip.containsMouse) || !appWindow.toolTip.visible) && !mouseArea.containsMouse) {
|
||||||
tipItem.visible = false
|
appWindow.toolTip.visible = false
|
||||||
dropdown.expanded = false
|
dropdown.expanded = false
|
||||||
currentIndex = -1
|
currentIndex = -1
|
||||||
timer.stop()
|
timer.stop()
|
||||||
|
@ -148,6 +148,10 @@ Item {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
onClicked: {
|
||||||
|
optionClicked(currentIndex)
|
||||||
|
}
|
||||||
|
|
||||||
onExited: timer.start()
|
onExited: timer.start()
|
||||||
preventStealing: true
|
preventStealing: true
|
||||||
z: 1
|
z: 1
|
||||||
|
@ -181,7 +185,6 @@ Item {
|
||||||
height: 30
|
height: 30
|
||||||
color: containsMouse ? "#F0EEEE" : "#DBDBDB"
|
color: containsMouse ? "#F0EEEE" : "#DBDBDB"
|
||||||
//radius: index === repeater.count - 1 ? 5 : 0
|
//radius: index === repeater.count - 1 ? 5 : 0
|
||||||
|
|
||||||
Rectangle {
|
Rectangle {
|
||||||
anchors.left: parent.left
|
anchors.left: parent.left
|
||||||
anchors.top: parent.top
|
anchors.top: parent.top
|
||||||
|
@ -207,15 +210,18 @@ Item {
|
||||||
|
|
||||||
onContainsMouseChanged: {
|
onContainsMouseChanged: {
|
||||||
if(containsMouse) {
|
if(containsMouse) {
|
||||||
var pos = rootItem.mapFromItem(delegate, 30, -20)
|
var pos = rootItem.mapFromItem(delegate, 30, -25)
|
||||||
tipItem.text = name
|
appWindow.toolTip.text = name
|
||||||
tipItem.x = pos.x + appWindow.x
|
appWindow.toolTip.x = pos.x - appWindow.toolTip.width
|
||||||
if(tipItem.height > 30)
|
// if(appWindow.toolTip.height > 30)
|
||||||
pos.y -= tipItem.height - 30
|
// pos.y -= appWindow.toolTip.height - 30
|
||||||
tipItem.y = pos.y + appWindow.y
|
appWindow.toolTip.y = pos.y
|
||||||
tipItem.visible = true
|
appWindow.toolTip.visible = true
|
||||||
|
appWindow.toolTip.z = 3
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
11
main.cpp
11
main.cpp
|
@ -46,6 +46,8 @@
|
||||||
#include "model/TransactionHistoryModel.h"
|
#include "model/TransactionHistoryModel.h"
|
||||||
#include "model/TransactionHistorySortFilterModel.h"
|
#include "model/TransactionHistorySortFilterModel.h"
|
||||||
#include "daemon/DaemonManager.h"
|
#include "daemon/DaemonManager.h"
|
||||||
|
#include "AddressBook.h"
|
||||||
|
#include "model/AddressBookModel.h"
|
||||||
|
|
||||||
|
|
||||||
int main(int argc, char *argv[])
|
int main(int argc, char *argv[])
|
||||||
|
@ -92,6 +94,13 @@ int main(int argc, char *argv[])
|
||||||
|
|
||||||
qmlRegisterUncreatableType<DaemonManager>("moneroComponents.DaemonManager", 1, 0, "DaemonManager",
|
qmlRegisterUncreatableType<DaemonManager>("moneroComponents.DaemonManager", 1, 0, "DaemonManager",
|
||||||
"DaemonManager can't be instantiated directly");
|
"DaemonManager can't be instantiated directly");
|
||||||
|
|
||||||
|
qmlRegisterUncreatableType<AddressBookModel>("moneroComponents.AddressBookModel", 1, 0, "AddressBookModel",
|
||||||
|
"AddressBookModel can't be instantiated directly");
|
||||||
|
|
||||||
|
qmlRegisterUncreatableType<AddressBook>("moneroComponents.AddressBook", 1, 0, "AddressBook",
|
||||||
|
"AddressBook can't be instantiated directly");
|
||||||
|
|
||||||
qRegisterMetaType<PendingTransaction::Priority>();
|
qRegisterMetaType<PendingTransaction::Priority>();
|
||||||
qRegisterMetaType<TransactionInfo::Direction>();
|
qRegisterMetaType<TransactionInfo::Direction>();
|
||||||
qRegisterMetaType<TransactionHistoryModel::TransactionInfoRole>();
|
qRegisterMetaType<TransactionHistoryModel::TransactionInfoRole>();
|
||||||
|
@ -152,7 +161,7 @@ int main(int argc, char *argv[])
|
||||||
QObject::connect(eventFilter, SIGNAL(mousePressed(QVariant,QVariant,QVariant)), rootObject, SLOT(mousePressed(QVariant,QVariant,QVariant)));
|
QObject::connect(eventFilter, SIGNAL(mousePressed(QVariant,QVariant,QVariant)), rootObject, SLOT(mousePressed(QVariant,QVariant,QVariant)));
|
||||||
QObject::connect(eventFilter, SIGNAL(mouseReleased(QVariant,QVariant,QVariant)), rootObject, SLOT(mouseReleased(QVariant,QVariant,QVariant)));
|
QObject::connect(eventFilter, SIGNAL(mouseReleased(QVariant,QVariant,QVariant)), rootObject, SLOT(mouseReleased(QVariant,QVariant,QVariant)));
|
||||||
|
|
||||||
// WalletManager::instance()->setLogLevel(WalletManager::LogLevel_Max);
|
//WalletManager::instance()->setLogLevel(WalletManager::LogLevel_Max);
|
||||||
|
|
||||||
return app.exec();
|
return app.exec();
|
||||||
}
|
}
|
||||||
|
|
32
main.qml
32
main.qml
|
@ -60,6 +60,7 @@ ApplicationWindow {
|
||||||
property bool daemonSynced: false
|
property bool daemonSynced: false
|
||||||
property int maxWindowHeight: (Screen.height < 900)? 720 : 800;
|
property int maxWindowHeight: (Screen.height < 900)? 720 : 800;
|
||||||
property bool daemonRunning: false
|
property bool daemonRunning: false
|
||||||
|
property alias toolTip: toolTip
|
||||||
|
|
||||||
// true if wallet ever synchronized
|
// true if wallet ever synchronized
|
||||||
property bool walletInitialized : false
|
property bool walletInitialized : false
|
||||||
|
@ -999,6 +1000,37 @@ ApplicationWindow {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// new ToolTip
|
||||||
|
Rectangle {
|
||||||
|
id: toolTip
|
||||||
|
property alias text: content.text
|
||||||
|
width: content.width + 12
|
||||||
|
height: content.height + 17
|
||||||
|
color: "#FF6C3C"
|
||||||
|
//radius: 3
|
||||||
|
visible:false;
|
||||||
|
|
||||||
|
Image {
|
||||||
|
id: tip
|
||||||
|
anchors.top: parent.bottom
|
||||||
|
anchors.right: parent.right
|
||||||
|
anchors.rightMargin: 5
|
||||||
|
source: "../images/tip.png"
|
||||||
|
}
|
||||||
|
|
||||||
|
Text {
|
||||||
|
id: content
|
||||||
|
anchors.horizontalCenter: parent.horizontalCenter
|
||||||
|
y: 6
|
||||||
|
lineHeight: 0.7
|
||||||
|
font.family: "Arial"
|
||||||
|
font.pixelSize: 12
|
||||||
|
font.letterSpacing: -1
|
||||||
|
color: "#FFFFFF"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
onClosing: {
|
onClosing: {
|
||||||
// Close wallet
|
// Close wallet
|
||||||
|
|
|
@ -32,7 +32,9 @@ HEADERS += \
|
||||||
src/QR-Code-generator/BitBuffer.hpp \
|
src/QR-Code-generator/BitBuffer.hpp \
|
||||||
src/QR-Code-generator/QrCode.hpp \
|
src/QR-Code-generator/QrCode.hpp \
|
||||||
src/QR-Code-generator/QrSegment.hpp \
|
src/QR-Code-generator/QrSegment.hpp \
|
||||||
src/daemon/DaemonManager.h
|
src/daemon/DaemonManager.h \
|
||||||
|
src/model/AddressBookModel.h \
|
||||||
|
src/libwalletqt/AddressBook.h
|
||||||
|
|
||||||
|
|
||||||
SOURCES += main.cpp \
|
SOURCES += main.cpp \
|
||||||
|
@ -52,7 +54,9 @@ SOURCES += main.cpp \
|
||||||
src/QR-Code-generator/BitBuffer.cpp \
|
src/QR-Code-generator/BitBuffer.cpp \
|
||||||
src/QR-Code-generator/QrCode.cpp \
|
src/QR-Code-generator/QrCode.cpp \
|
||||||
src/QR-Code-generator/QrSegment.cpp \
|
src/QR-Code-generator/QrSegment.cpp \
|
||||||
src/daemon/DaemonManager.cpp
|
src/daemon/DaemonManager.cpp \
|
||||||
|
src/model/AddressBookModel.cpp \
|
||||||
|
src/libwalletqt/AddressBook.cpp
|
||||||
|
|
||||||
lupdate_only {
|
lupdate_only {
|
||||||
SOURCES = *.qml \
|
SOURCES = *.qml \
|
||||||
|
|
|
@ -28,9 +28,13 @@
|
||||||
|
|
||||||
import QtQuick 2.0
|
import QtQuick 2.0
|
||||||
import "../components"
|
import "../components"
|
||||||
|
import moneroComponents.AddressBook 1.0
|
||||||
|
import moneroComponents.AddressBookModel 1.0
|
||||||
|
|
||||||
Rectangle {
|
Rectangle {
|
||||||
color: "#F0EEEE"
|
color: "#F0EEEE"
|
||||||
|
id: root
|
||||||
|
property var model
|
||||||
|
|
||||||
Text {
|
Text {
|
||||||
id: newEntryText
|
id: newEntryText
|
||||||
|
@ -66,6 +70,7 @@ Rectangle {
|
||||||
anchors.leftMargin: 17
|
anchors.leftMargin: 17
|
||||||
anchors.rightMargin: 17
|
anchors.rightMargin: 17
|
||||||
anchors.topMargin: 5
|
anchors.topMargin: 5
|
||||||
|
error: true;
|
||||||
}
|
}
|
||||||
|
|
||||||
Label {
|
Label {
|
||||||
|
@ -124,6 +129,27 @@ Rectangle {
|
||||||
releasedColor: "#FF6C3C"
|
releasedColor: "#FF6C3C"
|
||||||
pressedColor: "#FF4304"
|
pressedColor: "#FF4304"
|
||||||
text: qsTr("ADD")
|
text: qsTr("ADD")
|
||||||
|
enabled: checkInformation(addressLine.text, paymentIdLine.text, appWindow.persistentSettings.testnet)
|
||||||
|
|
||||||
|
onClicked: {
|
||||||
|
if (!currentWallet.addressBook.addRow(addressLine.text.trim(), paymentIdLine.text.trim(), descriptionLine.text)) {
|
||||||
|
informationPopup.title = qsTr("Error") + translationManager.emptyString;
|
||||||
|
// TODO: check currentWallet.addressBook.errorString() instead.
|
||||||
|
if(currentWallet.addressBook.errorCode() === AddressBook.Invalid_Address)
|
||||||
|
informationPopup.text = qsTr("Invalid address")
|
||||||
|
else if(currentWallet.addressBook.errorCode() === AddressBook.Invalid_Payment_Id)
|
||||||
|
informationPopup.text = qsTr("Invalid Payment ID")
|
||||||
|
else
|
||||||
|
informationPopup.text = qsTr("Can't create entry")
|
||||||
|
|
||||||
|
informationPopup.onCloseCallback = null
|
||||||
|
informationPopup.open();
|
||||||
|
} else {
|
||||||
|
addressLine.text = "";
|
||||||
|
paymentIdLine.text = "";
|
||||||
|
descriptionLine.text = "";
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Item {
|
Item {
|
||||||
|
@ -170,9 +196,10 @@ Rectangle {
|
||||||
|
|
||||||
ListModel {
|
ListModel {
|
||||||
id: columnsModel
|
id: columnsModel
|
||||||
ListElement { columnName: qsTr("Address") + translationManager.emptyString; columnWidth: 148 }
|
// ListElement { columnName: qsTr("Address") + translationManager.emptyString; columnWidth: 148 }
|
||||||
ListElement { columnName: qsTr("Payment ID") + translationManager.emptyString; columnWidth: 148 }
|
// ListElement { columnName: qsTr("Payment ID") + translationManager.emptyString; columnWidth: 148 }
|
||||||
ListElement { columnName: qsTr("Description") + translationManager.emptyString; columnWidth: 148 }
|
// ListElement { columnName: qsTr("Description") + translationManager.emptyString; columnWidth: 148 }
|
||||||
|
//
|
||||||
}
|
}
|
||||||
|
|
||||||
TableHeader {
|
TableHeader {
|
||||||
|
@ -223,7 +250,30 @@ Rectangle {
|
||||||
anchors.leftMargin: 14
|
anchors.leftMargin: 14
|
||||||
anchors.rightMargin: 14
|
anchors.rightMargin: 14
|
||||||
onContentYChanged: flickableScroll.flickableContentYChanged()
|
onContentYChanged: flickableScroll.flickableContentYChanged()
|
||||||
model: testModel
|
model: root.model
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function checkInformation(address, payment_id, testnet) {
|
||||||
|
address = address.trim()
|
||||||
|
payment_id = payment_id.trim()
|
||||||
|
|
||||||
|
var address_ok = walletManager.addressValid(address, testnet)
|
||||||
|
var payment_id_ok = payment_id.length == 0 || walletManager.paymentIdValid(payment_id)
|
||||||
|
var ipid = walletManager.paymentIdFromAddress(address, testnet)
|
||||||
|
if (ipid.length > 0 && payment_id.length > 0)
|
||||||
|
payment_id_ok = false
|
||||||
|
|
||||||
|
addressLine.error = !address_ok
|
||||||
|
paymentIdLine.error = !payment_id_ok
|
||||||
|
|
||||||
|
return address_ok && payment_id_ok
|
||||||
|
}
|
||||||
|
|
||||||
|
function onPageCompleted() {
|
||||||
|
console.log("adress book");
|
||||||
|
root.model = currentWallet.addressBookModel;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -264,15 +264,6 @@ Rectangle {
|
||||||
anchors.topMargin: 5
|
anchors.topMargin: 5
|
||||||
}
|
}
|
||||||
|
|
||||||
function checkAddressAndPaymentID(address, payment_id, testnet) {
|
|
||||||
if (!walletManager.addressValid(address, testnet))
|
|
||||||
return false
|
|
||||||
var ipid = walletManager.paymentIdFromAddress(address, testnet)
|
|
||||||
if (ipid.length > 0)
|
|
||||||
return payment_id === ""
|
|
||||||
return payment_id === "" || walletManager.paymentIdValid(payment_id)
|
|
||||||
}
|
|
||||||
|
|
||||||
function checkInformation(amount, address, payment_id, testnet) {
|
function checkInformation(amount, address, payment_id, testnet) {
|
||||||
address = address.trim()
|
address = address.trim()
|
||||||
payment_id = payment_id.trim()
|
payment_id = payment_id.trim()
|
||||||
|
@ -404,4 +395,11 @@ Rectangle {
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Popuplate fields from addressbook.
|
||||||
|
function sendTo(address, paymentId, description){
|
||||||
|
addressLine.text = address
|
||||||
|
paymentIdLine.text = paymentId
|
||||||
|
descriptionLine.text = description
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
70
src/libwalletqt/AddressBook.cpp
Normal file
70
src/libwalletqt/AddressBook.cpp
Normal file
|
@ -0,0 +1,70 @@
|
||||||
|
#include "AddressBook.h"
|
||||||
|
#include <QDebug>
|
||||||
|
|
||||||
|
AddressBook::AddressBook(Monero::AddressBook *abImpl,QObject *parent)
|
||||||
|
: QObject(parent), m_addressBookImpl(abImpl)
|
||||||
|
{
|
||||||
|
qDebug(__FUNCTION__);
|
||||||
|
getAll();
|
||||||
|
}
|
||||||
|
|
||||||
|
QString AddressBook::errorString() const
|
||||||
|
{
|
||||||
|
return QString::fromStdString(m_addressBookImpl->errorString());
|
||||||
|
}
|
||||||
|
|
||||||
|
int AddressBook::errorCode() const
|
||||||
|
{
|
||||||
|
return m_addressBookImpl->errorCode();
|
||||||
|
}
|
||||||
|
|
||||||
|
QList<Monero::AddressBookRow*> AddressBook::getAll(bool update) const
|
||||||
|
{
|
||||||
|
qDebug(__FUNCTION__);
|
||||||
|
|
||||||
|
emit refreshStarted();
|
||||||
|
|
||||||
|
if(update)
|
||||||
|
m_rows.clear();
|
||||||
|
|
||||||
|
if (m_rows.empty()){
|
||||||
|
for (auto &abr: m_addressBookImpl->getAll()) {
|
||||||
|
m_rows.append(abr);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
emit refreshFinished();
|
||||||
|
return m_rows;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
Monero::AddressBookRow * AddressBook::getRow(int index) const
|
||||||
|
{
|
||||||
|
return m_rows.at(index);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool AddressBook::addRow(const QString &address, const QString &payment_id, const QString &description) const
|
||||||
|
{
|
||||||
|
// virtual bool addRow(const std::string &dst_addr , const std::string &payment_id, const std::string &description) = 0;
|
||||||
|
bool r = m_addressBookImpl->addRow(address.toStdString(), payment_id.toStdString(), description.toStdString());
|
||||||
|
|
||||||
|
if(r)
|
||||||
|
getAll(true);
|
||||||
|
|
||||||
|
return r;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool AddressBook::deleteRow(int rowId) const
|
||||||
|
{
|
||||||
|
bool r = m_addressBookImpl->deleteRow(rowId);
|
||||||
|
|
||||||
|
// Fetch new data from wallet2.
|
||||||
|
getAll(true);
|
||||||
|
|
||||||
|
return r;
|
||||||
|
}
|
||||||
|
|
||||||
|
quint64 AddressBook::count() const
|
||||||
|
{
|
||||||
|
return m_rows.size();
|
||||||
|
}
|
50
src/libwalletqt/AddressBook.h
Normal file
50
src/libwalletqt/AddressBook.h
Normal file
|
@ -0,0 +1,50 @@
|
||||||
|
#ifndef ADDRESSBOOK_H
|
||||||
|
#define ADDRESSBOOK_H
|
||||||
|
|
||||||
|
#include <wallet/wallet2_api.h>
|
||||||
|
#include <QObject>
|
||||||
|
#include <QList>
|
||||||
|
#include <QDateTime>
|
||||||
|
|
||||||
|
namespace Monero {
|
||||||
|
class AddressBook;
|
||||||
|
}
|
||||||
|
class AddressBookRow;
|
||||||
|
|
||||||
|
class AddressBook : public QObject
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
public:
|
||||||
|
Q_INVOKABLE QList<Monero::AddressBookRow*> getAll(bool update = false) const;
|
||||||
|
Q_INVOKABLE Monero::AddressBookRow * getRow(int index) const;
|
||||||
|
Q_INVOKABLE bool addRow(const QString &address, const QString &payment_id, const QString &description) const;
|
||||||
|
Q_INVOKABLE bool deleteRow(int rowId) const;
|
||||||
|
quint64 count() const;
|
||||||
|
Q_INVOKABLE QString errorString() const;
|
||||||
|
Q_INVOKABLE int errorCode() const;
|
||||||
|
|
||||||
|
enum ErrorCode {
|
||||||
|
Status_Ok,
|
||||||
|
General_Error,
|
||||||
|
Invalid_Address,
|
||||||
|
Invalid_Payment_Id
|
||||||
|
};
|
||||||
|
|
||||||
|
Q_ENUM(ErrorCode);
|
||||||
|
|
||||||
|
|
||||||
|
signals:
|
||||||
|
void refreshStarted() const;
|
||||||
|
void refreshFinished() const;
|
||||||
|
|
||||||
|
|
||||||
|
public slots:
|
||||||
|
|
||||||
|
private:
|
||||||
|
explicit AddressBook(Monero::AddressBook * abImpl, QObject *parent);
|
||||||
|
friend class Wallet;
|
||||||
|
Monero::AddressBook * m_addressBookImpl;
|
||||||
|
mutable QList<Monero::AddressBookRow*> m_rows;
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif // ADDRESSBOOK_H
|
|
@ -23,7 +23,8 @@ class PendingTransaction : public QObject
|
||||||
public:
|
public:
|
||||||
enum Status {
|
enum Status {
|
||||||
Status_Ok = Monero::PendingTransaction::Status_Ok,
|
Status_Ok = Monero::PendingTransaction::Status_Ok,
|
||||||
Status_Error = Monero::PendingTransaction::Status_Error
|
Status_Error = Monero::PendingTransaction::Status_Error,
|
||||||
|
Status_Critical = Monero::PendingTransaction::Status_Critical
|
||||||
};
|
};
|
||||||
Q_ENUM(Status)
|
Q_ENUM(Status)
|
||||||
|
|
||||||
|
|
|
@ -1,8 +1,10 @@
|
||||||
#include "Wallet.h"
|
#include "Wallet.h"
|
||||||
#include "PendingTransaction.h"
|
#include "PendingTransaction.h"
|
||||||
#include "TransactionHistory.h"
|
#include "TransactionHistory.h"
|
||||||
|
#include "AddressBook.h"
|
||||||
#include "model/TransactionHistoryModel.h"
|
#include "model/TransactionHistoryModel.h"
|
||||||
#include "model/TransactionHistorySortFilterModel.h"
|
#include "model/TransactionHistorySortFilterModel.h"
|
||||||
|
#include "model/AddressBookModel.h"
|
||||||
#include "wallet/wallet2_api.h"
|
#include "wallet/wallet2_api.h"
|
||||||
|
|
||||||
#include <QFile>
|
#include <QFile>
|
||||||
|
@ -11,6 +13,8 @@
|
||||||
#include <QUrl>
|
#include <QUrl>
|
||||||
#include <QTimer>
|
#include <QTimer>
|
||||||
#include <QtConcurrent/QtConcurrent>
|
#include <QtConcurrent/QtConcurrent>
|
||||||
|
#include <QList>
|
||||||
|
#include <QVector>
|
||||||
|
|
||||||
namespace {
|
namespace {
|
||||||
static const int DAEMON_BLOCKCHAIN_HEIGHT_CACHE_TTL_SECONDS = 10;
|
static const int DAEMON_BLOCKCHAIN_HEIGHT_CACHE_TTL_SECONDS = 10;
|
||||||
|
@ -48,7 +52,6 @@ public:
|
||||||
|
|
||||||
virtual void updated()
|
virtual void updated()
|
||||||
{
|
{
|
||||||
qDebug() << __FUNCTION__;
|
|
||||||
emit m_wallet->updated();
|
emit m_wallet->updated();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -324,6 +327,22 @@ TransactionHistorySortFilterModel *Wallet::historyModel() const
|
||||||
return m_historySortFilterModel;
|
return m_historySortFilterModel;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
AddressBook *Wallet::addressBook() const
|
||||||
|
{
|
||||||
|
return m_addressBook;
|
||||||
|
}
|
||||||
|
|
||||||
|
AddressBookModel *Wallet::addressBookModel() const
|
||||||
|
{
|
||||||
|
|
||||||
|
if (!m_addressBookModel) {
|
||||||
|
Wallet * w = const_cast<Wallet*>(this);
|
||||||
|
m_addressBookModel = new AddressBookModel(w,m_addressBook);
|
||||||
|
}
|
||||||
|
|
||||||
|
return m_addressBookModel;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
QString Wallet::generatePaymentId() const
|
QString Wallet::generatePaymentId() const
|
||||||
{
|
{
|
||||||
|
@ -437,6 +456,8 @@ Wallet::Wallet(Monero::Wallet *w, QObject *parent)
|
||||||
, m_walletImpl(w)
|
, m_walletImpl(w)
|
||||||
, m_history(nullptr)
|
, m_history(nullptr)
|
||||||
, m_historyModel(nullptr)
|
, m_historyModel(nullptr)
|
||||||
|
, m_addressBook(nullptr)
|
||||||
|
, m_addressBookModel(nullptr)
|
||||||
, m_daemonBlockChainHeight(0)
|
, m_daemonBlockChainHeight(0)
|
||||||
, m_daemonBlockChainHeightTtl(DAEMON_BLOCKCHAIN_HEIGHT_CACHE_TTL_SECONDS)
|
, m_daemonBlockChainHeightTtl(DAEMON_BLOCKCHAIN_HEIGHT_CACHE_TTL_SECONDS)
|
||||||
, m_daemonBlockChainTargetHeight(0)
|
, m_daemonBlockChainTargetHeight(0)
|
||||||
|
@ -444,6 +465,7 @@ Wallet::Wallet(Monero::Wallet *w, QObject *parent)
|
||||||
, m_connectionStatusTtl(WALLET_CONNECTION_STATUS_CACHE_TTL_SECONDS)
|
, m_connectionStatusTtl(WALLET_CONNECTION_STATUS_CACHE_TTL_SECONDS)
|
||||||
{
|
{
|
||||||
m_history = new TransactionHistory(m_walletImpl->history(), this);
|
m_history = new TransactionHistory(m_walletImpl->history(), this);
|
||||||
|
m_addressBook = new AddressBook(m_walletImpl->addressBook(), this);
|
||||||
m_walletImpl->setListener(new WalletListenerImpl(this));
|
m_walletImpl->setListener(new WalletListenerImpl(this));
|
||||||
m_connectionStatus = Wallet::ConnectionStatus_Disconnected;
|
m_connectionStatus = Wallet::ConnectionStatus_Disconnected;
|
||||||
// start cache timers
|
// start cache timers
|
||||||
|
@ -456,6 +478,9 @@ Wallet::Wallet(Monero::Wallet *w, QObject *parent)
|
||||||
Wallet::~Wallet()
|
Wallet::~Wallet()
|
||||||
{
|
{
|
||||||
qDebug("~Wallet: Closing wallet");
|
qDebug("~Wallet: Closing wallet");
|
||||||
|
|
||||||
delete m_history;
|
delete m_history;
|
||||||
|
|
||||||
Monero::WalletManagerFactory::getWalletManager()->closeWallet(m_walletImpl);
|
Monero::WalletManagerFactory::getWalletManager()->closeWallet(m_walletImpl);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -15,6 +15,8 @@ namespace Monero {
|
||||||
class TransactionHistory;
|
class TransactionHistory;
|
||||||
class TransactionHistoryModel;
|
class TransactionHistoryModel;
|
||||||
class TransactionHistorySortFilterModel;
|
class TransactionHistorySortFilterModel;
|
||||||
|
class AddressBook;
|
||||||
|
class AddressBookModel;
|
||||||
|
|
||||||
class Wallet : public QObject
|
class Wallet : public QObject
|
||||||
{
|
{
|
||||||
|
@ -32,13 +34,17 @@ class Wallet : public QObject
|
||||||
Q_PROPERTY(QString paymentId READ paymentId WRITE setPaymentId)
|
Q_PROPERTY(QString paymentId READ paymentId WRITE setPaymentId)
|
||||||
Q_PROPERTY(TransactionHistorySortFilterModel * historyModel READ historyModel NOTIFY historyModelChanged)
|
Q_PROPERTY(TransactionHistorySortFilterModel * historyModel READ historyModel NOTIFY historyModelChanged)
|
||||||
Q_PROPERTY(QString path READ path)
|
Q_PROPERTY(QString path READ path)
|
||||||
|
Q_PROPERTY(AddressBookModel * addressBookModel READ addressBookModel)
|
||||||
|
Q_PROPERTY(AddressBook * addressBook READ addressBook)
|
||||||
|
|
||||||
|
|
||||||
public:
|
public:
|
||||||
|
|
||||||
|
|
||||||
enum Status {
|
enum Status {
|
||||||
Status_Ok = Monero::Wallet::Status_Ok,
|
Status_Ok = Monero::Wallet::Status_Ok,
|
||||||
Status_Error = Monero::Wallet::Status_Error
|
Status_Error = Monero::Wallet::Status_Error,
|
||||||
|
Status_Critical = Monero::Wallet::Status_Critical
|
||||||
};
|
};
|
||||||
|
|
||||||
Q_ENUM(Status)
|
Q_ENUM(Status)
|
||||||
|
@ -159,6 +165,12 @@ public:
|
||||||
//! returns transaction history model
|
//! returns transaction history model
|
||||||
TransactionHistorySortFilterModel *historyModel() const;
|
TransactionHistorySortFilterModel *historyModel() const;
|
||||||
|
|
||||||
|
//! returns Address book
|
||||||
|
AddressBook *addressBook() const;
|
||||||
|
|
||||||
|
//! returns adress book model
|
||||||
|
AddressBookModel *addressBookModel() const;
|
||||||
|
|
||||||
//! generate payment id
|
//! generate payment id
|
||||||
Q_INVOKABLE QString generatePaymentId() const;
|
Q_INVOKABLE QString generatePaymentId() const;
|
||||||
|
|
||||||
|
@ -227,6 +239,8 @@ private:
|
||||||
int m_connectionStatusTtl;
|
int m_connectionStatusTtl;
|
||||||
mutable QTime m_connectionStatusTime;
|
mutable QTime m_connectionStatusTime;
|
||||||
mutable bool m_initialized;
|
mutable bool m_initialized;
|
||||||
|
AddressBook * m_addressBook;
|
||||||
|
mutable AddressBookModel * m_addressBookModel;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
|
76
src/model/AddressBookModel.cpp
Normal file
76
src/model/AddressBookModel.cpp
Normal file
|
@ -0,0 +1,76 @@
|
||||||
|
#include "AddressBookModel.h"
|
||||||
|
#include "AddressBook.h"
|
||||||
|
#include <QDebug>
|
||||||
|
#include <QHash>
|
||||||
|
#include <wallet/wallet2_api.h>
|
||||||
|
|
||||||
|
AddressBookModel::AddressBookModel(QObject *parent, AddressBook *addressBook)
|
||||||
|
: QAbstractListModel(parent) , m_addressBook(addressBook)
|
||||||
|
{
|
||||||
|
qDebug(__FUNCTION__);
|
||||||
|
connect(m_addressBook,SIGNAL(refreshStarted()),this,SLOT(startReset()));
|
||||||
|
connect(m_addressBook,SIGNAL(refreshFinished()),this,SLOT(endReset()));
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
void AddressBookModel::startReset(){
|
||||||
|
qDebug(__FUNCTION__);
|
||||||
|
beginResetModel();
|
||||||
|
}
|
||||||
|
void AddressBookModel::endReset(){
|
||||||
|
qDebug(__FUNCTION__);
|
||||||
|
endResetModel();
|
||||||
|
}
|
||||||
|
|
||||||
|
int AddressBookModel::rowCount(const QModelIndex &parent) const
|
||||||
|
{
|
||||||
|
return m_addressBook->count();
|
||||||
|
}
|
||||||
|
|
||||||
|
QVariant AddressBookModel::data(const QModelIndex &index, int role) const
|
||||||
|
{
|
||||||
|
if (!index.isValid())
|
||||||
|
return QVariant();
|
||||||
|
|
||||||
|
if (index.row() < 0 || (unsigned)index.row() >= m_addressBook->count()) {
|
||||||
|
return QVariant();
|
||||||
|
}
|
||||||
|
|
||||||
|
Monero::AddressBookRow * ar = m_addressBook->getRow(index.row());
|
||||||
|
|
||||||
|
QVariant result = "";
|
||||||
|
switch (role) {
|
||||||
|
case AddressBookAddressRole:
|
||||||
|
result = QString::fromStdString(ar->getAddress());
|
||||||
|
break;
|
||||||
|
case AddressBookDescriptionRole:
|
||||||
|
result = QString::fromStdString(ar->getDescription());
|
||||||
|
break;
|
||||||
|
case AddressBookPaymentIdRole:
|
||||||
|
result = QString::fromStdString(ar->getPaymentId());
|
||||||
|
break;
|
||||||
|
case AddressBookRowIdRole:
|
||||||
|
// Qt doesnt support size_t overload type casting
|
||||||
|
result.setValue(ar->getRowId());
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool AddressBookModel::deleteRow(int row)
|
||||||
|
{
|
||||||
|
m_addressBook->deleteRow(row);
|
||||||
|
}
|
||||||
|
|
||||||
|
QHash<int, QByteArray> AddressBookModel::roleNames() const
|
||||||
|
{
|
||||||
|
QHash<int, QByteArray> roleNames = QAbstractListModel::roleNames();
|
||||||
|
roleNames.insert(AddressBookAddressRole, "address");
|
||||||
|
roleNames.insert(AddressBookPaymentIdRole, "paymentId");
|
||||||
|
roleNames.insert(AddressBookDescriptionRole, "description");
|
||||||
|
roleNames.insert(AddressBookRowIdRole, "rowId");
|
||||||
|
|
||||||
|
|
||||||
|
return roleNames;
|
||||||
|
}
|
37
src/model/AddressBookModel.h
Normal file
37
src/model/AddressBookModel.h
Normal file
|
@ -0,0 +1,37 @@
|
||||||
|
#ifndef ADDRESSBOOKMODEL_H
|
||||||
|
#define ADDRESSBOOKMODEL_H
|
||||||
|
|
||||||
|
#include <QAbstractListModel>
|
||||||
|
|
||||||
|
class AddressBook;
|
||||||
|
|
||||||
|
class AddressBookModel : public QAbstractListModel
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
|
||||||
|
public:
|
||||||
|
enum AddressBookRowRole {
|
||||||
|
AddressBookRole = Qt::UserRole + 1, // for the AddressBookRow object;
|
||||||
|
AddressBookAddressRole,
|
||||||
|
AddressBookDescriptionRole,
|
||||||
|
AddressBookPaymentIdRole,
|
||||||
|
AddressBookRowIdRole,
|
||||||
|
};
|
||||||
|
Q_ENUM(AddressBookRowRole)
|
||||||
|
|
||||||
|
AddressBookModel(QObject *parent, AddressBook * addressBook);
|
||||||
|
|
||||||
|
int rowCount(const QModelIndex &parent = QModelIndex()) const override;
|
||||||
|
QVariant data(const QModelIndex &index, int role = Qt::DisplayRole) const override;
|
||||||
|
Q_INVOKABLE bool deleteRow(int row);
|
||||||
|
virtual QHash<int, QByteArray> roleNames() const override;
|
||||||
|
|
||||||
|
public slots:
|
||||||
|
void startReset();
|
||||||
|
void endReset();
|
||||||
|
|
||||||
|
private:
|
||||||
|
AddressBook * m_addressBook;
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif // ADDRESSBOOKMODEL_H
|
Loading…
Reference in a new issue