gamja/components/buffer.js

521 lines
12 KiB
JavaScript
Raw Normal View History

import { html, Component } from "../lib/index.js";
import linkify from "../lib/linkify.js";
import * as irc from "../lib/irc.js";
import { strip as stripANSI } from "../lib/ansi.js";
import { BufferType, getNickURL, getChannelURL, getMessageURL } from "../state.js";
import Membership from "./membership.js";
function djb2(s) {
2021-06-10 12:11:11 -04:00
let hash = 5381;
for (let i = 0; i < s.length; i++) {
hash = (hash << 5) + hash + s.charCodeAt(i);
hash = hash >>> 0; // convert to uint32
}
return hash;
}
function Nick(props) {
function handleClick(event) {
event.preventDefault();
2020-06-25 12:45:41 -04:00
props.onClick();
}
2021-06-10 12:11:11 -04:00
let colorIndex = djb2(props.nick) % 16 + 1;
return html`
2020-07-15 12:47:33 -04:00
<a href=${getNickURL(props.nick)} class="nick nick-${colorIndex}" onClick=${handleClick}>${props.nick}</a>
`;
}
2020-07-15 12:47:33 -04:00
function Timestamp({ date, url }) {
if (!date) {
return html`<spam class="timestamp">--:--:--</span>`;
2020-06-25 12:45:41 -04:00
}
2021-06-10 12:11:11 -04:00
let hh = date.getHours().toString().padStart(2, "0");
let mm = date.getMinutes().toString().padStart(2, "0");
let ss = date.getSeconds().toString().padStart(2, "0");
let timestamp = `${hh}:${mm}:${ss}`;
return html`
2020-07-15 12:47:33 -04:00
<a href=${url} class="timestamp" onClick=${(event) => event.preventDefault()}>${timestamp}</a>
`;
}
2021-05-28 08:52:31 -04:00
/**
* Check whether a message can be folded.
*
* Unimportant and noisy messages that may clutter the discussion should be
* folded.
*/
function canFoldMessage(msg) {
switch (msg.command) {
case "JOIN":
case "PART":
case "QUIT":
case "NICK":
return true;
}
return false;
}
class LogLine extends Component {
shouldComponentUpdate(nextProps) {
return this.props.message !== nextProps.message;
}
render() {
2021-06-10 12:11:11 -04:00
let msg = this.props.message;
let buf = this.props.buffer;
let server = this.props.server;
2021-06-10 12:11:11 -04:00
let onNickClick = this.props.onNickClick;
let onChannelClick = this.props.onChannelClick;
function createNick(nick) {
return html`
<${Nick} nick=${nick} onClick=${() => onNickClick(nick)}/>
`;
}
function createChannel(channel) {
function onClick(event) {
event.preventDefault();
onChannelClick(channel);
}
return html`
<a href=${getChannelURL(channel)} onClick=${onClick}>
${channel}
</a>
`;
}
2021-06-10 12:11:11 -04:00
let lineClass = "";
let content;
2021-06-24 12:01:24 -04:00
let invitee;
switch (msg.command) {
case "NOTICE":
case "PRIVMSG":
let target = msg.params[0];
2021-06-10 12:11:11 -04:00
let text = msg.params[1];
2021-06-10 12:11:11 -04:00
let ctcp = irc.parseCTCP(msg);
if (ctcp) {
if (ctcp.command == "ACTION") {
lineClass = "me-tell";
2021-05-31 22:39:35 -04:00
content = html`* ${createNick(msg.prefix.name)} ${linkify(stripANSI(ctcp.param), onChannelClick)}`;
} else {
content = html`
${createNick(msg.prefix.name)} has sent a CTCP command: ${ctcp.command} ${ctcp.param}
`;
}
} else {
lineClass = "talk";
2021-06-10 12:11:11 -04:00
let prefix = "<", suffix = ">";
if (msg.command == "NOTICE") {
prefix = suffix = "-";
}
2021-05-31 22:39:35 -04:00
content = html`${prefix}${createNick(msg.prefix.name)}${suffix} ${linkify(stripANSI(text), onChannelClick)}`;
}
let status = null;
let allowedPrefixes = server.isupport.get("STATUSMSG");
if (target !== buf.name && allowedPrefixes) {
let parts = irc.parseTargetPrefix(target, allowedPrefixes);
if (parts.name === buf.name) {
content = [html`(<${Membership} value=${parts.prefix}/>)`, " ", content];
}
}
if (msg.isHighlight) {
lineClass += " highlight";
}
break;
2021-05-28 12:47:40 -04:00
case "JOIN":
content = html`
${createNick(msg.prefix.name)} has joined
`;
break;
case "PART":
content = html`
${createNick(msg.prefix.name)} has left
`;
break;
case "QUIT":
content = html`
${createNick(msg.prefix.name)} has quit
`;
break;
case "NICK":
2021-06-10 12:11:11 -04:00
let newNick = msg.params[0];
2021-05-28 12:47:40 -04:00
content = html`
${createNick(msg.prefix.name)} is now known as ${createNick(newNick)}
`;
break;
case "KICK":
content = html`
${createNick(msg.params[1])} was kicked by ${createNick(msg.prefix.name)} (${msg.params.slice(2)})
`;
break;
case "MODE":
content = html`
* ${createNick(msg.prefix.name)} sets mode ${msg.params.slice(1).join(" ")}
`;
break;
case "TOPIC":
2021-06-10 12:11:11 -04:00
let topic = msg.params[1];
content = html`
2021-05-31 22:39:35 -04:00
${createNick(msg.prefix.name)} changed the topic to: ${linkify(stripANSI(topic), onChannelClick)}
`;
break;
case "INVITE":
2021-06-24 12:01:24 -04:00
invitee = msg.params[0];
2021-06-10 12:11:11 -04:00
let channel = msg.params[1];
// TODO: instead of checking buffer type, check if invitee is our nick
if (buf.type === BufferType.SERVER) {
lineClass = "talk";
content = html`
You have been invited to ${createChannel(channel)} by ${createNick(msg.prefix.name)}
`;
} else {
content = html`
${createNick(msg.prefix.name)} has invited ${createNick(invitee)} to the channel
`;
}
break;
2021-06-24 12:01:24 -04:00
case irc.RPL_INVITING:
invitee = msg.params[1];
content = html`${createNick(invitee)} has been invited to the channel`;
break;
2021-05-28 04:46:32 -04:00
case irc.RPL_MOTD:
lineClass = "motd";
2021-06-10 06:09:07 -04:00
content = linkify(stripANSI(msg.params[1]), onChannelClick);
2021-05-28 04:46:32 -04:00
break;
default:
if (irc.isError(msg.command) && msg.command != irc.ERR_NOMOTD) {
lineClass = "error";
}
2021-05-28 12:47:40 -04:00
content = html`${msg.command} ${msg.params.join(" ")}`;
2020-06-29 08:29:31 -04:00
}
return html`
<div class="logline ${lineClass}" data-key=${msg.key}>
<${Timestamp} date=${new Date(msg.tags.time)} url=${getMessageURL(buf, msg)}/>
${" "}
${content}
</div>
`;
}
}
2021-05-28 12:47:40 -04:00
function createNickList(nicks, createNick) {
if (nicks.length === 0) {
return null;
} else if (nicks.length === 1) {
return createNick(nicks[0]);
}
2021-06-10 12:11:11 -04:00
let l = nicks.slice(0, nicks.length - 1).map((nick, i) => {
2021-05-28 12:47:40 -04:00
if (i === 0) {
return createNick(nick);
} else {
return [", ", createNick(nick)];
}
});
l.push(" and ");
l.push(createNick(nicks[nicks.length - 1]));
return l;
}
2021-05-28 08:52:31 -04:00
class FoldGroup extends Component {
shouldComponentUpdate(nextProps) {
return this.props.messages[0] !== nextProps.messages[0] ||
this.props.messages[this.props.messages.length - 1] !== nextProps.messages[nextProps.messages.length - 1];
2021-05-28 08:52:31 -04:00
}
render() {
2021-06-10 12:11:11 -04:00
let msgs = this.props.messages;
let buf = this.props.buffer;
2021-05-28 08:52:31 -04:00
2021-06-10 12:11:11 -04:00
let onNickClick = this.props.onNickClick;
2021-05-28 08:52:31 -04:00
function createNick(nick) {
return html`
<${Nick} nick=${nick} onClick=${() => onNickClick(nick)}/>
`;
}
2021-06-10 12:11:11 -04:00
let byCommand = {
2021-05-28 12:47:40 -04:00
"JOIN": [],
"PART": [],
"QUIT": [],
"NICK": [],
};
msgs.forEach((msg) => {
byCommand[msg.command].push(msg);
});
2021-06-10 12:11:11 -04:00
let first = true;
let content = [];
2021-05-28 12:47:40 -04:00
["JOIN", "PART", "QUIT"].forEach((cmd) => {
if (byCommand[cmd].length === 0) {
return;
}
2021-06-10 12:11:11 -04:00
let plural = byCommand[cmd].length > 1;
let action;
2021-05-28 12:47:40 -04:00
switch (cmd) {
case "JOIN":
action = plural ? "have joined" : "has joined";
break;
case "PART":
action = plural ? "have left" : "has left";
break;
case "QUIT":
action = plural ? "have quit" : "has quit";
break;
}
if (first) {
first = false;
} else {
content.push(", ");
}
2021-06-10 12:11:11 -04:00
let nicks = byCommand[cmd].map((msg) => msg.prefix.name);
2021-05-28 12:47:40 -04:00
content.push(createNickList(nicks, createNick));
content.push(" " + action);
});
byCommand["NICK"].forEach((msg) => {
if (first) {
2021-05-31 08:08:30 -04:00
first = false;
2021-05-28 08:52:31 -04:00
} else {
2021-05-28 12:47:40 -04:00
content.push(", ");
2021-05-28 08:52:31 -04:00
}
2021-05-28 12:47:40 -04:00
2021-06-10 12:11:11 -04:00
let newNick = msg.params[0];
2021-05-28 12:47:40 -04:00
content.push(html`
${createNick(msg.prefix.name)} is now known as ${createNick(newNick)}
`);
2021-05-28 08:52:31 -04:00
});
2021-06-10 12:11:11 -04:00
let lastMsg = msgs[msgs.length - 1];
let firstDate = new Date(msgs[0].tags.time);
let lastDate = new Date(lastMsg.tags.time);
let timestamp = html`
<${Timestamp} date=${firstDate} url=${getMessageURL(buf, msgs[0])}/>
`;
if (lastDate - firstDate > 60 * 100) {
timestamp = [
timestamp,
" — ",
html`
<${Timestamp} date=${lastDate} url=${getMessageURL(buf, lastMsg)}/>
`,
];
}
2021-05-28 08:52:31 -04:00
return html`
<div class="logline" data-key=${msgs[0].key}>
${timestamp}
2021-05-28 08:52:31 -04:00
${" "}
${content}
</div>
`;
}
}
// Workaround for https://bugs.chromium.org/p/chromium/issues/detail?id=481856
2021-06-10 12:11:11 -04:00
let notificationsSupported = false;
if (window.Notification) {
notificationsSupported = true;
if (Notification.permission === "default") {
try {
new Notification("");
} catch (err) {
if (err.name === "TypeError") {
notificationsSupported = false;
}
}
}
}
2020-06-29 05:50:42 -04:00
class NotificationNagger extends Component {
state = { nag: false };
constructor(props) {
super(props);
this.handleClick = this.handleClick.bind(this);
this.state.nag = this.shouldNag();
}
shouldNag() {
return notificationsSupported && Notification.permission === "default";
2020-06-29 05:50:42 -04:00
}
handleClick(event) {
event.preventDefault();
Notification.requestPermission((permission) => {
this.setState({ nag: this.shouldNag() });
});
}
render() {
if (!this.state.nag) {
return null;
}
return html`
<div class="logline">
<${Timestamp}/>
2020-06-29 05:50:42 -04:00
${" "}
<a href="#" onClick=${this.handleClick}>Turn on desktop notifications</a> to get notified about new messages
</div>
`;
}
}
class DateSeparator extends Component {
constructor(props) {
super(props);
}
shouldComponentUpdate(nextProps) {
return this.props.date.getTime() !== nextProps.date.getTime();
}
render() {
2021-06-10 12:11:11 -04:00
let date = this.props.date;
let YYYY = date.getFullYear().toString().padStart(4, "0");
let MM = (date.getMonth() + 1).toString().padStart(2, "0");
let DD = date.getDate().toString().padStart(2, "0");
let text = `${YYYY}-${MM}-${DD}`;
return html`
<div class="separator date-separator">
${text}
</div>
`;
}
}
function UnreadSeparator(props) {
return html`<div class="separator unread-separator">New messages</div>`;
}
function sameDate(d1, d2) {
return d1.getFullYear() === d2.getFullYear() && d1.getMonth() === d2.getMonth() && d1.getDate() === d2.getDate();
}
export default class Buffer extends Component {
shouldComponentUpdate(nextProps) {
return this.props.buffer !== nextProps.buffer;
}
render() {
2021-06-10 12:11:11 -04:00
let buf = this.props.buffer;
let server = this.props.server;
if (!buf) {
return null;
}
2021-06-10 12:11:11 -04:00
let children = [];
if (buf.type == BufferType.SERVER) {
children.push(html`<${NotificationNagger}/>`);
}
2021-06-10 12:11:11 -04:00
let onChannelClick = this.props.onChannelClick;
let onNickClick = this.props.onNickClick;
2021-05-28 08:52:31 -04:00
function createLogLine(msg) {
return html`
<${LogLine}
key=${"msg-" + msg.key}
message=${msg}
buffer=${buf}
server=${server}
2021-05-31 22:39:35 -04:00
onChannelClick=${onChannelClick}
2021-05-28 08:52:31 -04:00
onNickClick=${onNickClick}
/>
`;
}
function createFoldGroup(msgs) {
2021-05-28 09:51:39 -04:00
// Filter out PART → JOIN pairs
2021-06-10 12:11:11 -04:00
let partIndexes = new Map();
let keep = [];
2021-05-28 09:51:39 -04:00
msgs.forEach((msg, i) => {
if (msg.command === "PART" || msg.command === "QUIT") {
partIndexes.set(msg.prefix.name, i);
}
if (msg.command === "JOIN" && partIndexes.has(msg.prefix.name)) {
keep[partIndexes.get(msg.prefix.name)] = false;
partIndexes.delete(msg.prefix.name);
keep.push(false);
} else {
keep.push(true);
}
});
msgs = msgs.filter((msg, i) => keep[i]);
2021-05-28 08:52:31 -04:00
if (msgs.length === 0) {
return null;
} else if (msgs.length === 1) {
return createLogLine(msgs[0]);
}
return html`
<${FoldGroup}
key=${"fold-" + msgs[0].key + "-" + msgs[msgs.length - 1].key}
2021-05-28 08:52:31 -04:00
messages=${msgs}
buffer=${buf}
server=${server}
2021-05-28 08:52:31 -04:00
onNickClick=${onNickClick}
/>
`;
}
2021-06-10 12:11:11 -04:00
let hasUnreadSeparator = false;
let prevDate = new Date();
let foldMessages = [];
buf.messages.forEach((msg) => {
2021-06-10 12:11:11 -04:00
let sep = [];
2021-05-28 08:52:31 -04:00
if (!hasUnreadSeparator && buf.type != BufferType.SERVER && buf.lastReadReceipt && msg.tags.time > buf.lastReadReceipt.time) {
2021-05-28 08:52:31 -04:00
sep.push(html`<${UnreadSeparator} key="unread"/>`);
hasUnreadSeparator = true;
}
2021-06-10 12:11:11 -04:00
let date = new Date(msg.tags.time);
if (!sameDate(prevDate, date)) {
2021-05-28 08:52:31 -04:00
sep.push(html`<${DateSeparator} key=${"date-" + date} date=${date}/>`);
}
prevDate = date;
2021-05-28 08:52:31 -04:00
if (sep.length > 0) {
children.push(createFoldGroup(foldMessages));
children.push(sep);
foldMessages = [];
}
// TODO: consider checking the time difference too
if (canFoldMessage(msg)) {
foldMessages.push(msg);
return;
}
if (foldMessages.length > 0) {
children.push(createFoldGroup(foldMessages));
foldMessages = [];
}
children.push(createLogLine(msg));
});
2021-05-28 08:52:31 -04:00
children.push(createFoldGroup(foldMessages));
return html`
<div class="logline-list">
${children}
</div>
`;
}
}