forked from cory/tildefriends
Added some edit/save/back keybindings.
git-svn-id: https://www.unprompted.com/svn/projects/tildefriends/trunk@3308 ed5197a5-7fde-0310-b194-c3ffbd925b24
This commit is contained in:
parent
b908f63424
commit
40b0de6c15
@ -11,7 +11,7 @@ var gGeolocatorWatch;
|
|||||||
|
|
||||||
var kMaxCommandHistory = 16;
|
var kMaxCommandHistory = 16;
|
||||||
|
|
||||||
function enter(event) {
|
function keydown(event) {
|
||||||
if (event.keyCode == 13) {
|
if (event.keyCode == 13) {
|
||||||
gCommandHistory.push(document.getElementById("input").value);
|
gCommandHistory.push(document.getElementById("input").value);
|
||||||
if (gCommandHistory.length > kMaxCommandHistory) {
|
if (gCommandHistory.length > kMaxCommandHistory) {
|
||||||
@ -19,20 +19,23 @@ function enter(event) {
|
|||||||
}
|
}
|
||||||
send();
|
send();
|
||||||
event.preventDefault();
|
event.preventDefault();
|
||||||
} else if (event.keyCode == 38) {
|
} else if (event.keyCode == 38 && !event.altKey) {
|
||||||
if (gCommandHistory.length) {
|
if (gCommandHistory.length) {
|
||||||
var input = document.getElementById("input");
|
var input = document.getElementById("input");
|
||||||
gCommandHistory.unshift(input.value);
|
gCommandHistory.unshift(input.value);
|
||||||
input.value = gCommandHistory.pop();
|
input.value = gCommandHistory.pop();
|
||||||
event.preventDefault();
|
event.preventDefault();
|
||||||
}
|
}
|
||||||
} else if (event.keyCode == 40) {
|
} else if (event.keyCode == 40 && !event.altKey) {
|
||||||
if (gCommandHistory.length) {
|
if (gCommandHistory.length) {
|
||||||
var input = document.getElementById("input");
|
var input = document.getElementById("input");
|
||||||
gCommandHistory.push(input.value);
|
gCommandHistory.push(input.value);
|
||||||
input.value = gCommandHistory.shift();
|
input.value = gCommandHistory.shift();
|
||||||
event.preventDefault();
|
event.preventDefault();
|
||||||
}
|
}
|
||||||
|
} else if (event.keyCode == 69 && event.altKey) {
|
||||||
|
window.location.href = url() + "/edit";
|
||||||
|
event.preventDefault();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -576,7 +579,7 @@ window.addEventListener("load", function() {
|
|||||||
Notification.requestPermission();
|
Notification.requestPermission();
|
||||||
}
|
}
|
||||||
var input = document.getElementById("input");
|
var input = document.getElementById("input");
|
||||||
input.addEventListener("keydown", enter);
|
input.addEventListener("keydown", keydown);
|
||||||
input.focus();
|
input.focus();
|
||||||
window.addEventListener("hashchange", hashChange);
|
window.addEventListener("hashchange", hashChange);
|
||||||
window.addEventListener("focus", focus);
|
window.addEventListener("focus", focus);
|
||||||
|
@ -11,6 +11,15 @@ window.addEventListener("load", function() {
|
|||||||
'showTrailingSpace': true,
|
'showTrailingSpace': true,
|
||||||
});
|
});
|
||||||
gBackup = gEditor.getValue();
|
gBackup = gEditor.getValue();
|
||||||
|
gEditor.focus();
|
||||||
|
});
|
||||||
|
|
||||||
|
window.addEventListener("keydown", function(event) {
|
||||||
|
if (event.keyCode == 83 && event.altKey) {
|
||||||
|
save();
|
||||||
|
} else if (event.keyCode == 66 && event.altKey) {
|
||||||
|
back();
|
||||||
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
function explodePath() {
|
function explodePath() {
|
||||||
|
@ -1,195 +1,195 @@
|
|||||||
"use strict";
|
"use strict";
|
||||||
|
|
||||||
//! {"category": "work in progress"}
|
//! {"category": "work in progress", "require": ["libdocument", "liblist", "ui"]}
|
||||||
|
|
||||||
class Log {
|
let libdocument = require("libdocument");
|
||||||
constructor(name, capacity) {
|
let liblist = require("liblist");
|
||||||
this._name = name;
|
let libui = require("ui");
|
||||||
this._capacity = capacity || -1;
|
|
||||||
|
class Blog {
|
||||||
|
constructor() {
|
||||||
|
this._list = liblist.ListStore("blog:list");
|
||||||
|
this._documents = libdocument.DocumentStore("blog:posts");
|
||||||
}
|
}
|
||||||
|
|
||||||
append(item) {
|
async renderIndex() {
|
||||||
var log = this;
|
|
||||||
return database.get(log._name + "_head").catch(function(error) {
|
|
||||||
return 0;
|
|
||||||
}).then(function(head) {
|
|
||||||
var newHead = (parseInt(head) || 0) + 1;
|
|
||||||
var actions = [
|
|
||||||
database.set(log._name + "_" + newHead.toString(), JSON.stringify(item)),
|
|
||||||
database.set(log._name + "_head", newHead),
|
|
||||||
];
|
|
||||||
if (log._capacity >= 0) {
|
|
||||||
actions.push(log.truncate(newHead - log._capacity));
|
|
||||||
}
|
|
||||||
return Promise.all(actions);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
truncate(end) {
|
|
||||||
var log = this;
|
|
||||||
return database.get(log._name + "_" + end.toString()).then(function(item) {
|
|
||||||
if (item) {
|
|
||||||
return database.remove(log._name + "_" + end.toString()).then(function() {
|
|
||||||
return log.truncate(end - 1);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
get(count, start, result) {
|
|
||||||
var log = this;
|
|
||||||
if (!count) {
|
|
||||||
count = 10;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!start) {
|
|
||||||
return database.get(log._name + "_head").then(function(head) {
|
|
||||||
if (head !== undefined) {
|
|
||||||
return log.get(count, parseInt(head));
|
|
||||||
} else {
|
|
||||||
return [];
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
var promises = [];
|
|
||||||
promises.length = count;
|
|
||||||
for (var i = 0; i < count; i++) {
|
|
||||||
promises[i] = database.get(log._name + "_" + (start - i).toString());
|
|
||||||
}
|
|
||||||
return Promise.all(promises);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
/*
|
|
||||||
|
|
||||||
if (imports.terminal) {
|
|
||||||
core.register("onSubmit", function(message) {
|
|
||||||
core.broadcast(message);
|
|
||||||
});
|
|
||||||
|
|
||||||
core.register("onMessage", function(from, message) {
|
|
||||||
terminal.print(JSON.stringify(message));
|
|
||||||
});
|
|
||||||
|
|
||||||
terminal.print("Hello, world!");
|
|
||||||
var log = new Log("events");
|
|
||||||
|
|
||||||
function dump() {
|
|
||||||
return log.get().then(function(data) {
|
|
||||||
terminal.print(JSON.stringify(data));
|
|
||||||
}).catch(function(error) {
|
|
||||||
terminal.print(error);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
core.register("onInput", function(input) {
|
|
||||||
log.append({message: input}).then(dump);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
core.register("onAtom", function(query) {
|
|
||||||
return "hello, world!";
|
|
||||||
});
|
|
||||||
|
|
||||||
*/
|
|
||||||
|
|
||||||
terminal.setEcho(false);
|
|
||||||
|
|
||||||
var gBlog = new Log("blog");
|
|
||||||
|
|
||||||
core.register("onInput", function(input) {
|
|
||||||
if (input == "new post") {
|
|
||||||
startNewPost();
|
|
||||||
} else if (input == "submit") {
|
|
||||||
submitNewPost().then(function() {
|
|
||||||
core.unregister("onWindowMessage", onWindowMessage);
|
|
||||||
renderBlog();
|
|
||||||
});
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
function renderBlog() {
|
|
||||||
terminal.split([
|
|
||||||
{name: "terminal"},
|
|
||||||
]);
|
|
||||||
terminal.select("terminal");
|
|
||||||
|
|
||||||
terminal.print("Blog Test");
|
|
||||||
if (core.user.credentials.permissions.authenticated) {
|
|
||||||
terminal.print({command: "new post"});
|
|
||||||
}
|
|
||||||
|
|
||||||
gBlog.get(10).then(function(entries) {
|
|
||||||
for (var i = 0; i < entries.length; i++) {
|
|
||||||
var entry = JSON.parse(entries[i]);
|
|
||||||
terminal.print({style: "font-weight: bold", value: entry.title});
|
|
||||||
terminal.print(entry.entry);
|
|
||||||
terminal.print();
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
var gNewPost;
|
|
||||||
|
|
||||||
function submitNewPost() {
|
|
||||||
return gBlog.append(gNewPost);
|
|
||||||
}
|
|
||||||
|
|
||||||
function onWindowMessage(message) {
|
|
||||||
gNewPost = message.message;
|
|
||||||
terminal.cork();
|
terminal.cork();
|
||||||
terminal.select("right");
|
try {
|
||||||
|
terminal.split([{name: "terminal"}]);
|
||||||
terminal.clear();
|
terminal.clear();
|
||||||
terminal.print({style: "font-width: x-large", value: message.message.title});
|
let posts = await this._list.get(-1, -10);
|
||||||
terminal.print(message.message.entry);
|
for (let i = 0; i < posts.length; i++) {
|
||||||
terminal.print({command: "submit"});
|
let post = await this._documents.get(posts[i]);
|
||||||
terminal.uncork();
|
if (post) {
|
||||||
}
|
let formatted = this.formatPost(post);
|
||||||
|
for (let j in formatted) {
|
||||||
function startNewPost() {
|
terminal.print(formatted[j]);
|
||||||
core.register("onWindowMessage", onWindowMessage);
|
|
||||||
terminal.split([
|
|
||||||
{
|
|
||||||
type: "horizontal",
|
|
||||||
children: [
|
|
||||||
{name: "left", grow: "0", shrink: "0", basis: "50%"},
|
|
||||||
{name: "right", grow: "0", shrink: "0", basis: "50%"},
|
|
||||||
],
|
|
||||||
}
|
}
|
||||||
]);
|
}
|
||||||
terminal.select("left");
|
}
|
||||||
terminal.print({iframe: `<html>
|
if (core.user.credentials.permissions.administration) {
|
||||||
|
terminal.print({command: JSON.stringify({action: "new"}), value: "new post"});
|
||||||
|
}
|
||||||
|
} finally {
|
||||||
|
terminal.uncork();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
formatPost(post) {
|
||||||
|
let result = [
|
||||||
|
[{style: "font-size: xx-large", value: post.title}],
|
||||||
|
[{style: "font-size: x-small", value: post.author}, " ", {style: "font-size: x-small", value: post.created}],
|
||||||
|
post.body,
|
||||||
|
];
|
||||||
|
if (core.user.credentials.permissions.administration) {
|
||||||
|
result[0].push({command: JSON.stringify({action: "edit", post: post.name}), value: "edit"});
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
async submitPost(post) {
|
||||||
|
let now = new Date();
|
||||||
|
let oldPost = await this._documents.get(post.name);
|
||||||
|
if (!oldPost) {
|
||||||
|
post.created = now;
|
||||||
|
post.author = core.user.name;
|
||||||
|
this._list.push(post.name);
|
||||||
|
} else {
|
||||||
|
for (let key in oldPost) {
|
||||||
|
if (!post[key]) {
|
||||||
|
post[key] = oldPost[key];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
post.modified = now;
|
||||||
|
await this._documents.set(post.name, post);
|
||||||
|
}
|
||||||
|
|
||||||
|
async deletePost(name) {
|
||||||
|
await this._documents.set(name, null);
|
||||||
|
}
|
||||||
|
|
||||||
|
async handleCommand(command) {
|
||||||
|
if (command.action == "edit") {
|
||||||
|
await this.edit(command.post);
|
||||||
|
} else if (command.action == "new") {
|
||||||
|
await this.edit(null);
|
||||||
|
} else if (command.action == "delete") {
|
||||||
|
await this.deletePost(command.post);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async edit(page) {
|
||||||
|
terminal.cork();
|
||||||
|
try {
|
||||||
|
let self = this;
|
||||||
|
self._post = await this._documents.get(page);
|
||||||
|
|
||||||
|
if (!this._onWindowMessage) {
|
||||||
|
this._onWindowMessage = function(event) {
|
||||||
|
let message = event.message;
|
||||||
|
if (message == "load") {
|
||||||
|
terminal.postMessageToIframe("iframe", self._post);
|
||||||
|
} else if (message.action == "save") {
|
||||||
|
self.submitPost(message.post).then(self.renderIndex.bind(self)).catch(terminal.print);
|
||||||
|
} else if (message.action == "delete") {
|
||||||
|
self.deletePost(message.post).then(self.renderIndex.bind(self)).catch(terminal.print);
|
||||||
|
} else if (message.action == "back") {
|
||||||
|
self.renderIndex.bind(self)().catch(terminal.print);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
core.register("onWindowMessage", this._onWindowMessage);
|
||||||
|
}
|
||||||
|
|
||||||
|
terminal.split([{name: "terminal", type: "vertical"}]);
|
||||||
|
terminal.clear();
|
||||||
|
terminal.print({iframe: `
|
||||||
|
<!DOCTYPE html>
|
||||||
|
<html>
|
||||||
<head>
|
<head>
|
||||||
|
<script src="//cdnjs.cloudflare.com/ajax/libs/codemirror/5.13.2/codemirror.min.js"></script>
|
||||||
|
<link rel="stylesheet" href="//cdnjs.cloudflare.com/ajax/libs/codemirror/5.13.2/codemirror.min.css"></link>
|
||||||
<style>
|
<style>
|
||||||
html, body, textarea {
|
html, body, #contents {
|
||||||
position: relative;
|
|
||||||
width: 100%;
|
|
||||||
height: 100%;
|
height: 100%;
|
||||||
margin: 0;
|
margin: 0;
|
||||||
padding: 0;
|
padding: 0;
|
||||||
overflow: hidden;
|
|
||||||
}
|
}
|
||||||
textarea {
|
body {
|
||||||
overflow: auto;
|
display: flex;
|
||||||
resize: none;
|
flex-direction: column;
|
||||||
|
color: #fff;
|
||||||
|
}
|
||||||
|
.CodeMirror {
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
}
|
||||||
|
.CodeMirror-scroll {
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<div>
|
||||||
|
<input type="button" id="back" value="Back" onclick="back()">
|
||||||
|
<input type="button" id="save" value="Save" onclick="save()">
|
||||||
|
<input type="button" id="delete" value="Delete" onclick="deletePost()">
|
||||||
|
</div>
|
||||||
|
<div><label for="name">Name:</label> <input type="text" id="name"></div>
|
||||||
|
<div><label for="title">Title:</label> <input type="text" id="title"></div>
|
||||||
|
<textarea id="contents" style="width: 100%; height: 100%"></textarea>
|
||||||
<script>
|
<script>
|
||||||
function textChanged() {
|
var gEditor;
|
||||||
|
window.addEventListener("message", function(event) {
|
||||||
|
var message = event.data;
|
||||||
|
console.debug(message);
|
||||||
|
gEditor.setValue(message.body || "");
|
||||||
|
document.getElementById("name").value = message.name || "untitled";
|
||||||
|
document.getElementById("title").value = message.title || "untitled";
|
||||||
|
});
|
||||||
|
window.addEventListener("load", function() {
|
||||||
|
gEditor = CodeMirror.fromTextArea(document.getElementById("contents"), {
|
||||||
|
lineNumbers: true
|
||||||
|
});
|
||||||
|
//gEditor.on("change", textChanged);
|
||||||
|
parent.postMessage("load", "*");
|
||||||
|
});
|
||||||
|
function back() {
|
||||||
|
parent.postMessage({action: "back"}, "*");
|
||||||
|
}
|
||||||
|
function save() {
|
||||||
parent.postMessage({
|
parent.postMessage({
|
||||||
|
action: "save",
|
||||||
|
post: {
|
||||||
|
name: document.getElementById("name").value,
|
||||||
title: document.getElementById("title").value,
|
title: document.getElementById("title").value,
|
||||||
entry: document.getElementById("entry").value,
|
body: gEditor.getValue(),
|
||||||
|
},
|
||||||
|
}, "*");
|
||||||
|
}
|
||||||
|
function deletePost() {
|
||||||
|
parent.postMessage({
|
||||||
|
action: "delete",
|
||||||
|
post: document.getElementById("name").value,
|
||||||
}, "*");
|
}, "*");
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
</head>
|
|
||||||
<body>
|
|
||||||
<input type="text" id="title" style="width: 100%" oninput="textChanged()">
|
|
||||||
<textarea id="entry" oninput="textChanged()"></textarea>
|
|
||||||
</body>
|
</body>
|
||||||
</html>`, style: "overflow: hidden; position: relative; left: 0; top: 0; right: 0; bottom: 0"});
|
</html>
|
||||||
terminal.select("right");
|
`, name: "iframe", style: "flex: 1 1 auto; width: 100%; margin: 0; border: 0; padding: 0;"});
|
||||||
|
} finally {
|
||||||
|
terminal.uncork();
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
renderBlog();
|
terminal.setEcho(false);
|
||||||
|
let blog = new Blog();
|
||||||
|
blog.renderIndex().catch(terminal.print);
|
||||||
|
|
||||||
|
core.register("onInput", async function(input) {
|
||||||
|
try {
|
||||||
|
await blog.handleCommand(JSON.parse(input));
|
||||||
|
} catch (error) {
|
||||||
|
terminal.print(error);
|
||||||
|
}
|
||||||
|
});
|
Loading…
Reference in New Issue
Block a user