thelounge/client/js/helpers/contextMenu.ts

469 lines
9.7 KiB
TypeScript
Raw Normal View History

2019-11-09 23:21:34 +01:00
import socket from "../socket";
2020-03-16 18:58:40 +01:00
import eventbus from "../eventbus";
import type {ClientChan, ClientNetwork, ClientUser} from "../types";
import {switchToChannel} from "../router";
import {TypedStore} from "../store";
import useCloseChannel from "../hooks/use-close-channel";
type BaseContextMenuItem = {
label: string;
type: string;
class: string;
};
type ContextMenuItemWithAction = BaseContextMenuItem & {
action: () => void;
};
type ContextMenuItemWithLink = BaseContextMenuItem & {
link?: string;
};
type ContextMenuDividerItem = {
type: "divider";
};
export type ContextMenuItem =
| ContextMenuItemWithAction
| ContextMenuItemWithLink
| ContextMenuDividerItem;
export function generateChannelContextMenu(
channel: ClientChan,
network: ClientNetwork
): ContextMenuItem[] {
const closeChannel = useCloseChannel(channel);
2019-11-09 23:21:34 +01:00
const typeMap = {
lobby: "network",
channel: "chan",
query: "query",
special: "chan",
};
const closeMap = {
lobby: "Remove",
channel: "Leave",
query: "Close",
special: "Close",
};
let items: ContextMenuItem[] = [
2019-11-09 23:21:34 +01:00
{
label: channel.name,
type: "item",
class: typeMap[channel.type],
link: `/chan-${channel.id}`,
},
{
type: "divider",
},
];
// Add menu items for lobbies
if (channel.type === "lobby") {
items = [
...items,
{
label: "Edit this network…",
type: "item",
class: "edit",
link: `/edit-network/${network.uuid}`,
},
{
label: "Join a channel…",
type: "item",
class: "join",
action: () => (network.isJoinChannelShown = true),
},
{
label: "List all channels",
type: "item",
class: "list",
action: () =>
socket.emit("input", {
target: channel.id,
text: "/list",
}),
},
{
label: "List ignored users",
type: "item",
class: "list",
action: () =>
socket.emit("input", {
target: channel.id,
text: "/ignorelist",
}),
},
network.status.connected
? {
label: "Disconnect",
type: "item",
class: "disconnect",
action: () =>
socket.emit("input", {
target: channel.id,
text: "/disconnect",
}),
}
: {
label: "Connect",
type: "item",
class: "connect",
action: () =>
socket.emit("input", {
target: channel.id,
text: "/connect",
}),
},
];
}
// Add menu items for channels
if (channel.type === "channel") {
2019-11-23 15:26:20 +01:00
items.push({
label: "Edit topic",
type: "item",
class: "edit",
action() {
channel.editTopic = true;
switchToChannel(channel);
2019-11-09 23:21:34 +01:00
},
2019-11-23 15:26:20 +01:00
});
items.push({
label: "List banned users",
type: "item",
class: "list",
action() {
socket.emit("input", {
target: channel.id,
text: "/banlist",
});
2019-11-09 23:21:34 +01:00
},
2019-11-23 15:26:20 +01:00
});
2019-11-09 23:21:34 +01:00
}
// Add menu items for queries
if (channel.type === "query") {
2020-11-18 23:57:20 +01:00
items.push(
{
label: "User information",
type: "item",
class: "action-whois",
action() {
switchToChannel(channel);
2020-11-18 23:57:20 +01:00
socket.emit("input", {
target: channel.id,
text: "/whois " + channel.name,
});
},
2019-11-09 23:21:34 +01:00
},
2020-11-18 23:57:20 +01:00
{
label: "Ignore user",
type: "item",
class: "action-ignore",
action() {
socket.emit("input", {
target: channel.id,
text: "/ignore " + channel.name,
});
},
}
);
2019-11-09 23:21:34 +01:00
}
if (channel.type === "channel" || channel.type === "query") {
items.push({
label: "Clear history",
type: "item",
class: "clear-history",
action() {
2020-03-16 18:58:40 +01:00
eventbus.emit(
2020-02-25 10:16:05 +01:00
"confirm-dialog",
{
title: "Clear history",
text: `Are you sure you want to clear history for ${channel.name}? This cannot be undone.`,
button: "Clear history",
},
(result) => {
if (!result) {
return;
}
2020-02-25 10:16:05 +01:00
socket.emit("history:clear", {
target: channel.id,
});
}
);
},
});
}
const humanFriendlyChanTypeMap: Record<string, string> = {
lobby: "network",
channel: "channel",
query: "conversation",
};
// We don't allow the muting of ChanType.SPECIAL channels
const mutableChanTypes = Object.keys(humanFriendlyChanTypeMap);
if (mutableChanTypes.includes(channel.type)) {
const chanType = humanFriendlyChanTypeMap[channel.type];
items.push({
label: channel.muted ? `Unmute ${chanType}` : `Mute ${chanType}`,
type: "item",
class: "mute",
action() {
socket.emit("mute:change", {
target: channel.id,
setMutedTo: !channel.muted,
});
},
});
}
2019-11-09 23:21:34 +01:00
// Add close menu item
items.push({
label: closeMap[channel.type],
type: "item",
class: "close",
action() {
closeChannel();
2019-11-09 23:21:34 +01:00
},
});
return items;
}
export function generateInlineChannelContextMenu(
store: TypedStore,
chan: string,
network: ClientNetwork
): ContextMenuItem[] {
const join = () => {
const channel = network.channels.find((c) => c.name === chan);
if (channel) {
switchToChannel(channel);
}
if (store.state.activeChannel) {
socket.emit("input", {
target: store.state.activeChannel.channel.id,
text: "/join " + chan,
});
} else {
// eslint-disable-next-line no-console
console.error("Unable to join channel: activeChannel is undefined");
}
};
const channel = network.channels.find((c) => c.name === chan);
if (channel) {
return [
{
label: "Go to channel",
type: "item",
class: "chan",
link: `/chan-${channel.id}`,
},
];
}
return [
{
label: "Join channel",
type: "item",
class: "join",
action: join,
},
];
}
export function generateUserContextMenu(
store: TypedStore,
channel: ClientChan,
network: ClientNetwork,
user: Pick<ClientUser, "nick" | "modes">
): ContextMenuItem[] {
const currentChannelUser: ClientUser | Record<string, never> = channel
2020-04-24 09:20:40 +02:00
? channel.users.find((u) => u.nick === network.nick) || {}
: {};
2019-11-09 23:21:34 +01:00
const whois = () => {
2020-04-24 09:20:40 +02:00
const chan = network.channels.find((c) => c.name === user.nick);
2019-11-09 23:21:34 +01:00
if (chan) {
switchToChannel(chan);
2019-11-09 23:21:34 +01:00
}
socket.emit("input", {
target: channel.id,
text: "/whois " + user.nick,
});
};
const items: ContextMenuItem[] = [
2019-11-09 23:21:34 +01:00
{
label: user.nick,
type: "item",
class: "user",
action: whois,
},
{
type: "divider",
},
{
label: "User information",
type: "item",
class: "action-whois",
action: whois,
},
2020-11-18 23:57:20 +01:00
{
label: "Ignore user",
type: "item",
class: "action-ignore",
action() {
socket.emit("input", {
target: channel.id,
text: "/ignore " + user.nick,
});
},
},
2019-11-09 23:21:34 +01:00
{
label: "Direct messages",
type: "item",
class: "action-query",
action() {
const chan = store.getters.findChannelOnCurrentNetwork(user.nick);
2019-11-09 23:21:34 +01:00
if (chan) {
switchToChannel(chan);
2019-11-09 23:21:34 +01:00
}
socket.emit("input", {
target: channel.id,
text: "/query " + user.nick,
});
},
},
];
// Bail because we're in a query or we don't have a special mode.
if (!currentChannelUser.modes || currentChannelUser.modes.length < 1) {
return items;
}
// Names of the standard modes we are able to change
const modeCharToName = {
"~": "owner",
"&": "admin",
"@": "operator",
"%": "half-op",
"+": "voice",
};
// Labels for the mode changes. For example .rev({mode: "a", symbol: "&"}) => 'Revoke admin (-a)'
const modeTextTemplate = {
revoke(m: {symbol: string; mode: string}) {
const name = modeCharToName[m.symbol];
if (typeof name !== "string") {
return "";
}
const res = name ? `Revoke ${name} (-${m.mode})` : `Mode -${m.mode}`;
return res;
},
give(m: {symbol: string; mode: string}) {
const name = modeCharToName[m.symbol];
if (typeof name !== "string") {
return "";
}
const res = name ? `Give ${name} (+${m.mode})` : `Mode +${m.mode}`;
return res;
},
};
2019-11-09 23:21:34 +01:00
const networkModeSymbols = network.serverOptions.PREFIX.symbols;
2019-11-09 23:21:34 +01:00
/**
* Determine whether the prefix of mode p1 has access to perform actions on p2.
*
* EXAMPLE:
* compare('@', '@') => true
* compare('&', '@') => true
* compare('+', '~') => false
* @param {string} p1 The mode performing an action
* @param {string} p2 The target mode
*
* @return {boolean} whether p1 can perform an action on p2
*/
function compare(p1: string, p2: string): boolean {
// The modes ~ and @ can perform actions on their own mode. The others on modes below.
return "~@".indexOf(p1) > -1
? networkModeSymbols.indexOf(p1) <= networkModeSymbols.indexOf(p2)
: networkModeSymbols.indexOf(p1) < networkModeSymbols.indexOf(p2);
}
network.serverOptions.PREFIX.prefix.forEach((mode) => {
if (!compare(currentChannelUser.modes[0], mode.symbol)) {
// Our highest mode is below the current mode. Bail.
return;
}
if (!user.modes.includes(mode.symbol)) {
// The target doesn't already have this mode, therefore we can set it.
2019-11-09 23:21:34 +01:00
items.push({
label: modeTextTemplate.give(mode),
2019-11-09 23:21:34 +01:00
type: "item",
class: "action-set-mode",
2019-11-09 23:21:34 +01:00
action() {
socket.emit("input", {
target: channel.id,
text: "/mode +" + mode.mode + " " + user.nick,
2019-11-09 23:21:34 +01:00
});
},
});
} else {
items.push({
label: modeTextTemplate.revoke(mode),
2019-11-09 23:21:34 +01:00
type: "item",
class: "action-revoke-mode",
2019-11-09 23:21:34 +01:00
action() {
socket.emit("input", {
target: channel.id,
text: "/mode -" + mode.mode + " " + user.nick,
2019-11-09 23:21:34 +01:00
});
},
});
}
});
2019-11-09 23:21:34 +01:00
// Determine if we are half-op or op depending on the network modes so we can kick.
if (!compare(networkModeSymbols.indexOf("%") > -1 ? "%" : "@", currentChannelUser.modes[0])) {
// Check if the target user has no mode or a mode lower than ours.
if (user.modes.length === 0 || compare(currentChannelUser.modes[0], user.modes[0])) {
2019-11-09 23:21:34 +01:00
items.push({
label: "Kick",
2019-11-09 23:21:34 +01:00
type: "item",
class: "action-kick",
2019-11-09 23:21:34 +01:00
action() {
socket.emit("input", {
target: channel.id,
text: "/kick " + user.nick,
2019-11-09 23:21:34 +01:00
});
},
});
}
}
return items;
}