2023-11-02 00:29:07 +00:00
|
|
|
import {LitElement, html, unsafeHTML} from './lit-all.min.js';
|
2023-10-30 00:22:30 +00:00
|
|
|
import * as tfrpc from '/static/tfrpc.js';
|
|
|
|
|
|
|
|
class TfWikiDocElement extends LitElement {
|
|
|
|
static get properties() {
|
|
|
|
return {
|
|
|
|
whoami: {type: String},
|
2023-11-14 17:38:48 +00:00
|
|
|
wiki: {type: String},
|
2023-10-30 00:22:30 +00:00
|
|
|
value: {type: Object},
|
|
|
|
blob: {type: String},
|
|
|
|
blob_original: {type: String},
|
2023-11-01 22:21:42 +00:00
|
|
|
blob_for_value: {type: String},
|
2023-11-02 00:29:07 +00:00
|
|
|
is_editing: {type: Boolean},
|
2023-10-30 00:22:30 +00:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
constructor() {
|
|
|
|
super();
|
|
|
|
}
|
|
|
|
|
2023-11-02 00:29:07 +00:00
|
|
|
markdown(md) {
|
|
|
|
var reader = new commonmark.Parser({safe: true});
|
|
|
|
var writer = new commonmark.HtmlRenderer();
|
|
|
|
var parsed = reader.parse(md || '');
|
2023-11-14 17:38:48 +00:00
|
|
|
let walker = parsed.walker();
|
|
|
|
let event;
|
|
|
|
while ((event = walker.next())) {
|
|
|
|
let node = event.node;
|
|
|
|
if (event.entering) {
|
|
|
|
if (node.type === 'link') {
|
|
|
|
if (node.destination.indexOf(':') == -1 &&
|
|
|
|
node.destination.indexOf('/') == -1) {
|
|
|
|
node.destination = `#${this.wiki?.name}/${node.destination}`;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-11-02 00:29:07 +00:00
|
|
|
return writer.render(parsed);
|
|
|
|
}
|
|
|
|
|
2023-10-30 00:22:30 +00:00
|
|
|
async load_blob() {
|
2023-11-08 01:58:02 +00:00
|
|
|
let blob = await tfrpc.rpc.get_blob(this.value?.blob);
|
|
|
|
if (blob.endsWith('.box')) {
|
|
|
|
let d = await tfrpc.rpc.try_decrypt(this.whoami, blob);
|
|
|
|
if (d) {
|
|
|
|
blob = d;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
this.blob = blob;
|
|
|
|
this.blob_original = blob;
|
2023-10-30 00:22:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
on_edit(event) {
|
|
|
|
this.blob = event.srcElement.value;
|
|
|
|
}
|
|
|
|
|
2023-11-02 00:29:07 +00:00
|
|
|
on_discard(event) {
|
|
|
|
this.blob = this.blob_original;
|
|
|
|
this.is_editing = false;
|
|
|
|
}
|
|
|
|
|
2023-11-05 23:25:55 +00:00
|
|
|
async append_message(draft) {
|
2023-11-08 01:58:02 +00:00
|
|
|
let blob = this.blob;
|
|
|
|
if (draft) {
|
|
|
|
blob = await tfrpc.rpc.encrypt(this.whoami, this.value.editors, blob);
|
|
|
|
}
|
|
|
|
let id = await tfrpc.rpc.store_blob(blob);
|
2023-11-05 23:25:55 +00:00
|
|
|
let message = {
|
|
|
|
type: 'wiki-doc',
|
|
|
|
key: this.value.id,
|
|
|
|
parent: this.value.parent,
|
|
|
|
blob: id,
|
|
|
|
};
|
|
|
|
if (draft) {
|
|
|
|
message.recps = this.value.editors;
|
|
|
|
print(message);
|
|
|
|
message = await tfrpc.rpc.encrypt(this.whoami, this.value.editors, JSON.stringify(message));
|
|
|
|
}
|
|
|
|
print(message);
|
|
|
|
await tfrpc.rpc.appendMessage(this.whoami, message);
|
2023-11-02 00:45:20 +00:00
|
|
|
this.is_editing = false;
|
|
|
|
}
|
|
|
|
|
2023-11-05 23:25:55 +00:00
|
|
|
async on_save_draft() {
|
|
|
|
return this.append_message(true);
|
|
|
|
}
|
|
|
|
|
2023-11-02 00:29:07 +00:00
|
|
|
async on_publish() {
|
2023-11-05 23:25:55 +00:00
|
|
|
return this.append_message(false);
|
2023-10-30 00:22:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
render() {
|
2023-11-01 22:21:42 +00:00
|
|
|
let value = JSON.stringify(this.value);
|
|
|
|
if (this.blob_for_value != value) {
|
|
|
|
this.blob_for_value = value;
|
|
|
|
this.blob = undefined;
|
|
|
|
this.blob_original = undefined;
|
2023-10-30 00:22:30 +00:00
|
|
|
this.load_blob();
|
|
|
|
}
|
2023-11-02 00:29:07 +00:00
|
|
|
let self = this;
|
2023-10-30 00:22:30 +00:00
|
|
|
return html`
|
2023-11-02 00:29:07 +00:00
|
|
|
<div style="display: inline-flex; flex-direction: row">
|
|
|
|
<button ?disabled=${!this.whoami || this.is_editing} @click=${() => self.is_editing = true}>Edit</button>
|
2023-11-02 00:45:20 +00:00
|
|
|
<button ?disabled=${this.blob == this.blob_original} @click=${this.on_save_draft}>Save Draft</button>
|
2023-11-02 01:19:14 +00:00
|
|
|
<button ?disabled=${this.blob == this.blob_original && !this.value?.draft} @click=${this.on_publish}>Publish</button>
|
2023-11-02 00:29:07 +00:00
|
|
|
<button ?disabled=${!this.is_editing} @click=${this.on_discard}>Discard</button>
|
|
|
|
</div>
|
2023-11-01 23:39:34 +00:00
|
|
|
<div style="display: flex; flex-direction: row">
|
|
|
|
<textarea
|
2023-11-02 00:29:07 +00:00
|
|
|
?hidden=${!this.is_editing}
|
2023-11-01 23:39:34 +00:00
|
|
|
style="flex: 1 1; min-height: 10em"
|
|
|
|
@input=${this.on_edit} .value=${this.blob ?? ''}></textarea>
|
2023-11-02 00:29:07 +00:00
|
|
|
<div style="flex: 1 1">${unsafeHTML(this.markdown(this.blob))}</div>
|
2023-11-01 23:39:34 +00:00
|
|
|
</div>
|
2023-10-30 00:22:30 +00:00
|
|
|
`;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
customElements.define('tf-wiki-doc', TfWikiDocElement);
|