Mastodon-Circles/create-circle.js

883 lines
24 KiB
JavaScript
Raw Normal View History

2023-07-19 17:16:22 -06:00
/**
*
* @param {RequestInfo | URL} url
* @param {{ body?: any } & RequestInit?} options
*/
async function apiRequest(url, options = null)
{
console.log(`Fetching :: ${url}`);
if (options && options.body) {
options.body = JSON.stringify(options.body);
}
2022-11-19 08:56:09 -07:00
2023-07-19 17:16:22 -06:00
return await fetch(url, options ?? {})
.then(response => {
if (response.ok) {
return response;
}
throw new Error(`Error fetching ${url}: ${response.status} ${response.statusText}`);
})
2023-07-19 17:16:22 -06:00
.then(response => response.json())
.catch(error => {
console.error(`Error fetching ${url}: ${error}`);
return null;
});
2022-11-19 08:56:09 -07:00
}
2023-08-31 12:49:04 -06:00
function Handle(name, instance) {
let handleObj = Object.create(Handle.prototype);
handleObj.name = name;
handleObj.instance = instance;
handleObj._baseInstance = null;
handleObj._apiInstance = null;
handleObj.profileUrl = null;
2023-08-31 12:49:04 -06:00
return handleObj;
2023-08-31 12:49:04 -06:00
}
2023-07-19 17:16:22 -06:00
Object.defineProperty(Handle.prototype, "baseInstance", {
get: function () {
return this._baseInstance || this.instance;
}
});
Object.defineProperty(Handle.prototype, "apiInstance", {
get: function () {
return this._apiInstance || this.instance;
}
});
Object.defineProperty(Handle.prototype, "baseHandle", {
get: function () {
return this.name + "@" + this.baseInstance;
}
});
Handle.prototype.toString = function () {
return this.name + "@" + this.instance;
};
2023-07-19 17:16:22 -06:00
/**
* @returns {Promise<Handle>} The handle WebFingered, or the original on fail
2023-07-19 17:16:22 -06:00
*/
Handle.prototype.webFinger = async function () {
if (this._baseInstance) {
return this;
}
let url = `https://${this.instance}/.well-known/webfinger?` + new URLSearchParams({
resource: `acct:${this}`
});
let webFinger = await apiRequest(url);
if (!webFinger)
return this;
let acct = webFinger["subject"];
if (typeof acct !== "string")
return this;
if (acct.startsWith("acct:")) {
acct = acct.substring("acct:".length);
}
let baseHandle = parseHandle(acct);
baseHandle._baseInstance = baseHandle.instance;
baseHandle.instance = this.instance;
const links = webFinger["links"];
if (!Array.isArray(links)) {
return baseHandle;
}
const selfLink = links.find(link => link["rel"] === "self");
if (!selfLink) {
return baseHandle;
}
try {
const url = new URL(selfLink["href"])
baseHandle._apiInstance = url.hostname;
} catch (e) {
console.error(`Error parsing WebFinger self link ${selfLink["href"]}: ${e}`);
}
const profileLink = links.find(link => link["rel"] === "http://webfinger.net/rel/profile-page");
if (profileLink?.["href"]) {
try {
baseHandle.profileUrl = new URL(profileLink["href"]);
} catch (e) {
console.error(`Error parsing WebFinger profile page link ${profileLink["href"]}: ${e}`);
}
}
return baseHandle;
};
2023-07-19 17:16:22 -06:00
/**
* @typedef {{
* id: string,
* avatar: string,
* bot: boolean,
* name: string,
* handle: Handle,
* }} FediUser
*/
2023-08-31 12:49:04 -06:00
/**
* @typedef {FediUser & {conStrength: number}} RatedUser
*/
2023-07-19 17:16:22 -06:00
/**
* @typedef {{
* id: string,
* replies: number,
* renotes: number,
* favorites: number,
2023-07-20 08:19:53 -06:00
* extra_reacts: boolean,
2023-07-19 17:16:22 -06:00
* instance: string,
* author?: FediUser,
* }} Note
*/
class ApiClient {
/**
* @param {string} instance
*/
constructor(instance) {
this._instance = instance;
}
/**
*
* @param instance
* @returns {Promise<ApiClient>}
*/
static async getClient(instance) {
if (instanceTypeCache.has(instance)) {
return instanceTypeCache.get(instance);
}
2023-07-30 13:56:16 -06:00
let url = `https://${instance}/.well-known/nodeinfo`;
2023-07-19 17:16:22 -06:00
let nodeInfo = await apiRequest(url);
if (!nodeInfo || !Array.isArray(nodeInfo.links)) {
const client = new MastodonApiClient(instance);
instanceTypeCache.set(instance, client);
return client;
}
const { links } = nodeInfo;
let apiLink = links.find(link => link.rel === "http://nodeinfo.diaspora.software/ns/schema/2.1");
if (!apiLink) {
apiLink = links.find(link => link.rel === "http://nodeinfo.diaspora.software/ns/schema/2.0");
}
if (!apiLink) {
console.error(`No NodeInfo API found for ${instance}}`);
const client = new MastodonApiClient(instance);
instanceTypeCache.set(instance, client);
return client;
}
let apiResponse = await apiRequest(apiLink.href);
if (!apiResponse) {
2023-07-30 13:56:16 -06:00
// Guess from API endpoints
const misskeyMeta = await apiRequest(`https://${instance}/api/meta`, {
method: "POST",
headers: {
"Content-Type": "application/json"
},
body: {}
});
if (misskeyMeta) {
const client = new MisskeyApiClient(instance);
instanceTypeCache.set(instance, client);
return client;
}
2023-07-19 17:16:22 -06:00
const client = new MastodonApiClient(instance);
instanceTypeCache.set(instance, client);
return client;
}
let { software } = apiResponse;
software.name = software.name.toLowerCase();
2023-07-19 17:29:50 -06:00
if (software.name.includes("misskey") ||
software.name.includes("calckey") ||
software.name.includes("foundkey") ||
software.name.includes("magnetar") ||
software.name.includes("firefish")) {
2023-07-19 17:16:22 -06:00
const client = new MisskeyApiClient(instance);
instanceTypeCache.set(instance, client);
return client;
}
2023-07-20 08:19:53 -06:00
let features = apiResponse?.metadata?.features;
if (Array.isArray(features) && features.includes("pleroma_api")) {
const has_emoji_reacts = features.includes("pleroma_emoji_reactions");
const client = new PleromaApiClient(instance, has_emoji_reacts);
instanceTypeCache.set(instance, client);
return client;
}
2023-07-19 17:16:22 -06:00
const client = new MastodonApiClient(instance);
instanceTypeCache.set(instance, client);
return client;
}
/**
* @param {Handle} handle
*
* @return {Promise<FediUser>}
*/
async getUserIdFromHandle(handle){ throw new Error("Not implemented"); }
/**
* @param {FediUser} user
*
* return {Promise<Note[]>}
*/
async getNotes(user){ throw new Error("Not implemented"); }
/**
* @param {Note} note
*
* return {Promise<FediUser[] | null>}
*/
async getRenotes(note){ throw new Error("Not implemented"); }
/**
* @param {Note} note
*
* return {Promise<Note[] | null>}
*/
async getReplies(note){ throw new Error("Not implemented"); }
/**
* @param {Note} note
2023-07-20 08:19:53 -06:00
* @param {boolean} extra_reacts
2023-07-19 17:16:22 -06:00
*
* return {Promise<FediUser[] | null>}
*/
2023-07-20 08:19:53 -06:00
async getFavs(note, extra_reacts) { throw new Error("Not implemented"); }
2023-07-19 17:52:39 -06:00
/**
* @return string
*/
getClientName() { throw new Error("Not implemented"); }
2022-11-19 08:56:09 -07:00
}
2023-07-19 17:16:22 -06:00
class MastodonApiClient extends ApiClient {
/**
* @param {string} instance
*/
constructor(instance) {
super(instance);
}
async getUserIdFromHandle(handle) {
const url = `https://${this._instance}/api/v1/accounts/lookup?acct=${handle.baseHandle}`;
let response = await apiRequest(url, null);
if (!response) {
const url = `https://${this._instance}/api/v1/accounts/lookup?acct=${handle}`;
response = await apiRequest(url, null);
}
2023-07-19 17:16:22 -06:00
if (!response) {
return null;
}
return {
id: response.id,
avatar: response.avatar,
bot: response.bot,
name: response["display_name"],
handle: handle,
};
}
async getNotes(user) {
const url = `https://${this._instance}/api/v1/accounts/${user.id}/statuses?exclude_replies=true&exclude_reblogs=true&limit=40`;
const response = await apiRequest(url, null);
if (!response) {
return null;
}
return response.map(note => ({
id: note.id,
replies: note["replies_count"] || 0,
renotes: note["reblogs_count"] || 0,
favorites: note["favourites_count"],
2023-07-20 08:19:53 -06:00
// Actually a Pleroma/Akkoma thing
extra_reacts: note?.["pleroma"]?.["emoji_reactions"]?.length > 0,
2023-07-19 17:16:22 -06:00
instance: this._instance,
author: user
}));
}
async getRenotes(note) {
const url = `https://${this._instance}/api/v1/statuses/${note.id}/reblogged_by`;
const response = await apiRequest(url);
if (!response) {
return null;
}
return response.map(user => ({
id: user.id,
avatar: user.avatar,
bot: user.bot,
name: user["display_name"],
handle: parseHandle(user["acct"], note.instance)
}));
}
async getReplies(noteIn) {
const url = `https://${this._instance}/api/v1/statuses/${noteIn.id}/context`;
const response = await apiRequest(url);
if (!response) {
return null;
}
return response["descendants"].map(note => {
2023-07-19 17:16:22 -06:00
let handle = parseHandle(note["account"]["acct"], noteIn.instance);
return {
id: note.id,
replies: note["replies_count"] || 0,
renotes: note["reblogs_count"] || 0,
favorites: note["favourites_count"],
2023-07-20 08:19:53 -06:00
// Actually a Pleroma/Akkoma thing
extra_reacts: note?.["pleroma"]?.["emoji_reactions"]?.length > 0,
2023-07-19 17:16:22 -06:00
instance: handle.instance,
author: {
id: note["account"]["id"],
bot: note["account"]["bot"],
name: note["account"]["display_name"],
avatar: note["account"]["avatar"],
handle: handle
}
};
});
}
2023-07-20 08:19:53 -06:00
async getFavs(note, extra_reacts) {
2023-07-19 17:16:22 -06:00
const url = `https://${this._instance}/api/v1/statuses/${note.id}/favourited_by`;
const response = await apiRequest(url);
if (!response) {
return null;
}
return response.map(user => ({
id: user.id,
avatar: user.avatar,
bot: user.bot,
name: user["display_name"],
handle: parseHandle(user["acct"], note.instance)
}));
}
2023-07-19 17:52:39 -06:00
getClientName() {
return "mastodon";
}
2022-11-19 08:56:09 -07:00
}
2023-07-20 08:19:53 -06:00
class PleromaApiClient extends MastodonApiClient {
/**
* @param {string} instance
* @param {boolean} emoji_reacts
*/
constructor(instance, emoji_reacts) {
super(instance);
this._emoji_reacts = emoji_reacts;
}
async getFavs(note, extra_reacts) {
// Pleroma/Akkoma supports both favs and emoji reacts
// with several emoji reacts per users being possible.
// Coalesce them and count every user only once
let favs = await super.getFavs(note);
if (!this._emoji_reacts || !extra_reacts)
return favs;
/**
* @type {Map<string, FediUser>}
*/
let users = new Map();
if (favs !== null) {
favs.forEach(u => {
users.set(u.id, u);
});
}
const url = `https://${this._instance}/api/v1/pleroma/statuses/${note.id}/reactions`;
const response = await apiRequest(url) ?? [];
for (const reaction of response) {
reaction["accounts"]
.map(account => ({
id: account["id"],
avatar: account["avatar"],
bot: account["bot"],
name: account["display_name"],
handle: parseHandle(account["acct"], note.instance)
}))
.forEach(u => {
if(!users.has(u.id))
users.set(u.id, u);
})
}
return Array.from(users.values());
}
getClientName() {
return "pleroma";
}
}
2023-07-19 17:16:22 -06:00
class MisskeyApiClient extends ApiClient {
/**
* @param {string} instance
*/
constructor(instance) {
super(instance);
}
async getUserIdFromHandle(handle) {
2023-07-30 13:56:16 -06:00
const lookupUrl = `https://${this._instance}/api/users/search-by-username-and-host`;
const lookup = await apiRequest(lookupUrl, {
method: "POST",
headers: {
"Content-Type": "application/json"
},
body: {
username: handle.name,
host: null
}
});
let id = null;
2023-07-30 14:05:08 -06:00
for (const user of Array.isArray(lookup) ? lookup : []) {
const isLocal = user?.["host"] === handle.instance ||
user?.["host"] === handle.baseInstance ||
this._instance === handle.apiInstance && user?.["host"] === null;
if (isLocal && user?.["username"] === handle.name && user["id"]) {
2023-07-30 13:56:16 -06:00
id = user["id"];
break;
}
}
2023-07-19 17:16:22 -06:00
const url = `https://${this._instance}/api/users/show`;
const response = await apiRequest(url, {
method: "POST",
2023-07-30 13:56:16 -06:00
headers: {
"Content-Type": "application/json"
},
2023-07-19 17:16:22 -06:00
body: {
2023-07-30 13:56:16 -06:00
id: id ? id : undefined,
2023-07-19 17:16:22 -06:00
username: handle.name
}
});
if (!response) {
return null;
}
2022-11-19 08:56:09 -07:00
2023-07-19 17:16:22 -06:00
return {
id: response.id,
avatar: response["avatarUrl"],
bot: response["isBot"],
name: response["name"],
handle: handle,
};
2022-11-19 08:56:09 -07:00
}
2022-11-20 05:03:18 -07:00
2023-07-19 17:16:22 -06:00
async getNotes(user) {
const url = `https://${this._instance}/api/users/notes`;
const response = await apiRequest(url, {
method: "POST",
2023-07-30 13:56:16 -06:00
headers: {
"Content-Type": "application/json"
},
2023-07-19 17:16:22 -06:00
body: {
userId: user.id,
limit: 70,
reply: false,
renote: false,
}
});
if (!response) {
return null;
}
return response.map(note => ({
id: note.id,
replies: note["repliesCount"],
renotes: note["renoteCount"],
favorites: Object.values(note["reactions"]).reduce((a, b) => a + b, 0),
2023-07-20 08:19:53 -06:00
extra_reacts: false,
2023-07-19 17:16:22 -06:00
instance: this._instance,
author: user
}));
}
async getRenotes(note) {
const url = `https://${this._instance}/api/notes/renotes`;
const response = await apiRequest(url, {
method: "POST",
2023-07-30 13:56:16 -06:00
headers: {
"Content-Type": "application/json"
},
2023-07-19 17:16:22 -06:00
body: {
noteId: note.id,
limit: 50,
}
});
if (!response) {
return null;
}
return response.map(renote => ({
id: renote["user"]["id"],
avatar: renote["user"]["avatarUrl"],
bot: renote["user"]["isBot"] || false,
name: renote["user"]["name"],
2023-07-31 03:34:11 -06:00
handle: parseHandle(renote["user"]["username"], renote["user"]["host"] ?? this._instance)
2023-07-19 17:16:22 -06:00
}));
}
async getReplies(note) {
const url = `https://${this._instance}/api/notes/replies`;
const response = await apiRequest(url, {
method: "POST",
2023-07-30 13:56:16 -06:00
headers: {
"Content-Type": "application/json"
},
2023-07-19 17:16:22 -06:00
body: {
noteId: note.id,
limit: 100,
}
});
if (!response) {
return null;
}
return response.map(reply => {
2023-07-31 03:34:11 -06:00
const handle = parseHandle(reply["user"]["username"], reply["user"]["host"] ?? this._instance);
2023-07-19 17:16:22 -06:00
return {
id: reply.id,
replies: reply["repliesCount"],
renotes: reply["renoteCount"],
favorites: Object.values(reply["reactions"]).reduce((a, b) => a + b, 0),
2023-07-20 08:19:53 -06:00
extra_reacts: false,
2023-07-19 17:16:22 -06:00
instance: handle.instance,
author: {
id: reply["user"]["id"],
avatar: reply["user"]["avatarUrl"],
bot: reply["user"]["isBot"] || false,
name: reply["user"]["name"],
handle: handle
}
};
});
}
2023-07-20 08:19:53 -06:00
async getFavs(note, extra_reacts) {
2023-07-19 17:16:22 -06:00
const url = `https://${this._instance}/api/notes/reactions`;
const response = await apiRequest(url, {
method: "POST",
2023-07-30 13:56:16 -06:00
headers: {
"Content-Type": "application/json"
},
2023-07-19 17:16:22 -06:00
body: {
noteId: note.id,
limit: 100,
}
});
if (!response) {
return null;
}
return response.map(reaction => ({
id: reaction["user"]["id"],
avatar: reaction["user"]["avatarUrl"],
bot: reaction["user"]["isBot"] || false,
name: reaction["user"]["name"],
2023-07-31 03:34:11 -06:00
handle: parseHandle(reaction["user"]["username"], reaction["user"]["host"] ?? this._instance),
2023-07-19 17:16:22 -06:00
}));
}
2023-07-19 17:52:39 -06:00
getClientName() {
return "misskey";
}
2022-11-19 08:56:09 -07:00
}
2023-08-31 12:49:04 -06:00
/** @type {Map<string, ApiClient>} */
2023-07-19 17:16:22 -06:00
let instanceTypeCache = new Map();
/**
* @param {string} fediHandle
* @param {string} fallbackInstance
*
* @returns {Handle}
*/
function parseHandle(fediHandle, fallbackInstance = "") {
if (fediHandle.charAt(0) === '@')
fediHandle = fediHandle.substring(1);
fediHandle = fediHandle.replaceAll(" ", "");
const [name, instance] = fediHandle.split("@", 2);
2023-08-31 12:49:04 -06:00
return new Handle(name, instance || fallbackInstance);
2022-11-19 08:56:09 -07:00
}
2023-07-19 17:16:22 -06:00
async function circleMain() {
2023-07-19 17:52:39 -06:00
let progress = document.getElementById("outInfo");
2023-07-19 17:16:22 -06:00
2023-07-19 17:52:39 -06:00
const generateBtn = document.getElementById("generateButton");
2023-07-19 17:16:22 -06:00
2023-07-19 17:52:39 -06:00
generateBtn.style.display = "none";
let fediHandle = document.getElementById("txt_mastodon_handle");
const selfUser = await parseHandle(fediHandle.value).webFinger();
2023-07-19 17:52:39 -06:00
let form = document.getElementById("generateForm");
let backend = form.backend;
for (const radio of backend) {
radio.disabled = true;
}
fediHandle.disabled = true;
let client;
switch (backend.value) {
case "mastodon":
client = new MastodonApiClient(selfUser.apiInstance);
2023-07-19 17:52:39 -06:00
break;
2023-07-20 08:19:53 -06:00
case "pleroma":
client = new PleromaApiClient(selfUser.apiInstance, true);
2023-07-20 08:19:53 -06:00
break;
2023-07-19 17:52:39 -06:00
case "misskey":
client = new MisskeyApiClient(selfUser.apiInstance);
2023-07-19 17:52:39 -06:00
break;
default:
progress.innerText = "Detecting instance...";
client = await ApiClient.getClient(selfUser.apiInstance);
2023-07-19 17:52:39 -06:00
backend.value = client.getClientName();
break;
}
2023-07-19 17:16:22 -06:00
progress.innerText = "Fetching your user...";
const user = await client.getUserIdFromHandle(selfUser);
2022-11-19 09:53:53 -07:00
2023-07-19 17:16:22 -06:00
if (!user) {
alert("Something went horribly wrong, couldn't fetch your user.");
2023-07-19 17:52:39 -06:00
fediHandle.disabled = false;
for (const radio of backend) {
radio.disabled = false;
}
generateBtn.style.display = "inline";
progress.innerText = "";
2023-07-19 17:16:22 -06:00
return;
2022-11-19 09:53:53 -07:00
}
2022-11-19 12:01:33 -07:00
2023-07-19 17:16:22 -06:00
progress.innerText = "Fetching your latest posts...";
2022-11-19 09:53:53 -07:00
2023-07-19 17:16:22 -06:00
const notes = await client.getNotes(user);
if (!notes) {
alert("Something went horribly wrong, couldn't fetch your notes.");
return;
2022-11-19 08:56:09 -07:00
}
2022-11-19 12:01:33 -07:00
2023-07-19 17:16:22 -06:00
/**
* @type {Map<string, RatedUser>}
*/
let connectionList = new Map();
await processNotes(client, connectionList, notes);
showConnections(user, connectionList);
2022-11-19 08:56:09 -07:00
}
2023-07-19 17:16:22 -06:00
/**
* @param {ApiClient} client
* @param {Map<string, RatedUser>} connectionList
* @param {Note[]} notes
*/
async function processNotes(client, connectionList, notes) {
let progress = document.getElementById("outInfo");
let counter = 0;
let total = notes.length;
2022-11-19 08:56:09 -07:00
2023-07-19 17:16:22 -06:00
for (const note of notes) {
progress.innerText = `Processing :3 (${counter}/${total}) `;
await evaluateNote(client, connectionList, note);
counter++;
2022-11-19 08:56:09 -07:00
}
}
2023-07-19 17:16:22 -06:00
/**
@param {ApiClient} client
* @param {Map<string, RatedUser>} connectionList
* @param {Note} note
*/
async function evaluateNote(client, connectionList, note) {
2023-07-20 08:19:53 -06:00
if (note.favorites > 0 || note.extra_reacts) {
await client.getFavs(note, note.extra_reacts).then(users => {
2023-07-19 17:16:22 -06:00
if (!users)
return;
users.forEach(user => {
incConnectionValue(connectionList, user, 1.0);
});
}).catch(() => {});
}
if (note.renotes > 0) {
await client.getRenotes(note).then(users => {
if (!users)
return;
users.forEach(user => {
incConnectionValue(connectionList, user, 1.3);
});
}).catch(() => {});
}
await client.getReplies(note).then(replies => {
if (!replies)
return [];
replies.forEach(reply => {
incConnectionValue(connectionList, reply.author, 1.1);
});
return replies;
}).catch(() => {});
2022-11-19 08:56:09 -07:00
}
2023-07-19 17:16:22 -06:00
/**
* @param {Map<string, RatedUser>} connectionList
* @param {FediUser} user
* @param {number} plus
*/
function incConnectionValue(connectionList, user, plus) {
if (user.bot)
return;
2022-11-19 08:56:09 -07:00
2023-07-19 17:16:22 -06:00
if (!connectionList.has(user.id)) {
connectionList.set(user.id, {
conStrength: 0,
...user
});
}
2022-11-19 08:56:09 -07:00
2023-07-19 17:16:22 -06:00
connectionList.get(user.id).conStrength += plus;
}
/**
* @param {FediUser} localUser
* @param {Map<string, RatedUser>} connectionList
*/
function showConnections(localUser, connectionList) {
if (connectionList.has(localUser.id))
connectionList.delete(localUser.id);
2022-11-19 09:53:53 -07:00
// Sort dict into Array items
2023-07-19 17:16:22 -06:00
const items = [...connectionList.values()].sort((first, second) => second.conStrength - first.conStrength);
2022-11-20 06:51:58 -07:00
// Also export the Username List
2023-07-19 17:16:22 -06:00
let usersDivs = [
document.getElementById("ud1"),
document.getElementById("ud2"),
document.getElementById("ud3")
];
usersDivs.forEach((div) => div.innerHTML = "")
2023-08-31 15:41:36 -06:00
const [inner, middle, outer] = usersDivs;
inner.innerHTML = "<div><h3>Inner Circle</h3></div>";
middle.innerHTML = "<div><h3>Middle Circle</h3></div>";
outer.innerHTML = "<div><h3>Outer Circle</h3></div>";
for (let i= 0; i < items.length; i++) {
const newUser = document.createElement("a");
newUser.className = "userItem";
newUser.innerText = items[i].handle.name;
newUser.title = items[i].name;
// I'm so sorry
newUser.href = "javascript:void(0)";
2023-08-31 15:41:36 -06:00
const handle = items[i].handle;
newUser.onclick = async () => {
const fingeredHandle = await handle.webFinger();
if (fingeredHandle.profileUrl)
window.open(fingeredHandle.profileUrl, "_blank");
else
alert("Could not the profile URL for " + fingeredHandle.baseHandle);
};
2023-08-31 15:41:36 -06:00
const newUserHost = document.createElement("span");
newUserHost.className = "userHost";
newUserHost.innerText = "@" + items[i].handle.instance;
newUser.appendChild(newUserHost);
const newUserImg = document.createElement("img");
newUserImg.src = items[i].avatar;
newUserImg.title = newUserImg.alt = stripName(items[i].name || items[i].handle.name) + "'s avatar";
newUserImg.className = "userImg";
newUserImg.onerror = () => {
newUserImg.alt = "";
};
2023-08-31 15:41:36 -06:00
newUser.prepend(newUserImg);
2023-07-19 17:16:22 -06:00
2022-11-20 06:51:58 -07:00
let udNum = 0;
if (i > numb[0]) udNum = 1;
2023-07-19 17:16:22 -06:00
if (i > numb[0] + numb[1]) udNum = 2;
2022-11-20 06:51:58 -07:00
usersDivs[udNum].appendChild(newUser);
2022-11-20 05:03:18 -07:00
}
2023-08-31 15:41:36 -06:00
usersDivs.forEach((div) => {
const items = div.querySelectorAll(".userItem");
2022-11-19 08:56:09 -07:00
2023-08-31 15:41:36 -06:00
for (let i = 0; i < items.length - 1; i++) {
const item = items[i];
item.innerHTML += ", ";
}
});
const outDiv = document.getElementById("outDiv");
outDiv.style.display = "block";
document.getElementById("outSelfUser").innerText = stripName(localUser.name || localUser.handle.name);
render(items, localUser);
2022-11-19 08:56:09 -07:00
}
2023-08-31 15:41:36 -06:00
function stripName(name) {
return name.replaceAll(/:[a-zA-Z0-9_]+:/g, "").trim();
}