You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
319 lines
9.5 KiB
319 lines
9.5 KiB
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 = 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() |
|
} |
|
} |
|
}
|
|
|