2014-01-05 20:58:25 +00:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2013, Stefan Brand <seiichiro@seiichiro0185.org>
|
|
|
|
* 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. The names of the contributors may not 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
import QtQuick 2.0
|
|
|
|
import Sailfish.Silica 1.0
|
|
|
|
import "../lib/storage.js" as DB
|
|
|
|
import "../lib/crypto.js" as OTP
|
|
|
|
|
|
|
|
Page {
|
2014-01-06 19:08:16 +00:00
|
|
|
id: mainPage
|
|
|
|
|
|
|
|
ListModel {
|
|
|
|
id: otpListModel
|
|
|
|
}
|
|
|
|
|
|
|
|
// This holds the time of the last update of the page as Unix Timestamp (in Milliseconds)
|
2014-01-10 18:20:52 +00:00
|
|
|
property double lastUpdated: 0
|
2014-01-06 19:08:16 +00:00
|
|
|
|
|
|
|
// Add an entry to the list
|
2014-01-09 21:12:55 +00:00
|
|
|
function appendOTP(title, secret, type, counter, fav) {
|
2014-01-12 17:03:23 +00:00
|
|
|
otpListModel.append({"secret": secret, "title": title, "fav": fav, "type": type, "counter": counter, "otp": "------"});
|
2014-01-06 19:08:16 +00:00
|
|
|
}
|
|
|
|
|
2014-01-10 18:20:52 +00:00
|
|
|
// Hand favorite over to the cover
|
2014-01-12 17:03:23 +00:00
|
|
|
function setCoverOTP(title, secret, type) {
|
2014-01-10 18:20:52 +00:00
|
|
|
appWin.coverTitle = title
|
|
|
|
appWin.coverSecret = secret
|
2014-01-12 17:03:23 +00:00
|
|
|
appWin.coverType = type
|
|
|
|
if (secret = "") {
|
|
|
|
appWin.coverOTP = "";
|
|
|
|
} else if (type == "HOTP") {
|
|
|
|
appWin.coverOTP = "------";
|
|
|
|
}
|
2014-01-10 18:20:52 +00:00
|
|
|
}
|
|
|
|
|
2014-01-06 19:08:16 +00:00
|
|
|
// Reload the List of OTPs from storage
|
|
|
|
function refreshOTPList() {
|
2014-01-09 21:12:55 +00:00
|
|
|
otpList.visible = false;
|
2014-01-06 19:08:16 +00:00
|
|
|
otpListModel.clear();
|
|
|
|
DB.getOTP();
|
|
|
|
refreshOTPValues();
|
2014-01-09 21:12:55 +00:00
|
|
|
otpList.visible = true;
|
2014-01-06 19:08:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Calculate new OTPs for every entry
|
|
|
|
function refreshOTPValues() {
|
|
|
|
// get seconds from current Date
|
|
|
|
var curDate = new Date();
|
|
|
|
var seconds = curDate.getSeconds();
|
|
|
|
|
|
|
|
// Iterate over all List entries
|
|
|
|
for (var i=0; i<otpListModel.count; i++) {
|
2014-01-12 17:03:23 +00:00
|
|
|
if (otpListModel.get(i).type == "TOTP") {
|
|
|
|
// Only update on full 30 / 60 Seconds or if last run of the Functions is more than 2s in the past (e.g. app was in background)
|
|
|
|
if (otpListModel.get(i).otp == "------" || seconds == 30 || seconds == 0 || (curDate.getTime() - lastUpdated > 2000)) {
|
|
|
|
var curOTP = OTP.calcOTP(otpListModel.get(i).secret, "TOTP")
|
|
|
|
otpListModel.setProperty(i, "otp", curOTP);
|
|
|
|
}
|
|
|
|
} else if (appWin.coverType == "HOTP" && (curDate.getTime() - lastUpdated > 2000) && otpListModel.get(i).fav == 1) {
|
|
|
|
// If we are coming back from the CoverPage update OTP value if current favourite is HOTP
|
|
|
|
otpListModel.setProperty(i, "otp", appWin.coverOTP);
|
2014-01-06 19:08:16 +00:00
|
|
|
}
|
2014-01-05 20:58:25 +00:00
|
|
|
}
|
|
|
|
|
2014-01-06 19:08:16 +00:00
|
|
|
// Update the Progressbar
|
|
|
|
updateProgress.value = 29 - (seconds % 30)
|
|
|
|
// Set lastUpdate property
|
|
|
|
lastUpdated = curDate.getTime();
|
|
|
|
}
|
|
|
|
|
|
|
|
Timer {
|
2014-01-12 17:03:23 +00:00
|
|
|
interval: 500
|
2014-01-09 21:12:55 +00:00
|
|
|
// Timer only runs when app is acitive and we have entries
|
|
|
|
running: Qt.application.active && otpListModel.count
|
2014-01-06 19:08:16 +00:00
|
|
|
repeat: true
|
|
|
|
onTriggered: refreshOTPValues();
|
|
|
|
}
|
|
|
|
|
|
|
|
SilicaFlickable {
|
|
|
|
anchors.fill: parent
|
|
|
|
|
|
|
|
PullDownMenu {
|
|
|
|
MenuItem {
|
|
|
|
text: "About"
|
|
|
|
onClicked: pageStack.push(Qt.resolvedUrl("About.qml"))
|
|
|
|
}
|
|
|
|
MenuItem {
|
|
|
|
text: "Add OTP"
|
|
|
|
onClicked: pageStack.push(Qt.resolvedUrl("AddOTP.qml"), {parentPage: mainPage})
|
|
|
|
}
|
2014-01-05 20:58:25 +00:00
|
|
|
}
|
|
|
|
|
2014-01-06 19:08:16 +00:00
|
|
|
ProgressBar {
|
|
|
|
id: updateProgress
|
|
|
|
width: parent.width
|
|
|
|
maximumValue: 29
|
|
|
|
anchors.top: parent.top
|
|
|
|
anchors.topMargin: 48
|
2014-01-09 21:12:55 +00:00
|
|
|
// Only show when there are enries
|
|
|
|
visible: otpListModel.count
|
2014-01-05 20:58:25 +00:00
|
|
|
}
|
|
|
|
|
2014-01-06 19:08:16 +00:00
|
|
|
SilicaListView {
|
|
|
|
id: otpList
|
|
|
|
header: PageHeader {
|
|
|
|
title: "SailOTP"
|
|
|
|
}
|
|
|
|
anchors.fill: parent
|
|
|
|
model: otpListModel
|
|
|
|
width: parent.width
|
2014-01-05 20:58:25 +00:00
|
|
|
|
2014-01-06 19:08:16 +00:00
|
|
|
ViewPlaceholder {
|
|
|
|
enabled: otpList.count == 0
|
|
|
|
text: "Nothing here"
|
|
|
|
hintText: "Pull down to add a OTP"
|
|
|
|
}
|
2014-01-05 20:58:25 +00:00
|
|
|
|
2014-01-06 19:08:16 +00:00
|
|
|
delegate: ListItem {
|
|
|
|
id: otpListItem
|
|
|
|
menu: otpContextMenu
|
|
|
|
contentHeight: Theme.itemSizeMedium
|
2014-01-09 21:12:55 +00:00
|
|
|
width: parent.width
|
2014-01-05 20:58:25 +00:00
|
|
|
|
2014-01-10 18:26:46 +00:00
|
|
|
function remove() {
|
|
|
|
// Show 5s countdown, then delete from DB and List
|
2014-01-06 19:08:16 +00:00
|
|
|
remorseAction("Deleting", function() { DB.removeOTP(title, secret); otpListModel.remove(index) })
|
2014-01-05 20:58:25 +00:00
|
|
|
}
|
|
|
|
|
2014-01-10 18:26:46 +00:00
|
|
|
onClicked: {
|
|
|
|
Clipboard.text = otp
|
|
|
|
}
|
|
|
|
|
2014-01-06 19:08:16 +00:00
|
|
|
ListView.onRemove: animateRemoval()
|
|
|
|
Rectangle {
|
2014-01-09 21:12:55 +00:00
|
|
|
id: listRow
|
|
|
|
width: parent.width
|
2014-01-06 19:08:16 +00:00
|
|
|
anchors.horizontalCenter: parent.horizontalCenter
|
|
|
|
|
2014-01-10 05:38:38 +00:00
|
|
|
IconButton {
|
|
|
|
icon.source: fav == 1 ? "image://theme/icon-m-favorite-selected" : "image://theme/icon-m-favorite"
|
2014-01-09 21:12:55 +00:00
|
|
|
anchors.left: parent.left
|
|
|
|
onClicked: {
|
2014-01-10 18:20:52 +00:00
|
|
|
if (fav == 0) {
|
|
|
|
DB.setFav(title, secret)
|
2014-01-12 17:03:23 +00:00
|
|
|
setCoverOTP(title, secret, type)
|
|
|
|
if (type == "HOTP") appWin.coverOTP = otp
|
2014-01-10 18:20:52 +00:00
|
|
|
for (var i=0; i<otpListModel.count; i++) {
|
|
|
|
if (i != index) {
|
|
|
|
otpListModel.setProperty(i, "fav", 0);
|
|
|
|
} else {
|
|
|
|
otpListModel.setProperty(i, "fav", 1);
|
|
|
|
}
|
2014-01-09 21:12:55 +00:00
|
|
|
}
|
2014-01-10 18:20:52 +00:00
|
|
|
} else {
|
|
|
|
DB.resetFav(title, secret)
|
2014-01-12 17:03:23 +00:00
|
|
|
setCoverOTP("SailOTP", "", "")
|
2014-01-10 18:20:52 +00:00
|
|
|
otpListModel.setProperty(index, "fav", 0);
|
2014-01-10 05:38:38 +00:00
|
|
|
}
|
2014-01-09 21:12:55 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Column {
|
|
|
|
anchors.horizontalCenter: parent.horizontalCenter
|
|
|
|
|
|
|
|
Label {
|
2014-01-06 19:08:16 +00:00
|
|
|
id: otpLabel
|
|
|
|
text: model.title
|
|
|
|
color: Theme.secondaryColor
|
|
|
|
anchors.horizontalCenter: parent.horizontalCenter
|
2014-01-09 21:12:55 +00:00
|
|
|
}
|
2014-01-06 19:08:16 +00:00
|
|
|
|
2014-01-09 21:12:55 +00:00
|
|
|
Label {
|
2014-01-06 19:08:16 +00:00
|
|
|
id: otpValue
|
|
|
|
text: model.otp
|
|
|
|
color: Theme.highlightColor
|
|
|
|
font.pixelSize: Theme.fontSizeLarge
|
2014-01-09 21:12:55 +00:00
|
|
|
anchors.horizontalCenter: parent.horizontalCenter
|
|
|
|
}
|
2014-01-06 19:08:16 +00:00
|
|
|
}
|
2014-01-05 20:58:25 +00:00
|
|
|
}
|
|
|
|
|
2014-01-12 17:03:23 +00:00
|
|
|
// Show an update button on HTOP-Type Tokens
|
|
|
|
IconButton {
|
|
|
|
icon.source: "image://theme/icon-m-refresh"
|
|
|
|
anchors.right: parent.right
|
|
|
|
visible: type == "HOTP" ? true : false
|
|
|
|
onClicked: {
|
|
|
|
otpListModel.setProperty(index, "counter", DB.getCounter(title, secret, true));
|
|
|
|
otpListModel.setProperty(index, "otp", OTP.calcOTP(secret, "HOTP", counter));
|
|
|
|
if (fav == 1) appWin.coverOTP = otp;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-01-06 19:08:16 +00:00
|
|
|
Component {
|
|
|
|
id: otpContextMenu
|
|
|
|
ContextMenu {
|
2014-01-08 18:13:53 +00:00
|
|
|
MenuItem {
|
|
|
|
text: "Edit"
|
|
|
|
onClicked: {
|
2014-01-12 17:03:23 +00:00
|
|
|
pageStack.push(Qt.resolvedUrl("AddOTP.qml"), {parentPage: mainPage, paramLabel: title, paramKey: secret, paramType: type, paramCounter: DB.getCounter(title, secret, false)})
|
2014-01-08 18:13:53 +00:00
|
|
|
}
|
|
|
|
}
|
2014-01-06 19:08:16 +00:00
|
|
|
MenuItem {
|
|
|
|
text: "Delete"
|
|
|
|
onClicked: remove()
|
2014-01-05 20:58:25 +00:00
|
|
|
}
|
2014-01-06 19:08:16 +00:00
|
|
|
}
|
2014-01-05 20:58:25 +00:00
|
|
|
}
|
2014-01-06 19:08:16 +00:00
|
|
|
}
|
|
|
|
VerticalScrollDecorator{}
|
|
|
|
|
|
|
|
Component.onCompleted: {
|
|
|
|
// Load list of OTP-Entries
|
|
|
|
refreshOTPList();
|
|
|
|
}
|
2014-01-05 20:58:25 +00:00
|
|
|
}
|
2014-01-06 19:08:16 +00:00
|
|
|
}
|
2014-01-05 20:58:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|