2018-03-07 02:01:53 +02:00
// Copyright (c) 2018, The Monero Project
//
// All rights reserved.
//
// Redistribution and use in source and binary forms, with or without modification, are
// permitted provided that the following conditions are met:
//
// 1. Redistributions of source code must retain the above copyright notice, this list of
// conditions and the following disclaimer.
//
// 2. Redistributions in binary form must reproduce the above copyright notice, this list
// of conditions and the following disclaimer in the documentation and/or other
// materials provided with the distribution.
//
// 3. Neither the name of the copyright holder nor the names of its contributors may be
// used to endorse or promote products derived from this software without specific
// prior written permission.
//
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY
// EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
// MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL
// THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
// PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
// INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
// STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF
// THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
2019-04-11 04:17:29 +03:00
import QtQuick 2.9
2018-03-07 02:01:53 +02:00
import QtQuick . Controls 1.4
import QtQuick . Controls . Styles 1.4
import QtQuick . Layouts 1.1
import QtQuick . Dialogs 1.2
2019-02-01 23:20:38 +02:00
import "../components" as MoneroComponents
2018-03-07 02:01:53 +02:00
import moneroComponents . Clipboard 1.0
Rectangle {
2018-04-22 21:07:31 +03:00
property alias panelHeight: mainLayout . height
2018-03-28 22:12:00 +03:00
color: "transparent"
2018-03-07 02:01:53 +02:00
Clipboard { id: clipboard }
function validHex32 ( s ) {
if ( s . length != 64 )
return false
for ( var i = 0 ; i < s . length ; ++ i )
if ( "0123456789abcdefABCDEF" . indexOf ( s [ i ] ) == - 1 )
return false
return true
}
2018-09-30 11:12:44 +03:00
function validUnsigned ( s ) {
if ( s . length == 0 )
return false
for ( var i = 0 ; i < s . length ; ++ i )
if ( "0123456789" . indexOf ( s [ i ] ) == - 1 )
return false
return true
}
2018-03-07 02:01:53 +02:00
function validRing ( str , relative ) {
var outs = str . split ( " " ) ;
if ( outs . length == 0 )
return false
for ( var i = 1 ; i < outs . length ; ++ i ) {
if ( relative ) {
if ( outs [ i ] <= 0 )
return false
}
else {
if ( outs [ i ] <= outs [ i - 1 ] )
return false
}
}
return true
}
/* main layout */
ColumnLayout {
id: mainLayout
2019-02-01 23:20:38 +02:00
Layout.fillWidth: true
2019-09-06 01:11:12 +03:00
anchors.margins: 20
2021-03-20 11:12:15 +02:00
anchors.topMargin: 0
2018-03-07 02:01:53 +02:00
anchors.left: parent . left
anchors.top: parent . top
anchors.right: parent . right
2019-04-25 22:09:23 +03:00
spacing: 20
2018-03-07 02:01:53 +02:00
MessageDialog {
id: sharedRingDBDialog
standardButtons: StandardButton . Ok
}
2019-02-01 23:20:38 +02:00
MoneroComponents . Label {
2018-04-22 21:07:31 +03:00
id: signTitleLabel
2019-04-25 22:09:23 +03:00
fontSize: 24
2018-04-22 21:07:31 +03:00
text: qsTr ( "Shared RingDB" ) + translationManager . emptyString
}
2019-04-11 04:17:29 +03:00
MoneroComponents . TextPlain {
2018-04-22 21:07:31 +03:00
text: qsTr ( "This page allows you to interact with the shared ring database. " +
2018-03-07 02:01:53 +02:00
"This database is meant for use by Monero wallets as well as wallets from Monero clones which reuse the Monero keys." ) + translationManager . emptyString
wrapMode: Text . Wrap
2018-04-22 21:07:31 +03:00
Layout.fillWidth: true
2019-02-01 23:20:38 +02:00
font.family: MoneroComponents . Style . fontRegular . name
2019-04-25 22:09:23 +03:00
font.pixelSize: 14
2019-02-01 23:20:38 +02:00
color: MoneroComponents . Style . defaultFontColor
2018-03-07 02:01:53 +02:00
}
2019-02-01 23:20:38 +02:00
MoneroComponents . LabelSubheader {
2018-04-22 21:07:31 +03:00
Layout.fillWidth: true
2018-03-07 02:01:53 +02:00
textFormat: Text . RichText
text: "<style type='text/css'>a {text-decoration: none; color: #FF6C3C; font-size: 14px;}</style>" +
2018-10-18 21:44:13 +03:00
qsTr ( "Outputs marked as spent" ) + " <a href='#'>" + qsTr ( "Help" ) + "</a>" + translationManager . emptyString
2018-03-07 02:01:53 +02:00
onLinkActivated: {
2018-10-18 21:44:13 +03:00
sharedRingDBDialog . title = qsTr ( "Outputs marked as spent" ) + translationManager . emptyString ;
2018-03-07 02:01:53 +02:00
sharedRingDBDialog . text = qsTr (
"In order to obscure which inputs in a Monero transaction are being spent, a third party should not be able " +
"to tell which inputs in a ring are already known to be spent. Being able to do so would weaken the protection " +
"afforded by ring signatures. If all but one of the inputs are known to be already spent, then the input being " +
"actually spent becomes apparent, thereby nullifying the effect of ring signatures, one of the three main layers " +
"of privacy protection Monero uses.<br>" +
"To help transactions avoid those inputs, a list of known spent ones can be used to avoid using them in new " +
"transactions. Such a list is maintained by the Monero project and is available on the getmonero.org website, " +
"and you can import this list here.<br>" +
"Alternatively, you can scan the blockchain (and the blockchain of key-reusing Monero clones) yourself " +
2018-10-18 21:44:13 +03:00
"using the monero-blockchain-mark-spent-outputs tool to create a list of known spent outputs.<br>"
2019-04-16 17:35:30 +03:00
) + translationManager . emptyString
2018-03-07 02:01:53 +02:00
sharedRingDBDialog . icon = StandardIcon . Information
sharedRingDBDialog . open ( )
}
2018-04-22 21:07:31 +03:00
}
2019-04-11 04:17:29 +03:00
MoneroComponents . TextPlain {
2018-04-22 21:07:31 +03:00
textFormat: Text . RichText
2019-02-01 23:20:38 +02:00
font.family: MoneroComponents . Style . fontRegular . name
2019-04-25 22:09:23 +03:00
font.pixelSize: 14
2018-04-22 21:07:31 +03:00
text: qsTr ( "This sets which outputs are known to be spent, and thus not to be used as privacy placeholders in ring signatures. " ) +
qsTr ( "You should only have to load a file when you want to refresh the list. Manual adding/removing is possible if needed." ) + translationManager . emptyString
wrapMode: Text . Wrap
Layout.fillWidth: true ;
2019-02-01 23:20:38 +02:00
color: MoneroComponents . Style . defaultFontColor
2018-03-07 02:01:53 +02:00
}
2018-04-22 21:07:31 +03:00
ColumnLayout {
Layout.fillWidth: true
Layout.topMargin: 12
2018-03-07 02:01:53 +02:00
FileDialog {
id: loadBlackballFileDialog
2018-10-18 21:44:13 +03:00
title: qsTr ( "Please choose a file from which to load outputs to mark as spent" ) + translationManager . emptyString ;
2018-03-07 02:01:53 +02:00
folder: "file://"
2019-02-01 23:20:38 +02:00
nameFilters: [ "*" ]
2018-03-07 02:01:53 +02:00
onAccepted: {
loadBlackballFileLine . text = walletManager . urlToLocalPath ( loadBlackballFileDialog . fileUrl )
}
}
2019-02-01 23:20:38 +02:00
MoneroComponents . LineEdit {
id: loadBlackballFileLine
2018-04-22 21:07:31 +03:00
Layout.fillWidth: true
2019-04-25 22:09:23 +03:00
fontSize: 16
placeholderFontSize: 16
2019-02-01 23:20:38 +02:00
placeholderText: qsTr ( "Path to file" ) + "..." + translationManager . emptyString
2019-04-25 22:09:23 +03:00
labelFontSize: 14
2019-02-01 23:20:38 +02:00
labelText: qsTr ( "Filename with outputs to mark as spent" ) + ":" + translationManager . emptyString
copyButton: true
readOnly: false
2018-03-07 02:01:53 +02:00
}
2018-04-22 21:07:31 +03:00
RowLayout {
2018-03-07 02:01:53 +02:00
Layout.fillWidth: true
2018-04-22 21:07:31 +03:00
Layout.topMargin: 18
2019-02-01 23:20:38 +02:00
MoneroComponents . StandardButton {
2018-04-22 21:07:31 +03:00
id: selectBlackballFileButton
text: qsTr ( "Browse" ) + translationManager . emptyString
enabled: true
small: true
2018-03-07 02:01:53 +02:00
onClicked: {
2018-04-22 21:07:31 +03:00
loadBlackballFileDialog . open ( )
2018-03-07 02:01:53 +02:00
}
}
2019-02-01 23:20:38 +02:00
MoneroComponents . StandardButton {
2018-04-22 21:07:31 +03:00
id: loadBlackballFileButton
text: qsTr ( "Load" ) + translationManager . emptyString
small: true
enabled: ! ! appWindow . currentWallet && loadBlackballFileLine . text !== ""
onClicked: appWindow . currentWallet . blackballOutputs ( walletManager . urlToLocalPath ( loadBlackballFileDialog . fileUrl ) , true )
}
2018-03-07 02:01:53 +02:00
}
}
2019-02-01 23:20:38 +02:00
GridLayout {
Layout.fillWidth: true
2019-04-25 22:09:23 +03:00
columnSpacing: 20
2018-04-22 21:07:31 +03:00
2019-02-01 23:20:38 +02:00
MoneroComponents . LineEdit {
id: blackballOutputAmountLine
Layout.fillWidth: true
2019-04-25 22:09:23 +03:00
fontSize: 16
labelFontSize: 14
2019-02-01 23:20:38 +02:00
labelText: qsTr ( "Or manually mark a single output as spent/unspent:" ) + translationManager . emptyString
2019-04-25 22:09:23 +03:00
placeholderFontSize: 16
2019-02-01 23:20:38 +02:00
placeholderText: qsTr ( "Paste output amount" ) + "..." + translationManager . emptyString
readOnly: false
validator: IntValidator { bottom: 0 }
2018-03-07 02:01:53 +02:00
}
2019-02-01 23:20:38 +02:00
MoneroComponents . LineEdit {
id: blackballOutputOffsetLine
2018-04-22 21:07:31 +03:00
Layout.fillWidth: true
2019-04-25 22:09:23 +03:00
fontSize: 16
labelFontSize: 14
2019-02-01 23:20:38 +02:00
labelText: " "
2019-04-25 22:09:23 +03:00
placeholderFontSize: 16
2019-02-01 23:20:38 +02:00
placeholderText: qsTr ( "Paste output offset" ) + "..." + translationManager . emptyString
readOnly: false
validator: IntValidator { bottom: 0 }
}
}
2018-04-22 21:07:31 +03:00
2019-02-01 23:20:38 +02:00
RowLayout {
Layout.fillWidth: true
Layout.topMargin: 18
MoneroComponents . StandardButton {
id: blackballButton
text: qsTr ( "Mark as spent" ) + translationManager . emptyString
small: true
enabled: ! ! appWindow . currentWallet && validUnsigned ( blackballOutputAmountLine . text ) && validUnsigned ( blackballOutputOffsetLine . text )
onClicked: appWindow . currentWallet . blackballOutput ( blackballOutputAmountLine . text , blackballOutputOffsetLine . text )
}
2018-03-07 02:01:53 +02:00
2019-02-01 23:20:38 +02:00
MoneroComponents . StandardButton {
id: unblackballButton
text: qsTr ( "Mark as unspent" ) + translationManager . emptyString
small: true
enabled: ! ! appWindow . currentWallet && validUnsigned ( blackballOutputAmountLine . text ) && validUnsigned ( blackballOutputOffsetLine . text )
onClicked: appWindow . currentWallet . unblackballOutput ( blackballOutputAmountLine . text , blackballOutputOffsetLine . text )
2018-03-07 02:01:53 +02:00
}
}
2019-02-01 23:20:38 +02:00
MoneroComponents . LabelSubheader {
2018-04-22 21:07:31 +03:00
Layout.fillWidth: true
2019-04-25 22:09:23 +03:00
Layout.topMargin: 24
2018-03-07 02:01:53 +02:00
textFormat: Text . RichText
text: "<style type='text/css'>a {text-decoration: none; color: #FF6C3C; font-size: 14px;}</style>" +
2018-04-25 15:54:04 +03:00
qsTr ( "Rings" ) + " <a href='#'>" + qsTr ( "Help" ) + "</a>" + translationManager . emptyString
2018-03-07 02:01:53 +02:00
onLinkActivated: {
sharedRingDBDialog . title = qsTr ( "Rings" ) + translationManager . emptyString ;
sharedRingDBDialog . text = qsTr (
"In order to avoid nullifying the protection afforded by Monero's ring signatures, an output should not " +
"be spent with different rings on different blockchains. While this is normally not a concern, it can become one " +
2018-05-08 18:58:58 +03:00
"when a key-reusing Monero clone allows you to spend existing outputs. In this case, you need to ensure this " +
2018-03-07 02:01:53 +02:00
"existing outputs uses the same ring on both chains.<br>" +
"This will be done automatically by Monero and any key-reusing software which is not trying to actively strip " +
"you of your privacy.<br>" +
"If you are using a key-reusing Monero clone too, and this clone does not include this protection, you can still " +
"ensure your transactions are protected by spending on the clone first, then manually adding the ring on this page, " +
"which allows you to then spend your Monero safely.<br>" +
"If you do not use a key-reusing Monero clone without these safety features, then you do not need to do anything " +
"as it is all automated.<br>"
2019-04-16 17:35:30 +03:00
) + translationManager . emptyString
2018-03-07 02:01:53 +02:00
sharedRingDBDialog . icon = StandardIcon . Information
sharedRingDBDialog . open ( )
}
2018-04-22 21:07:31 +03:00
}
2019-04-11 04:17:29 +03:00
MoneroComponents . TextPlain {
2018-04-22 21:07:31 +03:00
textFormat: Text . RichText
2019-02-01 23:20:38 +02:00
font.family: MoneroComponents . Style . fontRegular . name
2019-04-25 22:09:23 +03:00
font.pixelSize: 14
2018-04-22 21:07:31 +03:00
text: qsTr ( "This records rings used by outputs spent on Monero on a key reusing chain, so that the same ring may be reused to avoid privacy issues." ) + translationManager . emptyString
wrapMode: Text . Wrap
Layout.fillWidth: true ;
2019-02-01 23:20:38 +02:00
color: MoneroComponents . Style . defaultFontColor
2018-03-07 02:01:53 +02:00
}
2019-02-01 23:20:38 +02:00
MoneroComponents . LineEdit {
id: keyImageLine
Layout.fillWidth: true
2019-04-25 22:09:23 +03:00
fontSize: 16
labelFontSize: 14
2019-02-01 23:20:38 +02:00
labelText: qsTr ( "Key image" ) + ":" + translationManager . emptyString
2019-04-25 22:09:23 +03:00
placeholderFontSize: 16
2019-02-01 23:20:38 +02:00
placeholderText: qsTr ( "Paste key image" ) + "..." + translationManager . emptyString
readOnly: false
copyButton: true
2018-04-22 21:07:31 +03:00
}
2018-03-07 02:01:53 +02:00
2018-04-22 21:07:31 +03:00
GridLayout {
2019-04-25 22:09:23 +03:00
Layout.topMargin: 12
2019-09-06 01:11:12 +03:00
columns: 2
2019-04-25 22:09:23 +03:00
columnSpacing: 32
2018-04-22 21:07:31 +03:00
ColumnLayout {
RowLayout {
2019-02-01 23:20:38 +02:00
MoneroComponents . LineEdit {
2018-04-22 21:07:31 +03:00
id: getRingLine
Layout.fillWidth: true
2019-04-25 22:09:23 +03:00
fontSize: 16
labelFontSize: 14
2018-04-22 21:07:31 +03:00
labelText: qsTr ( "Get ring" ) + ":" + translationManager . emptyString
readOnly: true
copyButton: true
}
}
RowLayout {
Layout.fillWidth: true
Layout.topMargin: 18
2019-02-01 23:20:38 +02:00
MoneroComponents . StandardButton {
2018-04-22 21:07:31 +03:00
id: getRingButton
text: qsTr ( "Get Ring" ) + translationManager . emptyString
small: true
enabled: ! ! appWindow . currentWallet && validHex32 ( keyImageLine . text )
onClicked: {
var ring = appWindow . currentWallet . getRing ( keyImageLine . text )
if ( ring === "" ) {
2019-04-16 17:35:30 +03:00
getRingLine . text = qsTr ( "No ring found" ) + translationManager . emptyString ;
2018-04-22 21:07:31 +03:00
}
else {
getRingLine . text = ring ;
}
2018-03-07 02:01:53 +02:00
}
}
}
}
2018-04-22 21:07:31 +03:00
ColumnLayout {
RowLayout {
2019-02-01 23:20:38 +02:00
MoneroComponents . LineEdit {
2018-04-22 21:07:31 +03:00
id: setRingLine
Layout.fillWidth: true
2019-04-25 22:09:23 +03:00
fontSize: 16
labelFontSize: 14
placeholderFontSize: 16
2018-04-22 21:07:31 +03:00
labelText: qsTr ( "Set ring" ) + ":" + translationManager . emptyString
readOnly: false
copyButton: true
2018-03-07 02:01:53 +02:00
}
}
2018-04-22 21:07:31 +03:00
RowLayout {
Layout.fillWidth: true
Layout.topMargin: 18
2019-02-01 23:20:38 +02:00
MoneroComponents . StandardButton {
2018-04-22 21:07:31 +03:00
id: setRingButton
text: qsTr ( "Set Ring" ) + translationManager . emptyString
small: true
enabled: ! ! appWindow . currentWallet && validHex32 ( keyImageLine . text ) && validRing ( setRingLine . text . trim ( ) , setRingRelative . checked )
onClicked: {
var outs = setRingLine . text . trim ( )
appWindow . currentWallet . setRing ( keyImageLine . text , outs , setRingRelative . checked )
2018-03-07 02:01:53 +02:00
}
}
}
}
}
2018-04-22 21:07:31 +03:00
GridLayout {
2019-04-25 22:09:23 +03:00
columnSpacing: 20
2019-09-06 01:11:12 +03:00
columns: 2
2018-04-22 21:07:31 +03:00
2019-02-01 23:20:38 +02:00
MoneroComponents . CheckBox {
2018-04-22 21:07:31 +03:00
id: segregatePreForkOutputs
checked: persistentSettings . segregatePreForkOutputs
text: qsTr ( "I intend to spend on key-reusing fork(s)" ) + translationManager . emptyString
onClicked: {
persistentSettings . segregatePreForkOutputs = segregatePreForkOutputs . checked
if ( appWindow . currentWallet ) {
appWindow . currentWallet . segregatePreForkOutputs ( segregatePreForkOutputs . checked )
2018-03-07 02:01:53 +02:00
}
}
}
2018-04-22 21:07:31 +03:00
2019-02-01 23:20:38 +02:00
MoneroComponents . CheckBox {
2018-04-22 21:07:31 +03:00
id: keyReuseMitigation2
checked: persistentSettings . keyReuseMitigation2
text: qsTr ( "I might want to spend on key-reusing fork(s)" ) + translationManager . emptyString
2018-03-07 02:01:53 +02:00
onClicked: {
2018-04-22 21:07:31 +03:00
persistentSettings . keyReuseMitigation2 = keyReuseMitigation2 . checked
if ( appWindow . currentWallet ) {
appWindow . currentWallet . keyReuseMitigation2 ( keyReuseMitigation2 . checked )
}
2018-03-07 02:01:53 +02:00
}
}
2018-03-16 13:55:56 +02:00
2019-02-01 23:20:38 +02:00
MoneroComponents . CheckBox {
2018-04-22 21:07:31 +03:00
id: setRingRelative
checked: true
text: qsTr ( "Relative" ) + translationManager . emptyString
2018-03-16 13:55:56 +02:00
}
}
2018-04-22 21:07:31 +03:00
2019-04-11 04:17:29 +03:00
GridLayout {
2018-03-16 13:55:56 +02:00
id: segregationHeightRow
2019-01-07 17:59:14 +02:00
Layout.fillWidth: true
2019-04-25 22:09:23 +03:00
Layout.topMargin: 17
2019-09-06 01:11:12 +03:00
columns: 2
2019-04-25 22:09:23 +03:00
columnSpacing: 32
2018-03-16 13:55:56 +02:00
2019-02-01 23:20:38 +02:00
MoneroComponents . LineEdit {
2018-03-16 13:55:56 +02:00
id: segregationHeightLine
2019-04-11 04:17:29 +03:00
property bool edited: false
2019-02-01 23:20:38 +02:00
Layout.fillWidth: true
2019-04-25 22:09:23 +03:00
placeholderFontSize: 16
labelFontSize: 14
2019-04-11 04:17:29 +03:00
labelText: qsTr ( "Set segregation height:" ) + translationManager . emptyString
2018-03-16 13:55:56 +02:00
validator: IntValidator { bottom: 0 }
2019-02-01 23:20:38 +02:00
readOnly: false
2018-03-16 13:55:56 +02:00
onEditingFinished: {
persistentSettings . segregationHeight = segregationHeightLine . text
if ( appWindow . currentWallet ) {
appWindow . currentWallet . segregationHeight ( segregationHeightLine . text )
}
2019-04-11 04:17:29 +03:00
// @TODO: LineEdit should visually be able show that an action
// has been completed due to modification of the text
2018-03-16 13:55:56 +02:00
}
}
2019-04-11 04:17:29 +03:00
Item {
Layout.fillWidth: true
}
2018-03-16 13:55:56 +02:00
}
2018-03-07 02:01:53 +02:00
}
function onPageCompleted ( ) {
console . log ( "RingDB page loaded" ) ;
2018-03-16 13:55:56 +02:00
appWindow . currentWallet . segregatePreForkOutputs ( persistentSettings . segregatePreForkOutputs )
appWindow . currentWallet . segregationHeight ( persistentSettings . segregationHeight )
segregationHeightLine . text = persistentSettings . segregationHeight
appWindow . currentWallet . keyReuseMitigation2 ( persistentSettings . keyReuseMitigation2 )
2018-03-07 02:01:53 +02:00
}
}