2024-07-05 01:27:29 +00:00
|
|
|
import util from "./util"
|
2024-07-16 19:43:35 +00:00
|
|
|
|
2024-07-05 01:27:29 +00:00
|
|
|
import { Message, Author } from "./message"
|
|
|
|
import { MessageThread } from "./thread"
|
2024-07-16 19:43:35 +00:00
|
|
|
import { AdapterState } from "./adapter"
|
2024-08-03 16:52:33 +00:00
|
|
|
import { BatchTimer } from "./batch-timer"
|
2024-07-05 01:27:29 +00:00
|
|
|
|
|
|
|
export class AdapterElement extends HTMLElement {
|
|
|
|
static observedAttributes = [ "data-latest", "data-view", "data-viewing" ]
|
|
|
|
|
|
|
|
private _latest: string = "" ;
|
|
|
|
private _view: string = "";
|
|
|
|
private _name: string = ""
|
|
|
|
private _viewing: string = "";
|
|
|
|
|
2024-08-03 16:52:33 +00:00
|
|
|
private _convoyBatchTimer = new BatchTimer((ids: string[])=>{
|
|
|
|
let url = `/api/adapters/${this._name}/fetch?entity_type=convoy`;
|
|
|
|
for (let id of ids) {
|
|
|
|
url += `&entity_id=${id}`;
|
|
|
|
}
|
|
|
|
util.authorizedFetch("GET", url, null)
|
|
|
|
});
|
|
|
|
|
2024-07-05 01:27:29 +00:00
|
|
|
// TODO: use visibility of the thread to organize into DMs and public threads
|
|
|
|
private _threads: MessageThread[] = [];
|
|
|
|
private _orphans: Message[] = [];
|
2024-08-31 17:01:31 +00:00
|
|
|
private _boosts: Message[] = [];
|
2024-07-05 01:27:29 +00:00
|
|
|
|
|
|
|
constructor() {
|
|
|
|
super();
|
|
|
|
}
|
|
|
|
|
|
|
|
connectedCallback() {
|
|
|
|
const name = this.getAttribute("data-name");
|
|
|
|
this._name = name ?? "";
|
|
|
|
this.buildThreads();
|
|
|
|
}
|
|
|
|
|
2024-08-04 17:48:51 +00:00
|
|
|
attributeChangedCallback(attr: string, prev: string, next: string) {
|
|
|
|
switch (attr) {
|
|
|
|
case "data-viewing":
|
|
|
|
if (next != prev) {
|
|
|
|
this._viewing = next;
|
2024-07-05 01:27:29 +00:00
|
|
|
}
|
2024-08-04 17:48:51 +00:00
|
|
|
// probably only needed for profile view; we might be able to move this to child components
|
|
|
|
// keep it simple for now, but eventually we will need a way to force _view to update
|
|
|
|
break;
|
|
|
|
case "data-view":
|
|
|
|
if (!next) {
|
|
|
|
this._view = "index";
|
|
|
|
} else {
|
|
|
|
this._view = next;
|
|
|
|
}
|
|
|
|
switch (this._view) {
|
|
|
|
case "index":
|
|
|
|
this.setIdxView();
|
|
|
|
this.populateIdxView();
|
|
|
|
break;
|
|
|
|
case "thread":
|
|
|
|
this.setThreadView();
|
|
|
|
this.populateThreadView();
|
|
|
|
break;
|
|
|
|
case "profile":
|
|
|
|
this.setProfileView();
|
|
|
|
this.populateProfileView();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case "data-latest":
|
|
|
|
let datastore = AdapterState._instance.data.get(this._name);
|
|
|
|
if (!datastore) {
|
|
|
|
// this shouldn't be possible
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (prev != next && next) {
|
|
|
|
this._latest = next;
|
|
|
|
}
|
|
|
|
const latestMsg = datastore.messages.get(this._latest);
|
|
|
|
if (latestMsg) {
|
|
|
|
const rootId = this.placeMsg(this._latest);
|
|
|
|
// if rootId is null, this is an orphan and we don't need to actually do any UI updates yet
|
|
|
|
if (rootId) {
|
|
|
|
switch (this._view) {
|
|
|
|
case "index":
|
|
|
|
this.updateIdxView(this._latest, rootId);
|
|
|
|
break;
|
|
|
|
case "thread":
|
|
|
|
// if the the message is part of this thread, update it
|
|
|
|
case "profile":
|
|
|
|
// if the message is from this user, show it in their profile
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
const latestAuthor = datastore.profileCache.get(this._latest);
|
|
|
|
if (latestAuthor) {
|
|
|
|
switch (this._view) {
|
|
|
|
case "index":
|
|
|
|
const threadsByThisAuthor = this._threads.filter(t=>t.root.data.author == this._latest);
|
|
|
|
for (let t of threadsByThisAuthor) {
|
|
|
|
let tse = this.querySelector(`underbbs-thread-summary[data-msg='${t.root.data.id}']`)
|
|
|
|
if (tse) {
|
|
|
|
tse.setAttribute("data-author", this._latest);
|
|
|
|
}
|
2024-07-16 19:43:35 +00:00
|
|
|
}
|
2024-08-31 17:01:31 +00:00
|
|
|
// also update any boosts by this author
|
2024-08-04 17:48:51 +00:00
|
|
|
case "thread":
|
|
|
|
case "profile":
|
|
|
|
break;
|
|
|
|
}
|
2024-07-16 19:43:35 +00:00
|
|
|
}
|
2024-07-05 01:27:29 +00:00
|
|
|
}
|
2024-08-04 17:48:51 +00:00
|
|
|
break;
|
2024-07-05 01:27:29 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
setIdxView() {
|
2024-08-31 17:01:31 +00:00
|
|
|
this.innerHTML = "<ul id='boost_carousel'></ul><ul id='dm_list'></ul><ul id='public_list'></ul>"
|
2024-07-05 01:27:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
setThreadView() {
|
|
|
|
let html = `<a href="#${this._name}">← return to index</a>`;
|
|
|
|
html += "<ul id='msg_list'></ul>";
|
|
|
|
this.innerHTML = html;
|
|
|
|
}
|
|
|
|
|
|
|
|
setProfileView() {
|
2024-07-16 19:43:35 +00:00
|
|
|
let profile_bar = util.$("profile_bar");
|
2024-07-05 01:27:29 +00:00
|
|
|
if (profile_bar) {
|
|
|
|
// clear any previous data
|
|
|
|
} else {
|
|
|
|
// insert the profileSidebar into the dom
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
populateIdxView() {
|
2024-08-31 17:01:31 +00:00
|
|
|
// populate boost carousel
|
|
|
|
|
2024-07-05 01:27:29 +00:00
|
|
|
// skip dm list for now
|
|
|
|
// public/unified list
|
2024-07-16 19:43:35 +00:00
|
|
|
const pl = util.$("public_list");
|
2024-07-05 01:27:29 +00:00
|
|
|
if (pl) {
|
2024-07-05 05:23:56 +00:00
|
|
|
let html = "";
|
2024-07-16 19:43:35 +00:00
|
|
|
for (const t of this._threads.sort((a: MessageThread, b: MessageThread) => b.latest - a.latest)) {
|
|
|
|
html +=`<li><underbbs-thread-summary data-len="${t.messageCount}" data-adapter="${t.root.data.adapter}" data-msg="${t.root.data.id}" data-created="${t.created}" data-latest="${t.latest}"></underbbs-thread-summary></li>`;
|
2024-07-05 01:27:29 +00:00
|
|
|
}
|
2024-07-05 05:23:56 +00:00
|
|
|
pl.innerHTML = html;
|
2024-07-05 01:27:29 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
updateIdxView(latest: string, rootId: string) {
|
2024-08-04 17:48:51 +00:00
|
|
|
const threadSelector = `underbbs-thread-summary[data-msg='${rootId}']`
|
|
|
|
const existingThread = document.querySelector(threadSelector);
|
2024-07-05 01:27:29 +00:00
|
|
|
const thread = this._threads.find(t=>t.root.data.id == rootId);
|
|
|
|
if (existingThread && thread) {
|
2024-07-16 19:43:35 +00:00
|
|
|
existingThread.setAttribute("data-latest", `${thread.latest}`);
|
2024-07-05 01:27:29 +00:00
|
|
|
existingThread.setAttribute("data-len", `${thread.messageCount}`);
|
|
|
|
existingThread.setAttribute("data-new", "true");
|
|
|
|
} else {
|
2024-08-31 17:01:31 +00:00
|
|
|
// if latest is a boost, put it in the carousel
|
|
|
|
|
2024-07-05 01:27:29 +00:00
|
|
|
// unified/public list for now
|
2024-07-16 19:43:35 +00:00
|
|
|
const pl = util.$("public_list");
|
2024-07-05 01:27:29 +00:00
|
|
|
if (pl && thread) {
|
2024-07-16 19:43:35 +00:00
|
|
|
const li = document.createElement("li");
|
|
|
|
li.innerHTML = `<underbbs-thread-summary data-len="1" data-adapter="${thread.root.data.adapter}" data-msg="${thread.root.data.id}" data-latest="${thread.latest}" data-created="${thread.created}"></underbbs-thread-summary>`;
|
|
|
|
let nextThread: Element | null = null;
|
|
|
|
for (let i = 0; i < pl.children.length; i++) {
|
|
|
|
const c = pl.children.item(i);
|
|
|
|
const latest = c?.children.item(0)?.getAttribute("data-latest")
|
|
|
|
if (latest && parseInt(latest) < thread.latest) {
|
|
|
|
nextThread = c;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (nextThread) {
|
2024-08-04 17:48:51 +00:00
|
|
|
nextThread.insertAdjacentElement('beforebegin', li);
|
|
|
|
} else {
|
|
|
|
pl.append(li);
|
2024-07-16 19:43:35 +00:00
|
|
|
}
|
2024-07-05 01:27:29 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
populateThreadView() {
|
|
|
|
}
|
|
|
|
|
|
|
|
populateProfileView() {
|
|
|
|
}
|
|
|
|
|
|
|
|
buildThreads() {
|
2024-07-16 19:43:35 +00:00
|
|
|
const datastore = AdapterState._instance.data.get(this._name);
|
|
|
|
if (!datastore) {
|
|
|
|
util.errMsg(this._name + " has no datastore!");
|
|
|
|
return;
|
|
|
|
}
|
2024-07-05 01:27:29 +00:00
|
|
|
// make multiple passes over the store until every message is either
|
2024-08-31 17:01:31 +00:00
|
|
|
// placed in a thread, the boost carousel, or orphaned and waiting for its parent to be returned
|
2024-07-05 01:27:29 +00:00
|
|
|
do{
|
|
|
|
for (let k of datastore.messages.keys()) {
|
|
|
|
this.placeMsg(k);
|
|
|
|
}
|
|
|
|
} while (this._threads.reduce((sum: number, thread: MessageThread)=>{
|
|
|
|
return sum + thread.messageCount;
|
2024-08-31 17:01:31 +00:00
|
|
|
}, 0) + this._boosts.length + this._orphans.length < datastore.messages.size);
|
2024-07-05 01:27:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
placeMsg(k: string): string | null {
|
2024-07-16 19:43:35 +00:00
|
|
|
const datastore = AdapterState._instance.data.get(this._name);
|
|
|
|
if (!datastore) {
|
|
|
|
util.errMsg(this._name + " has no datastore!");
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
const msg = datastore.messages.get(k);
|
|
|
|
if (!msg) {
|
|
|
|
util.errMsg(`message [${this._name}:${k}] doesn't exist`);
|
|
|
|
return null;
|
|
|
|
}
|
2024-08-31 17:01:31 +00:00
|
|
|
if (msg.renoteId) {
|
|
|
|
// fetch the referent thread and put the boost in the carousel
|
|
|
|
this._convoyBatchTimer.queue(msg.renoteId, 2000);
|
|
|
|
if (!this._boosts.some(m=>m.id == msg.id)) {
|
|
|
|
this._boosts.push(msg);
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
2024-07-05 05:23:56 +00:00
|
|
|
for (let t of this._threads) {
|
|
|
|
// avoid processing nodes again on subsequent passes
|
2024-07-16 19:43:35 +00:00
|
|
|
if (!msg || t.findNode(t.root, msg.id)) {
|
2024-07-05 05:23:56 +00:00
|
|
|
return null;
|
|
|
|
}
|
2024-08-31 17:01:31 +00:00
|
|
|
|
2024-07-05 05:23:56 +00:00
|
|
|
if (msg.replyTo) {
|
|
|
|
let x = t.addReply(msg.replyTo, msg);
|
2024-07-05 01:27:29 +00:00
|
|
|
if (x) {
|
|
|
|
// after adding, we try to adopt some orphans
|
|
|
|
const orphanChildren = this._orphans.filter(m=>m.replyTo == k);
|
|
|
|
for (let o of orphanChildren) {
|
|
|
|
let adopted = this.placeMsg(o.id);
|
|
|
|
if (adopted) {
|
|
|
|
this._orphans.splice(this._orphans.indexOf(o), 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return t.root.data.id;
|
2024-07-05 05:23:56 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// if we made it this far, this message doesn't go in any existing thread
|
|
|
|
|
|
|
|
// if it doesn't have a parent, we can make a new thread with it
|
|
|
|
if (!msg.replyTo) {
|
2024-07-05 01:27:29 +00:00
|
|
|
this._threads.push(new MessageThread(msg));
|
2024-07-16 19:43:35 +00:00
|
|
|
// after adding, we try to adopt some orphans
|
|
|
|
const orphanChildren = this._orphans.filter(m=>m.replyTo == k);
|
|
|
|
for (let o of orphanChildren) {
|
|
|
|
let adopted = this.placeMsg(o.id);
|
|
|
|
if (adopted) {
|
|
|
|
this._orphans.splice(this._orphans.indexOf(o), 1);
|
|
|
|
}
|
|
|
|
}
|
2024-07-05 05:23:56 +00:00
|
|
|
return msg.id;
|
|
|
|
}
|
|
|
|
|
2024-07-06 18:43:25 +00:00
|
|
|
// then, we should check if its parent is an orphan
|
|
|
|
const orphanedParent = this._orphans.find(o=>o.id == msg.replyTo);
|
|
|
|
if (orphanedParent) {
|
|
|
|
// then, try to place them both
|
|
|
|
|
|
|
|
if (this.placeMsg(orphanedParent.id)) {
|
|
|
|
this._orphans.splice(this._orphans.indexOf(orphanedParent), 1);
|
2024-07-16 19:43:35 +00:00
|
|
|
return this.placeMsg(k);
|
2024-07-06 18:43:25 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-07-05 05:23:56 +00:00
|
|
|
// otherwise we can orphan it and try to fill it in later
|
|
|
|
if (this._orphans.filter(o=>o.id == msg.id).length == 0) {
|
|
|
|
this._orphans.push(msg);
|
2024-07-07 03:13:18 +00:00
|
|
|
if (msg.replyTo) {
|
2024-08-03 16:52:33 +00:00
|
|
|
this._convoyBatchTimer.queue(k, 2000);
|
2024-07-07 03:13:18 +00:00
|
|
|
}
|
|
|
|
|
2024-07-05 01:27:29 +00:00
|
|
|
}
|
2024-07-05 05:23:56 +00:00
|
|
|
return null;
|
2024-07-05 01:27:29 +00:00
|
|
|
}
|
|
|
|
}
|