2021-06-22 06:56:37 -04:00
|
|
|
import IconButton from "../../ui/IconButton";
|
2021-06-20 06:05:12 -04:00
|
|
|
import LineDivider from "../../ui/LineDivider";
|
2021-06-19 10:29:04 -04:00
|
|
|
import { mapChannelWithUnread } from "./common";
|
|
|
|
import styled, { css } from "styled-components";
|
|
|
|
import ServerIcon from "../../common/ServerIcon";
|
2021-06-20 06:05:12 -04:00
|
|
|
import { Children } from "../../../types/Preact";
|
2021-06-22 06:56:37 -04:00
|
|
|
import { PlusCircle } from "@styled-icons/feather";
|
2021-06-19 10:29:04 -04:00
|
|
|
import PaintCounter from "../../../lib/PaintCounter";
|
2021-06-19 17:37:12 -04:00
|
|
|
import { attachContextMenu } from 'preact-context-menu';
|
2021-06-20 06:05:12 -04:00
|
|
|
import { connectState } from "../../../redux/connector";
|
2021-06-24 11:22:45 -04:00
|
|
|
import { useLocation, useParams } from "react-router-dom";
|
2021-06-20 06:05:12 -04:00
|
|
|
import { Unreads } from "../../../redux/reducers/unreads";
|
2021-06-24 11:22:45 -04:00
|
|
|
import ConditionalLink from "../../../lib/ConditionalLink";
|
2021-06-20 06:05:12 -04:00
|
|
|
import { Channel, Servers } from "revolt.js/dist/api/objects";
|
2021-06-24 11:22:45 -04:00
|
|
|
import { LastOpened } from "../../../redux/reducers/last_opened";
|
2021-06-22 06:56:37 -04:00
|
|
|
import { isTouchscreenDevice } from "../../../lib/isTouchscreenDevice";
|
|
|
|
import { useIntermediate } from "../../../context/intermediate/Intermediate";
|
2021-06-20 06:05:12 -04:00
|
|
|
import { useChannels, useForceUpdate, useServers } from "../../../context/revoltjs/hooks";
|
|
|
|
|
|
|
|
import logoSVG from '../../../assets/logo.svg';
|
2021-06-19 10:29:04 -04:00
|
|
|
|
|
|
|
function Icon({ children, unread, size }: { children: Children, unread?: 'mention' | 'unread', size: number }) {
|
|
|
|
return (
|
|
|
|
<svg
|
|
|
|
width={size}
|
|
|
|
height={size}
|
|
|
|
aria-hidden="true"
|
|
|
|
viewBox="0 0 32 32"
|
|
|
|
>
|
|
|
|
<foreignObject x="0" y="0" width="32" height="32">
|
|
|
|
{ children }
|
|
|
|
</foreignObject>
|
|
|
|
{unread === 'unread' && (
|
|
|
|
<circle
|
|
|
|
cx="27"
|
|
|
|
cy="27"
|
|
|
|
r="5"
|
|
|
|
fill={"white"}
|
|
|
|
/>
|
|
|
|
)}
|
|
|
|
{unread === 'mention' && (
|
|
|
|
<circle
|
|
|
|
cx="27"
|
|
|
|
cy="27"
|
|
|
|
r="5"
|
|
|
|
fill={"red"}
|
|
|
|
/>
|
|
|
|
)}
|
|
|
|
</svg>
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
const ServersBase = styled.div`
|
|
|
|
width: 52px;
|
|
|
|
height: 100%;
|
|
|
|
display: flex;
|
|
|
|
flex-direction: column;
|
2021-06-22 06:56:37 -04:00
|
|
|
|
|
|
|
${ isTouchscreenDevice && css`
|
|
|
|
padding-bottom: 50px;
|
|
|
|
` }
|
2021-06-19 10:29:04 -04:00
|
|
|
`;
|
|
|
|
|
|
|
|
const ServerList = styled.div`
|
|
|
|
flex-grow: 1;
|
|
|
|
display: flex;
|
|
|
|
overflow-y: scroll;
|
|
|
|
padding-bottom: 48px;
|
|
|
|
flex-direction: column;
|
|
|
|
border-inline-end: 2px solid var(--sidebar-active);
|
|
|
|
|
|
|
|
scrollbar-width: none;
|
|
|
|
|
|
|
|
> :first-child > svg {
|
|
|
|
margin: 6px 0 6px 4px;
|
|
|
|
}
|
|
|
|
|
|
|
|
&::-webkit-scrollbar {
|
|
|
|
width: 0px;
|
|
|
|
}
|
|
|
|
`;
|
|
|
|
|
|
|
|
const ServerEntry = styled.div<{ active: boolean, invert?: boolean }>`
|
|
|
|
height: 44px;
|
|
|
|
padding: 4px;
|
|
|
|
margin: 2px 0 2px 4px;
|
|
|
|
|
|
|
|
border-top-left-radius: 4px;
|
|
|
|
border-bottom-left-radius: 4px;
|
|
|
|
|
|
|
|
img {
|
|
|
|
width: 32px;
|
|
|
|
height: 32px;
|
|
|
|
}
|
|
|
|
|
|
|
|
${ props => props.active && css`
|
|
|
|
background: var(--sidebar-active);
|
|
|
|
` }
|
|
|
|
|
|
|
|
${ props => props.active && props.invert && css`
|
|
|
|
img {
|
|
|
|
filter: saturate(0) brightness(10);
|
|
|
|
}
|
|
|
|
` }
|
|
|
|
`;
|
|
|
|
|
|
|
|
interface Props {
|
|
|
|
unreads: Unreads;
|
2021-06-24 11:22:45 -04:00
|
|
|
lastOpened: LastOpened;
|
2021-06-19 10:29:04 -04:00
|
|
|
}
|
|
|
|
|
2021-06-24 11:22:45 -04:00
|
|
|
export function ServerListSidebar({ unreads, lastOpened }: Props) {
|
2021-06-19 10:29:04 -04:00
|
|
|
const ctx = useForceUpdate();
|
|
|
|
const activeServers = useServers(undefined, ctx) as Servers.Server[];
|
|
|
|
const channels = (useChannels(undefined, ctx) as Channel[])
|
|
|
|
.map(x => mapChannelWithUnread(x, unreads));
|
|
|
|
|
|
|
|
const unreadChannels = channels.filter(x => x.unread)
|
|
|
|
.map(x => x._id);
|
|
|
|
|
|
|
|
const servers = activeServers.map(server => {
|
|
|
|
let alertCount = 0;
|
|
|
|
for (let id of server.channels) {
|
|
|
|
let channel = channels.find(x => x._id === id);
|
|
|
|
if (channel?.alertCount) {
|
|
|
|
alertCount += channel.alertCount;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return {
|
|
|
|
...server,
|
|
|
|
unread: (typeof server.channels.find(x => unreadChannels.includes(x)) !== 'undefined' ?
|
|
|
|
( alertCount > 0 ? 'mention' : 'unread' ) : undefined) as 'mention' | 'unread' | undefined,
|
|
|
|
alertCount
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
const path = useLocation().pathname;
|
|
|
|
const { server: server_id } = useParams<{ server?: string }>();
|
|
|
|
const server = servers.find(x => x!._id == server_id);
|
|
|
|
|
2021-06-22 06:56:37 -04:00
|
|
|
const { openScreen } = useIntermediate();
|
2021-06-19 10:29:04 -04:00
|
|
|
|
|
|
|
let homeUnread: 'mention' | 'unread' | undefined;
|
|
|
|
let alertCount = 0;
|
|
|
|
for (let x of channels) {
|
|
|
|
if (((x.channel_type === 'DirectMessage' && x.active) || x.channel_type === 'Group') && x.unread) {
|
|
|
|
homeUnread = 'unread';
|
|
|
|
alertCount += x.alertCount ?? 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (alertCount > 0) homeUnread = 'mention';
|
2021-06-24 11:22:45 -04:00
|
|
|
const homeActive = typeof server === 'undefined' && !path.startsWith('/invite');
|
2021-06-19 10:29:04 -04:00
|
|
|
|
|
|
|
return (
|
|
|
|
<ServersBase>
|
|
|
|
<ServerList>
|
2021-06-24 11:22:45 -04:00
|
|
|
<ConditionalLink active={homeActive} to={lastOpened.home ? `/channel/${lastOpened.home}` : '/'}>
|
|
|
|
<ServerEntry invert active={homeActive}>
|
2021-06-19 10:29:04 -04:00
|
|
|
<Icon size={36} unread={homeUnread}>
|
2021-06-20 06:05:12 -04:00
|
|
|
<img src={logoSVG} />
|
2021-06-19 10:29:04 -04:00
|
|
|
</Icon>
|
|
|
|
</ServerEntry>
|
2021-06-24 11:22:45 -04:00
|
|
|
</ConditionalLink>
|
2021-06-19 10:29:04 -04:00
|
|
|
<LineDivider />
|
|
|
|
{
|
2021-06-24 11:22:45 -04:00
|
|
|
servers.map(entry => {
|
|
|
|
const active = entry!._id === server?._id;
|
|
|
|
const id = lastOpened[entry!._id];
|
|
|
|
|
|
|
|
return (
|
|
|
|
<ConditionalLink active={active} to={`/server/${entry!._id}` + (id ? `/channel/${id}` : '')}>
|
|
|
|
<ServerEntry
|
|
|
|
active={active}
|
|
|
|
onContextMenu={attachContextMenu('Menu', { server: entry!._id })}>
|
|
|
|
<Icon size={36} unread={entry.unread}>
|
|
|
|
<ServerIcon size={32} target={entry} />
|
|
|
|
</Icon>
|
|
|
|
</ServerEntry>
|
|
|
|
</ConditionalLink>
|
|
|
|
)
|
|
|
|
})
|
2021-06-19 10:29:04 -04:00
|
|
|
}
|
2021-06-22 06:56:37 -04:00
|
|
|
<IconButton onClick={() => openScreen({ id: 'special_input', type: 'create_server' })}>
|
|
|
|
<PlusCircle size={36} />
|
|
|
|
</IconButton>
|
2021-06-19 10:29:04 -04:00
|
|
|
<PaintCounter small />
|
|
|
|
</ServerList>
|
|
|
|
</ServersBase>
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
export default connectState(
|
|
|
|
ServerListSidebar,
|
|
|
|
state => {
|
|
|
|
return {
|
2021-06-24 11:22:45 -04:00
|
|
|
unreads: state.unreads,
|
|
|
|
lastOpened: state.lastOpened
|
2021-06-19 10:29:04 -04:00
|
|
|
};
|
|
|
|
}
|
|
|
|
);
|