2018-12-27 22:38:14 +03:00
import QtQuick 2.5
2018-10-16 18:50:58 +03:00
import Sailfish . Silica 1.0
2019-03-11 23:53:04 +03:00
import "../js/showdown/dist/showdown.js" as ShowDown
2018-10-16 18:50:58 +03:00
2018-11-18 13:25:28 +03:00
Dialog {
id: noteDialog
2018-11-30 20:40:03 +03:00
2019-05-14 00:32:37 +03:00
property int id
property int modified
property string title
property string category
property string content
property bool favorite
property string etag
property bool error
property string errorMessage
2018-11-30 20:40:03 +03:00
2018-11-27 11:58:21 +03:00
property var showdown: ShowDown . showdown
2018-11-30 17:12:16 +03:00
property var converter: new showdown . Converter (
2018-12-04 17:22:57 +03:00
{ simplifiedAutoLink: true ,
excludeTrailingPunctuationFromURLs: true ,
2018-12-23 03:59:01 +03:00
parseImgDimensions: true ,
2018-12-04 17:22:57 +03:00
strikethrough: true ,
2018-11-30 17:12:16 +03:00
tables: true ,
2018-12-03 22:33:37 +03:00
tasklists: false , // this is handled by the function parseContent() because LinkedLabel HTML support is to basic
2018-12-23 03:59:01 +03:00
smoothLivePreview: true ,
2018-11-30 17:12:16 +03:00
simpleLineBreaks: true ,
emoji: true } )
2018-11-18 13:25:28 +03:00
2018-12-20 20:53:29 +03:00
2018-12-04 01:08:41 +03:00
acceptDestination: Qt . resolvedUrl ( "EditPage.qml" )
2019-06-21 18:49:54 +03:00
acceptDestinationProperties: (
{ id: id ,
modified: modified ,
title: title ,
category: category ,
content: content ,
favorite: favorite ,
etag: etag ,
error: error ,
2019-11-15 01:15:33 +03:00
errorMessage: errorMessage
} )
2018-12-20 20:53:29 +03:00
onAccepted: {
2019-11-12 01:23:41 +03:00
//acceptDestinationInstance.note = note
2018-12-20 20:53:29 +03:00
acceptDestinationInstance . reloadContent ( )
}
onStatusChanged: {
if ( status === DialogStatus . Opened ) {
2019-11-12 01:23:41 +03:00
//notesApi.getNoteFromApi(id)
2018-12-20 20:53:29 +03:00
}
}
2018-12-04 01:08:41 +03:00
Component.onCompleted: {
2019-11-12 01:23:41 +03:00
console . log ( title )
2018-12-20 20:53:29 +03:00
parseContent ( )
2018-12-04 01:08:41 +03:00
}
2018-12-07 02:30:18 +03:00
function reloadContent ( ) {
2019-11-12 01:23:41 +03:00
//notesApi.getNoteFromApi(id)
/ * n o t e = n o t e s A p i . g e t N o t e ( i d )
2019-05-14 00:32:37 +03:00
dialogHeader . title = title
favoriteButton . selected = favorite
categoryField . text = category
modifiedDetail . modified = modified
2019-02-24 18:40:47 +03:00
parseContent ( ) * /
2018-12-07 02:30:18 +03:00
}
2018-12-03 22:33:37 +03:00
function parseContent ( ) {
2019-11-12 01:23:41 +03:00
//note = notesApi.getNoteFromApi(id, false)
2019-05-14 00:32:37 +03:00
var convertedText = converter . makeHtml ( content )
2018-11-28 16:05:36 +03:00
var occurence = - 1
2018-12-05 18:38:07 +03:00
convertedText = convertedText . replace ( /^<li>(<p>)?\[ \] (.*)(<.*)$/gmi ,
function ( match , p1 , p2 , p3 , offset ) {
2018-11-28 16:05:36 +03:00
occurence ++
2018-12-11 15:34:17 +03:00
return '<li class="tasklist"><a class="checkbox" href="tasklist:checkbox_' + occurence + '">' + ( p1 ? p1 : "" ) + '☐ ' + p2 + '</a>' + p3
2018-11-28 16:05:36 +03:00
} )
occurence = - 1
2018-12-05 18:38:07 +03:00
convertedText = convertedText . replace ( /^<li>(<p>)?\[[xX]\] (.*)(<.*)$/gmi ,
function ( match , p1 , p2 , p3 , offset ) {
2018-11-28 16:05:36 +03:00
occurence ++
2018-12-11 15:34:17 +03:00
return '<li class="tasklist"><a class="checkbox" href="tasklist:uncheckbox_' + occurence + '">' + ( p1 ? p1 : "" ) + '☑ ' + p2 + '</a>' + p3
2018-11-28 16:05:36 +03:00
} )
2018-12-23 03:59:01 +03:00
convertedText = convertedText . replace ( /<table>/gmi , "<table border='1' cellpadding='" + Theme . paddingMedium + "'>" )
2018-12-26 23:51:20 +03:00
convertedText = convertedText . replace ( /<del>(.*)<\/del>/gmi , function ( match , p1 ) { return "<s>" + p1 + "</s>" } )
convertedText = convertedText . replace ( /<hr \/>/gmi , "<p><img width=\"" + contentLabel . width + "\" height=\"1\" src=\"\" /></p>" )
2018-12-11 15:34:17 +03:00
contentLabel . text = "<style>\n" +
2018-12-27 22:38:14 +03:00
"p,ul,ol,table,img { margin: " + Theme . paddingLarge + "px 0px; }\n" +
2018-12-04 17:22:57 +03:00
"a:link { color: " + Theme . primaryColor + "; }\n" +
2018-12-11 15:34:17 +03:00
"a.checkbox { text-decoration: none; padding: " + Theme . paddingSmall + "px; display: inline-block; }\n" +
"li.tasklist { font-size:large; margin: " + Theme . paddingMedium + "px 0px; }\n" +
"table { border-color: " + Theme . secondaryColor + "; }\n" +
"</style>\n" + convertedText
2018-12-27 22:38:14 +03:00
//if (debug) console.log(contentLabel.text)
2018-11-24 21:34:01 +03:00
}
2018-10-16 18:50:58 +03:00
SilicaFlickable {
anchors.fill: parent
2018-12-13 01:10:45 +03:00
contentHeight: mainColumn . height + Theme . paddingLarge
2018-10-16 18:50:58 +03:00
Column {
2018-11-26 19:21:33 +03:00
id: mainColumn
2018-10-21 02:44:23 +03:00
width: parent . width
2018-11-26 19:21:33 +03:00
2018-11-24 21:34:01 +03:00
RemorsePopup {
id: remorse
onTriggered: pageStack . pop ( )
}
PullDownMenu {
2019-11-12 01:23:41 +03:00
busy: notesApi . busy
2018-11-24 21:34:01 +03:00
MenuItem {
text: qsTr ( "Delete" )
2019-11-12 01:23:41 +03:00
onClicked: remorse . execute ( "Deleting" , function ( ) { notesApi . deleteNote ( id ) } )
2018-11-24 21:34:01 +03:00
}
MenuItem {
text: enabled ? qsTr ( "Reload" ) : qsTr ( "Updating..." )
2019-11-12 01:23:41 +03:00
enabled: ! notesApi . busy
2020-01-11 15:32:26 +03:00
onClicked: notesApi . getNote ( id )
2018-11-24 21:34:01 +03:00
}
MenuLabel {
2018-12-07 02:30:18 +03:00
visible: appSettings . currentAccount . length >= 0
text: qsTr ( "Last update" ) + ": " + (
2020-01-04 16:53:50 +03:00
new Date ( account . update ) . valueOf ( ) !== 0 ?
new Date ( account . update ) . toLocaleString ( Qt . locale ( ) , Locale . ShortFormat ) :
2018-12-07 02:30:18 +03:00
qsTr ( "never" ) )
2018-11-24 21:34:01 +03:00
}
}
2018-10-16 18:50:58 +03:00
2018-11-18 13:25:28 +03:00
DialogHeader {
2018-11-18 17:01:32 +03:00
id: dialogHeader
2020-01-11 15:32:26 +03:00
//title: noteDialog.title
2018-11-18 13:25:28 +03:00
acceptText: qsTr ( "Edit" )
cancelText: qsTr ( "Notes" )
2018-12-23 03:59:01 +03:00
}
BusyIndicator {
parent: dialogHeader . extraContent
anchors.verticalCenter: parent . verticalCenter
anchors.horizontalCenter: parent . horizontalCenter
size: BusyIndicatorSize . Medium
2019-11-12 01:23:41 +03:00
running: notesApi . busy
2018-10-16 18:50:58 +03:00
}
2018-11-26 19:21:33 +03:00
Column {
width: parent . width
spacing: Theme . paddingLarge
2020-01-11 15:32:26 +03:00
/ * S e p a r a t o r {
2018-12-07 02:30:18 +03:00
width: parent . width
color: Theme . primaryColor
horizontalAlignment: Qt . AlignHCenter
2020-01-11 15:32:26 +03:00
} * /
2018-12-07 02:30:18 +03:00
2018-11-26 19:21:33 +03:00
LinkedLabel {
id: contentLabel
x: Theme . horizontalPageMargin
width: parent . width - 2 * x
2018-12-04 17:22:57 +03:00
textFormat: Text . RichText
linkColor: Theme . primaryColor
2018-11-28 16:05:36 +03:00
defaultLinkActions: false
onLinkActivated: {
2018-12-05 19:09:02 +03:00
//console.log(link)
2018-11-28 16:05:36 +03:00
var occurence = - 1
2019-05-14 00:32:37 +03:00
var newContent = content
2018-11-28 16:05:36 +03:00
if ( /^tasklist:checkbox_(\d+)$/m . test ( link ) ) {
2018-12-05 18:38:07 +03:00
newContent = newContent . replace ( /- \[ \] (.*)$/gm ,
2018-11-28 16:05:36 +03:00
function ( match , p1 , offset , string ) {
occurence ++
if ( occurence === parseInt ( link . split ( '_' ) [ 1 ] ) ) {
2018-12-04 13:44:06 +03:00
return ( appSettings . useCapitalX ? '- [X] ' : '- [x] ' ) + p1 }
2018-12-03 22:33:37 +03:00
else { return match }
2018-11-28 16:05:36 +03:00
} )
2019-05-14 00:32:37 +03:00
content = newContent
2018-12-03 22:33:37 +03:00
parseContent ( )
2019-11-12 01:23:41 +03:00
notesApi . updateNote ( id , { 'content' : content } )
2018-11-28 16:05:36 +03:00
}
else if ( /^tasklist:uncheckbox_(\d+)$/m . test ( link ) ) {
2018-12-05 18:38:07 +03:00
newContent = newContent . replace ( /- \[[xX]\] (.*)$/gm ,
2018-11-28 16:05:36 +03:00
function ( match , p1 , offset , string ) {
occurence ++
if ( occurence === parseInt ( link . split ( '_' ) [ 1 ] ) ) {
2018-12-03 22:33:37 +03:00
return '- [ ] ' + p1 }
else { return match }
2018-11-28 16:05:36 +03:00
} )
2019-05-14 00:32:37 +03:00
content = newContent
2018-12-03 22:33:37 +03:00
parseContent ( )
2019-11-12 01:23:41 +03:00
notesApi . updateNote ( id , { 'content' : content } )
2018-11-28 16:05:36 +03:00
}
else {
Qt . openUrlExternally ( link )
}
}
2018-11-26 19:21:33 +03:00
}
Separator {
width: parent . width
color: Theme . primaryColor
horizontalAlignment: Qt . AlignHCenter
}
2018-12-07 02:30:18 +03:00
Flow {
x: Theme . horizontalPageMargin
width: parent . width - 2 * x
spacing: Theme . paddingMedium
visible: opacity > 0.0
opacity: categoryField . focus ? 1.0 : 0.0
Behavior on opacity { FadeAnimator { } }
Repeater {
id: categoryRepeater
2019-11-12 01:23:41 +03:00
model: notesApi . categories
2018-12-07 02:30:18 +03:00
BackgroundItem {
id: categoryBackground
width: categoryRectangle . width
height: categoryRectangle . height
Rectangle {
id: categoryRectangle
width: categoryLabel . width + Theme . paddingLarge
height: categoryLabel . height + Theme . paddingSmall
color: "transparent"
border.color: Theme . highlightColor
radius: height / 4
Label {
id: categoryLabel
anchors.centerIn: parent
text: modelData
color: categoryBackground . highlighted ? Theme.highlightColor : Theme . primaryColor
font.pixelSize: Theme . fontSizeSmall
2018-12-06 15:37:29 +03:00
}
}
2018-12-07 02:30:18 +03:00
onClicked: categoryField . text = modelData
2018-12-06 15:37:29 +03:00
}
}
2018-12-07 02:30:18 +03:00
}
}
Row {
x: Theme . horizontalPageMargin
width: parent . width - x
IconButton {
id: favoriteButton
2019-05-14 00:32:37 +03:00
property bool selected: favorite
2018-12-07 02:30:18 +03:00
width: Theme . iconSizeMedium
icon.source: ( selected ? "image://theme/icon-m-favorite-selected?" : "image://theme/icon-m-favorite?" ) +
( favoriteButton . highlighted ? Theme.secondaryHighlightColor : Theme . secondaryColor )
onClicked: {
2020-03-29 19:03:05 +03:00
notesApi . updateNote ( id , { 'favorite' : ! favorite , 'modified' : new Date ( ) . valueOf ( ) / 1000 } )
2018-12-03 22:33:37 +03:00
}
2018-12-07 02:30:18 +03:00
}
TextField {
id: categoryField
width: parent . width - favoriteButton . width
2019-05-14 00:32:37 +03:00
text: category
2018-12-07 02:30:18 +03:00
placeholderText: qsTr ( "No category" )
label: qsTr ( "Category" )
EnterKey.iconSource: "image://theme/icon-m-enter-accept"
EnterKey.onClicked: {
categoryField . focus = false
}
onFocusChanged: {
2019-05-14 00:32:37 +03:00
if ( focus === false && text !== category ) {
2020-03-29 19:03:05 +03:00
notesApi . updateNote ( id , { 'content' : content , 'category' : text , 'modified' : new Date ( ) . valueOf ( ) / 1000 } ) // This does not seem to work without adding the content
2018-12-07 02:30:18 +03:00
}
2018-12-04 17:22:57 +03:00
}
}
2018-10-16 18:50:58 +03:00
}
2018-12-07 02:30:18 +03:00
DetailItem {
id: modifiedDetail
label: qsTr ( "Modified" )
2019-11-15 01:15:33 +03:00
value: new Date ( noteDialog . modified * 1000 ) . toLocaleString ( Qt . locale ( ) , Locale . ShortFormat )
2018-12-07 02:30:18 +03:00
}
2018-10-16 18:50:58 +03:00
}
2018-10-21 02:44:23 +03:00
2018-10-16 18:50:58 +03:00
VerticalScrollDecorator { }
}
2018-12-04 19:24:45 +03:00
allowedOrientations: defaultAllowedOrientations
2018-10-16 18:50:58 +03:00
}