gamja/components/app.js

1302 lines
32 KiB
JavaScript
Raw Normal View History

import * as irc from "../lib/irc.js";
import Client from "../lib/client.js";
import Buffer from "./buffer.js";
import BufferList from "./buffer-list.js";
import BufferHeader from "./buffer-header.js";
import MemberList from "./member-list.js";
import ConnectForm from "./connect-form.js";
import JoinForm from "./join-form.js";
import Help from "./help.js";
2021-03-08 12:15:04 -05:00
import NetworkForm from "./network-form.js";
import Composer from "./composer.js";
import ScrollManager from "./scroll-manager.js";
2021-03-08 10:23:16 -05:00
import Dialog from "./dialog.js";
import { html, Component, createRef } from "../lib/index.js";
import { strip as stripANSI } from "../lib/ansi.js";
2021-06-04 12:03:03 -04:00
import { SERVER_BUFFER, BufferType, ReceiptType, ServerStatus, Unread, State } from "../state.js";
import commands from "../commands.js";
import { setup as setupKeybindings } from "../keybindings.js";
import * as store from "../store.js";
2020-06-26 04:35:38 -04:00
const baseConfig = {
server: {},
};
const configPromise = fetch("./config.json")
.then((resp) => {
if (resp.ok) {
return resp.json();
}
if (resp.status !== 404) {
console.error("Failed to fetch config: HTTP error:", resp.status, resp.statusText);
}
return {};
})
.catch((err) => {
console.error("Failed to fetch config:", err);
return {};
})
.then((config) => {
return {
...baseConfig,
...config,
};
});
2020-07-15 12:21:09 -04:00
const CHATHISTORY_MAX_SIZE = 4000;
function parseQueryString() {
2021-06-10 12:11:11 -04:00
let query = window.location.search.substring(1);
let params = {};
query.split('&').forEach((s) => {
if (!s) {
return;
}
2021-06-10 12:11:11 -04:00
let pair = s.split('=');
params[decodeURIComponent(pair[0])] = decodeURIComponent(pair[1] || "");
});
return params;
}
function fillConnectParams(params) {
2021-06-10 12:11:11 -04:00
let host = window.location.host || "localhost:8080";
let proto = "wss:";
if (window.location.protocol != "https:") {
proto = "ws:";
}
2021-06-10 12:11:11 -04:00
let path = window.location.pathname || "/";
if (!window.location.host) {
path = "/";
}
params = { ...params };
if (!params.url) {
params.url = proto + "//" + host + path + "socket";
}
if (params.url.startsWith("/")) {
params.url = proto + "//" + host + params.url;
}
if (params.url.indexOf("://") < 0) {
params.url = proto + "//" + params.url;
}
if (!params.username) {
params.username = params.nick;
}
if (!params.realname) {
params.realname = params.nick;
}
return params;
}
2020-07-15 12:21:09 -04:00
function debounce(f, delay) {
2021-06-10 12:11:11 -04:00
let timeout = null;
2020-07-15 12:21:09 -04:00
return (...args) => {
clearTimeout(timeout);
timeout = setTimeout(() => {
timeout = null;
f(...args);
}, delay);
};
}
function showNotification(title, options) {
if (!window.Notification || Notification.permission !== "granted") {
return new EventTarget();
}
// This can still fail due to:
// https://bugs.chromium.org/p/chromium/issues/detail?id=481856
try {
return new Notification(title, options);
} catch (err) {
console.error("Failed to show notification: ", err);
return new EventTarget();
}
}
export default class App extends Component {
state = {
connectParams: {
2021-05-31 12:04:02 -04:00
url: null,
pass: null,
username: null,
realname: null,
nick: null,
saslPlain: null,
autoconnect: false,
autojoin: [],
},
servers: new Map(),
buffers: new Map(),
bouncerNetworks: new Map(),
activeBuffer: null,
connectForm: true,
2021-03-08 10:23:16 -05:00
dialog: null,
error: null,
openPanels: {
bufferList: false,
memberList: false,
},
};
config = { ...baseConfig };
2021-01-22 11:36:53 -05:00
clients = new Map();
2020-06-29 03:06:47 -04:00
endOfHistory = new Map();
2020-07-15 12:21:09 -04:00
receipts = new Map();
buffer = createRef();
composer = createRef();
2021-05-25 06:40:33 -04:00
switchToChannel = null;
constructor(props) {
super(props);
this.handleConnectSubmit = this.handleConnectSubmit.bind(this);
2021-03-08 10:23:16 -05:00
this.handleJoinSubmit = this.handleJoinSubmit.bind(this);
this.handleBufferListClick = this.handleBufferListClick.bind(this);
this.toggleBufferList = this.toggleBufferList.bind(this);
this.toggleMemberList = this.toggleMemberList.bind(this);
this.handleComposerSubmit = this.handleComposerSubmit.bind(this);
2021-05-31 22:39:35 -04:00
this.handleChannelClick = this.handleChannelClick.bind(this);
2020-06-25 12:45:41 -04:00
this.handleNickClick = this.handleNickClick.bind(this);
2020-06-29 06:36:17 -04:00
this.autocomplete = this.autocomplete.bind(this);
2020-06-29 03:06:47 -04:00
this.handleBufferScrollTop = this.handleBufferScrollTop.bind(this);
2021-03-08 10:23:16 -05:00
this.handleDialogDismiss = this.handleDialogDismiss.bind(this);
2021-03-08 12:15:04 -05:00
this.handleAddNetworkClick = this.handleAddNetworkClick.bind(this);
2021-03-09 13:10:22 -05:00
this.handleNetworkSubmit = this.handleNetworkSubmit.bind(this);
this.handleNetworkRemove = this.handleNetworkRemove.bind(this);
this.dismissError = this.dismissError.bind(this);
2020-07-15 12:21:09 -04:00
this.saveReceipts = debounce(this.saveReceipts.bind(this), 500);
this.receipts = store.receipts.load();
2020-07-01 06:25:57 -04:00
configPromise.then((config) => {
this.handleConfig(config);
return config;
});
}
/**
* Handle configuration data and populate the connection parameters.
*
* The priority order is:
*
* - URL params
* - Saved parameters in local storage
* - Configuration data (fetched from the config.json file)
* - Default server URL constructed from the current URL location
*/
handleConfig(config) {
2021-06-10 12:11:11 -04:00
let connectParams = {};
if (config.server) {
2021-05-31 12:04:02 -04:00
connectParams.url = config.server.url;
if (Array.isArray(config.server.autojoin)) {
connectParams.autojoin = config.server.autojoin;
} else if (config.server.autojoin) {
connectParams.autojoin = [config.server.autojoin];
}
}
2021-06-10 12:11:11 -04:00
let autoconnect = store.autoconnect.load();
if (autoconnect) {
connectParams = {
...connectParams,
...autoconnect,
autoconnect: true,
};
}
2021-06-10 12:11:11 -04:00
let queryParams = parseQueryString();
if (typeof queryParams.server === "string") {
connectParams.url = queryParams.server;
// When using a custom server, some configuration options don't
// make sense anymore.
config.server.auth = null;
}
if (typeof queryParams.nick === "string") {
connectParams.nick = queryParams.nick;
}
if (typeof queryParams.channels == "string") {
connectParams.autojoin = queryParams.channels.split(",");
2020-07-15 12:21:09 -04:00
}
if (window.location.hash) {
connectParams.autojoin = window.location.hash.split(",");
}
this.config = config;
this.setState((state) => {
return {
connectParams: {
...state.connectParams,
...connectParams,
},
};
});
if (connectParams.autoconnect) {
this.setState({ connectForm: false });
this.connect(connectParams);
}
}
dismissError(event) {
event.preventDefault();
this.setState({ error: null });
}
setServerState(id, updater, callback) {
2021-01-21 13:01:50 -05:00
this.setState((state) => {
2021-06-04 12:03:03 -04:00
return State.updateServer(state, id, updater);
2021-01-21 13:01:50 -05:00
}, callback);
}
2021-01-21 14:41:44 -05:00
setBufferState(id, updater, callback) {
this.setState((state) => {
2021-06-04 12:03:03 -04:00
return State.updateBuffer(state, id, updater);
}, callback);
}
createBuffer(serverID, name) {
2021-06-10 12:11:11 -04:00
let id = null;
this.setState((state) => {
2021-06-10 12:11:11 -04:00
let client = this.clients.get(serverID);
let updated;
2021-06-04 12:37:34 -04:00
[id, updated] = State.createBuffer(state, name, serverID, client);
return updated;
});
return id;
}
2021-01-21 16:15:33 -05:00
switchBuffer(id) {
2021-06-10 12:11:11 -04:00
let buf;
2021-01-21 14:41:44 -05:00
this.setState((state) => {
2021-06-04 12:03:03 -04:00
buf = State.getBuffer(state, id);
2021-01-21 14:41:44 -05:00
if (!buf) {
return;
}
return { activeBuffer: buf.id };
}, () => {
2021-01-21 16:15:33 -05:00
if (!buf) {
return;
}
2021-06-10 12:11:11 -04:00
let lastReadReceipt = this.getReceipt(buf.name, ReceiptType.READ);
2021-01-21 16:15:33 -05:00
// TODO: only mark as read if user scrolled at the bottom
this.setBufferState(buf.id, {
unread: Unread.NONE,
lastReadReceipt,
});
if (this.buffer.current) {
this.buffer.current.focus();
}
2020-07-15 12:21:09 -04:00
2021-01-21 16:15:33 -05:00
if (buf.messages.length == 0) {
2020-07-15 12:21:09 -04:00
return;
}
2021-06-10 12:11:11 -04:00
let lastMsg = buf.messages[buf.messages.length - 1];
2021-01-21 16:15:33 -05:00
this.setReceipt(buf.name, ReceiptType.READ, lastMsg);
2020-07-15 12:21:09 -04:00
});
}
saveReceipts() {
store.receipts.put(this.receipts);
2020-07-15 12:21:09 -04:00
}
getReceipt(target, type) {
2021-06-10 12:11:11 -04:00
let receipts = this.receipts.get(target);
2020-07-15 12:21:09 -04:00
if (!receipts) {
return undefined;
}
return receipts[type];
}
hasReceipt(target, type, msg) {
2021-06-10 12:11:11 -04:00
let receipt = this.getReceipt(target, type);
2020-07-15 12:21:09 -04:00
return receipt && msg.tags.time <= receipt.time;
}
setReceipt(target, type, msg) {
2021-06-10 12:11:11 -04:00
let receipt = this.getReceipt(target, type);
2020-07-15 12:21:09 -04:00
if (this.hasReceipt(target, type, msg)) {
return;
}
// TODO: this doesn't trigger a redraw
2020-07-15 12:21:09 -04:00
this.receipts.set(target, {
...this.receipts.get(target),
[type]: { time: msg.tags.time },
});
2020-07-15 12:21:09 -04:00
this.saveReceipts();
}
latestReceipt(type) {
2021-06-10 12:11:11 -04:00
let last = null;
this.receipts.forEach((receipts, target) => {
2021-06-10 12:11:11 -04:00
let delivery = receipts[type];
if (target == "*" || !delivery || !delivery.time) {
return;
}
if (!last || delivery.time > last.time) {
last = delivery;
}
});
return last;
}
addMessage(serverID, bufName, msg) {
2021-06-10 12:11:11 -04:00
let client = this.clients.get(serverID);
2021-01-22 11:36:53 -05:00
2021-06-23 13:52:45 -04:00
// Treat server-wide broadcasts as highlights. They're sent by server
// operators and can contain important information.
msg.isHighlight = irc.isHighlight(msg, client.nick, client.cm) || irc.isServerBroadcast(msg);
if (!msg.tags) {
msg.tags = {};
}
2020-06-29 03:06:47 -04:00
if (!msg.tags.time) {
msg.tags.time = irc.formatDate(new Date());
}
2021-06-10 12:11:11 -04:00
let isDelivered = this.hasReceipt(bufName, ReceiptType.DELIVERED, msg);
let isRead = this.hasReceipt(bufName, ReceiptType.READ, msg);
2020-07-15 12:21:09 -04:00
// TODO: messages coming from infinite scroll shouldn't trigger notifications
2021-06-24 12:04:26 -04:00
if (client.isMyNick(msg.prefix.name)) {
isRead = true;
}
2021-06-10 12:11:11 -04:00
let msgUnread = Unread.NONE;
2020-07-15 12:21:09 -04:00
if ((msg.command == "PRIVMSG" || msg.command == "NOTICE") && !isRead) {
2021-06-10 12:57:57 -04:00
let target = msg.params[0];
2021-06-10 12:11:11 -04:00
let text = msg.params[1];
2020-06-29 05:50:42 -04:00
2021-06-10 12:11:11 -04:00
let kind;
if (msg.isHighlight) {
2020-06-29 05:08:47 -04:00
msgUnread = Unread.HIGHLIGHT;
2020-06-29 05:50:42 -04:00
kind = "highlight";
2021-06-10 12:57:57 -04:00
} else if (client.isMyNick(target)) {
2020-06-29 05:50:42 -04:00
msgUnread = Unread.HIGHLIGHT;
kind = "private message";
2020-06-29 05:08:47 -04:00
} else {
msgUnread = Unread.MESSAGE;
}
2020-06-29 05:50:42 -04:00
if (msgUnread == Unread.HIGHLIGHT && !isDelivered && !irc.parseCTCP(msg)) {
2021-06-10 12:11:11 -04:00
let title = "New " + kind + " from " + msg.prefix.name;
if (client.isChannel(bufName)) {
title += " in " + bufName;
2020-06-29 05:50:42 -04:00
}
2021-06-10 12:11:11 -04:00
let notif = showNotification(title, {
body: stripANSI(text),
2020-06-29 05:50:42 -04:00
requireInteraction: true,
tag: "msg," + msg.prefix.name + "," + bufName,
2020-06-29 05:50:42 -04:00
});
notif.addEventListener("click", () => {
// TODO: scroll to message
this.switchBuffer({ server: serverID, name: bufName });
2020-06-29 05:50:42 -04:00
});
}
2020-06-24 10:56:28 -04:00
}
if (msg.command === "INVITE" && client.isMyNick(msg.params[0])) {
msgUnread = Unread.HIGHLIGHT;
2021-06-10 12:11:11 -04:00
let channel = msg.params[1];
let notif = new Notification("Invitation to " + channel, {
body: msg.prefix.name + " has invited you to " + channel,
requireInteraction: true,
tag: "invite," + msg.prefix.name + "," + channel,
actions: [{
action: "accept",
title: "Accept",
}],
});
notif.addEventListener("click", (event) => {
if (event.action === "accept") {
this.setReceipt(bufName, ReceiptType.READ, msg);
this.open(channel, serverID);
} else {
// TODO: scroll to message
this.switchBuffer({ server: serverID, name: bufName });
}
});
}
2020-06-24 10:56:28 -04:00
if (!client.isMyNick(msg.prefix.name) && (msg.command != "PART" && msg.comand != "QUIT")) {
this.createBuffer(serverID, bufName);
}
2020-07-15 12:21:09 -04:00
this.setReceipt(bufName, ReceiptType.DELIVERED, msg);
2021-06-10 12:11:11 -04:00
let bufID = { server: serverID, name: bufName };
2021-06-04 13:07:14 -04:00
this.setState((state) => State.addMessage(state, msg, bufID));
this.setBufferState(bufID, (buf) => {
2020-07-15 12:21:09 -04:00
// TODO: set unread if scrolled up
2021-06-10 12:11:11 -04:00
let unread = buf.unread;
let lastReadReceipt = buf.lastReadReceipt;
if (this.state.activeBuffer != buf.id) {
2020-06-24 11:46:43 -04:00
unread = Unread.union(unread, msgUnread);
2020-07-15 12:21:09 -04:00
} else {
this.setReceipt(bufName, ReceiptType.READ, msg);
lastReadReceipt = this.getReceipt(bufName, ReceiptType.READ);
2020-06-24 10:56:28 -04:00
}
2021-06-04 13:07:14 -04:00
return { unread, lastReadReceipt };
});
}
2021-01-22 12:44:06 -05:00
connect(params) {
2021-06-10 12:11:11 -04:00
let serverID = null;
2021-01-21 13:01:50 -05:00
this.setState((state) => {
2021-06-10 12:11:11 -04:00
let update;
2021-06-10 04:54:33 -04:00
[serverID, update] = State.createServer(state);
return update;
2021-01-21 13:01:50 -05:00
});
this.setState({ connectParams: params });
2021-06-10 12:11:11 -04:00
let client = new Client(fillConnectParams(params));
this.clients.set(serverID, client);
2021-06-10 04:54:33 -04:00
this.setServerState(serverID, { status: client.status });
2021-01-22 11:36:53 -05:00
2021-01-22 12:29:22 -05:00
client.addEventListener("status", () => {
this.setServerState(serverID, { status: client.status });
if (client.status === Client.Status.REGISTERED) {
this.setState({ connectForm: false });
}
2021-01-21 13:01:50 -05:00
});
2021-01-22 11:36:53 -05:00
client.addEventListener("message", (event) => {
this.handleMessage(serverID, event.detail.message);
});
2021-01-22 11:36:53 -05:00
client.addEventListener("error", (event) => {
this.setState({ error: event.detail });
});
this.createBuffer(serverID, SERVER_BUFFER);
if (!this.state.activeBuffer) {
this.switchBuffer({ server: serverID, name: SERVER_BUFFER });
}
2021-05-25 06:40:33 -04:00
if (params.autojoin.length > 0) {
this.switchToChannel = params.autojoin[0];
}
2021-05-27 12:17:41 -04:00
if (this.config.server && typeof this.config.server.ping !== "undefined") {
client.setPingInterval(this.config.server.ping);
2021-05-27 12:17:41 -04:00
}
}
disconnect(serverID) {
if (!serverID) {
2021-06-04 12:03:03 -04:00
serverID = State.getActiveServerID(this.state);
}
2021-06-10 12:11:11 -04:00
let client = this.clients.get(serverID);
2021-01-22 11:36:53 -05:00
if (client) {
this.clients.delete(serverID);
2021-01-22 12:29:22 -05:00
client.disconnect();
2021-01-12 04:35:38 -05:00
}
}
reconnect(serverID) {
if (!serverID) {
2021-06-04 12:03:03 -04:00
serverID = State.getActiveServerID(this.state);
}
2021-06-10 12:11:11 -04:00
let client = this.clients.get(serverID);
2021-01-22 12:44:06 -05:00
if (client) {
client.reconnect();
}
2021-01-12 04:35:38 -05:00
}
serverFromBouncerNetwork(bouncerNetworkID) {
2021-06-10 12:11:11 -04:00
for (let [id, client] of this.clients) {
if (client.params.bouncerNetwork === bouncerNetworkID) {
return id;
}
}
return null;
}
handleMessage(serverID, msg) {
2021-06-10 12:11:11 -04:00
let client = this.clients.get(serverID);
let chatHistoryBatch = irc.findBatchByType(msg, "chathistory");
this.setState((state) => State.handleMessage(state, msg, serverID, client));
2021-06-10 12:11:11 -04:00
let target, channel, affectedBuffers;
switch (msg.command) {
case irc.RPL_WELCOME:
if (this.state.connectParams.autojoin.length > 0) {
2021-01-22 11:36:53 -05:00
client.send({
command: "JOIN",
params: [this.state.connectParams.autojoin.join(",")],
});
}
2021-06-10 12:11:11 -04:00
let lastReceipt = this.latestReceipt(ReceiptType.READ);
if (lastReceipt && lastReceipt.time && client.enabledCaps["draft/chathistory"] && (!client.enabledCaps["soju.im/bouncer-networks"] || client.params.bouncerNetwork)) {
2021-06-10 12:11:11 -04:00
let now = irc.formatDate(new Date());
client.fetchHistoryTargets(now, lastReceipt.time).then((targets) => {
targets.forEach((target) => {
2021-06-10 12:11:11 -04:00
let from = this.getReceipt(target, ReceiptType.READ);
if (!from) {
from = lastReceipt;
}
2021-06-10 12:11:11 -04:00
let to = { time: msg.tags.time || irc.formatDate(new Date()) };
this.fetchBacklog(client, target.name, from, to);
});
});
}
break;
case "MODE":
2021-06-10 12:11:11 -04:00
target = msg.params[0];
if (client.isChannel(target)) {
this.addMessage(serverID, target, msg);
}
break;
case "NOTICE":
case "PRIVMSG":
2021-06-10 12:11:11 -04:00
target = msg.params[0];
if (client.isMyNick(target)) {
if (client.cm(msg.prefix.name) === client.cm(client.serverPrefix.name)) {
target = SERVER_BUFFER;
} else {
target = msg.prefix.name;
}
}
2021-06-10 06:07:17 -04:00
if (msg.command === "NOTICE" && !State.getBuffer(this.state, { server: serverID, name: target })) {
// Don't open a new buffer if this is just a NOTICE
target = SERVER_BUFFER;
}
2021-06-10 12:11:11 -04:00
let allowedPrefixes = client.isupport.get("STATUSMSG");
if (allowedPrefixes) {
2021-06-10 12:11:11 -04:00
let parts = irc.parseTargetPrefix(target, allowedPrefixes);
if (client.isChannel(parts.name)) {
target = parts.name;
}
}
this.addMessage(serverID, target, msg);
break;
case "JOIN":
2021-06-10 12:11:11 -04:00
channel = msg.params[0];
if (!client.isMyNick(msg.prefix.name)) {
this.addMessage(serverID, channel, msg);
}
2021-05-25 06:40:33 -04:00
if (channel == this.switchToChannel) {
this.switchBuffer({ server: serverID, name: channel });
2021-05-25 06:40:33 -04:00
this.switchToChannel = null;
}
break;
case "PART":
2021-06-10 12:11:11 -04:00
channel = msg.params[0];
this.addMessage(serverID, channel, msg);
2020-07-15 12:21:09 -04:00
2021-06-04 13:45:51 -04:00
if (!chatHistoryBatch && client.isMyNick(msg.prefix.name)) {
this.receipts.delete(channel);
this.saveReceipts();
}
break;
case "KICK":
2021-06-10 12:11:11 -04:00
channel = msg.params[0];
this.addMessage(serverID, channel, msg);
break;
2020-07-08 12:39:24 -04:00
case "QUIT":
2021-06-10 12:11:11 -04:00
affectedBuffers = [];
2021-06-04 13:45:51 -04:00
if (chatHistoryBatch) {
affectedBuffers.push(chatHistoryBatch.params[0]);
} else {
this.setState((state) => {
2021-06-10 12:11:11 -04:00
let buffers = new Map(state.buffers);
2021-06-04 13:45:51 -04:00
state.buffers.forEach((buf) => {
if (buf.server != serverID) {
return;
}
if (!buf.members.has(msg.prefix.name) && client.cm(buf.name) !== client.cm(msg.prefix.name)) {
return;
}
2021-06-10 12:11:11 -04:00
let members = new irc.CaseMapMap(buf.members);
2021-06-04 13:45:51 -04:00
members.delete(msg.prefix.name);
2021-06-10 12:11:11 -04:00
let offline = client.cm(buf.name) === client.cm(msg.prefix.name);
2021-06-04 13:45:51 -04:00
buffers.set(buf.id, { ...buf, members, offline });
affectedBuffers.push(buf.name);
});
return { buffers };
2020-07-08 12:39:24 -04:00
});
2021-06-04 13:45:51 -04:00
}
affectedBuffers.forEach((name) => this.addMessage(serverID, name, msg));
2020-07-08 12:39:24 -04:00
break;
case "NICK":
2021-06-10 12:11:11 -04:00
let newNick = msg.params[0];
2021-06-10 12:11:11 -04:00
affectedBuffers = [];
2021-06-04 13:45:51 -04:00
if (chatHistoryBatch) {
affectedBuffers.push(chatHistoryBatch.params[0]);
} else {
this.setState((state) => {
2021-06-10 12:11:11 -04:00
let buffers = new Map(state.buffers);
2021-06-04 13:45:51 -04:00
state.buffers.forEach((buf) => {
if (buf.server != serverID) {
return;
}
if (!buf.members.has(msg.prefix.name)) {
return;
}
2021-06-10 12:11:11 -04:00
let members = new irc.CaseMapMap(buf.members);
2021-06-04 13:45:51 -04:00
members.set(newNick, members.get(msg.prefix.name));
members.delete(msg.prefix.name);
buffers.set(buf.id, { ...buf, members });
affectedBuffers.push(buf.name);
});
return { buffers };
});
2021-06-04 13:45:51 -04:00
}
affectedBuffers.forEach((name) => this.addMessage(serverID, name, msg));
break;
case "TOPIC":
2021-06-10 12:11:11 -04:00
channel = msg.params[0];
this.addMessage(serverID, channel, msg);
break;
case "INVITE":
2021-06-10 12:11:11 -04:00
channel = msg.params[1];
// TODO: find a more reliable way to do this
2021-06-10 12:11:11 -04:00
let bufName = channel;
2021-06-04 12:03:03 -04:00
if (!State.getBuffer(this.state, { server: serverID, name: channel })) {
bufName = SERVER_BUFFER;
}
this.addMessage(serverID, bufName, msg);
break;
case "BOUNCER":
if (msg.params[0] !== "NETWORK") {
break; // We're only interested in network updates
}
if (client.isupport.has("BOUNCER_NETID")) {
// This cn happen if the user has specified a network to bind
// to via other means, e.g. "<username>/<network>".
break;
}
2021-06-10 12:11:11 -04:00
let id = msg.params[1];
let attrs = null;
if (msg.params[2] !== "*") {
attrs = irc.parseTags(msg.params[2]);
}
2021-06-10 12:11:11 -04:00
let isNew = false;
this.setState((state) => {
2021-06-10 12:11:11 -04:00
let bouncerNetworks = new Map(state.bouncerNetworks);
if (!attrs) {
bouncerNetworks.delete(id);
} else {
2021-06-10 12:11:11 -04:00
let prev = bouncerNetworks.get(id);
isNew = prev === undefined;
attrs = { ...prev, ...attrs };
bouncerNetworks.set(id, attrs);
}
return { bouncerNetworks };
}, () => {
if (!attrs) {
2021-06-10 12:11:11 -04:00
let serverID = this.serverFromBouncerNetwork(id);
if (serverID) {
this.close({ server: serverID, name: SERVER_BUFFER });
}
} else if (isNew) {
this.connect({
...client.params,
bouncerNetwork: id,
});
}
});
break;
case irc.RPL_CHANNELMODEIS:
case irc.RPL_CREATIONTIME:
case irc.RPL_INVITELIST:
case irc.RPL_ENDOFINVITELIST:
case irc.RPL_EXCEPTLIST:
case irc.RPL_ENDOFEXCEPTLIST:
2021-06-03 04:11:48 -04:00
case irc.RPL_BANLIST:
case irc.RPL_ENDOFBANLIST:
2021-06-10 18:27:19 -04:00
case irc.RPL_QUIETLIST:
case irc.RPL_ENDOFQUIETLIST:
2021-06-10 12:11:11 -04:00
channel = msg.params[1];
this.addMessage(serverID, channel, msg);
2021-06-03 04:11:48 -04:00
break;
2021-06-24 12:01:24 -04:00
case irc.RPL_INVITING:
channel = msg.params[2];
this.addMessage(serverID, channel, msg);
break;
case irc.RPL_MYINFO:
case irc.RPL_ISUPPORT:
case irc.RPL_NOTOPIC:
case irc.RPL_TOPIC:
case irc.RPL_TOPICWHOTIME:
case irc.RPL_NAMREPLY:
case irc.RPL_ENDOFNAMES:
case "AWAY":
case "SETNAME":
case "CAP":
case "AUTHENTICATE":
2020-07-01 06:12:56 -04:00
case "PING":
2021-05-27 12:17:41 -04:00
case "PONG":
2020-07-15 12:21:09 -04:00
case "BATCH":
case "TAGMSG":
case "CHATHISTORY":
case "ACK":
// Ignore these
break;
default:
if (irc.isError(msg.command) && msg.command != irc.ERR_NOMOTD) {
2021-06-10 12:11:11 -04:00
let description = msg.params[msg.params.length - 1];
this.setState({ error: description });
}
this.addMessage(serverID, SERVER_BUFFER, msg);
}
}
handleConnectSubmit(connectParams) {
this.setState({ error: null });
if (connectParams.autoconnect) {
store.autoconnect.put(connectParams);
} else {
store.autoconnect.put(null);
}
2021-01-22 12:44:06 -05:00
this.connect(connectParams);
}
2021-05-31 22:39:35 -04:00
handleChannelClick(channel) {
2021-06-10 12:11:11 -04:00
let serverID = State.getActiveServerID(this.state);
let buf = State.getBuffer(this.state, { server: serverID, name: channel });
2021-05-31 22:39:35 -04:00
if (buf) {
this.switchBuffer(buf.id);
} else {
this.open(channel);
}
}
2020-06-25 12:45:41 -04:00
handleNickClick(nick) {
this.open(nick);
}
fetchBacklog(client, target, after, before) {
client.fetchHistoryBetween(target, after, before, CHATHISTORY_MAX_SIZE).catch((err) => {
this.setState({ error: "Failed to fetch history for '" + taregt + "': " + err });
this.receipts.delete(channel);
this.saveReceipts();
});
}
open(target, serverID) {
if (!serverID) {
2021-06-04 12:03:03 -04:00
serverID = State.getActiveServerID(this.state);
}
2021-06-10 12:11:11 -04:00
let client = this.clients.get(serverID);
if (client.isServer(target)) {
this.switchBuffer({ server: serverID });
} else if (client.isChannel(target)) {
this.switchToChannel = target;
2021-01-22 11:36:53 -05:00
client.send({ command: "JOIN", params: [target] });
} else {
2021-05-31 11:11:42 -04:00
client.who(target);
this.createBuffer(serverID, target);
this.switchBuffer({ server: serverID, name: target });
2020-06-25 12:45:41 -04:00
}
}
2021-01-22 04:26:53 -05:00
close(id) {
2021-06-10 12:11:11 -04:00
let buf = State.getBuffer(this.state, id);
2021-01-22 04:26:53 -05:00
if (!buf) {
return;
}
2021-06-10 12:11:11 -04:00
let client = this.clients.get(buf.server);
2021-01-22 04:26:53 -05:00
switch (buf.type) {
case BufferType.SERVER:
2021-03-10 04:59:39 -05:00
this.setState((state) => {
2021-06-10 12:11:11 -04:00
let buffers = new Map(state.buffers);
for (let [id, b] of state.buffers) {
if (b.server === buf.server) {
2021-03-10 04:59:39 -05:00
buffers.delete(id);
}
}
2021-06-10 12:11:11 -04:00
let activeBuffer = state.activeBuffer;
if (activeBuffer && state.buffers.get(activeBuffer).server === buf.server) {
2021-03-10 04:59:39 -05:00
if (buffers.size > 0) {
activeBuffer = buffers.keys().next().value;
} else {
activeBuffer = null;
}
}
return { buffers, activeBuffer };
2021-01-11 12:12:28 -05:00
});
2021-03-10 04:59:39 -05:00
2021-06-10 12:11:11 -04:00
let disconnectAll = client && !client.params.bouncerNetwork && client.enabledCaps["soju.im/bouncer-networks"];
2021-03-10 04:59:39 -05:00
this.disconnect(buf.server);
2021-03-10 04:59:39 -05:00
2021-01-22 04:41:28 -05:00
this.setState((state) => {
2021-06-10 12:11:11 -04:00
let servers = new Map(state.servers);
servers.delete(buf.server);
2021-06-10 12:11:11 -04:00
let connectForm = state.connectForm;
if (servers.size == 0) {
connectForm = true;
}
return { servers, connectForm };
2021-01-22 04:41:28 -05:00
});
2021-03-10 04:59:39 -05:00
if (disconnectAll) {
2021-06-10 12:11:11 -04:00
for (let serverID of this.clients.keys()) {
this.close({ server: serverID, name: SERVER_BUFFER });
2021-03-10 04:59:39 -05:00
}
}
// TODO: only clear local storage if this server is stored there
if (buf.server == 1) {
store.autoconnect.put(null);
}
2021-01-22 04:26:53 -05:00
break;
case BufferType.CHANNEL:
2021-01-22 11:36:53 -05:00
client.send({ command: "PART", params: [buf.name] });
2021-01-22 04:26:53 -05:00
// fallthrough
case BufferType.NICK:
this.switchBuffer({ name: SERVER_BUFFER });
this.setState((state) => {
2021-06-10 12:11:11 -04:00
let buffers = new Map(state.buffers);
2021-03-03 03:59:40 -05:00
buffers.delete(buf.id);
2021-01-22 04:26:53 -05:00
return { buffers };
});
2020-07-15 12:21:09 -04:00
2021-01-22 11:36:53 -05:00
this.receipts.delete(buf.name);
2021-01-22 04:26:53 -05:00
this.saveReceipts();
break;
}
}
executeCommand(s) {
2021-06-10 12:11:11 -04:00
let parts = s.split(" ");
let name = parts[0].toLowerCase().slice(1);
let args = parts.slice(1);
2020-07-13 11:22:24 -04:00
2021-06-10 12:11:11 -04:00
let cmd = commands[name];
2020-07-13 11:22:24 -04:00
if (!cmd) {
this.setState({ error: `Unknown command "${name}" (run "/help" to get a command list)` });
2020-07-13 11:22:24 -04:00
return;
}
try {
2021-03-08 11:25:00 -05:00
cmd.execute(this, args);
} catch (error) {
console.error(`Failed to execute command "${name}":`, error);
this.setState({ error: error.message });
}
}
2020-06-28 03:29:39 -04:00
privmsg(target, text) {
if (target == SERVER_BUFFER) {
this.setState({ error: "Cannot send message in server buffer" });
2020-06-28 03:29:39 -04:00
return;
}
2021-06-10 12:11:11 -04:00
let serverID = State.getActiveServerID(this.state);
let client = this.clients.get(serverID);
2021-01-22 11:36:53 -05:00
2021-06-10 12:11:11 -04:00
let msg = { command: "PRIVMSG", params: [target, text] };
2021-01-22 11:36:53 -05:00
client.send(msg);
2020-06-28 03:29:39 -04:00
2021-01-22 11:36:53 -05:00
if (!client.enabledCaps["echo-message"]) {
msg.prefix = { name: client.nick };
this.addMessage(serverID, target, msg);
2020-06-28 03:29:39 -04:00
}
}
handleComposerSubmit(text) {
if (!text) {
return;
}
if (text.startsWith("//")) {
text = text.slice(1);
} else if (text.startsWith("/")) {
this.executeCommand(text);
return;
}
2021-06-10 12:11:11 -04:00
let buf = this.state.buffers.get(this.state.activeBuffer);
2021-01-21 14:41:44 -05:00
if (!buf) {
return;
}
2021-01-21 14:41:44 -05:00
this.privmsg(buf.name, text);
}
handleBufferListClick(id) {
this.switchBuffer(id);
this.closeBufferList();
}
toggleBufferList() {
this.setState((state) => {
2021-06-10 12:11:11 -04:00
let openPanels = {
...state.openPanels,
bufferList: !state.openPanels.bufferList,
};
return { openPanels };
});
}
toggleMemberList() {
this.setState((state) => {
2021-06-10 12:11:11 -04:00
let openPanels = {
...state.openPanels,
memberList: !state.openPanels.memberList,
};
return { openPanels };
});
}
closeBufferList() {
this.setState((state) => {
2021-06-10 12:11:11 -04:00
let openPanels = {
...state.openPanels,
bufferList: false,
};
return { openPanels };
});
}
closeMemberList() {
this.setState((state) => {
2021-06-10 12:11:11 -04:00
let openPanels = {
...state.openPanels,
memberList: false,
};
return { openPanels };
});
}
handleJoinClick(serverID) {
this.setState({ dialog: "join", joinDialog: { server: serverID } });
2021-03-08 10:23:16 -05:00
}
2021-01-22 11:36:53 -05:00
2021-03-08 10:23:16 -05:00
handleJoinSubmit(data) {
2021-06-10 12:11:11 -04:00
let client = this.clients.get(this.state.joinDialog.server);
2021-03-08 10:23:16 -05:00
this.switchToChannel = data.channel;
2021-03-08 10:23:16 -05:00
client.send({ command: "JOIN", params: [data.channel] });
2021-01-22 11:36:53 -05:00
2021-03-08 10:23:16 -05:00
this.setState({ dialog: null, joinDialog: null });
2020-06-29 04:12:46 -04:00
}
2020-06-29 06:36:17 -04:00
autocomplete(prefix) {
2020-07-13 11:28:49 -04:00
function fromList(l, prefix) {
prefix = prefix.toLowerCase();
let repl = [];
2021-06-10 12:11:11 -04:00
for (let item of l) {
2020-07-13 11:28:49 -04:00
if (item.toLowerCase().startsWith(prefix)) {
repl.push(item);
2020-07-13 11:28:49 -04:00
}
}
return repl;
2020-06-29 06:36:17 -04:00
}
2020-07-13 11:28:49 -04:00
if (prefix.startsWith("/")) {
2021-06-10 12:11:11 -04:00
let repl = fromList(Object.keys(commands), prefix.slice(1));
return repl.map(cmd => "/" + cmd);
2020-07-13 11:28:49 -04:00
}
2021-06-10 12:11:11 -04:00
let buf = this.state.buffers.get(this.state.activeBuffer);
2021-01-21 14:41:44 -05:00
if (!buf || !buf.members) {
return [];
2020-06-29 06:36:17 -04:00
}
2020-07-13 11:28:49 -04:00
return fromList(buf.members.keys(), prefix);
2020-06-29 06:36:17 -04:00
}
openHelp() {
this.setState({ dialog: "help" });
}
2020-06-29 03:06:47 -04:00
handleBufferScrollTop() {
2021-06-10 12:11:11 -04:00
let buf = this.state.buffers.get(this.state.activeBuffer);
2021-01-21 14:41:44 -05:00
if (!buf || buf.type == BufferType.SERVER) {
2020-06-29 03:06:47 -04:00
return;
}
2021-01-22 11:36:53 -05:00
2021-06-10 12:11:11 -04:00
let client = this.clients.get(buf.server);
2021-01-22 11:36:53 -05:00
if (!client || !client.enabledCaps["draft/chathistory"] || !client.enabledCaps["server-time"]) {
2020-06-29 03:06:47 -04:00
return;
}
2021-01-22 09:49:22 -05:00
if (this.endOfHistory.get(buf.id)) {
2020-06-29 03:06:47 -04:00
return;
}
2021-06-10 12:11:11 -04:00
let before;
2020-06-29 03:06:47 -04:00
if (buf.messages.length > 0) {
before = buf.messages[0].tags["time"];
} else {
before = irc.formatDate(new Date());
}
2020-07-15 12:21:09 -04:00
// Avoids sending multiple CHATHISTORY commands in parallel
2021-01-22 09:49:22 -05:00
this.endOfHistory.set(buf.id, true);
2020-07-15 12:21:09 -04:00
client.fetchHistoryBefore(buf.name, before, 100).then((result) => {
2021-01-23 06:16:57 -05:00
this.endOfHistory.set(buf.id, !result.more);
2020-07-15 12:21:09 -04:00
});
2020-06-29 03:06:47 -04:00
}
2021-03-08 10:23:16 -05:00
handleDialogDismiss() {
this.setState({ dialog: null });
}
2021-03-08 12:15:04 -05:00
handleAddNetworkClick() {
2021-03-09 13:10:22 -05:00
this.setState({ dialog: "network", networkDialog: null });
2021-03-08 12:15:04 -05:00
}
handleManageNetworkClick(serverID) {
2021-06-10 12:11:11 -04:00
let server = this.state.servers.get(serverID);
let bouncerNetID = server.isupport.get("BOUNCER_NETID");
let bouncerNetwork = this.state.bouncerNetworks.get(bouncerNetID);
2021-03-09 13:10:22 -05:00
this.setState({
dialog: "network",
networkDialog: {
id: bouncerNetID,
params: bouncerNetwork,
},
});
}
handleNetworkSubmit(attrs) {
2021-06-10 12:11:11 -04:00
let client = this.clients.values().next().value;
2021-03-09 13:10:22 -05:00
if (this.state.networkDialog && this.state.networkDialog.id) {
if (Object.keys(attrs).length == 0) {
this.setState({ dialog: null });
return;
}
client.send({
command: "BOUNCER",
params: ["CHANGENETWORK", this.state.networkDialog.id, irc.formatTags(attrs)],
});
} else {
attrs = { ...attrs, tls: "1" };
client.send({
command: "BOUNCER",
params: ["ADDNETWORK", irc.formatTags(attrs)],
});
}
this.setState({ dialog: null, networkDialog: null });
}
handleNetworkRemove() {
2021-06-10 12:11:11 -04:00
let client = this.clients.values().next().value;
2021-03-09 13:10:22 -05:00
2021-03-08 12:15:04 -05:00
client.send({
command: "BOUNCER",
2021-03-09 13:10:22 -05:00
params: ["DELNETWORK", this.state.networkDialog.id],
2021-03-08 12:15:04 -05:00
});
2021-03-09 13:10:22 -05:00
this.setState({ dialog: null, networkDialog: null });
2021-03-08 12:15:04 -05:00
}
componentDidMount() {
2020-07-23 03:58:05 -04:00
setupKeybindings(this);
}
render() {
2021-06-10 12:11:11 -04:00
let activeBuffer = null, activeServer = null, activeBouncerNetwork = null;
let isBouncer = false;
if (this.state.buffers.get(this.state.activeBuffer)) {
activeBuffer = this.state.buffers.get(this.state.activeBuffer);
activeServer = this.state.servers.get(activeBuffer.server);
2021-03-08 12:15:04 -05:00
2021-06-10 12:11:11 -04:00
let activeClient = this.clients.get(activeBuffer.server);
2021-03-08 12:15:04 -05:00
isBouncer = activeClient && activeClient.enabledCaps["soju.im/bouncer-networks"];
2021-03-10 05:48:58 -05:00
2021-06-10 12:11:11 -04:00
let bouncerNetID = activeServer.isupport.get("BOUNCER_NETID");
2021-03-10 05:48:58 -05:00
if (bouncerNetID) {
activeBouncerNetwork = this.state.bouncerNetworks.get(bouncerNetID);
}
}
if (this.state.connectForm) {
2021-06-10 12:11:11 -04:00
let status = activeServer ? activeServer.status : ServerStatus.DISCONNECTED;
let connecting = status === ServerStatus.CONNECTING || status === ServerStatus.REGISTERING;
// TODO: using key=connectParams trashes the ConnectForm state on update
2021-01-21 13:01:50 -05:00
return html`
<section id="connect">
<${ConnectForm}
error=${this.state.error}
params=${this.state.connectParams}
auth=${this.config.server.auth}
connecting=${connecting}
onSubmit=${this.handleConnectSubmit}
key=${this.state.connectParams}
/>
2021-01-21 13:01:50 -05:00
</section>
`;
}
2021-06-10 12:11:11 -04:00
let bufferHeader = null;
2020-06-26 06:08:14 -04:00
if (activeBuffer) {
2020-06-26 09:16:07 -04:00
bufferHeader = html`
<section id="buffer-header">
<${BufferHeader}
buffer=${activeBuffer}
server=${activeServer}
2021-03-08 12:15:04 -05:00
isBouncer=${isBouncer}
2021-03-10 05:48:58 -05:00
bouncerNetwork=${activeBouncerNetwork}
2021-05-31 22:39:35 -04:00
onChannelClick=${this.handleChannelClick}
onClose=${() => this.close(activeBuffer)}
onJoin=${() => this.handleJoinClick(activeBuffer.server)}
2021-03-08 12:15:04 -05:00
onAddNetwork=${this.handleAddNetworkClick}
onManageNetwork=${() => this.handleManageNetworkClick(activeBuffer.server)}
/>
</section>
`;
}
2021-06-10 12:11:11 -04:00
let memberList = null;
2020-06-26 08:32:56 -04:00
if (activeBuffer && activeBuffer.type == BufferType.CHANNEL) {
memberList = html`
<section
id="member-list"
class=${this.state.openPanels.memberList ? "expand" : ""}
>
<button
class="expander"
onClick=${this.toggleMemberList}
>
<span></span>
<span></span>
</button>
<section>
<section id="member-list-header">
${activeBuffer.members.size} users
</section>
<${MemberList}
members=${activeBuffer.members}
onNickClick=${this.handleNickClick}
/>
</section>
2020-06-26 08:32:56 -04:00
</section>
`;
}
2021-06-10 12:11:11 -04:00
let dialog = null;
2021-03-08 10:23:16 -05:00
switch (this.state.dialog) {
2021-03-09 13:10:22 -05:00
case "network":
2021-06-10 12:11:11 -04:00
let title = this.state.networkDialog ? "Edit network" : "Add network";
2021-03-08 12:15:04 -05:00
dialog = html`
2021-03-09 13:10:22 -05:00
<${Dialog} title=${title} onDismiss=${this.handleDialogDismiss}>
<${NetworkForm}
onSubmit=${this.handleNetworkSubmit}
onRemove=${this.handleNetworkRemove}
params=${this.state.networkDialog ? this.state.networkDialog.params : null}
/>
2021-03-08 12:15:04 -05:00
</>
`;
break;
case "help":
dialog = html`
<${Dialog} title="Help" onDismiss=${this.handleDialogDismiss}>
<${Help}/>
</>
`;
break;
2021-03-08 10:23:16 -05:00
case "join":
dialog = html`
<${Dialog} title="Join channel" onDismiss=${this.handleDialogDismiss}>
<${JoinForm} onSubmit=${this.handleJoinSubmit}/>
2021-03-08 10:23:16 -05:00
</>
`;
break;
}
2021-06-10 12:11:11 -04:00
let error = null;
2021-03-08 09:05:43 -05:00
if (this.state.error) {
error = html`
<div id="error-msg">
${this.state.error}
${" "}
<button onClick=${this.dismissError}>×</button>
</div>
2021-03-08 09:05:43 -05:00
`;
}
2021-06-10 12:11:11 -04:00
let composerReadOnly = false;
if (activeBuffer && activeBuffer.type === BufferType.SERVER) {
composerReadOnly = true;
}
if (activeServer && activeServer.status !== ServerStatus.REGISTERED) {
composerReadOnly = true;
}
return html`
<section
id="buffer-list"
class=${this.state.openPanels.bufferList ? "expand" : ""}
>
<${BufferList}
buffers=${this.state.buffers}
servers=${this.state.servers}
bouncerNetworks=${this.state.bouncerNetworks}
2021-03-08 12:15:04 -05:00
isBouncer=${isBouncer}
activeBuffer=${this.state.activeBuffer}
onBufferClick=${this.handleBufferListClick}
/>
<button
class="expander"
onClick=${this.toggleBufferList}
>
<span></span>
<span></span>
</button>
</section>
2020-06-26 09:16:07 -04:00
${bufferHeader}
<${ScrollManager}
target=${this.buffer}
stickTo=".logline"
scrollKey=${this.state.activeBuffer}
onScrollTop=${this.handleBufferScrollTop}
>
<section id="buffer" ref=${this.buffer} tabindex="-1">
2021-05-31 22:39:35 -04:00
<${Buffer}
buffer=${activeBuffer}
server=${activeServer}
2021-05-31 22:39:35 -04:00
onChannelClick=${this.handleChannelClick}
onNickClick=${this.handleNickClick}/>
</section>
</>
2020-06-26 08:32:56 -04:00
${memberList}
<${Composer}
ref=${this.composer}
readOnly=${composerReadOnly}
onSubmit=${this.handleComposerSubmit}
autocomplete=${this.autocomplete}
/>
2021-03-08 10:23:16 -05:00
${dialog}
2021-03-08 09:05:43 -05:00
${error}
`;
}
}