2024-04-27 22:39:53 +00:00
|
|
|
import {Adapter} from "./adapter";
|
2024-04-28 18:16:23 +00:00
|
|
|
import {Message, Attachment} from "./message"
|
2024-06-29 17:04:08 +00:00
|
|
|
import util from "./util"
|
2024-04-22 00:02:44 +00:00
|
|
|
|
2024-06-29 17:04:08 +00:00
|
|
|
var $ = util.$
|
|
|
|
var _ = util._
|
2024-04-22 00:32:14 +00:00
|
|
|
|
2024-04-27 16:50:27 +00:00
|
|
|
function main():void {
|
2024-04-27 22:39:53 +00:00
|
|
|
const settings = _("settings", JSON.parse(localStorage.getItem("settings") ?? "{}"));
|
|
|
|
const adapters = _("adapters", []);
|
2024-04-27 16:50:27 +00:00
|
|
|
|
|
|
|
if (settings != null) {
|
2024-05-13 00:26:44 +00:00
|
|
|
for (let s of settings.adapters ?? []) {
|
2024-06-30 00:18:31 +00:00
|
|
|
|
2024-04-28 18:16:23 +00:00
|
|
|
}
|
|
|
|
if (adapters.length > 0) {
|
2024-06-30 00:18:31 +00:00
|
|
|
_("currentAdapter", adapters[0].nickname);
|
2024-04-28 18:16:23 +00:00
|
|
|
// update tabbar and tabcontent with first adapter
|
|
|
|
}
|
2024-04-22 00:02:44 +00:00
|
|
|
} else {
|
|
|
|
console.log("no settings exist for this client");
|
2024-04-27 22:39:53 +00:00
|
|
|
_("settings", { adapters: [] });
|
2024-04-28 18:16:23 +00:00
|
|
|
showSettings();
|
2024-04-22 00:02:44 +00:00
|
|
|
}
|
2024-05-13 00:26:44 +00:00
|
|
|
registerServiceWorker();
|
2024-04-22 00:02:44 +00:00
|
|
|
};
|
|
|
|
|
2024-05-13 00:26:44 +00:00
|
|
|
async function registerServiceWorker() {
|
|
|
|
if ("serviceWorker" in navigator) {
|
|
|
|
try {
|
|
|
|
const registration = await navigator.serviceWorker.register("/serviceWorker.js", {
|
|
|
|
scope: "/",
|
|
|
|
});
|
|
|
|
if (registration.installing) {
|
|
|
|
console.log("Service worker installing");
|
|
|
|
} else if (registration.waiting) {
|
|
|
|
console.log("Service worker installed");
|
|
|
|
} else if (registration.active) {
|
|
|
|
console.log("Service worker active");
|
|
|
|
}
|
|
|
|
} catch (error) {
|
|
|
|
console.error(`Registration failed with ${error}`);
|
|
|
|
}
|
|
|
|
const registration = await navigator.serviceWorker.ready;
|
|
|
|
(registration as any).sync.register("testdata").then((r:any)=>{console.log("but i will see this!")});
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2024-04-27 16:50:27 +00:00
|
|
|
function showSettings():void {
|
2024-04-22 00:32:14 +00:00
|
|
|
// tab bar hidden
|
2024-04-22 02:11:37 +00:00
|
|
|
const tabbar = $("tabbar");
|
2024-04-27 16:50:27 +00:00
|
|
|
if (tabbar) {
|
2024-04-22 00:32:14 +00:00
|
|
|
tabbar.style.display = "none";
|
2024-04-27 16:50:27 +00:00
|
|
|
}
|
2024-04-22 00:32:14 +00:00
|
|
|
|
|
|
|
// tabcontent to show settings ui
|
2024-04-22 02:11:37 +00:00
|
|
|
const tabcontent = $("tabcontent");
|
2024-05-13 00:26:44 +00:00
|
|
|
const adapters = _("adapters") as Adapter[] ?? [];
|
2024-04-22 00:32:14 +00:00
|
|
|
|
2024-04-27 16:50:27 +00:00
|
|
|
if (tabcontent) {
|
2024-04-28 18:16:23 +00:00
|
|
|
let html = "<p>this is our settings dialogue</p>";
|
|
|
|
html += "<button onclick='addAdapter()'>New</button>";
|
|
|
|
html += adapters.reduce((self: string, a: Adapter) => {
|
|
|
|
self += `<li><a href='#' onclick='editAdapter(${a.nickname})'>${a.nickname}</a></li>`
|
|
|
|
return self;
|
|
|
|
}, "<ul id='settings_adapterlist'>");
|
|
|
|
html += "</ul>";
|
|
|
|
html += "<button onclick='saveSettings()'>save</button>";
|
|
|
|
tabcontent.innerHTML = html;
|
2024-04-27 16:50:27 +00:00
|
|
|
}
|
2024-04-22 02:11:37 +00:00
|
|
|
}
|
|
|
|
|
2024-04-27 16:50:27 +00:00
|
|
|
function addAdapter(): void {
|
2024-04-22 02:11:37 +00:00
|
|
|
const tabcontent = $("tabcontent");
|
2024-04-27 16:50:27 +00:00
|
|
|
if (tabcontent) {
|
2024-04-28 18:16:23 +00:00
|
|
|
// dropdown for protocol
|
|
|
|
let html = "<select id='settings_newadapter_protocolselect' onchange='fillAdapterProtocolOptions()'>";
|
2024-06-23 04:02:15 +00:00
|
|
|
html += [ "nostr", "mastodon", "misskey" ].reduce((self, p)=>{
|
2024-04-28 18:16:23 +00:00
|
|
|
self += `<option value='${p}'>${p}</option>`;
|
|
|
|
return self;
|
|
|
|
}, "");
|
|
|
|
html += "</select>";
|
|
|
|
|
|
|
|
// nostr is the first protocol, so show its options by default
|
|
|
|
html += "<div id='settings_newadapter_protocoloptions'>";
|
|
|
|
html += " <label>nickname<input id='settings_newadapter_nickname'/></label>";
|
|
|
|
html += " <label>privkey<input id='settings_newadapter_nostr_privkey'/></label>";
|
|
|
|
html += " <label>default relays<input id='settings_newadapter_nostr_default_relays'/></label>";
|
|
|
|
html += "</div>";
|
|
|
|
|
|
|
|
html += "<button onclick='saveAdapter()'>Add</button>";
|
|
|
|
html += "<button onclick='showSettings()'>Back</button>";
|
2024-04-22 00:32:14 +00:00
|
|
|
|
2024-04-28 18:16:23 +00:00
|
|
|
tabcontent.innerHTML = html;
|
2024-04-27 16:50:27 +00:00
|
|
|
}
|
2024-04-22 02:11:37 +00:00
|
|
|
}
|
|
|
|
|
2024-04-27 16:50:27 +00:00
|
|
|
function fillAdapterProtocolOptions(): void {
|
|
|
|
const proto = $("settings_newadapter_protocolselect") as HTMLSelectElement;
|
|
|
|
|
2024-04-27 22:39:53 +00:00
|
|
|
let html = "";
|
|
|
|
|
|
|
|
switch(proto?.options[proto.selectedIndex].value) {
|
|
|
|
case "nostr":
|
|
|
|
html += " <label>nickname<input id='settings_newadapter_nickname'/></label>";
|
|
|
|
html += " <label>privkey<input id='settings_newadapter_nostr_privkey'/></label>";
|
|
|
|
html += " <label>default relays<input id='settings_newadapter_nostr_default_relays'/></label>";
|
|
|
|
break;
|
|
|
|
case "mastodon":
|
2024-06-23 04:02:15 +00:00
|
|
|
case "misskey":
|
2024-04-27 22:39:53 +00:00
|
|
|
html += " <label>nickname<input id='settings_newadapter_nickname'/></label>";
|
|
|
|
html += " <label>server<input id='settings_newadapter_masto_server'/></label>";
|
|
|
|
html += " <label>API key<input id='settings_newadapter_masto_apikey'/></label>";
|
|
|
|
break;
|
|
|
|
}
|
2024-04-27 16:50:27 +00:00
|
|
|
|
|
|
|
|
2024-04-27 22:39:53 +00:00
|
|
|
const div = $("settings_newadapter_protocoloptions");
|
|
|
|
if (div) {
|
|
|
|
div.innerHTML = html;
|
|
|
|
}
|
2024-04-22 00:32:14 +00:00
|
|
|
}
|
|
|
|
|
2024-04-27 16:50:27 +00:00
|
|
|
function saveSettings(): void {
|
2024-04-27 22:39:53 +00:00
|
|
|
const settings = _("settings");
|
2024-04-27 16:50:27 +00:00
|
|
|
if (settings) {
|
|
|
|
localStorage.setItem("settings", JSON.stringify(settings));
|
2024-04-22 00:32:14 +00:00
|
|
|
}
|
2024-04-22 02:11:37 +00:00
|
|
|
// tab bar hidden
|
|
|
|
const tabbar = $("tabbar");
|
2024-04-27 16:50:27 +00:00
|
|
|
if (tabbar) {
|
|
|
|
tabbar.style.display = "block";
|
|
|
|
}
|
2024-04-22 00:32:14 +00:00
|
|
|
|
|
|
|
// tabcontent to show settings ui
|
2024-04-22 02:11:37 +00:00
|
|
|
const tabcontent = $("tabcontent");
|
2024-04-27 16:50:27 +00:00
|
|
|
if (tabcontent) {
|
|
|
|
tabcontent.innerHTML = "";
|
|
|
|
}
|
2024-04-22 00:32:14 +00:00
|
|
|
}
|
|
|
|
|
2024-04-27 16:50:27 +00:00
|
|
|
function saveAdapter(): void {
|
|
|
|
let self: any = {};
|
2024-04-22 02:11:37 +00:00
|
|
|
// get selected adapter protocol
|
2024-04-27 16:50:27 +00:00
|
|
|
const proto = $("settings_newadapter_protocolselect") as HTMLSelectElement;
|
2024-04-22 02:11:37 +00:00
|
|
|
console.log(proto.options[proto.selectedIndex]);
|
|
|
|
|
2024-04-27 22:39:53 +00:00
|
|
|
|
|
|
|
const nickname = ($("settings_newadapter_nickname") as HTMLInputElement)?.value ?? "" ;
|
|
|
|
|
2024-04-22 02:11:37 +00:00
|
|
|
// switch protocol
|
|
|
|
switch (proto.options[proto.selectedIndex].value) {
|
|
|
|
case "nostr":
|
2024-04-27 16:50:27 +00:00
|
|
|
const privkey = ($("settings_newadapter_nostr_privkey") as HTMLInputElement)?.value ?? "";
|
|
|
|
const relays = ($("settings_newadapter_nostr_default_relays") as HTMLInputElement)?.value ?? "";
|
2024-04-22 02:11:37 +00:00
|
|
|
self = { nickname: nickname, protocol: "nostr", privkey: privkey, relays: relays.split(",").map(r=>r.trim()) };
|
|
|
|
break;
|
2024-04-27 22:39:53 +00:00
|
|
|
case "mastodon":
|
2024-06-23 04:02:15 +00:00
|
|
|
case "misskey":
|
2024-04-27 22:39:53 +00:00
|
|
|
const server = ($("settings_newadapter_masto_server") as HTMLInputElement)?.value ?? "";
|
|
|
|
const apiKey = ($("settings_newadapter_masto_apikey") as HTMLInputElement)?.value ?? "";
|
2024-06-23 04:02:15 +00:00
|
|
|
self = { nickname: nickname, protocol: proto.options[proto.selectedIndex].value, server: server, apiKey: apiKey };
|
2024-04-22 02:11:37 +00:00
|
|
|
break;
|
|
|
|
}
|
2024-04-27 22:39:53 +00:00
|
|
|
const settings = _("settings");
|
2024-06-23 04:02:15 +00:00
|
|
|
if (settings) {
|
2024-05-13 00:26:44 +00:00
|
|
|
if (!settings.adapters) {
|
|
|
|
settings.adapters = [];
|
|
|
|
}
|
2024-04-27 16:50:27 +00:00
|
|
|
settings.adapters.push(self);
|
2024-06-29 17:04:08 +00:00
|
|
|
|
2024-04-27 16:50:27 +00:00
|
|
|
localStorage.setItem("settings", JSON.stringify(settings));
|
|
|
|
showSettings();
|
|
|
|
}
|
2024-04-22 02:11:37 +00:00
|
|
|
}
|
|
|
|
|
2024-05-13 00:26:44 +00:00
|
|
|
|
|
|
|
let _conn: WebSocket | null = null;
|
|
|
|
|
2024-06-02 16:17:12 +00:00
|
|
|
async function authorizedFetch(method: string, uri: string, body: any): Promise<Response> {
|
|
|
|
const headers = new Headers()
|
|
|
|
headers.set('Authorization', 'Bearer ' + _("skey"))
|
|
|
|
return await fetch(uri, {
|
|
|
|
method: method,
|
|
|
|
headers: headers,
|
|
|
|
body: body,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2024-05-13 00:26:44 +00:00
|
|
|
function connect() {
|
2024-05-26 18:50:19 +00:00
|
|
|
|
2024-06-30 00:18:31 +00:00
|
|
|
var datastore: AdapterState = {}
|
|
|
|
datastore = _("datastore", datastore);
|
|
|
|
|
2024-05-19 20:42:28 +00:00
|
|
|
const wsProto = location.protocol == "https:" ? "wss" : "ws";
|
2024-06-02 16:17:12 +00:00
|
|
|
_conn = new WebSocket(`${wsProto}://${location.host}/subscribe`, "underbbs");
|
2024-05-19 20:56:25 +00:00
|
|
|
_conn.addEventListener("open", (e: any) => {
|
|
|
|
console.log("websocket connection opened");
|
|
|
|
console.log(JSON.stringify(e));
|
|
|
|
});
|
2024-05-26 18:50:19 +00:00
|
|
|
_conn.addEventListener("message", (e: any) => {
|
2024-06-02 16:17:12 +00:00
|
|
|
|
2024-06-29 17:04:08 +00:00
|
|
|
// debugging
|
2024-05-26 18:50:19 +00:00
|
|
|
console.log(e)
|
2024-06-29 17:04:08 +00:00
|
|
|
|
|
|
|
// now we'll figure out what to do with it
|
2024-06-02 16:17:12 +00:00
|
|
|
const data = JSON.parse(e.data);
|
|
|
|
if (data.key) {
|
|
|
|
_("skey", data.key)
|
|
|
|
authorizedFetch("POST", "/api/adapters", JSON.stringify(_("settings").adapters))
|
2024-06-29 17:04:08 +00:00
|
|
|
} else {
|
2024-06-30 00:18:31 +00:00
|
|
|
if (!datastore[data.adapter]) {
|
|
|
|
datastore[data.adapter] = new AdapterData(data.protocol);
|
|
|
|
}
|
|
|
|
|
|
|
|
// typeswitch on the incoming data type and fill the memory
|
|
|
|
switch (data.type) {
|
|
|
|
case "message":
|
|
|
|
datastore[data.adapter].messages[data.id] = <Message>data;
|
|
|
|
break;
|
|
|
|
case "author":
|
|
|
|
datastore[data.adapter].profileCache[data.id] = <Author>data;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
2024-06-29 17:04:08 +00:00
|
|
|
// if the adapter is active, inject the web components
|
|
|
|
// FOR HOTFETCHED DATA:
|
|
|
|
// before fetching, we can set properties on the DOM,
|
|
|
|
// so when those data return to us we know where to inject components!
|
2024-06-30 00:18:31 +00:00
|
|
|
if (_("currentAdapter") == data.adapter) {
|
|
|
|
// dive in and insert that shit in the dom
|
|
|
|
}
|
2024-06-02 16:17:12 +00:00
|
|
|
}
|
2024-05-26 18:50:19 +00:00
|
|
|
});
|
2024-05-19 20:56:25 +00:00
|
|
|
_conn.addEventListener("error", (e: any) => {
|
|
|
|
console.log("websocket connection error");
|
|
|
|
console.log(JSON.stringify(e));
|
|
|
|
});
|
2024-05-13 00:26:44 +00:00
|
|
|
_("websocket", _conn);
|
|
|
|
}
|
|
|
|
|
2024-04-27 22:39:53 +00:00
|
|
|
_("addAdapter", addAdapter);
|
|
|
|
_("saveAdapter", saveAdapter);
|
|
|
|
_("fillAdapterProtocolOptions", fillAdapterProtocolOptions);
|
|
|
|
_("showSettings", showSettings);
|
|
|
|
_("saveSettings", saveSettings);
|
2024-05-13 00:26:44 +00:00
|
|
|
_("connect", connect);
|
2024-04-27 16:50:27 +00:00
|
|
|
main();
|