mirror of https://github.com/espruino/BangleApps
Add UI for the TodoList app
parent
4d1cf766ef
commit
a20ea2df71
|
@ -2,39 +2,63 @@ Todo List
|
||||||
========
|
========
|
||||||
|
|
||||||
This is a simple Todo List application.
|
This is a simple Todo List application.
|
||||||
|
The content is loaded from a JSON file.
|
||||||
|
A task can be marked as completed or uncompleted.
|
||||||
|
|
||||||
data:image/s3,"s3://crabby-images/1f620/1f62083b013d4a0f2cc609ed109032b3a70b86ba" alt=""
|
data:image/s3,"s3://crabby-images/1f620/1f62083b013d4a0f2cc609ed109032b3a70b86ba" alt=""
|
||||||
|
|
||||||
The content is loaded from a JSON file.
|
Once installed, the list can be modified via the `Download data from app` icon in the [Bangle.js App Store](https://banglejs.com/apps/) (TodoList app).
|
||||||
You can mark a task as completed.
|
|
||||||
|
data:image/s3,"s3://crabby-images/6f366/6f366c586067a1f128281d0d6fe061224cc1302d" alt=""
|
||||||
|
|
||||||
|
|
||||||
JSON file content example:
|
JSON file content example:
|
||||||
```javascript
|
```javascript
|
||||||
[
|
[
|
||||||
{
|
{
|
||||||
name: "Pro",
|
"name": "Pro",
|
||||||
children: [
|
"children": [
|
||||||
{
|
{
|
||||||
name: "Read doc",
|
"name": "Read doc",
|
||||||
done: true,
|
"done": true,
|
||||||
children: [],
|
"children": []
|
||||||
}
|
}
|
||||||
],
|
]
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: "Pers",
|
"name": "Pers",
|
||||||
children: [
|
"children": [
|
||||||
{
|
{
|
||||||
name: "Grocery",
|
"name": "Grocery",
|
||||||
children: [
|
"children": [
|
||||||
{ name: "Milk", done: false, children: [] },
|
{
|
||||||
{ name: "Eggs", done: false, children: [] },
|
"name": "Milk",
|
||||||
{ name: "Cheese", done: false, children: [] },
|
"done": false,
|
||||||
],
|
"children": []
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "Eggs",
|
||||||
|
"done": false,
|
||||||
|
"children": []
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "Cheese",
|
||||||
|
"done": false,
|
||||||
|
"children": []
|
||||||
|
}
|
||||||
|
]
|
||||||
},
|
},
|
||||||
{ name: "Workout", done: false, children: [] },
|
{
|
||||||
{ name: "Learn Rust", done: false, children: [] },
|
"name": "Workout",
|
||||||
],
|
"done": false,
|
||||||
},
|
"children": []
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "Learn Rust",
|
||||||
|
"done": false,
|
||||||
|
"children": []
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
]
|
]
|
||||||
```
|
```
|
|
@ -0,0 +1,135 @@
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<link rel="stylesheet" href="../../css/spectre.min.css" />
|
||||||
|
|
||||||
|
<style type="text/css">
|
||||||
|
.alert {
|
||||||
|
padding: 20px;
|
||||||
|
background-color: #f44336; /* Red */
|
||||||
|
color: white;
|
||||||
|
margin-bottom: 15px;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<div id="info"></div>
|
||||||
|
|
||||||
|
<button id="btnReload" class="btn btn-primary">Reload from watch</button>
|
||||||
|
<button id="btnUpload" class="btn btn-primary">Upload to watch</button>
|
||||||
|
<button id="btnDownload" class="btn btn-primary">Download</button>
|
||||||
|
|
||||||
|
<pre id="todos" contenteditable></pre>
|
||||||
|
|
||||||
|
<script src="../../core/lib/interface.js"></script>
|
||||||
|
<script>
|
||||||
|
const fileTodoList = "todolist.json";
|
||||||
|
|
||||||
|
function errorFormat() {
|
||||||
|
var date = new Date();
|
||||||
|
var error =
|
||||||
|
'<p class="alert">' +
|
||||||
|
date.toUTCString() +
|
||||||
|
" : Wrong format, it should be JSON" +
|
||||||
|
"</p>";
|
||||||
|
return error;
|
||||||
|
}
|
||||||
|
|
||||||
|
function getEditableContent() {
|
||||||
|
return document.getElementById("todos").innerHTML.replace(/<[^>]*>/g, '');;
|
||||||
|
}
|
||||||
|
|
||||||
|
function isJsonString(str) {
|
||||||
|
try {
|
||||||
|
JSON.parse(str);
|
||||||
|
} catch (e) {
|
||||||
|
console.log(str)
|
||||||
|
console.log(e)
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
function uploadFile(fileid, contents) {
|
||||||
|
Puck.write(
|
||||||
|
`\x10(function() {
|
||||||
|
require("Storage").write("${fileid}",'${contents}');
|
||||||
|
Bluetooth.print("OK");
|
||||||
|
})()\n`,
|
||||||
|
(ret) => {
|
||||||
|
console.log("uploadFile", ret);
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Load settings JSON file from the watch.
|
||||||
|
*/
|
||||||
|
function loadTodos() {
|
||||||
|
document.getElementById("info").innerHTML = "";
|
||||||
|
Util.showModal("Loading...");
|
||||||
|
Puck.eval(`require('Storage').readJSON("${fileTodoList}")`, (data) => {
|
||||||
|
document.getElementById("todos").innerHTML = JSON.stringify(
|
||||||
|
data,
|
||||||
|
null,
|
||||||
|
2
|
||||||
|
);
|
||||||
|
Util.hideModal();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
/* Save settings as a JSON file on the watch.
|
||||||
|
*/
|
||||||
|
function uploadTodos() {
|
||||||
|
document.getElementById("info").innerHTML = "";
|
||||||
|
Util.showModal("Uploading...");
|
||||||
|
let jsonTodos = getEditableContent();
|
||||||
|
if (isJsonString(jsonTodos)) {
|
||||||
|
let shortJsonTodos = JSON.stringify(JSON.parse(jsonTodos));
|
||||||
|
uploadFile(fileTodoList, shortJsonTodos);
|
||||||
|
} else {
|
||||||
|
document.getElementById("info").innerHTML = errorFormat();
|
||||||
|
}
|
||||||
|
Util.hideModal();
|
||||||
|
}
|
||||||
|
|
||||||
|
function downloadTodos() {
|
||||||
|
document.getElementById("info").innerHTML = "";
|
||||||
|
Util.showModal("Downloading...");
|
||||||
|
let jsonTodos = getEditableContent();
|
||||||
|
if (isJsonString(jsonTodos)) {
|
||||||
|
var a = document.createElement("a"),
|
||||||
|
file = new Blob([jsonTodos], { type: "application/json" });
|
||||||
|
var url = URL.createObjectURL(file);
|
||||||
|
a.href = url;
|
||||||
|
a.download = fileTodoList;
|
||||||
|
document.body.appendChild(a);
|
||||||
|
a.click();
|
||||||
|
setTimeout(function () {
|
||||||
|
document.body.removeChild(a);
|
||||||
|
window.URL.revokeObjectURL(url);
|
||||||
|
}, 0);
|
||||||
|
} else {
|
||||||
|
document.getElementById("info").innerHTML = errorFormat();
|
||||||
|
}
|
||||||
|
Util.hideModal();
|
||||||
|
}
|
||||||
|
|
||||||
|
document
|
||||||
|
.getElementById("btnUpload")
|
||||||
|
.addEventListener("click", function () {
|
||||||
|
uploadTodos();
|
||||||
|
});
|
||||||
|
document
|
||||||
|
.getElementById("btnDownload")
|
||||||
|
.addEventListener("click", function () {
|
||||||
|
downloadTodos();
|
||||||
|
});
|
||||||
|
document
|
||||||
|
.getElementById("btnReload")
|
||||||
|
.addEventListener("click", function () {
|
||||||
|
loadTodos();
|
||||||
|
});
|
||||||
|
function onInit() {
|
||||||
|
loadTodos();
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
</body>
|
||||||
|
</html>
|
|
@ -10,6 +10,7 @@
|
||||||
"tags": "tool,todo",
|
"tags": "tool,todo",
|
||||||
"supports": ["BANGLEJS", "BANGLEJS2"],
|
"supports": ["BANGLEJS", "BANGLEJS2"],
|
||||||
"readme": "README.md",
|
"readme": "README.md",
|
||||||
|
"interface": "interface.html",
|
||||||
"storage": [
|
"storage": [
|
||||||
{ "name": "todolist.app.js", "url": "app.js" },
|
{ "name": "todolist.app.js", "url": "app.js" },
|
||||||
{ "name": "todolist.img", "url": "app-icon.js", "evaluate": true }
|
{ "name": "todolist.img", "url": "app-icon.js", "evaluate": true }
|
||||||
|
|
Binary file not shown.
After Width: | Height: | Size: 72 KiB |
Loading…
Reference in New Issue