dispatch/client/js/state/messages.js

672 lines
13 KiB
JavaScript
Raw Normal View History

import React from 'react';
import { createSelector } from 'reselect';
2018-04-25 03:36:27 +00:00
import has from 'lodash/has';
2018-04-05 23:46:22 +00:00
import {
findBreakpoints,
messageHeight,
linkify,
timestamp,
2018-12-14 13:24:23 +00:00
isChannel,
formatDate,
unix
2018-04-05 23:46:22 +00:00
} from 'utils';
import stringToRGB from 'utils/color';
import colorify from 'utils/colorify';
import createReducer from 'utils/createReducer';
2017-06-06 23:03:35 +00:00
import { getApp } from './app';
import { getSelectedTab } from './tab';
import * as actions from './actions';
export const getMessages = state => state.messages;
export const getSelectedMessages = createSelector(
getSelectedTab,
getMessages,
2018-04-25 03:36:27 +00:00
(tab, messages) => {
2020-06-15 08:58:51 +00:00
const target = tab.name || tab.network;
if (has(messages, [tab.network, target])) {
return messages[tab.network][target];
2018-04-25 03:36:27 +00:00
}
return [];
}
);
export const getHasMoreMessages = createSelector(
getSelectedMessages,
messages => {
2018-04-25 03:36:27 +00:00
const first = messages[0];
return first && first.next;
}
);
2020-06-15 08:58:51 +00:00
function init(state, network, tab) {
if (!state[network]) {
state[network] = {};
2018-04-25 03:36:27 +00:00
}
2020-06-15 08:58:51 +00:00
if (!state[network][tab]) {
state[network][tab] = [];
2018-04-25 03:36:27 +00:00
}
}
const collapsedEvents = ['join', 'part', 'quit'];
function shouldCollapse(msg1, msg2) {
return (
msg1.events &&
msg2.events &&
collapsedEvents.indexOf(msg1.events[0].type) !== -1 &&
collapsedEvents.indexOf(msg2.events[0].type) !== -1
);
}
const eventVerbs = {
join: 'joined the channel',
part: 'left the channel',
quit: 'quit'
};
function renderNick(nick, type = '') {
const style = {
color: stringToRGB(nick),
fontWeight: 400
};
return (
<span className="message-sender" style={style} key={`${nick} ${type}`}>
{nick}
</span>
);
}
function renderMore(count, type) {
return (
<span
className="message-events-more"
key={`more ${type}`}
>{`${count} more`}</span>
);
}
function renderEvent(event, type, nicks) {
const ending = eventVerbs[type];
if (nicks.length === 1) {
event.push(renderNick(nicks[0], type));
event.push(` ${ending}`);
}
if (nicks.length === 2) {
event.push(renderNick(nicks[0], type));
event.push(' and ');
event.push(renderNick(nicks[1], type));
event.push(` ${ending}`);
}
if (nicks.length > 2) {
event.push(renderNick(nicks[0], type));
event.push(', ');
event.push(renderNick(nicks[1], type));
event.push(' and ');
event.push(renderMore(nicks.length - 2, type));
event.push(` ${ending}`);
}
}
function renderEvents(events) {
const first = events[0];
if (first.type === 'nick') {
const [oldNick, newNick] = first.params;
return [renderNick(oldNick), ' changed nick to ', renderNick(newNick)];
}
2020-06-15 08:58:51 +00:00
if (first.type === 'kick') {
const [kicked, by] = first.params;
return [renderNick(by), ' kicked ', renderNick(kicked)];
}
if (first.type === 'topic') {
const [nick, newTopic] = first.params;
const topic = colorify(linkify(newTopic));
if (!topic) {
return [renderNick(nick), ' cleared the topic'];
}
const result = [renderNick(nick), ' changed the topic to: '];
if (Array.isArray(topic)) {
result.push(...topic);
} else {
result.push(topic);
}
return result;
}
const byType = {};
for (let i = events.length - 1; i >= 0; i--) {
const event = events[i];
const [nick] = event.params;
if (!byType[event.type]) {
byType[event.type] = [nick];
} else if (byType[event.type].indexOf(nick) === -1) {
byType[event.type].push(nick);
}
}
const result = [];
if (byType.join) {
renderEvent(result, 'join', byType.join);
}
if (byType.part) {
if (result.length > 1) {
result[result.length - 1] += ', ';
}
renderEvent(result, 'part', byType.part);
}
if (byType.quit) {
if (result.length > 1) {
result[result.length - 1] += ', ';
}
renderEvent(result, 'quit', byType.quit);
}
return result;
}
2018-12-14 13:24:23 +00:00
let nextID = 0;
function initMessage(
state,
message,
2020-06-15 08:58:51 +00:00
network,
tab,
wrapWidth,
charWidth,
windowWidth,
prepend
) {
2020-06-15 08:58:51 +00:00
const messages = state[network][tab];
if (messages.length > 0 && !prepend) {
const lastMessage = messages[messages.length - 1];
if (shouldCollapse(lastMessage, message)) {
lastMessage.events.push(message.events[0]);
lastMessage.content = renderEvents(lastMessage.events);
[lastMessage.breakpoints, lastMessage.length] = findBreakpoints(
lastMessage.content
);
lastMessage.height = messageHeight(
lastMessage,
wrapWidth,
charWidth,
6 * charWidth,
windowWidth
);
return false;
}
}
if (message.time) {
message.date = new Date(message.time * 1000);
} else {
message.date = new Date();
}
message.time = timestamp(message.date);
if (!message.id) {
message.id = nextID;
nextID++;
}
if (tab.charAt(0) === '#') {
message.channel = true;
}
if (message.events) {
message.type = 'info';
message.content = renderEvents(message.events);
} else {
message.content = message.content || '';
// Collapse multiple adjacent spaces into a single one
message.content = message.content.replace(/\s\s+/g, ' ');
if (message.content.indexOf('\x01ACTION') === 0) {
const { from } = message;
message.from = null;
message.type = 'action';
message.content = from + message.content.slice(7, -1);
}
}
if (!message.events) {
message.content = colorify(linkify(message.content));
}
[message.breakpoints, message.length] = findBreakpoints(message.content);
message.height = messageHeight(
message,
wrapWidth,
charWidth,
6 * charWidth,
windowWidth
);
message.indent = 6 * charWidth;
return true;
}
2018-12-14 13:24:23 +00:00
function createDateMessage(date) {
const message = {
id: nextID,
type: 'date',
content: formatDate(date),
height: 40
};
nextID++;
return message;
}
function isSameDay(d1, d2) {
return (
d1.getDate() === d2.getDate() &&
d1.getMonth() === d2.getMonth() &&
d1.getFullYear() === d2.getFullYear()
);
}
function reducerPrependMessages(
state,
messages,
2020-06-15 08:58:51 +00:00
network,
tab,
wrapWidth,
charWidth,
windowWidth
) {
const msgs = [];
for (let i = 0; i < messages.length; i++) {
const message = messages[i];
initMessage(
state,
message,
2020-06-15 08:58:51 +00:00
network,
tab,
wrapWidth,
charWidth,
windowWidth,
true
);
if (i > 0 && !isSameDay(messages[i - 1].date, message.date)) {
msgs.push(createDateMessage(message.date));
}
msgs.push(message);
}
2020-06-15 08:58:51 +00:00
const m = state[network][tab];
if (m.length > 0) {
const lastNewMessage = msgs[msgs.length - 1];
const firstMessage = m[0];
if (
firstMessage.date &&
!isSameDay(firstMessage.date, lastNewMessage.date)
) {
msgs.push(createDateMessage(firstMessage.date));
}
}
m.unshift(...msgs);
}
2020-06-15 08:58:51 +00:00
function reducerAddMessage(message, network, tab, state) {
const messages = state[network][tab];
2018-12-14 13:24:23 +00:00
if (messages.length > 0) {
const lastMessage = messages[messages.length - 1];
if (lastMessage.date && !isSameDay(lastMessage.date, message.date)) {
messages.push(createDateMessage(message.date));
2018-12-14 13:24:23 +00:00
}
}
messages.push(message);
2018-12-14 13:24:23 +00:00
}
2018-04-25 03:36:27 +00:00
export default createReducer(
{},
{
[actions.ADD_MESSAGE](
state,
2020-06-15 08:58:51 +00:00
{ network, tab, message, wrapWidth, charWidth, windowWidth }
) {
2020-06-15 08:58:51 +00:00
init(state, network, tab);
const shouldAdd = initMessage(
state,
message,
2020-06-15 08:58:51 +00:00
network,
tab,
wrapWidth,
charWidth,
windowWidth
);
if (shouldAdd) {
2020-06-15 08:58:51 +00:00
reducerAddMessage(message, network, tab, state);
}
2018-04-25 03:36:27 +00:00
},
[actions.ADD_MESSAGES](
state,
2020-06-15 08:58:51 +00:00
{ network, tab, messages, prepend, wrapWidth, charWidth, windowWidth }
) {
if (prepend) {
2020-06-15 08:58:51 +00:00
init(state, network, tab);
reducerPrependMessages(
state,
messages,
2020-06-15 08:58:51 +00:00
network,
tab,
wrapWidth,
charWidth,
windowWidth
);
} else {
if (!messages[0].tab) {
2020-06-15 08:58:51 +00:00
init(state, network, tab);
}
2018-04-25 03:36:27 +00:00
messages.forEach(message => {
if (message.tab) {
2020-06-15 08:58:51 +00:00
init(state, network, message.tab);
}
const shouldAdd = initMessage(
state,
message,
2020-06-15 08:58:51 +00:00
network,
message.tab || tab,
wrapWidth,
charWidth,
windowWidth
);
if (shouldAdd) {
2020-06-15 08:58:51 +00:00
reducerAddMessage(message, network, message.tab || tab, state);
}
2018-04-25 03:36:27 +00:00
});
}
2018-04-25 03:36:27 +00:00
},
2020-06-15 08:58:51 +00:00
[actions.DISCONNECT](state, { network }) {
delete state[network];
2018-04-25 03:36:27 +00:00
},
2020-06-15 08:58:51 +00:00
[actions.PART](state, { network, channels }) {
channels.forEach(channel => delete state[network][channel]);
2018-04-25 03:36:27 +00:00
},
2020-06-15 08:58:51 +00:00
[actions.CLOSE_PRIVATE_CHAT](state, { network, nick }) {
delete state[network][nick];
},
2020-06-15 08:58:51 +00:00
[actions.socket.CHANNEL_FORWARD](state, { network, old }) {
if (state[network]) {
delete state[network][old];
}
},
2018-04-25 03:36:27 +00:00
[actions.UPDATE_MESSAGE_HEIGHT](
state,
{ wrapWidth, charWidth, windowWidth }
) {
2020-06-15 08:58:51 +00:00
Object.keys(state).forEach(network =>
Object.keys(state[network]).forEach(target =>
state[network][target].forEach(message => {
2018-12-14 13:24:23 +00:00
if (message.type === 'date') {
return;
}
2018-04-25 03:36:27 +00:00
message.height = messageHeight(
message,
wrapWidth,
charWidth,
6 * charWidth,
windowWidth
);
})
)
2018-04-25 03:36:27 +00:00
);
2018-05-25 21:54:36 +00:00
},
2020-06-15 08:58:51 +00:00
[actions.socket.NETWORKS](state, { data }) {
2018-05-25 21:54:36 +00:00
if (data) {
data.forEach(({ host }) => {
state[host] = {};
});
}
2018-04-25 03:36:27 +00:00
}
}
2018-04-25 03:36:27 +00:00
);
2015-01-17 01:37:21 +00:00
2020-06-15 08:58:51 +00:00
export function getMessageTab(network, to) {
if (!to || to === '*' || (!isChannel(to) && to.indexOf('.') !== -1)) {
2020-06-15 08:58:51 +00:00
return network;
}
return to;
}
2017-05-02 21:21:25 +00:00
export function fetchMessages() {
return (dispatch, getState) => {
const state = getState();
2018-04-25 03:36:27 +00:00
const first = getSelectedMessages(state)[0];
2017-05-02 21:21:25 +00:00
if (!first) {
return;
}
const tab = state.tab.selected;
2020-05-19 22:30:44 +00:00
if (tab.name) {
2017-05-02 21:21:25 +00:00
dispatch({
type: actions.FETCH_MESSAGES,
socket: {
type: 'fetch_messages',
data: {
2020-06-15 08:58:51 +00:00
network: tab.network,
2017-05-02 21:21:25 +00:00
channel: tab.name,
next: first.id
}
}
});
}
};
}
2020-06-15 08:58:51 +00:00
export function addFetchedMessages(network, tab) {
return {
type: actions.ADD_FETCHED_MESSAGES,
2020-06-15 08:58:51 +00:00
network,
tab
};
}
export function updateMessageHeight(wrapWidth, charWidth, windowWidth) {
2017-05-12 08:51:37 +00:00
return {
2016-02-16 21:43:25 +00:00
type: actions.UPDATE_MESSAGE_HEIGHT,
2017-05-12 08:51:37 +00:00
wrapWidth,
charWidth,
windowWidth
2017-05-12 08:51:37 +00:00
};
2015-12-28 23:34:32 +00:00
}
2020-06-15 08:58:51 +00:00
export function sendMessage(content, to, network) {
2016-02-16 21:43:25 +00:00
return (dispatch, getState) => {
const state = getState();
const { wrapWidth, charWidth, windowWidth } = getApp(state);
2016-02-16 21:43:25 +00:00
dispatch({
2017-05-15 03:57:12 +00:00
type: actions.ADD_MESSAGE,
2020-06-15 08:58:51 +00:00
network,
tab: to,
message: {
2020-06-15 08:58:51 +00:00
from: state.networks[network].nick,
content
},
wrapWidth,
charWidth,
windowWidth,
2016-02-16 21:43:25 +00:00
socket: {
type: 'message',
2020-06-15 08:58:51 +00:00
data: { content, to, network }
2016-02-16 21:43:25 +00:00
}
});
2016-02-16 21:43:25 +00:00
};
}
2020-06-15 08:58:51 +00:00
export function addMessage(message, network, to) {
const tab = getMessageTab(network, to);
2015-01-17 01:37:21 +00:00
return (dispatch, getState) => {
const { wrapWidth, charWidth, windowWidth } = getApp(getState());
2018-04-05 23:46:22 +00:00
dispatch({
type: actions.ADD_MESSAGE,
2020-06-15 08:58:51 +00:00
network,
2018-04-05 23:46:22 +00:00
tab,
message,
wrapWidth,
charWidth,
windowWidth
2018-04-05 23:46:22 +00:00
});
};
2015-12-28 23:34:32 +00:00
}
2015-01-17 01:37:21 +00:00
2020-06-15 08:58:51 +00:00
export function addMessages(messages, network, to, prepend, next) {
const tab = getMessageTab(network, to);
2015-12-28 23:34:32 +00:00
2016-02-16 21:43:25 +00:00
return (dispatch, getState) => {
const state = getState();
2017-05-02 21:21:25 +00:00
if (next) {
messages[0].id = next;
messages[0].next = true;
2017-05-02 21:21:25 +00:00
}
const { wrapWidth, charWidth, windowWidth } = getApp(state);
2016-02-16 21:43:25 +00:00
dispatch({
type: actions.ADD_MESSAGES,
2020-06-15 08:58:51 +00:00
network,
tab,
2017-05-02 21:21:25 +00:00
messages,
prepend,
wrapWidth,
charWidth,
windowWidth
2016-02-16 21:43:25 +00:00
});
2015-12-28 23:34:32 +00:00
};
}
2020-06-15 08:58:51 +00:00
export function addEvent(network, tab, type, ...params) {
return addMessage(
{
type: 'info',
events: [
{
type,
params,
time: unix()
}
]
},
2020-06-15 08:58:51 +00:00
network,
tab
);
}
2020-06-15 08:58:51 +00:00
export function broadcastEvent(network, channels, type, ...params) {
const now = unix();
return addMessages(
channels.map(channel => ({
type: 'info',
tab: channel,
events: [
{
type,
params,
time: now
}
]
})),
2020-06-15 08:58:51 +00:00
network
);
}
2020-06-15 08:58:51 +00:00
export function broadcast(message, network, channels) {
2018-04-05 23:46:22 +00:00
return addMessages(
channels.map(channel => ({
tab: channel,
content: message,
type: 'info'
})),
2020-06-15 08:58:51 +00:00
network
2018-04-05 23:46:22 +00:00
);
2015-12-28 23:34:32 +00:00
}
2020-06-15 08:58:51 +00:00
export function print(message, network, channel, type) {
2015-12-28 23:34:32 +00:00
if (Array.isArray(message)) {
2018-04-05 23:46:22 +00:00
return addMessages(
message.map(line => ({
content: line,
type
})),
2020-06-15 08:58:51 +00:00
network,
2018-04-05 23:46:22 +00:00
channel
);
2015-12-28 23:34:32 +00:00
}
2018-04-05 23:46:22 +00:00
return addMessage(
{
content: message,
type
},
2020-06-15 08:58:51 +00:00
network,
2018-04-05 23:46:22 +00:00
channel
);
2015-12-28 23:34:32 +00:00
}
2020-06-15 08:58:51 +00:00
export function inform(message, network, channel) {
return print(message, network, channel, 'info');
}
2020-06-15 08:58:51 +00:00
export function runCommand(command, channel, network) {
2015-12-28 23:34:32 +00:00
return {
type: actions.COMMAND,
command,
channel,
2020-06-15 08:58:51 +00:00
network
2015-12-28 23:34:32 +00:00
};
}
2016-01-27 19:48:47 +00:00
2020-06-15 08:58:51 +00:00
export function raw(message, network) {
2016-01-27 19:48:47 +00:00
return {
type: actions.RAW,
message,
2020-06-15 08:58:51 +00:00
network,
2016-01-27 19:48:47 +00:00
socket: {
type: 'raw',
2020-06-15 08:58:51 +00:00
data: { message, network }
2016-01-27 19:48:47 +00:00
}
};
}