322 lines
9.5 KiB
QML
322 lines
9.5 KiB
QML
import QtQuick 2.6
|
|
import QtQuick.Controls 2.0
|
|
import QtQuick.Layouts 1.0
|
|
import QtQuick.Controls.Material 2.0
|
|
import QtQml.Models 2.1
|
|
|
|
import org.electrum 1.0
|
|
|
|
import "controls"
|
|
|
|
Pane {
|
|
id: rootItem
|
|
|
|
function clear() {
|
|
recipient.text = ''
|
|
amount.text = ''
|
|
}
|
|
|
|
GridLayout {
|
|
id: form
|
|
width: parent.width
|
|
rowSpacing: constants.paddingSmall
|
|
columnSpacing: constants.paddingSmall
|
|
columns: 4
|
|
|
|
BalanceSummary {
|
|
Layout.columnSpan: 4
|
|
Layout.alignment: Qt.AlignHCenter
|
|
}
|
|
|
|
Label {
|
|
text: qsTr('Recipient')
|
|
}
|
|
|
|
TextArea {
|
|
id: recipient
|
|
Layout.columnSpan: 2
|
|
Layout.fillWidth: true
|
|
font.family: FixedFont
|
|
wrapMode: Text.Wrap
|
|
placeholderText: qsTr('Paste address or invoice')
|
|
onTextChanged: {
|
|
if (activeFocus)
|
|
invoice.recipient = text
|
|
}
|
|
}
|
|
|
|
RowLayout {
|
|
spacing: 0
|
|
ToolButton {
|
|
icon.source: '../../icons/paste.png'
|
|
icon.height: constants.iconSizeMedium
|
|
icon.width: constants.iconSizeMedium
|
|
onClicked: invoice.recipient = AppController.clipboardToText()
|
|
}
|
|
ToolButton {
|
|
icon.source: '../../icons/qrcode.png'
|
|
icon.height: constants.iconSizeMedium
|
|
icon.width: constants.iconSizeMedium
|
|
scale: 1.2
|
|
onClicked: {
|
|
var page = app.stack.push(Qt.resolvedUrl('Scan.qml'))
|
|
page.onFound.connect(function() {
|
|
invoice.recipient = page.scanData
|
|
})
|
|
}
|
|
}
|
|
}
|
|
|
|
Label {
|
|
text: qsTr('Amount')
|
|
}
|
|
|
|
TextField {
|
|
id: amount
|
|
font.family: FixedFont
|
|
placeholderText: qsTr('Amount')
|
|
Layout.preferredWidth: parent.width /2
|
|
inputMethodHints: Qt.ImhPreferNumbers
|
|
property Amount textAsSats
|
|
onTextChanged: {
|
|
textAsSats = Config.unitsToSats(amount.text)
|
|
if (amountFiat.activeFocus)
|
|
return
|
|
amountFiat.text = Daemon.fx.fiatValue(amount.textAsSats)
|
|
}
|
|
|
|
Connections {
|
|
target: Config
|
|
function onBaseUnitChanged() {
|
|
amount.text = amount.textAsSats != 0 ? Config.satsToUnits(amount.textAsSats) : ''
|
|
}
|
|
}
|
|
}
|
|
|
|
Label {
|
|
text: Config.baseUnit
|
|
color: Material.accentColor
|
|
Layout.fillWidth: true
|
|
}
|
|
|
|
Item { width: 1; height: 1 }
|
|
|
|
Item { width: 1; height: 1; visible: Daemon.fx.enabled }
|
|
|
|
TextField {
|
|
id: amountFiat
|
|
visible: Daemon.fx.enabled
|
|
font.family: FixedFont
|
|
Layout.preferredWidth: parent.width /2
|
|
placeholderText: qsTr('Amount')
|
|
inputMethodHints: Qt.ImhPreferNumbers
|
|
onTextChanged: {
|
|
if (amountFiat.activeFocus)
|
|
amount.text = text == '' ? '' : Config.satsToUnits(Daemon.fx.satoshiValue(amountFiat.text))
|
|
}
|
|
}
|
|
|
|
Label {
|
|
visible: Daemon.fx.enabled
|
|
text: Daemon.fx.fiatCurrency
|
|
color: Material.accentColor
|
|
Layout.fillWidth: true
|
|
}
|
|
|
|
Item { visible: Daemon.fx.enabled ; height: 1; width: 1 }
|
|
|
|
Label {
|
|
text: qsTr('Description')
|
|
}
|
|
|
|
TextField {
|
|
id: message
|
|
font.family: FixedFont
|
|
placeholderText: qsTr('Message')
|
|
Layout.columnSpan: 3
|
|
Layout.fillWidth: true
|
|
}
|
|
|
|
RowLayout {
|
|
Layout.columnSpan: 4
|
|
Layout.alignment: Qt.AlignHCenter
|
|
spacing: constants.paddingMedium
|
|
|
|
Button {
|
|
text: qsTr('Save')
|
|
enabled: invoice.invoiceType != Invoice.Invalid
|
|
onClicked: {
|
|
Daemon.currentWallet.create_invoice(recipient.text, amount.text, message.text)
|
|
}
|
|
}
|
|
|
|
Button {
|
|
text: qsTr('Pay now')
|
|
enabled: invoice.invoiceType != Invoice.Invalid // TODO && has funds
|
|
onClicked: {
|
|
var f_amount = parseFloat(amount.text)
|
|
if (isNaN(f_amount))
|
|
return
|
|
var dialog = confirmPaymentDialog.createObject(app, {
|
|
'address': recipient.text,
|
|
'satoshis': Config.unitsToSats(amount.text),
|
|
'message': message.text
|
|
})
|
|
dialog.open()
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
Frame {
|
|
verticalPadding: 0
|
|
horizontalPadding: 0
|
|
|
|
anchors {
|
|
top: form.bottom
|
|
topMargin: constants.paddingXLarge
|
|
left: parent.left
|
|
right: parent.right
|
|
bottom: parent.bottom
|
|
}
|
|
|
|
background: PaneInsetBackground {}
|
|
|
|
ColumnLayout {
|
|
spacing: 0
|
|
anchors.fill: parent
|
|
|
|
Item {
|
|
Layout.preferredHeight: hitem.height
|
|
Layout.preferredWidth: parent.width
|
|
Rectangle {
|
|
anchors.fill: parent
|
|
color: Qt.lighter(Material.background, 1.25)
|
|
}
|
|
RowLayout {
|
|
id: hitem
|
|
width: parent.width
|
|
Label {
|
|
text: qsTr('Send queue')
|
|
font.pixelSize: constants.fontSizeXLarge
|
|
}
|
|
}
|
|
}
|
|
|
|
ListView {
|
|
id: listview
|
|
Layout.fillHeight: true
|
|
Layout.fillWidth: true
|
|
clip: true
|
|
|
|
model: DelegateModel {
|
|
id: delegateModel
|
|
model: Daemon.currentWallet.invoiceModel
|
|
delegate: InvoiceDelegate {
|
|
onClicked: {
|
|
var dialog = confirmInvoiceDialog.createObject(app, {'invoice' : invoice, 'invoice_key': model.key})
|
|
dialog.open()
|
|
}
|
|
}
|
|
}
|
|
|
|
remove: Transition {
|
|
NumberAnimation { properties: 'scale'; to: 0.75; duration: 300 }
|
|
NumberAnimation { properties: 'opacity'; to: 0; duration: 300 }
|
|
}
|
|
removeDisplaced: Transition {
|
|
SequentialAnimation {
|
|
PauseAnimation { duration: 200 }
|
|
SpringAnimation { properties: 'y'; duration: 100; spring: 5; damping: 0.5; mass: 2 }
|
|
}
|
|
}
|
|
|
|
ScrollIndicator.vertical: ScrollIndicator { }
|
|
}
|
|
}
|
|
}
|
|
|
|
Component {
|
|
id: confirmPaymentDialog
|
|
ConfirmPaymentDialog {}
|
|
}
|
|
|
|
Component {
|
|
id: confirmInvoiceDialog
|
|
ConfirmInvoiceDialog {
|
|
onDoPay: {
|
|
if (invoice.invoiceType == Invoice.OnchainInvoice) {
|
|
var dialog = confirmPaymentDialog.createObject(rootItem, {
|
|
'address': invoice.address,
|
|
'satoshis': invoice.amount,
|
|
'message': invoice.message
|
|
})
|
|
dialog.open()
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
Connections {
|
|
target: Daemon.currentWallet
|
|
function onInvoiceStatusChanged(key, status) {
|
|
// TODO: status from?
|
|
//Daemon.currentWallet.invoiceModel.updateInvoice(key, status)
|
|
}
|
|
}
|
|
|
|
Connections {
|
|
target: Daemon.fx
|
|
function onQuotesUpdated() {
|
|
amountFiat.text = amount.text == ''
|
|
? ''
|
|
: Daemon.fx.fiatValue(Config.unitsToSats(amount.text))
|
|
}
|
|
}
|
|
|
|
// make clicking the dialog background move the scope away from textedit fields
|
|
// so the keyboard goes away
|
|
MouseArea {
|
|
anchors.fill: parent
|
|
z: -1000
|
|
onClicked: parkFocus.focus = true
|
|
FocusScope { id: parkFocus }
|
|
}
|
|
|
|
Invoice {
|
|
id: invoice
|
|
wallet: Daemon.currentWallet
|
|
onValidationError: {
|
|
if (recipient.activeFocus) {
|
|
// no popups when editing
|
|
return
|
|
}
|
|
var dialog = app.messageDialog.createObject(app, {'text': message })
|
|
dialog.open()
|
|
rootItem.clear()
|
|
}
|
|
onValidationWarning: {
|
|
if (code == 'no_channels') {
|
|
var dialog = app.messageDialog.createObject(app, {'text': message })
|
|
dialog.open()
|
|
// TODO: ask user to open a channel, if funds allow
|
|
// and maybe store invoice if expiry allows
|
|
}
|
|
}
|
|
onValidationSuccess: {
|
|
// address only -> fill form fields
|
|
// else -> show invoice confirmation dialog
|
|
if (invoiceType == Invoice.OnchainOnlyAddress)
|
|
recipient.text = invoice.recipient
|
|
else {
|
|
var dialog = confirmInvoiceDialog.createObject(rootItem, {'invoice': invoice})
|
|
dialog.open()
|
|
}
|
|
}
|
|
onInvoiceSaved: {
|
|
console.log('invoice got saved')
|
|
Daemon.currentWallet.invoiceModel.init_model()
|
|
}
|
|
}
|
|
}
|