Skip to content

Instantly share code, notes, and snippets.

@HonbraDev
Created March 13, 2021 01:14
Show Gist options
  • Save HonbraDev/d18121aa207c75a85ed9f8b888c311d8 to your computer and use it in GitHub Desktop.
Save HonbraDev/d18121aa207c75a85ed9f8b888c311d8 to your computer and use it in GitHub Desktop.
All TypeScript of DogeHouse front-end combined
import React from "react";
import ReactDOM from "react-dom";
import * as Sentry from "@sentry/react";
import ReactModal from "react-modal";
import "react-toastify/dist/ReactToastify.css";
import "./index.css";
import { init_i18n } from "./i18n";
import { Providers } from "./Providers";
import { App } from "./app/App";
init_i18n();
ReactModal.setAppElement("#root");
Sentry.init({
dsn: process.env.REACT_APP_SENTRY_DSN,
enabled: !!process.env.REACT_APP_SENTRY_DSN,
});
ReactDOM.render(
<React.StrictMode>
<Providers>
<App />
</Providers>
</React.StrictMode>,
document.getElementById("root")
);
import React, { useLayoutEffect, useState } from "react";
import { useQuery } from "react-query";
import { BrowserRouter } from "react-router-dom";
import {
auth_query,
createWebSocket,
wsAuthFetch,
wsend,
} from "../createWebsocket";
import { useCurrentRoomStore } from "../webrtc/stores/useCurrentRoomStore";
import { useMuteStore } from "../webrtc/stores/useMuteStore";
import { useSocketStatus } from "../webrtc/stores/useSocketStatus";
import { useVoiceStore } from "../webrtc/stores/useVoiceStore";
import { WebRtcApp } from "../webrtc/WebRtcApp";
import { CenterLayout } from "./components/CenterLayout";
import { DeviceNotSupported } from "./components/DeviceNotSupported";
import { MicPermissionBanner } from "./components/MicPermissionBanner";
import { PageWrapper } from "./components/PageWrapper";
import { WsKilledMessage } from "./components/WsKilledMessage";
import { RoomChat } from "./modules/room-chat/RoomChat";
import { Routes } from "./Routes";
import { roomToCurrentRoom } from "./utils/roomToCurrentRoom";
import { useSaveTokensFromQueryParams } from "./utils/useSaveTokensFromQueryParams";
import { useTokenStore } from "./utils/useTokenStore";
interface AppProps {}
export const App: React.FC<AppProps> = () => {
const isDeviceSupported = useVoiceStore((s) => !!s.device);
const hasTokens = useTokenStore((s) => !!s.accessToken && !!s.refreshToken);
const authIsGood = useSocketStatus((s) => s.status === "auth-good");
const wsKilledByServer = useSocketStatus(
(s) => s.status === "closed-by-server"
);
useState(() => (hasTokens ? createWebSocket() : null));
useLayoutEffect(() => {
if (hasTokens) {
createWebSocket();
}
// eslint-disable-next-line react-hooks/exhaustive-deps
}, [hasTokens]);
const { isLoading } = useQuery<any>(
auth_query,
() => {
const { accessToken, refreshToken } = useTokenStore.getState();
// I think this will probably only run in dev
console.log(
"AUTH_QUERY RUNNING, I HOPE YOU ARE NOT IN PROD LOL (nothing bad happens if you are, probably)"
);
return wsAuthFetch({
op: auth_query,
d: {
accessToken,
refreshToken,
reconnectToVoice: false,
currentRoomId: useCurrentRoomStore.getState().currentRoom?.id,
muted: useMuteStore.getState().muted,
platform: "web",
},
});
},
{
onSuccess: (d) => {
if (d?.currentRoom) {
useCurrentRoomStore
.getState()
.setCurrentRoom(() => roomToCurrentRoom(d.currentRoom));
wsend({ op: "get_current_room_users", d: {} });
}
},
enabled: hasTokens && authIsGood,
staleTime: Infinity,
}
);
useSaveTokensFromQueryParams();
if (isLoading) {
return null;
}
if (!isDeviceSupported) {
return <DeviceNotSupported />;
}
if (wsKilledByServer) {
return <WsKilledMessage />;
}
return (
<BrowserRouter>
<WebRtcApp />
<PageWrapper>
<CenterLayout>
<Routes />
<MicPermissionBanner />
</CenterLayout>
<RoomChat sidebar />
</PageWrapper>
</BrowserRouter>
);
};
import * as React from "react";
const paths = {
megaphone:
"M2 6.77l12.33-3.43.67.53v8.6l-.67.53-6.089-1.595a2.16 2.16 0 1 1-4.178-1.095L2 9.77l-.42-.53V7.3L2 6.77zm3.006 3.787a1.13 1.13 0 0 0-.04.242 1.17 1.17 0 0 0 2.288.347l-2.248-.589zM2.58 8.82L14 11.83V4.5L2.58 7.72v1.1z",
mute:
"M1.5 5h2.79l3.86-3.83.85.35v13l-.85.33L4.29 11H1.5l-.5-.5v-5l.5-.5zm3.35 5.17L8 13.31V2.73L4.85 5.85 4.5 6H2v4h2.5l.35.17zm9.381-4.108l.707.707L13.207 8.5l1.731 1.732-.707.707L12.5 9.207l-1.732 1.732-.707-.707L11.793 8.5 10.06 6.77l.707-.707 1.733 1.73 1.731-1.731z",
person:
"M8 2a1 1 0 110 2 1 1 0 010-2zm0-1a2 2 0 100 4 2 2 0 000-4zm1.23 4.49H6.77A1.77 1.77 0 005 7.26V9.9A1.06 1.06 0 006 11v2.33a1.2 1.2 0 001.2 1.2h1.6a1.2 1.2 0 001.2-1.24V11a1.06 1.06 0 001-1.1V7.26a1.77 1.77 0 00-1.77-1.77zM6 10V7.26a.76.76 0 01.77-.77h2.46a.76.76 0 01.77.77V10H9v3.31a.2.2 0 01-.2.2H7.2a.2.2 0 01-.2-.2V10H6z",
arrowLeft:
"M7 3.093l-5 5V8.8l5 5 .707-.707-4.146-4.147H14v-1H3.56L7.708 3.8 7 3.093z",
plus: "M14 7v1H8v6H7V8H1V7h6V1h1v6h6z",
close:
"M8 8.707l3.646 3.647.708-.707L8.707 8l3.647-3.646-.707-.708L8 7.293 4.354 3.646l-.707.708L7.293 8l-3.646 3.646.707.708L8 8.707z",
cog:
"M19.85 8.75l4.15.83v4.84l-4.15.83 2.35 3.52-3.43 3.43-3.52-2.35-.83 4.15H9.58l-.83-4.15-3.52 2.35-3.43-3.43 2.35-3.52L0 14.42V9.58l4.15-.83L1.8 5.23 5.23 1.8l3.52 2.35L9.58 0h4.84l.83 4.15 3.52-2.35 3.43 3.43-2.35 3.52zm-1.57 5.07l4-.81v-2l-4-.81-.54-1.3 2.29-3.43-1.43-1.43-3.43 2.29-1.3-.54-.81-4h-2l-.81 4-1.3.54-3.43-2.29-1.43 1.43L6.38 8.9l-.54 1.3-4 .81v2l4 .81.54 1.3-2.29 3.43 1.43 1.43 3.43-2.29 1.3.54.81 4h2l.81-4 1.3-.54 3.43 2.29 1.43-1.43-2.29-3.43.54-1.3zm-8.186-4.672A3.43 3.43 0 0 1 12 8.57 3.44 3.44 0 0 1 15.43 12a3.43 3.43 0 1 1-5.336-2.852zm.956 4.274c.281.188.612.288.95.288A1.7 1.7 0 0 0 13.71 12a1.71 1.71 0 1 0-2.66 1.422z",
triangle: "M8 4l4 6.905H4L8 4z",
dot: "M10 8a2 2 0 1 1-4 0 2 2 0 0 1 4 0z",
feedback:
"M7.549 10.078c.46.182.88.424 1.258.725.378.3.701.65.97 1.046a4.829 4.829 0 0 1 .848 2.714V15H9.75v-.438a3.894 3.894 0 0 0-1.155-2.782 4.054 4.054 0 0 0-1.251-.84 3.898 3.898 0 0 0-1.532-.315A3.894 3.894 0 0 0 3.03 11.78a4.06 4.06 0 0 0-.84 1.251c-.206.474-.31.985-.315 1.531V15H1v-.438a4.724 4.724 0 0 1 .848-2.713 4.918 4.918 0 0 1 2.229-1.77 2.994 2.994 0 0 1-.555-.493 3.156 3.156 0 0 1-.417-.602 2.942 2.942 0 0 1-.26-.683 3.345 3.345 0 0 1-.095-.739c0-.423.08-.82.24-1.189a3.095 3.095 0 0 1 1.626-1.627 3.067 3.067 0 0 1 2.386-.007 3.095 3.095 0 0 1 1.627 1.627 3.067 3.067 0 0 1 .157 1.928c-.06.237-.148.465-.266.684a3.506 3.506 0 0 1-.417.608c-.16.187-.345.35-.554.492zM5.812 9.75c.301 0 .584-.057.848-.17a2.194 2.194 0 0 0 1.162-1.163c.119-.269.178-.554.178-.854a2.138 2.138 0 0 0-.643-1.538 2.383 2.383 0 0 0-.697-.472 2.048 2.048 0 0 0-.848-.178c-.3 0-.583.057-.847.17a2.218 2.218 0 0 0-1.17 1.17c-.113.264-.17.547-.17.848 0 .3.057.583.17.847.115.264.27.497.466.697a2.168 2.168 0 0 0 1.552.643zM15 1v7h-1.75l-2.625 2.625V8H9.75v-.875h1.75v1.388l1.388-1.388h1.237v-5.25h-8.75v1.572a7.255 7.255 0 0 0-.438.069 2.62 2.62 0 0 0-.437.123V1H15z",
search:
"M15.25 0a8.25 8.25 0 0 0-6.18 13.72L1 22.88l1.12 1 8.05-9.12A8.251 8.251 0 1 0 15.25.01V0zm0 15a6.75 6.75 0 1 1 0-13.5 6.75 6.75 0 0 1 0 13.5z",
refresh:
"M5.563 2.516A6.001 6.001 0 0 0 8 14 6 6 0 0 0 9.832 2.285l-.302.953A5.002 5.002 0 0 1 8 13a5 5 0 0 1-2.88-9.088l.443-1.396z",
arrowRight:
"M9 13.887l5-5V8.18l-5-5-.707.707 4.146 4.147H2v1h10.44L8.292 13.18l.707.707z",
};
export function Codicon({
name,
...props
}: React.SVGProps<SVGSVGElement> & { name: keyof typeof paths }) {
return (
<svg
width={24}
height={24}
viewBox={name === "cog" || name === "search" ? "0 0 24 24" : "0 0 16 16"}
xmlns="http://www.w3.org/2000/svg"
fill="currentColor"
{...props}
>
<path d={paths[name]} />
</svg>
);
}
import * as React from "react";
export function PawPrint(props: React.SVGProps<SVGSVGElement>) {
return (
<svg
xmlns="http://www.w3.org/2000/svg"
viewBox="0 0 480.004 480.004"
width={24}
height={24}
fill="currentColor"
{...props}
>
<path d="M242.458 173.571c-52.852 0-94.947 25.925-121.733 74.971C92.922 299.45 89.92 358.941 89.92 382.156c0 23.728 4.575 45.537 12.882 61.41 10.223 19.534 26.238 30.292 45.095 30.292 34.276 0 51.596-17.422 65.512-31.42 11.417-11.485 18.441-17.983 30.007-17.983 13.339 0 19.57 6.163 29.002 15.49 13.483 13.335 30.263 29.93 72.095 29.93 31.152 0 50.505-33.612 50.505-87.719.001-80.644-41.418-208.585-152.56-208.585zm132.561 208.585c0 33.707-9.433 67.719-30.505 67.719-33.613 0-46.027-12.277-58.032-24.15-10.081-9.97-21.507-21.27-43.065-21.27-20.449 0-32.519 12.141-44.191 23.882-13.047 13.124-25.371 25.521-51.329 25.521-26.234 0-37.977-36.012-37.977-71.702 0-21.68 2.763-77.164 28.358-124.027 23.395-42.837 58.446-64.558 104.18-64.558 52.886 0 84.163 35.204 101.088 64.737 19.708 34.385 31.473 80.684 31.473 123.848zM132.998 186.134c-.821-17.752-8.14-36.206-20.609-51.964-17.141-21.66-40.882-34.591-63.508-34.591-11.521 0-22.034 3.498-30.405 10.118C5.728 119.78-.805 135.786.079 154.765c.827 17.746 8.151 36.197 20.622 51.956 17.145 21.664 40.894 34.598 63.527 34.598 11.507 0 22.007-3.49 30.38-10.104 12.737-10.1 19.267-26.11 18.39-45.081zm-30.801 29.397c-4.861 3.84-10.907 5.787-17.969 5.787-16.326 0-34.659-10.349-47.844-27.009-9.895-12.502-15.692-26.877-16.327-40.475-.576-12.369 3.268-22.473 10.825-28.45 4.871-3.852 10.926-5.805 17.999-5.805 16.319 0 34.644 10.347 47.825 27.001 19.32 24.416 21.828 55.997 5.491 68.951z" />
<path d="M461.54 109.696c-8.37-6.619-18.886-10.117-30.412-10.117-22.644 0-46.39 12.932-63.52 34.591-12.47 15.759-19.789 34.214-20.607 51.965-.875 18.974 5.662 34.984 18.414 45.088 8.355 6.604 18.855 10.095 30.366 10.095 22.309 0 46.656-13.259 63.533-34.598 12.468-15.754 19.788-34.207 20.611-51.957.881-18.973-5.646-34.976-18.385-45.067zm-17.909 84.614c-13.176 16.659-31.51 27.007-47.848 27.007-7.064 0-13.108-1.946-17.955-5.778-16.359-12.962-13.856-44.542 5.466-68.962 13.171-16.654 31.501-26.999 47.835-26.999 7.078 0 13.135 1.953 18 5.799 16.343 12.949 13.826 44.515-5.498 68.933zm-151.068-25.942c1.981.275 4.005.415 6.014.415h.004c30.268 0 58.456-31.354 65.568-72.932 4.49-26.235-.355-52.185-12.962-69.415-8.25-11.276-19.17-18.147-31.563-19.868a43.792 43.792 0 00-6.059-.422c-30.207 0-58.37 31.331-65.51 72.885-4.488 26.24.354 52.199 12.954 69.44 8.241 11.279 19.149 18.159 31.554 19.897zm-24.796-85.956c5.421-31.55 25.538-56.265 45.798-56.265 1.097 0 2.201.077 3.295.23 8.741 1.214 14.714 7.125 18.186 11.87 9.476 12.95 12.985 33.224 9.39 54.232-5.401 31.575-25.544 56.307-45.857 56.305-1.092 0-2.189-.076-3.251-.224-8.732-1.224-14.701-7.14-18.171-11.888-9.475-12.964-12.984-33.25-9.39-54.26zm-82.509 86.372c2.009 0 4.033-.14 6.031-.417 12.374-1.735 23.268-8.607 31.504-19.871 12.606-17.239 17.463-43.204 12.99-69.47-7.139-41.547-35.304-72.878-65.515-72.878-2.023 0-4.061.142-6.046.42-12.404 1.722-23.319 8.59-31.566 19.861-12.605 17.229-17.449 43.181-12.958 69.422 7.111 41.578 35.296 72.933 65.56 72.933zM148.796 38.239c3.469-4.742 9.438-10.648 18.189-11.863a23.65 23.65 0 013.282-.229c20.264 0 40.383 24.714 45.802 56.25 3.581 21.032.06 41.328-9.42 54.293-3.467 4.741-9.428 10.648-18.125 11.869a23.885 23.885 0 01-3.266.225c-20.308 0-40.446-24.732-45.847-56.306-3.597-21.015-.088-41.292 9.385-54.239z" />
</svg>
);
}
import * as React from "react";
export function Logo(props: React.SVGProps<SVGSVGElement>) {
return (
<svg
xmlns="http://www.w3.org/2000/svg"
xmlnsXlink="http://www.w3.org/1999/xlink"
viewBox="0 0 494.28 131.12"
{...props}
>
<g data-name="Layer 2">
<g data-name="Layer 1">
<path
d="M135.08 66.6h13.61c7.77 0 13 1.59 17.28 5.33 4.89 4.32 7.56 10.59 7.56 18.15s-2.74 14.61-7.63 19c-4 3.6-9.72 5.54-16.64 5.54h-14.18zm9.86 39.89h1.87c5.26 0 8.5-.93 11.31-3.31 3.46-3 5.4-7.49 5.4-12.82 0-9.57-5.83-15.76-15-15.76h-3.6zM215.07 98.29c0 9.79-8 17.49-18.22 17.49-10.51 0-18.07-7.41-18.07-17.78 0-10.59 7.42-17.86 18.22-17.86 10.51 0 18.07 7.63 18.07 18.15zM188 98c0 5.76 3.6 9.65 8.93 9.65s8.92-4 8.92-9.51c0-6-3.38-9.79-8.85-9.79s-9 3.81-9 9.65zM246.68 81.29h9v30.89c0 6.84-1.3 10.8-4.47 13.9s-8.13 5-13.82 5A22.24 22.24 0 01221 124.5l5.69-5.77a13.13 13.13 0 0010.3 4.61c6.12 0 9.65-3.6 9.65-9.86v-3.39a14.23 14.23 0 01-11.74 5.69c-9.14 0-15.77-7.41-15.77-17.57 0-10.8 6.7-18.07 16.71-18.07a12.51 12.51 0 0110.8 5.26zm-17.79 16.49c0 5.76 3.75 9.87 9 9.87a9.19 9.19 0 006.92-3.1 9.31 9.31 0 001.72-5.91c0-3.59-.79-6-2.44-7.77a9.25 9.25 0 00-6.27-2.59c-5.26 0-8.93 3.96-8.93 9.5zM272 100.16a8.9 8.9 0 0016.42 3.17l8.42 2.3a17.68 17.68 0 01-16.2 10.15c-10.58 0-18.14-7.49-18.14-17.85s7.27-17.79 17.56-17.79c10.66 0 18.08 7.49 18.08 18.22l-.07 1.8zm16.71-5.26a8.14 8.14 0 00-8.5-7.06c-4.39 0-7.35 2.52-8.21 7.06zM315.08 114.63h-9.87v-48h9.87v20.28h18.21V66.6h9.87v48h-9.87V95h-18.21zM386.86 98.29c0 9.79-8 17.49-18.22 17.49-10.51 0-18.07-7.41-18.07-17.78 0-10.59 7.42-17.86 18.22-17.86 10.51 0 18.07 7.63 18.07 18.15zM359.79 98c0 5.76 3.6 9.65 8.93 9.65s8.92-4 8.92-9.51c0-6-3.38-9.79-8.85-9.79s-9 3.81-9 9.65zM422.21 81.29v21.1c0 4.9-1 7.63-3.74 9.94a16.62 16.62 0 01-10.8 3.45c-4.61 0-8.57-1.29-11.16-3.67-2.38-2.16-3.53-5.33-3.53-9.72v-21.1h9.14v20.52c0 3.75 2.24 6.27 5.55 6.27s5.4-2.16 5.4-5.69v-21.1zM444.89 90.94v-.58c0-2-1.8-3.45-4.25-3.45-2.23 0-3.67 1.22-3.67 3a2.74 2.74 0 00.94 2.16c1 .8 1.08.8 5.68 2.23C451.08 96.63 454 99.58 454 105c0 6.48-5.19 10.8-13 10.8s-12.74-4-13.1-10.94h8.78c.44 2.88 1.73 4.17 4.4 4.17 2.16 0 3.67-1.29 3.67-3.16s-1.3-3-6.19-4.47c-7.2-2.16-10.44-5.4-10.44-10.65 0-6.34 5.11-10.59 12.88-10.59s12.25 4 12.46 10.8zM468.15 100.16a8.89 8.89 0 0016.41 3.17l8.43 2.3a17.69 17.69 0 01-16.2 10.15c-10.59 0-18.15-7.49-18.15-17.85s7.27-17.79 17.57-17.79c10.66 0 18.07 7.49 18.07 18.22l-.07 1.8zm16.7-5.26a8.13 8.13 0 00-8.5-7.06c-4.39 0-7.34 2.52-8.2 7.06zM129 53.63S106.59 46.59 65.28 0C53.63 8.38 33.52 39.22 0 53l6 8.33c2.5-.59 8.47-3.64 8.47-3.64l15.08 57H50.7S32.19 63 67 63c32.52 0 13.16 51.59 13.16 51.59h21.45l14.27-57.42a66.22 66.22 0 008.44 4.08z"
fill="#6b6659"
/>
<image
data-name="Layer 0"
width={474}
height={492}
transform="matrix(.13 0 0 .13 34.41 50.05)"
xlinkHref=""
/>
</g>
</g>
</svg>
);
}
import * as React from "react";
/* SVG from iconmonstr, https://iconmonstr.com/license/ */
export function TwitterIcon(props: React.SVGProps<SVGSVGElement>) {
return (
<svg
xmlns="http://www.w3.org/2000/svg"
width={24}
height={24}
fill="white"
viewBox="0 0 24 24"
>
<path d="M24 4.557c-.883.392-1.832.656-2.828.775 1.017-.609 1.798-1.574 2.165-2.724-.951.564-2.005.974-3.127 1.195-.897-.957-2.178-1.555-3.594-1.555-3.179 0-5.515 2.966-4.797 6.045-4.091-.205-7.719-2.165-10.148-5.144-1.29 2.213-.669 5.108 1.523 6.574-.806-.026-1.566-.247-2.229-.616-.054 2.281 1.581 4.415 3.949 4.89-.693.188-1.452.232-2.224.084.626 1.956 2.444 3.379 4.6 3.419-2.07 1.623-4.678 2.348-7.29 2.04 2.179 1.397 4.768 2.212 7.548 2.212 9.142 0 14.307-7.721 13.995-14.646.962-.695 1.797-1.562 2.457-2.549z" />
</svg>
);
}
import * as React from "react";
export function PeopleIcon(props: React.SVGProps<SVGSVGElement>) {
return (
<svg
xmlns="http://www.w3.org/2000/svg"
viewBox="0 0 24 24"
width={18}
height={18}
{...props}
>
<path d="M0 0h24v24H0z" fill="none" />
<path d="M16 11c1.66 0 2.99-1.34 2.99-3S17.66 5 16 5c-1.66 0-3 1.34-3 3s1.34 3 3 3zm-8 0c1.66 0 2.99-1.34 2.99-3S9.66 5 8 5C6.34 5 5 6.34 5 8s1.34 3 3 3zm0 2c-2.33 0-7 1.17-7 3.5V19h14v-2.5c0-2.33-4.67-3.5-7-3.5zm8 0c-.29 0-.62.02-.97.05 1.16.84 1.97 1.97 1.97 3.45V19h6v-2.5c0-2.33-4.67-3.5-7-3.5z" />
</svg>
);
}
import * as React from "react";
export function MuteIcon(props: React.SVGProps<SVGSVGElement>) {
return (
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 58 58" {...props}>
<path d="M44.5 28a1 1 0 00-1 1v6c0 7.72-6.28 14-14 14s-14-6.28-14-14v-6a1 1 0 10-2 0v6c0 8.485 6.644 15.429 15 15.949V56h-5a1 1 0 100 2h12a1 1 0 100-2h-5v-5.051c8.356-.52 15-7.465 15-15.949v-6a1 1 0 00-1-1z" />
<path d="M29.5 46c6.065 0 11-4.935 11-11V11c0-6.065-4.935-11-11-11s-11 4.935-11 11v24c0 6.065 4.935 11 11 11zm-9-35c0-4.963 4.038-9 9-9s9 4.037 9 9v24c0 4.963-4.038 9-9 9s-9-4.037-9-9V11zm31.707-6.707a.999.999 0 00-1.414 0l-9 9a.999.999 0 101.414 1.414l9-9a.999.999 0 000-1.414z" />
<path d="M37.207 20.707a.999.999 0 10-1.414-1.414l-14 14a.999.999 0 101.414 1.414l14-14zM12.793 42.293l-7 7a.999.999 0 101.414 1.414l7-7a.999.999 0 10-1.414-1.414z" />
</svg>
);
}
import * as React from "react";
/* SVG from iconmonstr, https://iconmonstr.com/license/ */
export function GitHubIcon(props: React.SVGProps<SVGSVGElement>) {
return (
<svg
xmlns="http://www.w3.org/2000/svg"
width={24}
height={24}
fill="white"
viewBox="0 0 24 24"
>
<path d="M12 0c-6.626 0-12 5.373-12 12 0 5.302 3.438 9.8 8.207 11.387.599.111.793-.261.793-.577v-2.234c-3.338.726-4.033-1.416-4.033-1.416-.546-1.387-1.333-1.756-1.333-1.756-1.089-.745.083-.729.083-.729 1.205.084 1.839 1.237 1.839 1.237 1.07 1.834 2.807 1.304 3.492.997.107-.775.418-1.305.762-1.604-2.665-.305-5.467-1.334-5.467-5.931 0-1.311.469-2.381 1.236-3.221-.124-.303-.535-1.524.117-3.176 0 0 1.008-.322 3.301 1.23.957-.266 1.983-.399 3.003-.404 1.02.005 2.047.138 3.006.404 2.291-1.552 3.297-1.23 3.297-1.23.653 1.653.242 2.874.118 3.176.77.84 1.235 1.911 1.235 3.221 0 4.609-2.807 5.624-5.479 5.921.43.372.823 1.102.823 2.222v3.293c0 .319.192.694.801.576 4.765-1.589 8.199-6.086 8.199-11.386 0-6.627-5.373-12-12-12z" />
</svg>
);
}
import React from "react";
import { Volume2 } from "react-feather";
import {
PossibleSoundEffect,
useSoundEffectStore,
} from "../modules/sound-effects/useSoundEffectStore";
import { Checkbox } from "./Checkbox";
import { useTypeSafeTranslation } from "../utils/useTypeSafeTranslation";
interface ChatSettingsProps {}
const capitalize = (s: string) =>
s.length ? s[0].toUpperCase() + s.slice(1) : s;
const camelToReg = (str: string) =>
str.replace(/[A-Z]/g, (letter) => ` ${letter}`);
export const SoundEffectSettings: React.FC<ChatSettingsProps> = () => {
const [
soundEffectSettings,
setSetting,
playSoundEffect,
] = useSoundEffectStore((x) => [x.settings, x.setSetting, x.playSoundEffect]);
const { t } = useTypeSafeTranslation();
return (
<>
<h1 className={`py-8 text-4xl`}>
{t("pages.soundEffectSettings.header")}
</h1>
{Object.keys(soundEffectSettings).map((k) => {
return (
<div className={`flex`} key={k}>
<Checkbox
value={soundEffectSettings[k as PossibleSoundEffect]}
label={capitalize(camelToReg(k))}
onChange={() =>
setSetting(
k as PossibleSoundEffect,
!soundEffectSettings[k as PossibleSoundEffect]
)
}
/>
<button
onClick={() => playSoundEffect(k as PossibleSoundEffect, true)}
className={`ml-2`}
>
<Volume2 />
</button>
</div>
);
})}
</>
);
};
import React from "react";
interface CircleButtonProps {
size?: number;
onClick: () => void;
title?: string;
}
export const CircleButton: React.FC<CircleButtonProps> = ({
size = 60,
title,
onClick,
children,
}) => {
return (
<button
style={{
height: size,
width: size,
}}
title={title}
className={`rounded-full border border-simple-gray-80 bg-simple-gray-2b flex items-center justify-center`}
onClick={onClick}
>
{children}
</button>
);
};
import React from "react";
import { MicOff } from "react-feather";
import { BaseUser } from "../types";
import { Avatar } from "./Avatar";
import GlassesDoge from "../../assets/glasses-doge.png";
import RegularDoge from "../../assets/regular-doge.png";
interface UserNodeProps {
u: BaseUser;
isMuted: boolean;
isMod: boolean;
isCreator: boolean;
isSpeaker: boolean;
isSpeaking?: boolean;
onClick: () => void;
}
export const UserNode: React.FC<UserNodeProps> = ({
u,
isMuted,
onClick,
isMod,
isSpeaker,
isCreator,
isSpeaking,
}) => {
let prefix = null;
if (isCreator) {
prefix = (
<img
src={GlassesDoge}
alt="room creator"
style={{ marginLeft: 4, marginBottom: 6 }}
className={`w-4 h-4 ml-1 mb-1.5`}
/>
);
} else if (isMod) {
prefix = (
<img
src={RegularDoge}
alt="room mod"
style={{ marginLeft: 4, marginBottom: 6 }}
className={`w-4 h-4 ml-1 mb-1.5`}
/>
);
}
return (
<button
className={`flex flex-col items-center`}
onClick={onClick}
key={u.id}
>
<div className={`relative`}>
<Avatar
usernameForErrorImg={u.username}
circle
size={70}
active={isSpeaking}
src={u.avatarUrl}
/>
{isMuted && (isCreator || isSpeaker) ? (
<div
className={`absolute -bottom-2 -right-2 bg-blue-500 rounded-full p-1`}
>
<MicOff color="white" size={16} name="mute" />
</div>
) : null}
</div>
<div className={`mt-2 flex w-full justify-center items-center truncate`}>
<span className={`text-sm truncate`}>
{(u.displayName || u.username).trim().split(" ")[0]}
</span>
{prefix}
</div>
</button>
);
};
import normalizeUrl from "normalize-url";
import React, { useEffect, useState } from "react";
import { useHistory } from "react-router-dom";
import { wsend } from "../../createWebsocket";
import { useCurrentRoomStore } from "../../webrtc/stores/useCurrentRoomStore";
import { linkRegex } from "../constants";
import { BaseUser, RoomUser } from "../types";
import { onFollowUpdater } from "../utils/onFollowUpdater";
import { useMeQuery } from "../utils/useMeQuery";
import { useTypeSafeTranslation } from "../utils/useTypeSafeTranslation";
import { Avatar } from "./Avatar";
import { Button } from "./Button";
import { EditProfileModal } from "./EditProfileModal";
interface UserProfileProps {
profile: RoomUser;
}
export const UserProfile: React.FC<UserProfileProps> = ({
profile: userProfile,
}) => {
const history = useHistory();
const { me } = useMeQuery();
const { setCurrentRoom } = useCurrentRoomStore();
// if you edit your profile, me will be updated so we want to use that
const profile: BaseUser | RoomUser =
me?.id === userProfile.id ? me : userProfile;
const [youAreFollowing, setYouAreFollowing] = useState(
"youAreFollowing" in profile ? profile.youAreFollowing : false
);
const _youAreFollowing =
"youAreFollowing" in profile && profile.youAreFollowing;
useEffect(() => {
if (_youAreFollowing) {
setYouAreFollowing(_youAreFollowing);
}
}, [_youAreFollowing]);
const [editProfileModalOpen, setEditProfileModalOpen] = useState(false);
const { t } = useTypeSafeTranslation();
return (
<>
<EditProfileModal
user={profile}
isOpen={editProfileModalOpen}
onRequestClose={() => setEditProfileModalOpen(false)}
/>
<div className={`mb-4 flex justify-between align-center`}>
<Avatar src={profile.avatarUrl} />
{me?.id === profile.id ? (
<div>
<Button
onClick={() => {
setEditProfileModalOpen(true);
}}
variant="small"
>
{t("pages.viewUser.editProfile")}
</Button>
</div>
) : null}
{me?.id === profile.id ||
userProfile.youAreFollowing === null ||
userProfile.youAreFollowing === undefined ? null : (
<div>
<Button
onClick={() => {
wsend({
op: "follow",
d: {
userId: profile.id,
value: !youAreFollowing,
},
});
setYouAreFollowing(!youAreFollowing);
onFollowUpdater(setCurrentRoom, me, profile);
}}
variant="small"
>
{youAreFollowing ? "following" : "follow"}
</Button>
</div>
)}
</div>
<div className={`font-semibold`}>{profile.displayName}</div>
<div className={`my-1 flex`}>
<div>@{profile.username}</div>
{me?.id !== profile.id && userProfile.followsYou ? (
<div className={`ml-2 text-simple-gray-3d`}>
{t("pages.viewUser.followsYou")}
</div>
) : null}
</div>
<div className={`flex my-4`}>
<button
onClick={() => {
wsend({
op: `fetch_follow_list`,
d: { isFollowing: false, userId: profile.id, cursor: 0 },
});
history.push(`/followers/${profile.id}`);
}}
className={`mr-3`}
>
<span className={`font-bold`}>{profile.numFollowers}</span>{" "}
{t("pages.viewUser.followers")}
</button>
<button
onClick={() => {
wsend({
op: `fetch_follow_list`,
d: { isFollowing: true, userId: profile.id, cursor: 0 },
});
history.push(`/following/${profile.id}`);
}}
>
<span className={`font-bold`}>{profile.numFollowing}</span>{" "}
{t("pages.viewUser.following")}
</button>
</div>
<div className="mb-4 whitespace-pre-wrap break-all">
{profile.bio?.split(" ").map((chunk, i) => {
if (linkRegex.test(chunk)) {
try {
return (
<a
key={i}
href={normalizeUrl(chunk)}
target="_blank"
rel="noreferrer"
className="text-blue-500 p-0 hover:underline"
>
{chunk}{" "}
</a>
);
} catch {}
}
return <span key={i}>{chunk} </span>;
})}
</div>
</>
);
};
import React, { useState } from "react";
import { Button } from "./Button";
import { useTypeSafeTranslation } from "../utils/useTypeSafeTranslation";
interface InviteButtonProps {
onClick: () => void;
}
export const InviteButton: React.FC<InviteButtonProps> = ({ onClick }) => {
const [invited, setInvited] = useState(false);
const { t } = useTypeSafeTranslation();
return (
<Button
onClick={() => {
onClick();
setInvited(true);
}}
disabled={invited}
variant="small"
>
{invited
? t("components.inviteButton.invited")
: t("components.inviteButton.inviteToRoom")}
</Button>
);
};
import React from "react";
import { useMutation, useQuery, useQueryClient } from "react-query";
import { wsFetch, wsMutation } from "../../createWebsocket";
import { PaginatedBaseUsers } from "../types";
import { Avatar } from "./Avatar";
import { Button } from "./Button";
import { useTypeSafeTranslation } from "../utils/useTypeSafeTranslation";
interface BlockedFromRoomUsersProps {}
export const GET_BLOCKED_FROM_ROOM_USERS = "get_blocked_from_room_users";
const UnbanButton = ({
userId,
offset,
}: {
userId: string;
offset: number;
}) => {
const queryClient = useQueryClient();
const { mutateAsync, isLoading } = useMutation(wsMutation, {
onSuccess: () => {
queryClient.setQueryData<PaginatedBaseUsers | undefined>(
[GET_BLOCKED_FROM_ROOM_USERS, offset],
(d) => {
if (!d) {
return d;
}
return {
...d,
users: d.users.filter((x) => x.id !== userId),
};
}
);
},
});
const { t } = useTypeSafeTranslation();
return (
<Button
loading={isLoading}
onClick={() => {
mutateAsync({ op: "unban_from_room", d: { userId } });
}}
variant={`small`}
>
{t("components.blockedFromRoomUsers.unban")}
</Button>
);
};
export const BlockedFromRoomUsersPage: React.FC<{
offset: number;
onLoadMore: (newOffset: number) => void;
isLastPage: boolean;
isOnlyPage: boolean;
}> = ({ offset, onLoadMore, isOnlyPage, isLastPage }) => {
const queryClient = useQueryClient();
const { isLoading, data } = useQuery<PaginatedBaseUsers>(
[GET_BLOCKED_FROM_ROOM_USERS, offset],
() =>
wsFetch<PaginatedBaseUsers>({
op: GET_BLOCKED_FROM_ROOM_USERS,
d: { offset },
}),
{ enabled: false }
);
const { t } = useTypeSafeTranslation();
if (isLoading) {
return <div className={`mt-8`}>{t("common.loading")}</div>;
}
if (isOnlyPage && data?.users.length === 0) {
return (
<div className={`mt-2`}>
{t("components.blockedFromRoomUsers.noBans")}
</div>
);
}
if (!data) {
return null;
}
return (
<>
{data.users.map((profile) => (
<div
className={`border-b border-solid border-simple-gray-3c flex py-4 px-2 items-center`}
key={profile.id}
>
<div>
<Avatar size={60} src={profile.avatarUrl} />
</div>
<div className={`ml-4 flex-1 mr-4`}>
<div className={`text-lg`}>{profile.displayName}</div>
<div style={{ color: "" }}>@{profile.username}</div>
</div>
<UnbanButton offset={offset} userId={profile.id} />
</div>
))}
{isLastPage && data.nextCursor ? (
<div className={`flex items-center justify-center mt-4`}>
<Button
variant="small"
onClick={() => {
queryClient.prefetchQuery(
[GET_BLOCKED_FROM_ROOM_USERS, data.nextCursor],
() =>
wsFetch<PaginatedBaseUsers>({
op: GET_BLOCKED_FROM_ROOM_USERS,
d: { offset: data.nextCursor },
}),
{ staleTime: 0 }
);
onLoadMore(data.nextCursor!);
}}
>
{t("common.loadMore")}
</Button>
</div>
) : null}
</>
);
};
export const BlockedFromRoomUsers: React.FC<BlockedFromRoomUsersProps> = ({}) => {
const [offsets, setOffsets] = React.useState([0]);
const { t } = useTypeSafeTranslation();
return (
<>
<div className={`mt-4`}>
<h1 className={`text-xl`}>
{t("components.blockedFromRoomUsers.header")}
</h1>
<div>
{offsets.map((offset, i) => (
<BlockedFromRoomUsersPage
key={offset}
offset={offset}
isLastPage={i === offsets.length - 1}
isOnlyPage={offsets.length === 1}
onLoadMore={(o) => setOffsets([...offsets, o])}
/>
))}
</div>
</div>
</>
);
};
import React from "react";
interface BodyWrapperProps {}
export const BodyWrapper: React.FC<BodyWrapperProps> = ({
children
}) => {
return (
<div className={`px-5`}>
{children}
</div>
);
};
import React from "react";
interface InputErrorMsgProps {}
export const InputErrorMsg: React.FC<InputErrorMsgProps> = ({ children }) => {
return <div className={`text-red-600`}>{children}</div>;
};
import React from "react";
import { Codicon } from "../svgs/Codicon";
import { CurrentRoom, Room } from "../types";
import { useMeQuery } from "../utils/useMeQuery";
interface RoomProps {
active?: boolean;
onClick: () => void;
room: Room | CurrentRoom;
currentRoomId: string | undefined;
}
export const RoomCard: React.FC<RoomProps> = ({
room,
onClick,
active,
currentRoomId,
}) => {
const { me } = useMeQuery();
let n = room.numPeopleInside;
const previewNodes = [];
let userList = room.peoplePreviewList;
if (currentRoomId === room.id && "users" in room) {
n = room.users.length;
userList = room.users;
}
for (let i = 0; i < Math.min(6, userList.length); i++) {
const p = userList[i];
if (p.id === me?.id && currentRoomId !== room.id) {
n--;
continue;
}
previewNodes.push(
<div
key={p.id}
className={`text-left text-simple-gray-d9 ${!i ? "mt-1.5" : "mt-0.5"}`}
>
{p.displayName?.slice(0, 50)}
</div>
);
if (i >= 4 && previewNodes.length >= 5) {
break;
}
}
return (
<div>
<button
onClick={onClick}
className={`w-full ${
active ? "bg-simple-gray-4d" : "bg-simple-gray-33"
} hover:bg-simple-gray-69 active:bg-simple-gray-23 py-2.5 px-5 rounded-lg`}
>
<div className={`flex text-white`}>
<div
style={{
display: "-webkit-box",
WebkitBoxOrient: "vertical",
WebkitLineClamp: 3,
}}
className={`text-left flex-1 text-xl text-simple-gray-d9 text-ellipsis overflow-hidden break-all`}
>
{room.name?.slice(0, 100)}
</div>
<div className={`flex items-center`}>
<Codicon name="person" /> {n}
</div>
</div>
{previewNodes}
</button>
</div>
);
};
import React from "react";
import { createWebSocket } from "../../createWebsocket";
import { useTypeSafeTranslation } from "../utils/useTypeSafeTranslation";
import { Button } from "./Button";
import { CenterLayout } from "./CenterLayout";
import { Wrapper } from "./Wrapper";
interface WsKilledMessageProps {}
export const WsKilledMessage: React.FC<WsKilledMessageProps> = ({}) => {
const { t } = useTypeSafeTranslation();
return (
<div className="flex items-center h-full justify-around">
<CenterLayout>
<Wrapper>
<div className={`px-4`}>
<div className={`mb-4 mt-8 text-xl`}>
{t("components.wsKilled.description")}
</div>
<Button
onClick={() => {
createWebSocket(true);
}}
>
{t("components.wsKilled.reconnect")}
</Button>
</div>
</Wrapper>
</CenterLayout>
</div>
);
};
import React from "react";
import { X } from "react-feather";
import { wsend } from "../../createWebsocket";
import { useCurrentRoomStore } from "../../webrtc/stores/useCurrentRoomStore";
import { renameRoomAndMakePrivate } from "../../webrtc/utils/renameRoomAndMakePrivate";
import { renameRoomAndMakePublic } from "../../webrtc/utils/renameRoomAndMakePublic";
import { BlockedFromRoomUsers } from "./BlockedFromRoomUsers";
import { Button } from "./Button";
import { Modal } from "./Modal";
import { useTypeSafeTranslation } from "../utils/useTypeSafeTranslation";
interface RoomSettingsModalProps {
open: boolean;
onRequestClose: () => void;
}
export const RoomSettingsModal: React.FC<RoomSettingsModalProps> = ({
open,
onRequestClose,
}) => {
const { currentRoom, setCurrentRoom } = useCurrentRoomStore();
const { t } = useTypeSafeTranslation();
return (
<Modal isOpen={open} onRequestClose={onRequestClose}>
<button
onClick={() => {
onRequestClose();
}}
className={`p-2 -ml-3`}
>
<X />
</button>
{currentRoom ? (
<>
<label className={`flex items-center my-8`} htmlFor="auto-speaker">
<input
checked={!currentRoom.autoSpeaker}
onChange={(e) => {
setCurrentRoom((cr) =>
!cr
? cr
: {
...cr,
autoSpeaker: !e.target.checked,
}
);
wsend({
op: "set_auto_speaker",
d: { value: !e.target.checked },
});
}}
id="auto-speaker"
type="checkbox"
/>
<span className={`ml-2`}>
{t("components.modals.roomSettingsModal.requirePermission")}
</span>
</label>
{currentRoom.isPrivate ? (
<Button
onClick={() => {
renameRoomAndMakePublic(currentRoom.name);
onRequestClose();
}}
>
{t("components.modals.roomSettingsModal.makePublic")}
</Button>
) : (
<Button
onClick={() => {
renameRoomAndMakePrivate(currentRoom.name);
onRequestClose();
}}
>
{t("components.modals.roomSettingsModal.makePrivate")}
</Button>
)}
{open ? <BlockedFromRoomUsers /> : null}
</>
) : null}
</Modal>
);
};
import * as React from "react";
import { Modal } from "./Modal";
import create from "zustand";
import { combine } from "zustand/middleware";
import { Button } from "./Button";
import { useTypeSafeTranslation } from "../utils/useTypeSafeTranslation";
interface Props {}
type Fn = () => void;
const useConfirmModalStore = create(
combine(
{
message: "",
onConfirm: undefined as undefined | Fn,
},
(set) => ({
close: () => set({ onConfirm: undefined, message: "" }),
set,
})
)
);
export const modalConfirm = (message: string, onConfirm: Fn) => {
useConfirmModalStore.getState().set({ onConfirm, message });
};
export const ConfirmModal: React.FC<Props> = () => {
const { onConfirm, message, close } = useConfirmModalStore();
const { t } = useTypeSafeTranslation();
return (
<Modal isOpen={!!onConfirm} onRequestClose={() => close()}>
<div>{message}</div>
<div className={`flex mt-12`}>
<Button
type="button"
onClick={close}
className={`mr-1.5`}
color="secondary"
>
{t("common.cancel")}
</Button>
<Button
onClick={() => {
close();
onConfirm?.();
}}
type="submit"
className={`ml-1.5`}
>
{t("common.yes")}
</Button>
</div>
</Modal>
);
};
import React from "react";
import { useHistory } from "react-router-dom";
import DogeHouse from "../../assets/dogehouse.png";
import { ArrowLeft } from "react-feather";
interface BackbarProps {
actuallyGoBack?: boolean;
}
export const Backbar: React.FC<BackbarProps> = ({
children,
actuallyGoBack,
}) => {
const history = useHistory();
return (
<div className={`sticky top-0 z-10 flex py-4 mb-12 border-b border-simple-gray-80 bg-simple-gray-26 h-20`}>
{actuallyGoBack ? (
<button
className={`hover:bg-blue-700 px-2`}
onClick={() => {
history.goBack();
}}
>
<ArrowLeft color="#fff" size={30} />
</button>
) : (
<button
className={`hover:bg-blue-700 px-2`}
onClick={() => {
history.push("/");
}}
>
<img className={`w-12`} src={DogeHouse} alt="dogehouse" />
</button>
)}
{children}
</div>
);
};
import React, { useState } from "react";
import { useMicPermErrorStore } from "../../webrtc/stores/useMicPermErrorStore";
import { sendVoice } from "../../webrtc/utils/sendVoice";
import { isIOS } from "../utils/isIOS";
import { Button } from "./Button";
import { useTypeSafeTranslation } from "../utils/useTypeSafeTranslation";
interface MicPermissionBannerProps {}
export const MicPermissionBanner: React.FC<MicPermissionBannerProps> = () => {
const { error, set } = useMicPermErrorStore();
const [count, setCount] = useState(0);
const { t } = useTypeSafeTranslation();
if (!error) {
return null;
}
return (
<div className={`p-4 bg-simple-gray-3c`}>
<div className={`font-semibold text-xl mb-4 bg-red-400`}>
{t("components.micPermissionBanner.permissionDenied")}
</div>
<div className="flex space-x-2">
<Button color="secondary" onClick={() => set({ error: false })}>
{t("components.micPermissionBanner.dismiss")}
</Button>
<Button
onClick={() => {
if (count < 2 && !isIOS()) {
sendVoice();
setCount((c) => c + 1);
} else {
window.location.reload();
}
}}
>
{t("components.micPermissionBanner.tryAgain")}
</Button>
</div>
</div>
);
};
import React, { useState } from "react";
import {
MessageSquare,
Mic,
MicOff,
PhoneMissed,
Settings,
UserPlus,
} from "react-feather";
import { useQueryClient } from "react-query";
import { useHistory, useLocation } from "react-router-dom";
import { wsend, wsFetch } from "../../createWebsocket";
import { useCurrentRoomStore } from "../../webrtc/stores/useCurrentRoomStore";
import { useMuteStore } from "../../webrtc/stores/useMuteStore";
import { useCurrentRoomInfo } from "../atoms";
import { RoomChat } from "../modules/room-chat/RoomChat";
import { useRoomChatMentionStore } from "../modules/room-chat/useRoomChatMentionStore";
import { useRoomChatStore } from "../modules/room-chat/useRoomChatStore";
import { useShouldFullscreenChat } from "../modules/room-chat/useShouldFullscreenChat";
import { PaginatedBaseUsers } from "../types";
import { GET_BLOCKED_FROM_ROOM_USERS } from "./BlockedFromRoomUsers";
import { modalConfirm } from "./ConfirmModal";
import { Footer } from "./Footer";
import { RoomSettingsModal } from "./RoomSettingsModal";
import { useTypeSafeTranslation } from "../utils/useTypeSafeTranslation";
import { KeybindListener } from './KeybindListener';
interface BottomVoiceControlProps {}
const iconSize = 24;
const iconColor = "#8C8C8C";
const buttonStyle = `px-2.5 text-simple-gray-8c text-sm flex-1`;
export const BottomVoiceControl: React.FC<BottomVoiceControlProps> = ({
children,
}) => {
const queryClient = useQueryClient();
const location = useLocation();
const history = useHistory();
const { currentRoom } = useCurrentRoomStore();
const { muted, setMute } = useMuteStore();
const { canSpeak, isCreator } = useCurrentRoomInfo();
const [settingsOpen, setSettingsOpen] = useState(false);
const [toggleOpen, newUnreadMessages] = useRoomChatStore((s) => [
s.toggleOpen,
s.newUnreadMessages,
]);
const { iAmMentioned } = useRoomChatMentionStore();
const fullscreenChatOpen = useShouldFullscreenChat();
const buttons = [];
const { t } = useTypeSafeTranslation();
if (currentRoom) {
buttons.push(
<button
className={buttonStyle}
key="leave-room"
onClick={() => {
modalConfirm(
t("components.bottomVoiceControl.confirmLeaveRoom"),
() => {
wsend({ op: "leave_room", d: {} });
if (location.pathname.startsWith("/room")) {
history.push("/");
}
}
);
}}
title={t("components.bottomVoiceControl.leaveCurrentRoomBtn")}
>
<PhoneMissed
className={`m-auto mb-1`}
size={iconSize}
color={iconColor}
/>
{t("components.bottomVoiceControl.leave")}
</button>,
<button
className={buttonStyle}
key="chat"
onClick={() => {
toggleOpen();
}}
>
<div className={`flex justify-center`}>
<div className={`relative`}>
<MessageSquare
className={`m-auto mb-1`}
size={iconSize}
color={iconColor}
/>
{newUnreadMessages ? (
<span
className={`absolute rounded-full w-2.5 h-2.5`}
style={{
backgroundColor: iAmMentioned ? "#ff3c00" : "#FF9900",
right: -2,
top: -1,
}}
/>
) : null}
</div>
</div>
Chat
</button>,
<button
className={buttonStyle}
key="invite"
onClick={() => {
wsend({ op: "fetch_invite_list", d: { cursor: 0 } });
history.push("/invite");
}}
title={t("components.bottomVoiceControl.inviteUsersToRoomBtn")}
>
<UserPlus className={`m-auto mb-1`} size={iconSize} color={iconColor} />
{t("components.bottomVoiceControl.invite")}
</button>
);
if (isCreator || canSpeak) {
buttons.push(
<button
className={buttonStyle}
key="mute"
onClick={() => {
wsend({
op: "mute",
d: { value: !muted },
});
setMute(!muted);
}}
title={t("components.bottomVoiceControl.toggleMuteMicBtn")}
>
{muted ? (
<MicOff
className={`m-auto mb-1`}
size={iconSize}
color={iconColor}
/>
) : (
<Mic className={`m-auto mb-1`} size={iconSize} color={iconColor} />
)}
{muted
? t("components.bottomVoiceControl.unmute")
: t("components.bottomVoiceControl.mute")}
</button>
);
}
if (isCreator) {
buttons.push(
<button
className={buttonStyle}
key="to-public-room"
onClick={() => {
queryClient.prefetchQuery(
[GET_BLOCKED_FROM_ROOM_USERS, 0],
() =>
wsFetch<PaginatedBaseUsers>({
op: GET_BLOCKED_FROM_ROOM_USERS,
d: { offset: 0 },
}),
{ staleTime: 0 }
);
setSettingsOpen(true);
}}
title={t("components.bottomVoiceControl.makeRoomPublicBtn")}
>
<Settings
className={`m-auto mb-1`}
size={iconSize}
color={iconColor}
/>
{t("components.bottomVoiceControl.settings")}
</button>
);
}
}
return (
<>
<RoomSettingsModal
open={settingsOpen}
onRequestClose={() => setSettingsOpen(false)}
/>
<div
className={`${
fullscreenChatOpen
? `fixed top-0 left-0 right-0 flex-col flex h-full`
: `sticky`
} bottom-0 w-full`}
>
{fullscreenChatOpen ? null : children}
<RoomChat sidebar={false} />
{currentRoom &&
!fullscreenChatOpen &&
!location.pathname.startsWith("/room") ? (
<button
onClick={() => history.push(`/room/${currentRoom.id}`)}
className={`bg-simple-gray-26 py-5 px-10 w-full flex`}
>
<span
className={`text-simple-gray-a6 overflow-hidden overflow-ellipsis font-semibold`}
>
{currentRoom.name}{" "}
</span>
<span className={`text-blue-500 ml-2`}>
{canSpeak
? t("components.bottomVoiceControl.speaker")
: t("components.bottomVoiceControl.listener")}
</span>
</button>
) : null}
<div
className={`border-simple-gray-80 bg-simple-gray-26 border-t w-full mt-auto p-5`}
>
{currentRoom ? (
<>
<KeybindListener />
<div className={`flex justify-around`}>{buttons}</div>
</>
) : (
<div className={`px-5`}>
<Footer />
</div>
)}
</div>
</div>
</>
);
};
import * as React from "react";
import { Modal } from "./Modal";
import create from "zustand";
import { combine } from "zustand/middleware";
import { Button } from "./Button";
import { useTypeSafeTranslation } from "../utils/useTypeSafeTranslation";
interface Props {}
const useAlertModalStore = create(
combine(
{
message: "",
isOpen: false,
},
(set) => ({
close: () => set({ isOpen: false, message: "" }),
set,
})
)
);
export const modalAlert = (message: string) => {
useAlertModalStore.getState().set({ isOpen: true, message });
};
export const AlertModal: React.FC<Props> = () => {
const { isOpen, message, close } = useAlertModalStore();
const { t } = useTypeSafeTranslation();
return (
<Modal isOpen={isOpen} onRequestClose={() => close()}>
<div>{message}</div>
<form
onSubmit={(e) => {
e.preventDefault();
close();
}}
>
<div className={`flex mt-12`}>
<Button type="submit">{t("common.ok")}</Button>
</div>
</form>
</Modal>
);
};
import React from "react";
interface VolumeSliderProps {
label?: boolean;
max?: string;
volume: number;
onVolume: (n: number) => void;
}
export const VolumeSlider: React.FC<VolumeSliderProps> = ({
label,
max = "100",
volume,
onVolume,
}) => {
return (
<div>
{label ? "volume: " : ""}
{volume}
<input
type="range"
min="1"
max={max}
value={volume}
onChange={(e) => {
const n = parseInt(e.target.value);
onVolume(n);
}}
/>
</div>
);
};
import React from "react";
import ReactModal from "react-modal";
import { wsend } from "../../createWebsocket";
import {
RoomChatMessage,
useRoomChatStore,
} from "../modules/room-chat/useRoomChatStore";
import { Codicon } from "../svgs/Codicon";
import { CurrentRoom, RoomUser } from "../types";
import { Button } from "./Button";
import { modalConfirm } from "./ConfirmModal";
import { UserProfile } from "./UserProfile";
import { UserVolumeSlider } from "./UserVolumeSlider";
import { useTypeSafeTranslation } from "../utils/useTypeSafeTranslation";
interface ProfileModalProps {
onClose: () => void;
profile: RoomUser | null | undefined;
isMe: boolean;
iAmCreator: boolean;
iAmMod: boolean;
room: CurrentRoom;
messageToBeDeleted?: RoomChatMessage | null;
}
const customStyles = {
overlay: {
backgroundColor: "rgba(0,0,0,.5)",
zIndex: 999,
},
content: {
top: "50%",
left: "50%",
right: "auto",
bottom: "auto",
marginRight: "-50%",
transform: "translate(-50%, -50%)",
backgroundColor: "#3c3c3c",
border: "none",
width: "100%",
maxWidth: 500,
},
};
export const ProfileModal: React.FC<ProfileModalProps> = ({
profile,
onClose,
isMe,
iAmCreator,
iAmMod,
room,
messageToBeDeleted,
}) => {
const bannedUserIdMap = useRoomChatStore((s) => s.bannedUserIdMap);
const { t } = useTypeSafeTranslation();
return (
<ReactModal
isOpen={!!profile}
contentLabel="profile"
style={customStyles}
onRequestClose={() => onClose()}
>
{profile ? (
<>
<div className={`mb-4 flex`}>
<button
onClick={() => {
onClose();
}}
className={`p-2 -ml-2`}
>
<Codicon width={24} height={24} name="close" />
</button>
{iAmCreator && !isMe ? (
<div className={`ml-auto`}>
<Button
variant="small"
onClick={() => {
modalConfirm(
t("components.modals.profileModal.blockUserConfirm"),
() => {
onClose();
wsend({
op: "block_user_and_from_room",
d: {
userId: profile.id,
},
});
}
);
}}
>
{t("components.modals.profileModal.blockUser")}
</Button>
</div>
) : null}
</div>
{/* Profile */}
<UserProfile profile={profile} />
{/* User volume */}
{!isMe && profile.roomPermissions?.isSpeaker ? (
<div className={`mb-4`}>
<UserVolumeSlider userId={profile.id} />
</div>
) : null}
{/* Make mod button */}
{!isMe && iAmCreator ? (
<>
<div className={`mb-4`}>
<Button
onClick={() => {
onClose();
wsend({
op: "change_mod_status",
d: {
userId: profile.id,
value: !profile.roomPermissions?.isMod,
},
});
}}
>
{profile.roomPermissions?.isMod
? t("components.modals.profileModal.unmod")
: t("components.modals.profileModal.makeMod")}
</Button>
</div>
</>
) : null}
{/* Add speaker button */}
{!isMe && (iAmCreator || iAmMod) && profile.id !== room.creatorId ? (
<>
{!profile.roomPermissions?.isSpeaker &&
profile.roomPermissions?.askedToSpeak ? (
<div className={`mb-4`}>
<Button
onClick={() => {
onClose();
wsend({
op: "add_speaker",
d: {
userId: profile.id,
},
});
}}
>
{t("components.modals.profileModal.addAsSpeaker")}
</Button>
</div>
) : null}
{/* Set listener */}
{profile.roomPermissions?.isSpeaker ? (
<div className={`mb-4`}>
<Button
onClick={() => {
onClose();
wsend({
op: "set_listener",
d: {
userId: profile.id,
},
});
}}
>
{t("components.modals.profileModal.moveToListener")}
</Button>
</div>
) : null}
{/* Ban from chat */}
{!(profile.id in bannedUserIdMap) ? (
<div className={`mb-4`}>
<Button
onClick={() => {
onClose();
wsend({
op: "ban_from_room_chat",
d: {
userId: profile.id,
},
});
}}
>
{t("components.modals.profileModal.banFromChat")}
</Button>
</div>
) : null}
{/* Block from room */}
<div className="mb-4">
<Button
onClick={() => {
onClose();
wsend({
op: "block_from_room",
d: {
userId: profile.id,
},
});
}}
>
{t("components.modals.profileModal.banFromRoom")}
</Button>
</div>
</>
) : null}
{isMe &&
!iAmCreator &&
(profile.roomPermissions?.askedToSpeak ||
profile.roomPermissions?.isSpeaker) ? (
<div className={`mb-4`}>
<Button
onClick={() => {
onClose();
wsend({
op: "set_listener",
d: {
userId: profile.id,
},
});
}}
>
{t("components.modals.profileModal.goBackToListener")}
</Button>
</div>
) : null}
{/* Delete message */}
{messageToBeDeleted ? (
<Button
color="red"
onClick={() => {
wsend({
op: "delete_room_chat_message",
d: {
messageId: messageToBeDeleted.id,
userId: messageToBeDeleted.userId,
},
});
!onClose || onClose();
}}
>
{t("components.modals.profileModal.deleteMessage")}
</Button>
) : null}
</>
) : null}
</ReactModal>
);
};
import React from "react";
import { useTypeSafeTranslation } from "../utils/useTypeSafeTranslation";
import { LanguageSelector } from "./LanguageSelector";
import { RegularAnchor } from "./RegularAnchor";
interface FooterProps {
isLogin?: boolean;
}
export const Footer: React.FC<FooterProps> = ({ isLogin }) => {
const { t } = useTypeSafeTranslation();
return (
<div className={`justify-between flex text-center`}>
{isLogin ? (
<RegularAnchor href="https://www.youtube.com/watch?v=hy-EhJ_tTQo">
{t("footer.link_1")}
</RegularAnchor>
) : null}
<RegularAnchor href="https://discord.gg/wCbKBZF9cV">
{t("footer.link_2")}
</RegularAnchor>
<RegularAnchor href="https://github.com/benawad/dogehouse/issues">
{t("footer.link_3")}
</RegularAnchor>
{/* cramps footer on mobile @todo think about how to incorporate this without cramping footer and making the footer really tall */}
{/* <RegularAnchor
href="https://github.com/benawad/dogehouse/blob/prod/CHANGELOG.md"
target="_blank"
rel="noreferrer"
>
What's new?
</RegularAnchor> */}
<LanguageSelector />
</div>
);
};
import React from "react";
import { CenterLayout } from "./CenterLayout";
import { RegularAnchor } from "./RegularAnchor";
import { Wrapper } from "./Wrapper";
import { useTypeSafeTranslation } from "../utils/useTypeSafeTranslation";
interface DeviceNotSupportedProps {}
export const DeviceNotSupported: React.FC<DeviceNotSupportedProps> = ({}) => {
const { t } = useTypeSafeTranslation();
return (
<div className="flex items-center h-full justify-around">
<CenterLayout>
<Wrapper>
<div className={`mb-4 mt-8 text-xl`}>
{t("components.deviceNotSupported.notSupported")}{" "}
<RegularAnchor href="https://github.com/benawad/dogehouse/issues">
{t("components.deviceNotSupported.linkText")}
</RegularAnchor>{" "}
{t("components.deviceNotSupported.addSupport")}
</div>
</Wrapper>
</CenterLayout>
</div>
);
};
import { useAtom } from "jotai";
import React, { useCallback, useEffect, useState } from "react";
import { volumeAtom } from "../shared-atoms";
import { useMicIdStore } from "../shared-stores";
import { Button } from "./Button";
import { MuteKeybind, PTTKeybind, ChatKeybind } from "./keyboard-shortcuts";
import { VolumeSlider } from "./VolumeSlider";
import { useTypeSafeTranslation } from "../utils/useTypeSafeTranslation";
interface VoiceSettingsProps {}
export const VoiceSettings: React.FC<VoiceSettingsProps> = () => {
const { micId, setMicId } = useMicIdStore();
const [volume, setVolume] = useAtom(volumeAtom);
const [devices, setDevices] = useState<Array<{ id: string; label: string }>>(
[]
);
const fetchMics = useCallback(() => {
navigator.mediaDevices.getUserMedia({ audio: true }).then(() => {
navigator.mediaDevices
?.enumerateDevices()
.then((devices) =>
setDevices(
devices
.filter(
(device) => device.kind === "audioinput" && device.deviceId
)
.map((device) => ({ id: device.deviceId, label: device.label }))
)
);
});
}, []);
const { t } = useTypeSafeTranslation();
useEffect(() => {
fetchMics();
}, [fetchMics]);
return (
<>
<div className={`mb-2`}>{t("pages.voiceSettings.mic")} </div>
{devices.length ? (
<select
className={`mb-4`}
value={micId}
onChange={(e) => setMicId(e.target.value)}
>
{devices.map(({ id, label }) => (
<option key={id} value={id}>
{label}
</option>
))}
</select>
) : (
<div className={`mb-4`}>{t("pages.voiceSettings.permissionError")}</div>
)}
<div>
<Button
variant="small"
onClick={() => {
fetchMics();
}}
>
{t("pages.voiceSettings.refresh")}
</Button>
</div>
<div className={`mt-8 mb-2`}>{t("pages.voiceSettings.volume")} </div>
<div className={`mb-8`}>
<VolumeSlider volume={volume} onVolume={(n) => setVolume(n)} />
</div>
<MuteKeybind className={`mb-4`} />
<PTTKeybind className={`mb-4`} />
<ChatKeybind />
</>
);
};
import React from "react";
interface PageWrapperProps {}
export const PageWrapper: React.FC<PageWrapperProps> = ({ children }) => {
return (
<div className={`mx-auto max-w-5xl w-full h-full flex relative`}>
{children}
</div>
);
};
import React, { useMemo } from "react";
import { GlobalHotKeys } from "react-hotkeys";
import { wsend } from "../../createWebsocket";
import { useKeyMapStore } from "../../webrtc/stores/useKeyMapStore";
import { useMuteStore } from "../../webrtc/stores/useMuteStore";
import { useRoomChatStore } from "../modules/room-chat/useRoomChatStore";
interface KeybindListenerProps {}
export const KeybindListener: React.FC<KeybindListenerProps> = ({}) => {
const { keyMap } = useKeyMapStore();
const [toggleOpen, newUnreadMessages] = useRoomChatStore((s) => [
s.toggleOpen,
s.newUnreadMessages,
]);
return (
<GlobalHotKeys
allowChanges={true}
keyMap={keyMap}
handlers={useMemo(
() => ({
MUTE: () => {
const { muted, setMute } = useMuteStore.getState();
wsend({
op: "mute",
d: { value: !muted },
});
setMute(!muted);
},
PTT: (e) => {
if (!e) return;
const { setMute } = useMuteStore.getState();
const mute = e.type === "keyup";
wsend({
op: "mute",
d: { value: mute },
});
setMute(mute);
},
CHAT: toggleOpen,
}),
[]
)}
/>
);
};
import * as React from "react";
import { Modal } from "./Modal";
import create from "zustand";
import { combine } from "zustand/middleware";
import { Button } from "./Button";
import { Input } from "./Input";
import { useTypeSafeTranslation } from "../utils/useTypeSafeTranslation";
interface Props {}
type Fn = (v: string) => void;
const usePromptModalStore = create(
combine(
{
message: "",
value: "",
onConfirm: undefined as undefined | Fn,
},
(set) => ({
close: () => set({ onConfirm: undefined, message: "", value: "" }),
set,
})
)
);
export const modalPrompt = (
message: string,
onConfirm: Fn,
defaultValue = ""
) => {
usePromptModalStore
.getState()
.set({ onConfirm, message, value: defaultValue });
};
export const PromptModal: React.FC<Props> = () => {
const { onConfirm, message, close, value, set } = usePromptModalStore();
const { t } = useTypeSafeTranslation();
return (
<Modal isOpen={!!onConfirm} onRequestClose={() => close()}>
<div className={`mb-4`}>{message}</div>
<form
onSubmit={(e) => {
e.preventDefault();
close();
onConfirm?.(value);
}}
>
<Input
autoFocus
value={value}
onChange={(e) => set({ value: e.target.value })}
/>
<div className={`flex mt-12`}>
<Button
type="button"
onClick={close}
className={`mr-3`}
color="secondary"
>
{t("common.cancel")}
</Button>
<Button type="submit" className={`ml-3`}>
{t("common.ok")}
</Button>
</div>
</form>
</Modal>
);
};
import React from "react";
import { wsend } from "../../createWebsocket";
import { CurrentRoom, BaseUser, RoomUser } from "../types";
import { UserNode } from "./UserNode";
interface RoomUserNodeProps {
u: RoomUser;
me?: BaseUser | null;
profile?: BaseUser | null;
room: CurrentRoom;
setUserProfileId: (s: string) => void;
muted: boolean;
}
export const RoomUserNode: React.FC<RoomUserNodeProps> = ({
u,
room,
me,
profile,
muted,
setUserProfileId,
}) => {
const isCreator = u.id === room.creatorId;
const isSpeaker = !!u.roomPermissions?.isSpeaker;
const canSpeak = isCreator || isSpeaker;
const isMuted = me?.id === u.id ? muted : room.muteMap[u.id];
return (
<UserNode
u={u}
isMuted={canSpeak && isMuted}
isCreator={isCreator}
isSpeaking={
canSpeak && u.id in room.activeSpeakerMap && !room.muteMap[u.id]
}
isMod={!!u.roomPermissions?.isMod}
isSpeaker={isSpeaker}
onClick={() => {
if (u.id === profile?.id) {
setUserProfileId("");
} else {
if (
(u.youAreFollowing === undefined || u.youAreFollowing === null) &&
me?.id !== u.id
) {
wsend({ op: "follow_info", d: { userId: u.id } });
}
setUserProfileId(u.id);
}
}}
/>
);
};
import { useField } from "formik";
import React from "react";
import { Input } from "../Input";
import { InputErrorMsg } from "../InputErrorMsg";
export const InputField: React.FC<
React.DetailedHTMLProps<
React.InputHTMLAttributes<HTMLInputElement>,
HTMLInputElement
> & {
name: string;
errorMsg?: string;
label?: string;
textarea?: boolean;
altErrorMsg?: string;
}
> = ({ label, textarea, errorMsg, ref: _, ...props }) => {
const [field, meta] = useField(props);
return (
<div>
{label ? <div className={`mb-2`}>{label}</div> : null}
<Input textarea={textarea} {...field} {...props} />
{meta.error && meta.touched ? (
<div className={`mt-1`}>
<InputErrorMsg>{errorMsg || meta.error}</InputErrorMsg>
</div>
) : null}
</div>
);
};
import React from "react";
interface FieldSpacerProps {}
export const FieldSpacer: React.FC<FieldSpacerProps> = ({}) => {
return <div className={`my-6`} />;
};
import React, { useState } from "react";
interface AvatarProps {
src: string;
size?: number;
active?: boolean;
circle?: boolean;
usernameForErrorImg?: string;
className?: string;
isOnline?: boolean;
}
export const Avatar: React.FC<AvatarProps> = ({
src,
size = 70,
active,
circle,
usernameForErrorImg,
className,
isOnline,
}) => {
const [error, setError] = useState(false);
return (
<div className="relative inline-block">
<img
alt="avatar"
onError={() => setError(true)}
width={size}
height={size}
style={active ? { boxShadow: "0 0 0 3px #60A5FA" } : undefined}
className={`${circle ? `rounded-full` : `rounded-3xl`} ${className}`}
src={
error && usernameForErrorImg
? `https://ui-avatars.com/api/?name=${usernameForErrorImg}`
: src
}
/>
{isOnline ? (
<span className="rounded-full w-4 h-4 bg-green-500 absolute right-0 bottom-0"></span>
) : null}
</div>
);
};
import React from "react";
import { useTranslation } from "react-i18next";
interface LanguageSelectorProps {
options?: Array<{ value: string; label: string }>;
}
export const LanguageSelector: React.FC<LanguageSelectorProps> = ({
options = [
{ value: "en", label: "en" },
{ value: "de", label: "de" },
{ value: "es", label: "es" },
{ value: "fr", label: "fr" },
{ value: "he", label: "he" },
{ value: "hu", label: "hu" },
{ value: "nb", label: "nb" },
{ value: "pt-BR", label: "pt-br" },
{ value: "pt-PT", label: "pt-pt" },
{ value: "tr", label: "tr" },
{ value: "zh-CN", label: "zh-cn" }
],
}) => {
const { i18n } = useTranslation();
return (
<select
value={i18n.language}
onChange={(e) => {
i18n.changeLanguage(e.target.value);
}}
>
{options.map((o) => (
<option key={o.value} value={o.value}>
{o.label}
</option>
))}
</select>
);
};
import { Form, Formik } from "formik";
import React from "react";
import { useHistory } from "react-router-dom";
import { wsend, wsFetch } from "../../createWebsocket";
import { useCurrentRoomStore } from "../../webrtc/stores/useCurrentRoomStore";
import { useRoomChatStore } from "../modules/room-chat/useRoomChatStore";
import { roomToCurrentRoom } from "../utils/roomToCurrentRoom";
import { showErrorToast } from "../utils/showErrorToast";
import { Button } from "./Button";
import { InputField } from "./form-fields/InputField";
import { Modal } from "./Modal";
import { useTypeSafeTranslation } from "../utils/useTypeSafeTranslation";
interface CreateRoomModalProps {
onRequestClose: () => void;
name?: string;
description?: string;
isPrivate?: boolean;
edit?: boolean;
}
export const CreateRoomModal: React.FC<CreateRoomModalProps> = ({
onRequestClose,
name: currentName,
description: currentDescription,
isPrivate,
edit,
}) => {
const { t } = useTypeSafeTranslation();
const history = useHistory();
return (
<Modal isOpen onRequestClose={onRequestClose}>
<Formik<{
name: string;
privacy: string;
description: string;
}>
initialValues={{
name: currentName || "",
description: currentDescription || "",
privacy: isPrivate ? "private" : "public",
}}
validateOnChange={false}
validateOnBlur={false}
validate={({ name, description }) => {
const errors: Record<string, string> = {};
if (name.length < 2 || name.length > 60) {
return {
name: t("components.modals.createRoomModal.nameError"),
};
} else if (description.length > 500) {
return {
description: t(
"components.modals.createRoomModal.descriptionError"
),
};
}
return errors;
}}
onSubmit={async ({ name, privacy, description }) => {
const resp = await wsFetch<any>({
op: edit ? "edit_room" : "create_room",
d: { name, privacy, description },
});
if (resp.error) {
showErrorToast(resp.error);
return;
} else if (resp.room) {
const { room } = resp;
console.log("new room voice server id: " + room.voiceServerId);
useRoomChatStore.getState().clearChat();
wsend({ op: "get_current_room_users", d: {} });
history.push("/room/" + room.id);
useCurrentRoomStore
.getState()
.setCurrentRoom(() => roomToCurrentRoom(room));
}
onRequestClose();
}}
>
{({ setFieldValue, values, isSubmitting }) => (
<Form>
<InputField
name="name"
maxLength={60}
placeholder={t("components.modals.createRoomModal.roomName")}
autoFocus
/>
<div className="mt-3">
<InputField
name="description"
maxLength={500}
placeholder={t(
"components.modals.createRoomModal.roomDescription"
)}
textarea
/>
</div>
<div className={`grid mt-8 items-start grid-cols-1`}>
<select
className={`border border-simple-gray-3c`}
value={values.privacy}
onChange={(e) => {
const v = e.target.value;
setFieldValue("privacy", v);
}}
>
<option value="public" className={`bg-simple-gray-3c`}>
{t("components.modals.createRoomModal.public")}
</option>
<option value="private" className={`bg-simple-gray-3c`}>
{t("components.modals.createRoomModal.private")}
</option>
</select>
</div>
<div className={`flex mt-12`}>
<Button
type="button"
onClick={onRequestClose}
className={`mr-1.5`}
color="secondary"
>
{t("common.cancel")}
</Button>
<Button loading={isSubmitting} type="submit" className={`ml-1.5`}>
{t("common.ok")}
</Button>
</div>
</Form>
)}
</Formik>
</Modal>
);
};
import React from "react";
export const CenterLayout: React.FC = ({ children }) => {
return (
<div
className={`max-w-screen-sm mx-auto w-full h-full flex flex-col relative`}
>
{children}
</div>
);
};
import React, { useEffect } from "react";
import { useMuteStore } from "../../webrtc/stores/useMuteStore";
import { useTypeSafeTranslation } from "../utils/useTypeSafeTranslation";
import { useCurrentRoomStore } from "../../webrtc/stores/useCurrentRoomStore";
interface MuteTitleUpdaterProps {}
export const MuteTitleUpdater: React.FC<MuteTitleUpdaterProps> = ({}) => {
const { muted } = useMuteStore();
const { currentRoom } = useCurrentRoomStore();
const { t } = useTypeSafeTranslation();
useEffect(() => {
if (muted && currentRoom) {
document.title = t("header.mutedTitle");
} else {
document.title = t("header.title");
}
}, [muted, t, currentRoom]);
return null;
};
import React, { useEffect, useMemo, useRef, useState } from "react";
import makeUrls, { CalendarEvent } from "./makeUrls";
type CalendarURLs = ReturnType<typeof makeUrls>;
const useAutoFocus = () => {
const elementRef = useRef<HTMLElement>(null);
useEffect(() => {
const previous = document.activeElement;
const element = elementRef.current;
if (element) {
element.focus();
}
if (previous instanceof HTMLElement) {
return () => previous.focus();
}
return undefined;
}, []);
return elementRef;
};
type OpenStateToggle = (event?: React.MouseEvent) => void;
const useOpenState = (initialOpen: boolean): [boolean, OpenStateToggle] => {
const [open, setOpen] = useState<boolean>(initialOpen);
const onToggle = () => setOpen((current) => !current);
useEffect(() => {
if (open) {
const onClose = () => setOpen(false);
document.addEventListener("click", onClose);
return () => document.removeEventListener("click", onClose);
}
return undefined;
}, [open, setOpen]);
return [open, onToggle];
};
type CalendarRef = HTMLAnchorElement;
type CalendarProps = {
children: React.ReactNode;
filename?: string;
href: string;
};
const Calendar = React.forwardRef<CalendarRef, CalendarProps>(
({ children, filename = false, href }, ref) => (
<a
ref={ref}
download={filename}
href={href}
target="_blank"
rel="noopener noreferrer"
>
{children}
</a>
)
);
type DropdownProps = {
filename: string;
onToggle: OpenStateToggle;
urls: CalendarURLs;
};
const Dropdown: React.FC<DropdownProps> = ({ filename, onToggle, urls }) => {
const ref = useAutoFocus() as React.RefObject<HTMLAnchorElement>;
const onKeyDown = (event: React.KeyboardEvent) => {
if (event.key === "Escape") {
onToggle();
}
};
return (
<div
className="chq-atc--dropdown"
onKeyDown={onKeyDown}
role="presentation"
>
<Calendar href={urls.ics} filename={filename} ref={ref}>
Apple Calendar
</Calendar>
<Calendar href={urls.google}>Google</Calendar>
<Calendar href={urls.ics} filename={filename}>
Outlook
</Calendar>
<Calendar href={urls.outlook}>Outlook Web App</Calendar>
<Calendar href={urls.yahoo}>Yahoo</Calendar>
</div>
);
};
type AddToCalendarProps = {
event: CalendarEvent;
open?: boolean;
filename?: string;
};
const AddToCalendar: React.FC<AddToCalendarProps> = ({
children = "Add to My Calendar",
event,
filename = "download",
open: initialOpen = false,
}) => {
const [open, onToggle] = useOpenState(initialOpen);
const urls = useMemo<CalendarURLs>(() => makeUrls(event), [event]);
return (
<div className="chq-atc">
{event && (
<button
type="button"
className="chq-atc--button inline"
onClick={onToggle}
>
<svg
className="chq-atc--button-svg"
width="20px"
height="20px"
viewBox="0 0 1024 1024"
>
<path d="M704 192v-64h-32v64h-320v-64h-32v64h-192v704h768v-704h-192z M864 864h-704v-480h704v480z M864 352h-704v-128h160v64h32v-64h320v64h32v-64h160v128z" />
</svg>{" "}
{children}
</button>
)}
{open && <Dropdown filename={filename} onToggle={onToggle} urls={urls} />}
</div>
);
};
export default AddToCalendar;
import * as React from "react";
import AddToCalendar from "./AddToCalendar";
import "../../../css/add-to-calendar-button.css";
import { useTypeSafeTranslation } from "../../utils/useTypeSafeTranslation";
interface ScheduledEvent {
event: ScheduledEventProps;
}
interface ScheduledEventProps {
name: string;
details: string;
location: string;
startsAt: string;
endsAt: string;
}
export const AddToCalendarButton: React.FC<ScheduledEvent> = ({ event }) => {
const { t } = useTypeSafeTranslation();
return (
<AddToCalendar
children={t("components.addToCalendar.add")}
event={event}
filename={event.name}
/>
);
};
import React from "react";
import { useTypeSafeTranslation } from "../utils/useTypeSafeTranslation";
interface LoadingProps {}
export const Loading: React.FC<LoadingProps> = () => {
const { t } = useTypeSafeTranslation();
return <div>{t("common.loading")}</div>;
};
import React from "react";
import { Check } from "react-feather";
interface ListItemProps {}
export const ListItem: React.FC<ListItemProps> = ({ children }) => {
return (
<li className={`my-2`}>
<span className={`inline-flex items-center`}>
<Check className={`h-6 w-6`} />
<p className={`ml-3`}>{children}</p>
</span>
</li>
);
};
import React, { useState } from "react";
import { Spinner } from "./Spinner";
import "../../css/doge-button.css";
export const Button: React.FC<
React.DetailedHTMLProps<
React.ButtonHTMLAttributes<HTMLButtonElement>,
HTMLButtonElement
> & {
variant?: "default" | "small" | "slim" | "follow";
color?: "default" | "red" | "secondary";
loading?: boolean;
dogeProbability?: number;
}
> = ({
children,
loading,
disabled,
color = "default",
variant = "default",
dogeProbability,
...props
}) => {
const [hasDoge, setHasDoge] = useState(false);
const [dogeXAnchor, setDogeXAnchor] = useState(0);
const [dogeHideTimeout, setDogeHideTimeout] = useState<NodeJS.Timeout | null>(
null
);
const [useTopDoge, setUseTopDoge] = useState(false);
const [useGlassesDoge, setUseGlassesDoge] = useState(false);
const maybeShowDoge = () => {
if (hasDoge) return;
if (dogeHideTimeout) {
clearTimeout(dogeHideTimeout);
setDogeHideTimeout(null);
setHasDoge(true);
} else if ((dogeProbability ?? 0) > Math.random()) {
setDogeXAnchor(Math.random() * 0.8 + 0.1);
setHasDoge(true);
setUseTopDoge(Math.random() < 0.5);
setUseGlassesDoge(Math.random() < 0.2);
}
};
const hideDoge = () => {
if (!hasDoge) return;
setHasDoge(false);
const timeout = setTimeout(() => {
setDogeHideTimeout(null);
}, 200);
setDogeHideTimeout(timeout);
};
const onMouseOver =
(dogeProbability ?? 0) > 0
? (event: React.MouseEvent<HTMLButtonElement, MouseEvent>) => {
maybeShowDoge();
if (props.onMouseOver) props.onMouseOver(event);
}
: props.onMouseOver;
const onMouseLeave =
(dogeProbability ?? 0) > 0
? (event: React.MouseEvent<HTMLButtonElement, MouseEvent>) => {
hideDoge();
if (props.onMouseLeave) props.onMouseLeave(event);
}
: props.onMouseLeave;
return (
<button
{...props}
onMouseOver={onMouseOver}
onMouseLeave={onMouseLeave}
disabled={loading || disabled}
className={`
rounded capitalize outline-none w-full flex items-center justify-center text-center text-white button-fix
${
color === "secondary"
? "bg-simple-gray-3a hover:bg-simple-gray-45"
: color === "red"
? "bg-red-600 hover:bg-red-400"
: "bg-blue-500 hover:bg-blue-400"
}
${
variant === "small"
? "py-1 px-2 w-max"
: variant === "slim"
? "max-w-md ml-auto mr-auto py-2.5 px-12"
: "py-2.5 px-1"
}
${props.className}
`}
>
{loading ? <Spinner /> : null}
{children}
{(dogeProbability ?? 0) > 0 ? (
<>
<div
className={`button-doge button-doge-top ${
hasDoge && useTopDoge ? "button-doge-shown" : ""
} ${useGlassesDoge ? "button-doge-cooler" : ""}`}
style={{
left: `${dogeXAnchor * 100}%`,
}}
/>
<div
className={`button-doge button-doge-bottom ${
hasDoge && !useTopDoge ? "button-doge-shown" : ""
} ${useGlassesDoge ? "button-doge-cooler" : ""}`}
style={{
left: `${dogeXAnchor * 100}%`,
}}
/>
</>
) : (
<></>
)}
</button>
);
};
import React from "react";
import ReactModal from "react-modal";
const customStyles = {
overlay: {
backgroundColor: "rgba(0,0,0,.5)",
zIndex: 999,
},
content: {
top: "50%",
left: "50%",
right: "auto",
bottom: "auto",
marginRight: "-50%",
transform: "translate(-50%, -50%)",
backgroundColor: "#262626",
border: "none",
maxHeight: "80vh",
width: "90%",
maxWidth: 500,
},
};
export const Modal: React.FC<ReactModal["props"]> = ({
children,
...props
}) => {
return (
<ReactModal shouldCloseOnEsc style={customStyles} {...props}>
{children}
</ReactModal>
);
};
import { Formik } from "formik";
import React from "react";
import { useMutation, useQueryClient } from "react-query";
import { object, pattern, size, string } from "superstruct";
import { auth_query, wsMutation } from "../../createWebsocket";
import { BaseUser } from "../types";
import { showErrorToast } from "../utils/showErrorToast";
import { validateStruct } from "../utils/validateStruct";
import { Button } from "./Button";
import { FieldSpacer } from "./form-fields/FieldSpacer";
import { InputField } from "./form-fields/InputField";
import { Modal } from "./Modal";
import { useTypeSafeTranslation } from "../utils/useTypeSafeTranslation";
const profileStruct = object({
displayName: size(string(), 2, 50),
username: pattern(string(), /^(\w){4,15}$/),
bio: size(string(), 0, 160),
avatarUrl: pattern(
string(),
/https?:\/\/(www\.|)(pbs.twimg.com\/profile_images\/(.*)\.(jpg|png|jpeg|webp)|avatars\.githubusercontent\.com\/u\/)/
),
});
interface Shared {
user: BaseUser;
onRequestClose: () => void;
}
interface EditProfileModalProps extends Shared {
isOpen: boolean;
}
const validateFn = validateStruct(profileStruct);
export const EditProfileModal: React.FC<EditProfileModalProps> = ({
isOpen,
onRequestClose,
user,
}) => {
const { mutateAsync, isLoading } = useMutation(wsMutation);
const queryClient = useQueryClient();
const { t } = useTypeSafeTranslation();
return (
<Modal isOpen={isOpen} onRequestClose={onRequestClose}>
{isOpen ? (
<Formik
initialValues={{
displayName: user.displayName,
username: user.username,
bio: user.bio,
avatarUrl: user.avatarUrl,
}}
validateOnChange={false}
validate={(values) => {
return validateFn({
...values,
displayName: values.displayName.trim(),
});
}}
onSubmit={async (data) => {
const { isUsernameTaken } = ((await mutateAsync({
op: "edit_profile",
d: { data },
})) as unknown) as { isUsernameTaken: boolean };
if (isUsernameTaken) {
showErrorToast(
t("components.modals.editProfileModal.usernameTaken")
);
} else {
queryClient.setQueryData<{ user: BaseUser } | undefined>(
auth_query,
(x) =>
!x
? x
: {
...x,
user: {
...x.user,
...data,
displayName: data.displayName.trim(),
},
}
);
onRequestClose();
}
}}
>
{({ handleSubmit }) => (
<div>
<InputField
errorMsg={t(
"components.modals.editProfileModal.avatarUrlError"
)}
label={t("components.modals.editProfileModal.avatarUrlLabel")}
name="avatarUrl"
/>
<FieldSpacer />
<InputField
errorMsg={t(
"components.modals.editProfileModal.displayNameError"
)}
label={t("components.modals.editProfileModal.displayNameLabel")}
name="displayName"
/>
<FieldSpacer />
<InputField
errorMsg={t("components.modals.editProfileModal.usernameError")}
label={t("components.modals.editProfileModal.usernameLabel")}
name="username"
/>
<FieldSpacer />
<InputField
errorMsg={t("components.modals.editProfileModal.bioError")}
label={t("components.modals.editProfileModal.bioLabel")}
textarea
name="bio"
/>
<div className={`flex mt-12`}>
<Button
type="button"
onClick={onRequestClose}
className={`mr-2`}
color="secondary"
>
{t("common.cancel")}
</Button>
<Button
type="button"
loading={isLoading}
onClick={() => handleSubmit()}
className={`ml-2`}
>
{t("common.save")}
</Button>
</div>
</div>
)}
</Formik>
) : null}
</Modal>
);
};
import React from "react";
interface CheckboxProps {
className?: string;
value?: boolean | undefined;
label?: string;
onChange?: (e?: any) => void;
}
export const Checkbox: React.FC<CheckboxProps> = ({
className,
value,
label,
...props
}) => {
return (
<div className={className}>
<label>
<input type="checkbox" checked={value} {...props} className={`mr-3`} />
{label}
</label>
</div>
);
};
import * as React from "react";
import { History } from "history";
import { Modal } from "./Modal";
import create from "zustand";
import { combine } from "zustand/middleware";
import { Button } from "./Button";
import { Avatar } from "./Avatar";
import { useSoundEffectStore } from "../modules/sound-effects/useSoundEffectStore";
import { wsend } from "../../createWebsocket";
import { useTypeSafeTranslation } from "../utils/useTypeSafeTranslation";
interface Props {}
type Fn = () => void;
export type JoinRoomModalType = "invite" | "someone_you_follow_created_a_room";
export type UserPreviewInfo = {
username: string;
displayName: string;
avatarUrl: string;
};
type Options = {
type: JoinRoomModalType;
roomId: string;
roomName: string;
onConfirm: Fn;
} & UserPreviewInfo;
const useConfirmModalStore = create(
combine(
{
options: null as null | Options,
},
(set) => ({
close: () => set({ options: null }),
set,
})
)
);
export const invitedToRoomConfirm = (
options: Omit<Options, "onConfirm">,
history: History
) => {
useSoundEffectStore.getState().playSoundEffect("roomInvite");
useConfirmModalStore.getState().set({
options: {
...options,
onConfirm: () => {
wsend({ op: "join_room", d: { roomId: options.roomId } });
history.push("/room/" + options.roomId);
},
},
});
};
export const InvitedToJoinRoomModal: React.FC<Props> = () => {
const { options, close } = useConfirmModalStore();
const { t } = useTypeSafeTranslation();
return (
<Modal isOpen={!!options} onRequestClose={() => close()}>
{options ? (
<>
<h1 className={`text-2xl mb-2`}>
{options.type === "someone_you_follow_created_a_room"
? t("components.modals.invitedToJoinRoomModal.newRoomCreated")
: t("components.modals.invitedToJoinRoomModal.roomInviteFrom")}
</h1>
<div className={`flex items-center`}>
<Avatar src={options.avatarUrl} />
<div className={`ml-2`}>
<div className={`font-semibold`}>{options.displayName}</div>
<div className={`my-1 flex`}>
<div>@{options.username}</div>
</div>
</div>
</div>
<div className={`mt-4`}>
{options.type === "someone_you_follow_created_a_room"
? t("components.modals.invitedToJoinRoomModal.justStarted")
: t(
"components.modals.invitedToJoinRoomModal.inviteReceived"
)}{" "}
<span className={`font-semibold`}>{options.roomName}</span>
{t("components.modals.invitedToJoinRoomModal.likeToJoin")}
</div>
</>
) : null}
<div className={`flex mt-8`}>
<Button
type="button"
onClick={close}
className={`mr-1.5`}
color="secondary"
>
{t("common.cancel")}
</Button>
<Button
onClick={() => {
close();
options?.onConfirm();
}}
type="submit"
className={`ml-1.5`}
>
{t("common.yes")}
</Button>
</div>
</Modal>
);
};
import React from "react";
import { useConsumerStore } from "../../webrtc/stores/useConsumerStore";
import { VolumeSlider } from "./VolumeSlider";
import { useTypeSafeTranslation } from "../utils/useTypeSafeTranslation";
interface UserVolumeSliderProps {
userId: string;
}
export const UserVolumeSlider: React.FC<UserVolumeSliderProps> = ({
userId,
}) => {
const { consumerMap, setVolume } = useConsumerStore();
const consumerInfo = consumerMap[userId];
const { t } = useTypeSafeTranslation();
if (!consumerInfo) {
return <div>{t("components.userVolumeSlider.noAudioMessage")}</div>;
}
return (
<VolumeSlider
label
max="200"
volume={consumerInfo.volume}
onVolume={(n) => setVolume(userId, n)}
/>
);
};
import React from "react";
import "../../css/spinner.css";
interface SpinnerProps {
centered?: boolean
}
export const Spinner: React.FC<SpinnerProps> = ({
centered
}) => {
return (
<svg
className={`${centered && `center-spinner`} animate-spin -ml-1 mr-3 h-5 w-5 text-white`}
xmlns="http://www.w3.org/2000/svg"
fill="none"
viewBox="0 0 24 24"
>
<circle
className={`opacity-25`}
cx="12"
cy="12"
r="10"
stroke="currentColor"
strokeWidth="4"
></circle>
<path
className={`opacity-75`}
fill="currentColor"
d="M4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4zm2 5.291A7.962 7.962 0 014 12H0c0 3.042 1.135 5.824 3 7.938l3-2.647z"
></path>
</svg>
);
};
import React from "react";
import { useHistory } from "react-router-dom";
import { useMeQuery } from "../utils/useMeQuery";
import { Avatar } from "./Avatar";
interface ProfileButtonProps {
size?: number;
circle?: boolean;
}
export const ProfileButton: React.FC<ProfileButtonProps> = ({
size = 41,
circle,
}) => {
const { me } = useMeQuery();
const history = useHistory();
return me ? (
<button onClick={() => history.push("/me")} className={`px-2.5`}>
<Avatar circle={circle} size={size} src={me.avatarUrl} />
</button>
) : null;
};
import React from "react";
interface WrapperProps {}
export const Wrapper: React.FC<WrapperProps> = ({ children }) => {
return <div className={`mb-auto`}>{children}</div>;
};
import React from "react";
export const RegularAnchor: React.FC<
React.DetailedHTMLProps<
React.AnchorHTMLAttributes<HTMLAnchorElement>,
HTMLAnchorElement
>
> = ({ children, ...props }) => {
return (
<a {...props} className={`text-blue-400 flex items-center`}>
{children}
</a>
);
};
import React, { forwardRef } from "react";
export const Input = forwardRef<
HTMLInputElement,
React.DetailedHTMLProps<
React.InputHTMLAttributes<HTMLInputElement>,
HTMLInputElement
> & { textarea?: boolean }
>(({ textarea, ...props }, ref) => {
const cn = `w-full py-2 px-3 text-white bg-simple-gray-3c`;
return textarea ? (
<textarea ref={ref as any} className={cn} {...(props as any)} />
) : (
<input ref={ref} className={cn} {...props} />
);
});
import React, { useEffect, useState } from "react";
import { recordKeyCombination } from "react-hotkeys";
import { useKeyMapStore } from "../../../webrtc/stores/useKeyMapStore";
import { Button } from "../Button";
interface ChatKeybindProps {
className?: string;
}
export const ChatKeybind: React.FC<ChatKeybindProps> = ({ className }) => {
const [count, setCount] = useState(0);
const [active, setActive] = useState(false);
const {
keyNames: { CHAT },
setChatKeybind,
} = useKeyMapStore();
useEffect(() => {
if (count > 0) {
const unsub = recordKeyCombination(({ id }) => {
setActive(false);
setChatKeybind(id as string);
});
return () => unsub();
}
}, [count, setChatKeybind]);
return (
<div className={`flex items-center ${className}`}>
<Button
variant="small"
onClick={() => {
setCount((c) => c + 1);
setActive(true);
}}
>
set keybind
</Button>
<div className={`ml-4`}>
toggle chat keybind:{" "}
<span className={`font-bold text-lg`}>
{active ? "listening" : CHAT}
</span>
</div>
</div>
);
};
import React, { useEffect, useState } from "react";
import { recordKeyCombination } from "react-hotkeys";
import { useKeyMapStore } from "../../../webrtc/stores/useKeyMapStore";
import { useTypeSafeTranslation } from "../../utils/useTypeSafeTranslation";
import { Button } from "../Button";
interface MuteKeybindProps {
className?: string;
}
export const MuteKeybind: React.FC<MuteKeybindProps> = ({ className }) => {
const [count, setCount] = useState(0);
const [active, setActive] = useState(false);
const { t } = useTypeSafeTranslation();
const {
keyNames: { MUTE },
setMuteKeybind,
} = useKeyMapStore();
useEffect(() => {
if (count > 0) {
const unsub = recordKeyCombination(({ id }) => {
setActive(false);
setMuteKeybind(id as string);
});
return () => unsub();
}
}, [count, setMuteKeybind]);
return (
<div className={`flex items-center ${className}`}>
<Button
variant="small"
onClick={() => {
setCount((c) => c + 1);
setActive(true);
}}
>
{t("components.keyboardShortcuts.setKeybind")}
</Button>
<div className={`ml-4`}>
{t("components.keyboardShortcuts.toggleMuteKeybind")}:{" "}
<span className={`font-bold text-lg`}>
{active ? t("components.keyboardShortcuts.listening") : MUTE}
</span>
</div>
</div>
);
};
import React, { useEffect, useState } from "react";
import { recordKeyCombination } from "react-hotkeys";
import { useKeyMapStore } from "../../../webrtc/stores/useKeyMapStore";
import { useTypeSafeTranslation } from "../../utils/useTypeSafeTranslation";
import { Button } from "../Button";
interface PTTKeybindProps {
className?: string;
}
export const PTTKeybind: React.FC<PTTKeybindProps> = ({ className }) => {
const [count, setCount] = useState(0);
const [active, setActive] = useState(false);
const { t } = useTypeSafeTranslation();
const {
keyNames: { PTT },
setPTTKeybind,
} = useKeyMapStore();
useEffect(() => {
if (count > 0) {
const unsub = recordKeyCombination(({ id }) => {
setActive(false);
setPTTKeybind(id as string);
});
return () => unsub();
}
}, [count, setPTTKeybind]);
return (
<div className={`flex items-center ${className}`}>
<Button
variant="small"
onClick={() => {
setCount((c) => c + 1);
setActive(true);
}}
>
{t("components.keyboardShortcuts.setKeybind")}
</Button>
<div className={`ml-4`}>
{t("components.keyboardShortcuts.togglePushToTalkKeybind")}:{" "}
<span className={`font-bold text-lg`}>
{active ? t("components.keyboardShortcuts.listening") : PTT}
</span>
</div>
</div>
);
};
import React from "react";
import { useTranslation } from "react-i18next";
import { Route, Switch } from "react-router-dom";
import { ScheduledRoomsPage } from "./modules/scheduled-rooms/ScheduledRoomsPage";
import { ViewScheduledRoomPage } from "./modules/scheduled-rooms/ViewScheduledRoomPage";
import { BanUsersPage } from "./pages/BanUsersPage";
import { FollowingOnlineList } from "./pages/FollowingOnlineList";
import { FollowListPage } from "./pages/FollowListPage";
import { Home } from "./pages/Home";
import { InviteList } from "./pages/InviteList";
import { Login } from "./pages/Login";
import { MyProfilePage } from "./pages/MyProfilePage";
import { NotFoundPage } from "./pages/NotFoundPage";
import { RoomPage } from "./pages/RoomPage";
import { SearchUsersPage } from "./pages/SearchUsersPage";
import { SoundEffectSettingsPage } from "./pages/SoundEffectSettingsPage";
import { ViewUserPage } from "./pages/ViewUserPage";
import { VoiceSettingsPage } from "./pages/VoiceSettingsPage";
import { useMainWsHandler } from "./useMainWsHandler";
import { useTokenStore } from "./utils/useTokenStore";
interface RoutesProps {}
export const Routes: React.FC<RoutesProps> = () => {
const hasTokens = useTokenStore((s) => !!s.accessToken && !!s.refreshToken);
useMainWsHandler();
const [, , ready] = useTranslation();
if (!ready) {
// this could become loading indicator
return null;
}
return (
<Switch>
{/* PUBLIC ROUTES */}
<Route
exact
path="/scheduled-room/:id"
component={ViewScheduledRoomPage}
/>
{/* PRIVATE ROUTES - login required */}
<Route path="/">
{!hasTokens ? (
<Login />
) : (
<Switch>
<Route exact path="/" component={Home} />
<Route
exact
path="/scheduled-rooms"
component={ScheduledRoomsPage}
/>
<Route exact path="/room/:id" component={RoomPage} />
<Route exact path="/user" component={ViewUserPage} />
<Route exact path="/me" component={MyProfilePage} />
<Route exact path="/invite" component={InviteList} />
<Route exact path="/search/users" component={SearchUsersPage} />
<Route exact path="/ban/users" component={BanUsersPage} />
<Route exact path="/voice-settings" component={VoiceSettingsPage} />
<Route
exact
path="/sound-effect-settings"
component={SoundEffectSettingsPage}
/>
<Route
exact
path="/following-online"
component={FollowingOnlineList}
/>
<Route
exact
path={["/followers/:userId", "/following/:userId"]}
component={FollowListPage}
/>
<Route component={NotFoundPage} />
</Switch>
)}
</Route>
</Switch>
);
};
import React, { useEffect } from "react";
import { useCurrentRoomStore } from "../../../webrtc/stores/useCurrentRoomStore";
import { Avatar } from "../../components/Avatar";
import { BaseUser } from "../../types";
import { useMeQuery } from "../../utils/useMeQuery";
import { useRoomChatMentionStore } from "./useRoomChatMentionStore";
import { useRoomChatStore } from "./useRoomChatStore";
interface RoomChatMentionsProps {}
export const RoomChatMentions: React.FC<RoomChatMentionsProps> = ({}) => {
const { currentRoom } = useCurrentRoomStore();
const { me } = useMeQuery();
const { message, setMessage } = useRoomChatStore();
const {
activeUsername,
setActiveUsername,
queriedUsernames,
setQueriedUsernames,
mentions,
setMentions,
} = useRoomChatMentionStore();
function addMention(m: BaseUser) {
setMentions([...mentions, m]);
setMessage(
message.substring(0, message.lastIndexOf("@") + 1) + m.username + " "
);
setQueriedUsernames([]);
// Re-focus input after mention was clicked
document.getElementById("room-chat-input")?.focus();
}
useEffect(() => {
// regex to match mention patterns
const mentionMatches = message.match(/^(?!.*\bRT\b)(?:.+\s)?#?@\w+/i);
// query usernames for matched patterns
if (mentionMatches && me && currentRoom) {
const mentionsList = mentionMatches[0].replace(/@|#/g, "").split(" ");
const useMention = mentionsList[mentionsList.length - 1];
// hide usernames list if user continues typing without selecting
if (message[message.lastIndexOf(useMention) + useMention.length]) {
setQueriedUsernames([]);
} else {
const usernameMatches = currentRoom.users.filter(
({ id, username, displayName }) =>
(username?.toLowerCase().includes(useMention?.toLowerCase()) ||
displayName?.toLowerCase().includes(useMention?.toLowerCase())) &&
!mentions.find((m: BaseUser) => m.id === id) &&
me.id !== id
);
const firstFive = usernameMatches.slice(0, 5);
setQueriedUsernames(firstFive);
if (firstFive.length) setActiveUsername(firstFive[0].id);
}
} else {
// Hide mentions when message is sent
setQueriedUsernames([]);
}
// Remove mention if user deleted text
setMentions(
mentions.filter((u) => {
return message.toLowerCase().indexOf(u.username?.toLowerCase()) !== -1;
})
);
// eslint-disable-next-line react-hooks/exhaustive-deps
}, [message]);
if (queriedUsernames.length) {
return (
<div className={`flex flex-col pb-1 bg-simple-gray-26`}>
{queriedUsernames.map((m) => (
<button
className={`flex py-3 items-center px-8 focus:outline-none ${
activeUsername === m.id ? "bg-blue-800" : ""
}`}
key={m.id}
onClick={() => addMention(m)}
>
<span className={`pr-3 inline`}>
<Avatar size={20} src={m.avatarUrl} />
</span>
<p className={`m-0 mt-1`}>
{m.displayName}
{m.displayName !== m.username ? `(${m.username})` : null}
</p>
</button>
))}
</div>
);
}
return <></>;
};
import React, { useState } from "react";
import { useCurrentRoomStore } from "../../../webrtc/stores/useCurrentRoomStore";
import { truncate } from "../../utils/truncate";
import { useTypeSafeTranslation } from "../../utils/useTypeSafeTranslation";
const MAX_COLLAPSED_CHARACTERS = 100;
interface RoomDescriptionProps {}
export const RoomDescription: React.FC<RoomDescriptionProps> = () => {
const [expanded, setExpanded] = useState(false);
const { currentRoom } = useCurrentRoomStore();
const { t } = useTypeSafeTranslation();
return currentRoom?.description ? (
<div className="p-3 rounded-lg m-3 bg-simple-gray-3a">
<p className="text-gray-400 mb-1">
{t("modules.roomChat.roomDescription")}
</p>
<p className="whitespace-pre-wrap break-all">
{expanded
? currentRoom.description
: truncate(currentRoom.description, MAX_COLLAPSED_CHARACTERS)}
<button
className="ml-1 text-blue-400 cursor-pointer hover:text-blue-300"
onClick={() => setExpanded(!expanded)}
>
{currentRoom.description?.length > MAX_COLLAPSED_CHARACTERS
? "show " + (expanded ? "less" : "more")
: null}
</button>
</p>
</div>
) : null;
};
import { Picker } from "emoji-mart";
import "emoji-mart/css/emoji-mart.css";
import React, { useRef, useState } from "react";
import { Smile } from "react-feather";
import { toast } from "react-toastify";
import { wsend } from "../../../createWebsocket";
import { useCurrentRoomStore } from "../../../webrtc/stores/useCurrentRoomStore";
import { modalAlert } from "../../components/AlertModal";
import { Button } from "../../components/Button";
import { Codicon } from "../../svgs/Codicon";
import { createChatMessage } from "../../utils/createChatMessage";
import { useMeQuery } from "../../utils/useMeQuery";
import { useTypeSafeTranslation } from "../../utils/useTypeSafeTranslation";
import { useRoomChatMentionStore } from "./useRoomChatMentionStore";
import { useRoomChatStore } from "./useRoomChatStore";
interface ChatInputProps {}
export const RoomChatInput: React.FC<ChatInputProps> = () => {
const { message, setMessage } = useRoomChatStore();
const {
setQueriedUsernames,
queriedUsernames,
mentions,
setMentions,
activeUsername,
setActiveUsername,
} = useRoomChatMentionStore();
const { currentRoom } = useCurrentRoomStore();
const { me } = useMeQuery();
const [isEmoji, setIsEmoji] = useState<boolean>(false);
const inputRef = useRef<HTMLInputElement>(null);
const [lastMessageTimestamp, setLastMessageTimestamp] = useState<number>(0);
const { t } = useTypeSafeTranslation();
let position: number = 0;
const navigateThroughQueriedUsers = (e: any) => {
// Use dom method, GlobalHotkeys apparently don't catch arrow-key events on inputs
if (
!["ArrowUp", "ArrowDown", "Enter"].includes(e.code) ||
!queriedUsernames.length
)
return;
e.preventDefault();
let changeToIndex = null;
const activeIndex = queriedUsernames.findIndex(
(username) => username.id === activeUsername
);
if (e.code === "ArrowUp") {
changeToIndex =
activeIndex === 0 ? queriedUsernames.length - 1 : activeIndex - 1;
} else if (e.code === "ArrowDown") {
changeToIndex =
activeIndex === queriedUsernames.length - 1 ? 0 : activeIndex + 1;
} else if (e.code === "Enter") {
const selected = queriedUsernames[activeIndex];
setMentions([...mentions, selected]);
setMessage(
`${message.substring(0, message.lastIndexOf("@") + 1)}${
selected.username
} `
);
setQueriedUsernames([]);
}
// navigate to next/prev mention suggestion item
if (changeToIndex !== null) {
setActiveUsername(queriedUsernames[changeToIndex]?.id);
}
};
const addEmoji = (emoji: any) => {
position =
(position === 0 ? inputRef!.current!.selectionStart : position + 2) || 0;
const newMsg = [
message.slice(0, position),
emoji.native,
message.slice(position),
].join("");
setMessage(newMsg);
};
const handleSubmit = (
e: React.FormEvent<HTMLFormElement> | React.MouseEvent<HTMLButtonElement>
) => {
e.preventDefault();
if (
!message ||
!message.trim() ||
!message.replace(/[\u200B-\u200D\uFEFF]/g, "")
)
return;
if (!me) return;
if (me.id in useRoomChatStore.getState().bannedUserIdMap) {
modalAlert(t("modules.roomChat.bannedAlert"));
return;
}
if (Date.now() - lastMessageTimestamp <= 1000) {
if (!toast.isActive("message-timeout")) {
toast(t("modules.roomChat.waitAlert"), {
toastId: "message-timeout",
type: "warning",
autoClose: 3000,
});
}
return;
}
const tmp = message;
setMessage("");
wsend({
op: "send_room_chat_msg",
d: createChatMessage(tmp, mentions, currentRoom?.users),
});
setQueriedUsernames([]);
setLastMessageTimestamp(Date.now());
};
return (
<form
onSubmit={handleSubmit}
className={`bg-simple-gray-26 pb-5 px-8 pt-1 flex flex-col`}
>
{isEmoji ? (
<Picker
set="apple"
onSelect={(emoji) => {
addEmoji(emoji);
}}
style={{
position: "relative",
width: "100%",
minWidth: "278px",
right: 0,
overflowY: "hidden",
outline: "none",
alignSelf: "flex-end",
margin: "0 0 8px 0",
}}
sheetSize={32}
theme="dark"
emojiTooltip={true}
showPreview={false}
showSkinTones={false}
i18n={{
search: t("modules.roomChat.search"),
categories: {
search: t("modules.roomChat.searchResults"),
recent: t("modules.roomChat.recent"),
},
}}
/>
) : null}
<div className="flex items-stretch">
<div className="flex-1 mr-2 lg:mr-0 items-end">
<input
maxLength={512}
placeholder={t("modules.roomChat.sendMessage")}
value={message}
onChange={(e) => setMessage(e.target.value)}
ref={inputRef}
className={`w-full white bg-simple-gray-59 px-4 py-2 rounded text-lg focus:outline-none pr-12`}
autoComplete="off"
onKeyDown={navigateThroughQueriedUsers}
onFocus={() => {
setIsEmoji(false);
position = 0;
}}
id="room-chat-input"
/>
<div
style={{
color: "rgb(167, 167, 167)",
display: "flex",
marginRight: 13,
marginTop: -35,
flexDirection: "row-reverse",
}}
className={`mt-3 right-12 cursor-pointer`}
onClick={() => {
setIsEmoji(!isEmoji);
position = 0;
}}
>
<Smile style={{ inlineSize: "23px" }}></Smile>
</div>
</div>
{/* Send button (mobile only) */}
<Button
onClick={handleSubmit}
variant="small"
className="lg:hidden"
style={{ padding: "10px 12px" }}
>
<Codicon name="arrowRight" />
</Button>
</div>
</form>
);
};
import React, { useEffect } from "react";
import { useMediaQuery } from "react-responsive";
import { useCurrentRoomStore } from "../../../webrtc/stores/useCurrentRoomStore";
import { useTypeSafeTranslation } from "../../utils/useTypeSafeTranslation";
import { RoomChatInput } from "./RoomChatInput";
import { RoomChatList } from "./RoomChatList";
import { RoomChatMentions } from "./RoomChatMentions";
import { RoomDescription } from "./RoomDescription";
import { useRoomChatStore } from "./useRoomChatStore";
interface ChatProps {
sidebar: boolean;
}
export const roomChatMediaQuery = "(min-width: 980px)";
export const RoomChat: React.FC<ChatProps> = ({ sidebar }) => {
const chatShouldBeSidebar = useMediaQuery({ query: roomChatMediaQuery });
const { currentRoom: room } = useCurrentRoomStore();
const [open, reset, toggleOpen] = useRoomChatStore((s) => [
s.open,
s.reset,
s.toggleOpen,
]);
const { t } = useTypeSafeTranslation();
useEffect(() => {
if (!room) {
reset();
}
}, [reset, room]);
if (
!open ||
(!chatShouldBeSidebar && sidebar) ||
(chatShouldBeSidebar && !sidebar)
) {
return null;
}
return (
<div
style={{
width: sidebar ? 340 : "100%",
}}
className={`flex flex-1 w-full overflow-y-auto`}
>
<div
style={{
width: sidebar ? 340 : "100%",
height: sidebar ? "100%" : undefined,
}}
className={`bg-simple-gray-26 flex flex-1 w-full flex-col ${
sidebar ? `fixed bottom-0` : ``
}`}
>
<button
onClick={() => toggleOpen()}
className={`bg-simple-gray-26 border-b border-simple-gray-80 text-white py-4 px-8 text-2xl flex items-center h-20`}
>
{t("modules.roomChat.title")}{" "}
<span className={`ml-2 text-simple-gray-a6`}>
{t("modules.roomChat.emotesSoon")}
</span>
</button>
<RoomDescription />
<RoomChatList />
<RoomChatMentions />
<RoomChatInput />
</div>
</div>
);
};
import normalizeUrl from "normalize-url";
import React, { useEffect, useRef, useState } from "react";
import ReactTooltip from "react-tooltip";
import { useCurrentRoomStore } from "../../../webrtc/stores/useCurrentRoomStore";
import { useCurrentRoomInfo } from "../../atoms";
import { Avatar } from "../../components/Avatar";
import { dateFormat } from "../../utils/dateFormat";
import { useMeQuery } from "../../utils/useMeQuery";
import { useTypeSafeTranslation } from "../../utils/useTypeSafeTranslation";
import { ProfileModalFetcher } from "./ProfileModalFetcher";
import { useRoomChatMentionStore } from "./useRoomChatMentionStore";
import { RoomChatMessage, useRoomChatStore } from "./useRoomChatStore";
interface ChatListProps {}
export const RoomChatList: React.FC<ChatListProps> = ({}) => {
const [profileId, setProfileId] = useState("");
const messages = useRoomChatStore((s) => s.messages);
const { me } = useMeQuery();
const { currentRoom: room } = useCurrentRoomStore();
const { isMod: iAmMod, isCreator: iAmCreator } = useCurrentRoomInfo();
const [
messageToBeDeleted,
setMessageToBeDeleted,
] = useState<RoomChatMessage | null>(null);
const bottomRef = useRef<null | HTMLDivElement>(null);
const chatListRef = useRef<null | HTMLDivElement>(null);
const {
isRoomChatScrolledToTop,
setIsRoomChatScrolledToTop,
} = useRoomChatStore();
const { t } = useTypeSafeTranslation();
// Only scroll into view if not manually scrolled to top
useEffect(() => {
isRoomChatScrolledToTop || bottomRef.current?.scrollIntoView();
});
return (
<div
className={`bg-simple-gray-26 px-8 pt-8 flex-1 overflow-y-auto flex-col flex chat-message-container`}
ref={chatListRef}
onScroll={() => {
if (!chatListRef.current) return;
const { scrollTop, offsetHeight, scrollHeight } = chatListRef.current;
const isOnBottom = scrollTop + offsetHeight === scrollHeight;
setIsRoomChatScrolledToTop(!isOnBottom);
if (isOnBottom) {
useRoomChatMentionStore.getState().resetIAmMentioned();
}
}}
>
{profileId ? (
<ProfileModalFetcher
userId={profileId}
messageToBeDeleted={messageToBeDeleted}
onClose={() => {
setProfileId("");
setMessageToBeDeleted(null);
}}
/>
) : null}
{messages
.slice()
.reverse()
.map((m) => (
<div
className="flex flex-col flex-shrink-0"
key={m.id}
data-tip={dateFormat(m.sentAt)}
>
{/* Whisper label */}
{m.isWhisper ? (
<p className="mb-0 text-xs text-gray-400 px-2 bg-simple-gray-3a w-16 rounded-t mt-1 text-center">
{t("modules.roomChat.whisper")}
</p>
) : null}
<div
className={`flex items-center px-1 ${
m.isWhisper
? "bg-simple-gray-3a py-1 rounded-b-lg rounded-tr-lg mb-1"
: ""
}`}
>
<div
className={`py-1 block break-words max-w-full items-start flex-1`}
key={m.id}
>
<span className={`pr-2`}>
<Avatar size={20} src={m.avatarUrl} className="inline" />
</span>
<button
onClick={() => {
setProfileId(m.userId);
setMessageToBeDeleted(
(me?.id === m.userId ||
iAmCreator ||
(iAmMod && room?.creatorId !== m.userId)) &&
!m.deleted
? m
: null
);
}}
className={`hover:underline focus:outline-none`}
style={{ textDecorationColor: m.color, color: m.color }}
>
{m.displayName}
</button>
<span className={`mr-1`}>: </span>
{m.deleted ? (
<span className="text-gray-500">
[message{" "}
{m.deleterId === m.userId ? "retracted" : "deleted"}]
</span>
) : (
m.tokens.map(({ t, v }, i) => {
switch (t) {
case "text":
return (
<span className={`flex-1 m-0`} key={i}>
{v}{" "}
</span>
);
case "mention":
return (
<button
onClick={() => {
setProfileId(v);
}}
key={i}
className={`hover:underline flex-1 focus:outline-none ml-1 mr-2 ${
v === me?.username
? "bg-blue-500 text-white px-2 rounded text-md"
: ""
}`}
style={{
textDecorationColor: m.color,
color: v === me?.username ? "" : m.color,
}}
>
@{v}{" "}
</button>
);
case "link":
return (
<a
target="_blank"
rel="noreferrer"
href={v}
className={`flex-1 hover:underline text-blue-500`}
key={i}
>
{normalizeUrl(v, { stripProtocol: true })}{" "}
</a>
);
default:
return null;
}
})
)}
</div>
</div>
<ReactTooltip />
</div>
))}
{messages.length === 0 ? (
<div>{t("modules.roomChat.welcomeMessage")}</div>
) : null}
<div className={`pb-6`} ref={bottomRef} />
<style>{`
.chat-message-container > :first-child {
margin-top: auto;
}
`}</style>
</div>
);
};
import React, { useLayoutEffect } from "react";
import { useQuery } from "react-query";
import { wsend, wsFetch } from "../../../createWebsocket";
import { useCurrentRoomStore } from "../../../webrtc/stores/useCurrentRoomStore";
import { useCurrentRoomInfo } from "../../atoms";
import { ProfileModal } from "../../components/ProfileModal";
import { RoomUser, UserWithFollowInfo } from "../../types";
import { useMeQuery } from "../../utils/useMeQuery";
import { RoomChatMessage } from "./useRoomChatStore";
interface ProfileModalFetcherProps {
userId: string;
onClose: () => void;
messageToBeDeleted?: RoomChatMessage | null;
}
export const ProfileModalFetcher: React.FC<ProfileModalFetcherProps> = ({
userId,
onClose,
messageToBeDeleted,
}) => {
const { currentRoom: room } = useCurrentRoomStore();
const { me } = useMeQuery();
const { isMod: iAmMod, isCreator: iAmCreator } = useCurrentRoomInfo();
const profileFromRoom: RoomUser | undefined = room?.users.find((x) =>
[x.id, x.username].includes(userId)
);
const { data: profileFromDB } = useQuery<UserWithFollowInfo>(
["get_user_profile", userId],
() =>
wsFetch<any>({
op: "get_user_profile",
d: { userId },
}),
{ enabled: !profileFromRoom }
);
const profile = profileFromRoom || profileFromDB;
useLayoutEffect(() => {
if (
profile &&
me &&
profile.id !== me.id &&
(profile.youAreFollowing === undefined ||
profile.youAreFollowing === null)
) {
wsend({ op: "follow_info", d: { userId: profile.id } });
}
}, [me, profile]);
if (!room) {
return null;
}
if (!profile) {
return null;
}
return (
<ProfileModal
iAmCreator={iAmCreator}
iAmMod={iAmMod}
isMe={profile?.id === me?.id}
room={room}
onClose={onClose}
profile={profile}
messageToBeDeleted={messageToBeDeleted}
/>
);
};
import React, { useState } from "react";
import { useQuery, useQueryClient } from "react-query";
import { useParams } from "react-router";
import { Link } from "react-router-dom";
import { BodyWrapper } from "../../components/BodyWrapper";
import { Wrapper } from "../../components/Wrapper";
import { NotFoundPage } from "../../pages/NotFoundPage";
import { Logo } from "../../svgs/Logo";
import { ScheduledRoom } from "../../types";
import { EditScheduleRoomModalController } from "./EditScheduleRoomModalController";
import { ScheduledRoomCard } from "./ScheduledRoomCard";
interface ViewScheduledRoomPageProps {}
type GetScheduledRoomById = { room: ScheduledRoom | null };
export const ViewScheduledRoomPage: React.FC<ViewScheduledRoomPageProps> = ({}) => {
const queryClient = useQueryClient();
const [deleted, setDeleted] = useState(false);
const { id } = useParams<{ id: string }>();
const key = `/scheduled-room/${id}`;
const { data, isLoading } = useQuery<
GetScheduledRoomById | { error: string }
>(key);
if (isLoading) {
return null;
}
if (!data || "error" in data || !data.room) {
return <NotFoundPage />;
}
return (
<div className={`flex flex-col flex-1`}>
<Wrapper>
<BodyWrapper>
<div className={`mb-10 mt-8`}>
<Link to="/">
<Logo />
</Link>
</div>
</BodyWrapper>
{deleted ? (
<div>deleted</div>
) : (
<EditScheduleRoomModalController
onScheduledRoom={(_editInfo, values, _resp) => {
queryClient.setQueryData<GetScheduledRoomById>(key, {
room: {
...data.room!,
name: values.name,
description: values.description,
scheduledFor: values.scheduledFor.toISOString(),
},
});
}}
>
{({ onEdit }) => (
<ScheduledRoomCard
info={data.room!}
onDeleteComplete={() => setDeleted(true)}
noCopyLinkButton
onEdit={() =>
onEdit({ scheduleRoomToEdit: data.room!, cursor: "" })
}
/>
)}
</EditScheduleRoomModalController>
)}
</Wrapper>
</div>
);
};
import React, { useState } from "react";
import { Link2 } from "react-feather";
import { copyTextToClipboard } from "../../utils/copyToClipboard";
import { useTypeSafeTranslation } from "../../utils/useTypeSafeTranslation";
interface CopyLinkButtonProps {
text: string;
}
export const CopyScheduleRoomLinkButton: React.FC<CopyLinkButtonProps> = ({
text,
}) => {
const [copied, setCopied] = useState(false);
const { t } = useTypeSafeTranslation();
return (
<button
type="button"
className="chq-atc--button flex"
onClick={() => {
if (copyTextToClipboard(text)) {
setCopied(true);
}
}}
>
<Link2 className="mr-2" />
{copied ? t("common.copied") : t("common.copyLink")}
</button>
);
};
import DateFnsUtils from "@date-io/date-fns";
import { createMuiTheme, ThemeProvider } from "@material-ui/core/styles";
import { DateTimePicker, MuiPickersUtilsProvider } from "@material-ui/pickers";
import { add } from "date-fns";
import { Form, Formik } from "formik";
import React from "react";
import { wsFetch } from "../../../createWebsocket";
import { Button } from "../../components/Button";
import { InputField } from "../../components/form-fields/InputField";
import { InputErrorMsg } from "../../components/InputErrorMsg";
import { Modal } from "../../components/Modal";
import { BaseUser } from "../../types";
import { showErrorToast } from "../../utils/showErrorToast";
import { useTypeSafeTranslation } from "../../utils/useTypeSafeTranslation";
export interface ScheduleRoomFormData {
name: string;
description: string;
cohosts: BaseUser[];
scheduledFor: Date;
}
interface CreateRoomModalProps {
editInfo?: { intialValues: ScheduleRoomFormData; id: string };
onScheduledRoom: (data: ScheduleRoomFormData, resp: any) => void;
onRequestClose: () => void;
}
const theme = createMuiTheme({
palette: {
type: "dark",
},
});
export const ScheduleRoomModal: React.FC<CreateRoomModalProps> = ({
onScheduledRoom,
onRequestClose,
editInfo,
}) => {
const { t } = useTypeSafeTranslation();
return (
<ThemeProvider theme={theme}>
<MuiPickersUtilsProvider utils={DateFnsUtils}>
<Modal isOpen onRequestClose={onRequestClose}>
<Formik<ScheduleRoomFormData>
initialValues={
editInfo?.intialValues || {
name: "",
description: "",
cohosts: [] as BaseUser[],
scheduledFor: add(new Date(), { days: 1 }),
}
}
validateOnChange={false}
validateOnBlur={false}
validate={({ name, scheduledFor }) => {
const errors: Record<string, string> = {};
if (name.length < 2) {
return {
name: t("modules.scheduledRooms.modal.minLength"),
};
}
if (scheduledFor.getTime() < new Date().getTime()) {
return {
scheduledFor: t("modules.scheduledRooms.modal.needsFuture"),
};
}
return errors;
}}
onSubmit={async (allData) => {
const { name, scheduledFor, ...data } = allData;
const scheduledForISO = scheduledFor.toISOString();
const resp = await wsFetch<any>(
editInfo
? {
op: "edit_scheduled_room",
d: {
id: editInfo.id,
data: {
name,
scheduledFor: scheduledForISO,
...data,
},
},
}
: {
op: "schedule_room",
d: {
name,
scheduledFor: scheduledForISO,
...data,
},
}
);
if (resp.error) {
showErrorToast(resp.d);
return;
} else {
onScheduledRoom(allData, resp);
}
onRequestClose();
}}
>
{({ setFieldValue, values, errors, isSubmitting }) => (
<Form>
<InputField
name="name"
maxLength={60}
placeholder={t("modules.scheduledRooms.modal.roomName")}
autoFocus
/>
<div className={`mt-8`}>
<DateTimePicker
value={values.scheduledFor}
minDate={new Date()}
maxDate={add(new Date(), { months: 1 })}
onChange={(x) => {
if (x) {
setFieldValue("scheduledFor", x);
}
}}
/>
{errors.scheduledFor ? (
<div className={`mt-1`}>
<InputErrorMsg>{errors.scheduledFor}</InputErrorMsg>
</div>
) : null}
<div className={`mt-8`}>
<InputField
textarea
placeholder="description"
name="description"
maxLength={200}
/>
</div>
</div>
<div className={`flex mt-12`}>
<Button
type="button"
onClick={onRequestClose}
className={`mr-1.5`}
color="secondary"
>
{t("common.cancel")}
</Button>
<Button
loading={isSubmitting}
type="submit"
className={`ml-1.5`}
>
{t("common.ok")}
</Button>
</div>
</Form>
)}
</Formik>
</Modal>
</MuiPickersUtilsProvider>
</ThemeProvider>
);
};
import React, { useState } from "react";
import { useQuery, useQueryClient } from "react-query";
import { wsFetch } from "../../../createWebsocket";
import { useSocketStatus } from "../../../webrtc/stores/useSocketStatus";
import { Backbar } from "../../components/Backbar";
import { BodyWrapper } from "../../components/BodyWrapper";
import { BottomVoiceControl } from "../../components/BottomVoiceControl";
import { Button } from "../../components/Button";
import { ProfileButton } from "../../components/ProfileButton";
import { Spinner } from "../../components/Spinner";
import { Wrapper } from "../../components/Wrapper";
import { ScheduledRoom, ScheduledRoomsInfo } from "../../types";
import { useMeQuery } from "../../utils/useMeQuery";
import { useTypeSafeTranslation } from "../../utils/useTypeSafeTranslation";
import { EditScheduleRoomModalController } from "./EditScheduleRoomModalController";
import { ScheduledRoomCard } from "./ScheduledRoomCard";
import { ScheduleRoomModal } from "./ScheduleRoomModal";
interface ScheduledRoomsPageProps {}
export const GET_SCHEDULED_ROOMS = "get_scheduled_rooms";
const Page = ({
onLoadMore,
cursor,
isLastPage,
isOnlyPage,
getOnlyMyScheduledRooms,
onEdit,
}: {
onEdit: (sr: { scheduleRoomToEdit: ScheduledRoom; cursor: string }) => void;
getOnlyMyScheduledRooms: boolean;
cursor: string;
isLastPage: boolean;
isOnlyPage: boolean;
onLoadMore: (o: string) => void;
}) => {
const queryClient = useQueryClient();
const { status } = useSocketStatus();
const { isLoading, data } = useQuery<ScheduledRoomsInfo>(
[GET_SCHEDULED_ROOMS, cursor, getOnlyMyScheduledRooms],
() =>
wsFetch<any>({
op: GET_SCHEDULED_ROOMS,
d: { cursor, getOnlyMyScheduledRooms },
}),
{ staleTime: Infinity, enabled: status === "auth-good" }
);
const { t } = useTypeSafeTranslation();
if (isLoading) {
return <Spinner />;
}
if (!data) {
return null;
}
if (isOnlyPage && data.scheduledRooms.length === 0) {
return (
<div className={`mt-8 text-xl ml-4`}>
{t("modules.scheduledRooms.noneFound")}
</div>
);
}
return (
<>
{data.scheduledRooms.map((r) => (
<div className={`mt-4`} key={r.id}>
<ScheduledRoomCard
onDeleteComplete={() => {
queryClient.setQueryData<ScheduledRoomsInfo>(
[GET_SCHEDULED_ROOMS, cursor, getOnlyMyScheduledRooms],
(d) => {
return {
scheduledRooms: (d?.scheduledRooms || []).filter(
(x) => x.id !== r.id
),
nextCursor: d?.nextCursor,
};
}
);
}}
onEdit={() => onEdit({ cursor, scheduleRoomToEdit: r })}
info={r}
/>
</div>
))}
{isLastPage && data.nextCursor ? (
<div className={`flex justify-center my-10`}>
<Button variant="small" onClick={() => onLoadMore(data.nextCursor!)}>
{t("common.loadMore")}
</Button>
</div>
) : null}
</>
);
};
export const ScheduledRoomsPage: React.FC<ScheduledRoomsPageProps> = ({}) => {
const queryClient = useQueryClient();
const [showScheduleRoomModal, setShowScheduleRoomModal] = useState(false);
const [{ cursors, getOnlyMyScheduledRooms }, setQueryState] = useState<{
cursors: string[];
getOnlyMyScheduledRooms: boolean;
}>({ cursors: [""], getOnlyMyScheduledRooms: false });
const { me } = useMeQuery();
const { t } = useTypeSafeTranslation();
return (
<div className={`flex flex-col flex-1`}>
<Wrapper>
<BodyWrapper>
<Backbar>
<h1
className={`font-xl flex-1 text-center flex items-center justify-center text-2xl`}
>
{t("modules.scheduledRooms.title")}
</h1>
<ProfileButton />
</Backbar>
<select
onChange={(e) => {
const newGetOnlyMyScheduledRooms = e.target.value === "true";
if (newGetOnlyMyScheduledRooms === getOnlyMyScheduledRooms) {
return;
}
queryClient.prefetchQuery(
[GET_SCHEDULED_ROOMS, "", newGetOnlyMyScheduledRooms],
() =>
wsFetch({
op: GET_SCHEDULED_ROOMS,
d: {
cursor: "",
getOnlyMyScheduledRooms: newGetOnlyMyScheduledRooms,
},
}),
{ staleTime: 0 }
);
setQueryState({
cursors: [""],
getOnlyMyScheduledRooms: newGetOnlyMyScheduledRooms,
});
}}
value={"" + getOnlyMyScheduledRooms}
>
<option value="false">
{t("modules.scheduledRooms.allRooms")}
</option>
<option value="true">{t("modules.scheduledRooms.myRooms")}</option>
</select>
<EditScheduleRoomModalController
onScheduledRoom={(editInfo, data, _resp) => {
queryClient.setQueryData<ScheduledRoomsInfo>(
[GET_SCHEDULED_ROOMS, editInfo.cursor, getOnlyMyScheduledRooms],
(d) => {
return {
scheduledRooms: (d?.scheduledRooms || []).map((x) =>
x.id === editInfo.scheduleRoomToEdit.id
? {
...x,
name: data.name,
description: data.description,
scheduledFor: data.scheduledFor.toISOString(),
}
: x
),
nextCursor: d?.nextCursor,
};
}
);
}}
>
{({ onEdit }) =>
cursors.map((cursor, i) => (
<Page
getOnlyMyScheduledRooms={getOnlyMyScheduledRooms}
onLoadMore={(o) =>
setQueryState({
cursors: [...cursors, o],
getOnlyMyScheduledRooms,
})
}
onEdit={onEdit}
isOnlyPage={cursors.length === 1}
isLastPage={cursors.length - 1 === i}
key={cursor}
cursor={cursor}
/>
))
}
</EditScheduleRoomModalController>
<div style={{ height: 40 }} />
</BodyWrapper>
</Wrapper>
<BottomVoiceControl>
<div className={`mb-8 flex px-5`}>
<Button
variant="slim"
dogeProbability={0.01}
onClick={() => {
setShowScheduleRoomModal(true);
}}
>
<h3 className={`text-2xl`}>
{t("modules.scheduledRooms.scheduleRoomHeader")}
</h3>
</Button>
</div>
</BottomVoiceControl>
{showScheduleRoomModal ? (
<ScheduleRoomModal
onScheduledRoom={(data, resp) => {
queryClient.setQueryData<ScheduledRoomsInfo>(
[GET_SCHEDULED_ROOMS, "", getOnlyMyScheduledRooms],
(d) => {
return {
scheduledRooms: [
{
roomId: null,
creator: me!,
creatorId: me!.id,
description: data.description,
id: resp.scheduledRoom.id,
name: data.name,
numAttending: 0,
scheduledFor: data.scheduledFor.toISOString(),
},
...(d?.scheduledRooms || []),
],
nextCursor: d?.nextCursor,
};
}
);
}}
onRequestClose={() => setShowScheduleRoomModal(false)}
/>
) : null}
</div>
);
};
import React, { useState } from "react";
import { ScheduledRoom } from "../../types";
import { ScheduleRoomFormData, ScheduleRoomModal } from "./ScheduleRoomModal";
type State = {
scheduleRoomToEdit: ScheduledRoom;
cursor: string;
};
interface EditScheduleRoomModalControllerProps {
onScheduledRoom: (
editInfo: State,
data: ScheduleRoomFormData,
resp: any
) => void;
children: (x: { onEdit: (y: State) => void }) => React.ReactNode;
}
export const EditScheduleRoomModalController: React.FC<EditScheduleRoomModalControllerProps> = ({
onScheduledRoom,
children,
}) => {
const [editInfo, setScheduleRoomToEdit] = useState<State | null>(null);
return (
<>
{editInfo ? (
<ScheduleRoomModal
editInfo={{
id: editInfo.scheduleRoomToEdit.id,
intialValues: {
cohosts: [],
description: editInfo.scheduleRoomToEdit.description,
name: editInfo.scheduleRoomToEdit.name,
scheduledFor: new Date(editInfo.scheduleRoomToEdit.scheduledFor),
},
}}
onScheduledRoom={(...vals) => onScheduledRoom(editInfo, ...vals)}
onRequestClose={() => setScheduleRoomToEdit(null)}
/>
) : null}
{children({ onEdit: setScheduleRoomToEdit })}
</>
);
};
import { differenceInMilliseconds, isPast, isToday } from "date-fns";
import React, { useEffect, useMemo, useState } from "react";
import { useMutation } from "react-query";
import { useHistory } from "react-router-dom";
import {
wsend,
wsMutation,
wsMutationThrowError,
} from "../../../createWebsocket";
import { useCurrentRoomStore } from "../../../webrtc/stores/useCurrentRoomStore";
import { AddToCalendarButton } from "../../components/add-to-calendar/AddToCalendarButton";
import { Avatar } from "../../components/Avatar";
import { Button } from "../../components/Button";
import { modalConfirm } from "../../components/ConfirmModal";
import { ScheduledRoom } from "../../types";
import { roomToCurrentRoom } from "../../utils/roomToCurrentRoom";
import { useMeQuery } from "../../utils/useMeQuery";
import { useTypeSafeTranslation } from "../../utils/useTypeSafeTranslation";
import { useRoomChatStore } from "../room-chat/useRoomChatStore";
import { CopyScheduleRoomLinkButton } from "./CopyScheduleRoomLinkButton";
interface ScheduledRoomCardProps {
onEdit: () => void;
onDeleteComplete: () => void;
info: ScheduledRoom;
noCopyLinkButton?: boolean;
}
export const ScheduledRoomCard: React.FC<ScheduledRoomCardProps> = ({
onEdit,
onDeleteComplete,
noCopyLinkButton,
info: { id, name, scheduledFor, creator, description, roomId },
}) => {
const history = useHistory();
const {
mutateAsync: mutateAsyncStartRoom,
isLoading: isLoadingStartRoom,
} = useMutation(wsMutationThrowError, {
onSuccess: ({ room }) => {
console.log("new room voice server id: " + room.voiceServerId);
useRoomChatStore.getState().clearChat();
wsend({ op: "get_current_room_users", d: {} });
history.push("/room/" + room.id);
useCurrentRoomStore
.getState()
.setCurrentRoom(() => roomToCurrentRoom(room));
},
});
const { mutateAsync, isLoading } = useMutation(wsMutation, {
onSuccess: () => {
onDeleteComplete();
},
});
const [, rerender] = useState(0);
const dt = useMemo(() => new Date(scheduledFor), [scheduledFor]);
useEffect(() => {
let done = false;
const id = setTimeout(() => {
done = true;
rerender((x) => x + 1);
}, differenceInMilliseconds(dt, new Date()) + 1000); // + 1 second to be safe
return () => {
if (!done) {
clearTimeout(id);
}
};
}, [dt]);
const { me } = useMeQuery();
const { t } = useTypeSafeTranslation();
const isCreator = me?.id === creator.id;
const url = window.location.origin + `/scheduled-room/${id}`;
return (
<div>
<div className={`w-full bg-simple-gray-33 py-2.5 px-5 rounded-lg`}>
<div className={`text-white`}>
<div className={`flex justify-between`}>
<div>
{isToday(dt)
? t("common.formattedIntlTime", { time: dt })
: t("common.formattedIntlDate", { date: dt })
}
</div>
<AddToCalendarButton
event={{
name: name,
details: description,
location: url,
startsAt: dt.toISOString(),
endsAt: new Date(
dt.getTime() + 1 * 60 * 60 * 1000
).toISOString(),
}}
/>
{isCreator ? (
<div className={`flex`}>
<Button variant="small" onClick={() => onEdit()}>
{t("common.edit")}
</Button>
<div className={`ml-4`}>
<Button
loading={isLoading}
variant="small"
onClick={() =>
modalConfirm(
"Are you sure you want to delete this scheduled room?",
() => {
mutateAsync({
op: "delete_scheduled_room",
d: { id },
});
}
)
}
>
{t("common.delete")}
</Button>
</div>
</div>
) : null}
</div>
<div className={`flex items-center my-4`}>
<Avatar size={25} src={creator.avatarUrl} />
<div
style={{
display: "-webkit-box",
WebkitBoxOrient: "vertical",
WebkitLineClamp: 3,
}}
className={`ml-2 text-left flex-1 text-xl text-simple-gray-d9 text-ellipsis overflow-hidden break-all`}
>
{name.slice(0, 100)}
</div>
{noCopyLinkButton ? null : (
<CopyScheduleRoomLinkButton text={url} />
)}
</div>
<div className={`break-all`}>
{creator.displayName}
{description ? ` | ` + description : ``}
</div>
{isPast(dt) ? (
<div className={`mt-4`}>
{isCreator ? (
<Button
loading={isLoadingStartRoom}
onClick={() => {
mutateAsyncStartRoom({
op: "create_room_from_scheduled_room",
d: {
id,
name,
description,
},
});
}}
>
{t("modules.scheduledRooms.startRoom")}
</Button>
) : (
roomId && (
<Button
loading={isLoadingStartRoom}
onClick={() => {
wsend({ op: "join_room", d: { roomId } });
history.push("/room/" + roomId);
}}
>
{t("common.joinRoom")}
</Button>
)
)}
</div>
) : null}
</div>
</div>
</div>
);
};
import React, { createContext } from "react";
import { soundEffects, useSoundEffectStore } from "./useSoundEffectStore";
const soundKeys = Object.keys(soundEffects);
export const SoundEffectContext = createContext<{
playSoundEffect: (name: keyof typeof soundEffects) => void;
}>({ playSoundEffect: () => {} });
export const SoundEffectPlayer: React.FC = ({}) => {
const add = useSoundEffectStore((x) => x.add);
return (
<>
{soundKeys.map((key) => (
<audio
preload="none"
controls={false}
key={key}
ref={(ref) => {
if (ref) {
ref.volume = 0.7;
add(key, ref);
}
}}
src={`/sound-effects/${
soundEffects[key as keyof typeof soundEffects]
}`}
/>
))}
</>
);
};
import React from "react";
import { useLocation } from "react-router-dom";
import { Backbar } from "../components/Backbar";
import { BodyWrapper } from "../components/BodyWrapper";
import { UserProfile } from "../components/UserProfile";
import { Wrapper } from "../components/Wrapper";
import { RoomUser } from "../types";
export const ViewUserPage = () => {
const { state } = useLocation<RoomUser>();
return (
<Wrapper>
<Backbar actuallyGoBack />
<BodyWrapper>
<UserProfile profile={state} />
</BodyWrapper>
</Wrapper>
);
};
import React, { useEffect } from "react";
import { Button } from "../components/Button";
import { Footer } from "../components/Footer";
import { Wrapper } from "../components/Wrapper";
import { apiBaseUrl, __prod__, __staging__ } from "../constants";
import { Logo } from "../svgs/Logo";
import { useTokenStore } from "../utils/useTokenStore";
import qs from "query-string";
import { showErrorToast } from "../utils/showErrorToast";
import { CenterLayout } from "../components/CenterLayout";
import { modalPrompt, PromptModal } from "../components/PromptModal";
import { AlertModal } from "../components/AlertModal";
import { ConfirmModal } from "../components/ConfirmModal";
import { BodyWrapper } from "../components/BodyWrapper";
import { ListItem } from "../components/ListItem";
import { GitHubIcon } from "../svgs/GitHubIcon";
import { TwitterIcon } from "../svgs/TwitterIcon";
import { useTypeSafeTranslation } from "../utils/useTypeSafeTranslation";
interface LoginProps {}
export const Login: React.FC<LoginProps> = () => {
const { t } = useTypeSafeTranslation();
useEffect(() => {
const { error } = qs.parse(window.location.search);
if (error && typeof error === "string") {
showErrorToast(error);
}
}, []);
return (
<CenterLayout>
<Wrapper>
<BodyWrapper>
<div className={`my-8`}>
<Logo />
</div>
<div className={`text-4xl mb-4 tracking-tight font-extrabold`}>
{t("pages.login.headerText")}
</div>
<ul className={`my-4 mb-10 text-xl`}>
<ListItem>{t("pages.login.featureText_1")}</ListItem>
<ListItem>{t("pages.login.featureText_2")}</ListItem>
<ListItem>{t("pages.login.featureText_3")}</ListItem>
<ListItem>
<a
href="https://github.com/benawad/dogehouse"
className={`p-0 text-blue-400`}
>
{t("pages.login.featureText_4")}
</a>
</ListItem>
<ListItem>{t("pages.login.featureText_5")}</ListItem>
<ListItem>{t("pages.login.featureText_6")}</ListItem>
</ul>
<div className={`mb-8`}>
<Button
variant="slim"
style={{ backgroundColor: "#333" }}
onClick={() =>
(window.location.href =
apiBaseUrl +
"/auth/github/web" +
(__staging__
? "?redirect_after_base=" + window.location.origin
: ""))
}
>
<span className={`inline-flex items-center`}>
<GitHubIcon className={`h-6 w-6`} />
<p className={`ml-3`}>{t("pages.login.loginGithub")}</p>
</span>
</Button>
</div>
{!__staging__ ? (
<Button
variant="slim"
style={{ backgroundColor: "#0C84CF" }}
onClick={() =>
(window.location.href =
apiBaseUrl +
"/auth/twitter/web" +
(process.env.REACT_APP_IS_STAGING === "true"
? "?redirect_after_base=" + window.location.origin
: ""))
}
>
<span className={`inline-flex items-center`}>
<TwitterIcon className={`h-6 w-6`} />
<p className={`ml-3`}>{t("pages.login.loginTwitter")}</p>
</span>
</Button>
) : null}
{!__prod__ ? (
<Button
variant="slim"
className={`m-8`}
onClick={() => {
modalPrompt("username", async (name) => {
if (!name) {
return;
}
const r = await fetch(
`${apiBaseUrl}/dev/test-info?username=` + name
);
const d = await r.json();
useTokenStore.getState().setTokens({
accessToken: d.accessToken,
refreshToken: d.refreshToken,
});
});
}}
>
{t("pages.login.createTestUser")}
</Button>
) : null}
</BodyWrapper>
</Wrapper>
<div className={`mb-6 px-5`}>
<Footer isLogin />
</div>
<AlertModal />
<PromptModal />
<ConfirmModal />
</CenterLayout>
);
};
import { useAtom } from "jotai";
import React from "react";
import { useHistory } from "react-router-dom";
import { wsend } from "../../createWebsocket";
import { useCurrentRoomStore } from "../../webrtc/stores/useCurrentRoomStore";
import { followingOnlineAtom } from "../atoms";
import { Avatar } from "../components/Avatar";
import { Backbar } from "../components/Backbar";
import { BodyWrapper } from "../components/BodyWrapper";
import { Button } from "../components/Button";
import { Wrapper } from "../components/Wrapper";
import { useTypeSafeTranslation } from "../utils/useTypeSafeTranslation";
interface FriendListProps {}
export const FollowingOnlineList: React.FC<FriendListProps> = () => {
const history = useHistory();
const [{ users, nextCursor }] = useAtom(followingOnlineAtom);
const { currentRoom } = useCurrentRoomStore();
const { t } = useTypeSafeTranslation();
return (
<Wrapper>
<Backbar />
<BodyWrapper>
<div className={`mb-4 text-2xl`}>
{t("pages.followingOnlineList.listHeader")}
</div>
{users.length === 0 ? <div>{t("common.noUsersFound")}</div> : null}
{users.map((u) => (
<div
className={`border-b border-solid border-simple-gray-3c flex py-4 px-2 items-center`}
key={u.id}
>
<button onClick={() => history.push(`/user`, u)}>
<Avatar src={u.avatarUrl} isOnline={u.online} />
</button>
<button
onClick={() => {
if (u.currentRoom) {
if (u.currentRoom.id !== currentRoom?.id) {
wsend({ op: "join_room", d: { roomId: u.currentRoom.id } });
}
history.push("/room/" + u.currentRoom.id);
}
}}
className={`ml-4 flex-1 text-left`}
>
<div className={`text-lg`}>
{u.displayName || "@" + u.username}
</div>
<div style={{ color: "" }}>
{u.currentRoom ? (
<span>
{t("pages.followingOnlineList.currentRoom")}{" "}
<b>{u.currentRoom.name}</b>
</span>
) : null}
</div>
</button>
{u.followsYou ? (
<div className={`ml-auto`}>
<Button
onClick={() => {
wsend({
op: "create-room",
d: {
roomName: "My Private Room",
value: "private",
userIdToInvite: u.id,
},
});
}}
variant="small"
>
{t("pages.followingOnlineList.startPrivateRoom")}
</Button>
</div>
) : null}
</div>
))}
{nextCursor ? (
<div className={`flex justify-center my-10`}>
<Button
variant="small"
onClick={() =>
wsend({
op: "fetch_following_online",
d: { cursor: nextCursor },
})
}
>
{t("common.loadMore")}
</Button>
</div>
) : null}
</BodyWrapper>
</Wrapper>
);
};
import React from "react";
import { useHistory } from "react-router-dom";
import { closeWebSocket, wsend } from "../../createWebsocket";
import { Backbar } from "../components/Backbar";
import { BodyWrapper } from "../components/BodyWrapper";
import { Button } from "../components/Button";
import { modalConfirm } from "../components/ConfirmModal";
import { UserProfile } from "../components/UserProfile";
import { Wrapper } from "../components/Wrapper";
import { useMeQuery } from "../utils/useMeQuery";
import { useTokenStore } from "../utils/useTokenStore";
import { useTypeSafeTranslation } from "../utils/useTypeSafeTranslation";
interface MyProfilePageProps {}
export const MyProfilePage: React.FC<MyProfilePageProps> = ({}) => {
const { me } = useMeQuery();
const history = useHistory();
const { t } = useTypeSafeTranslation();
return (
<Wrapper>
<Backbar actuallyGoBack>
<div className={`ml-auto flex items-center`}>
<Button
className={`m-2.5`}
onClick={() => {
modalConfirm("Are you sure you want to logout?", () => {
history.push("/");
closeWebSocket();
useTokenStore
.getState()
.setTokens({ accessToken: "", refreshToken: "" });
});
}}
variant="small"
>
{t("pages.myProfile.logout")}
</Button>
</div>
</Backbar>
<BodyWrapper>
{me ? (
<UserProfile profile={me} />
) : (
<div>{t("pages.myProfile.probablyLoading")}</div>
)}
<div className={`pt-6 flex`}>
<Button
style={{ marginRight: "10px" }}
variant="small"
onClick={() => history.push(`/voice-settings`)}
>
{t("pages.myProfile.voiceSettings")}
</Button>
<Button
variant="small"
onClick={() => history.push(`/sound-effect-settings`)}
>
{t("pages.myProfile.soundSettings")}
</Button>
</div>
<div className={`pt-6 flex`}>
<Button
variant="small"
color="red"
onClick={() => {
modalConfirm(
"Are you sure you want to permanently delete your account?",
() => {
wsend({ op: "delete_account", d: {} });
}
);
}}
>
{t("pages.myProfile.deleteAccount")}
</Button>
</div>
</BodyWrapper>
</Wrapper>
);
};
import React from "react";
import { Backbar } from "../components/Backbar";
import { BodyWrapper } from "../components/BodyWrapper";
import { VoiceSettings } from "../components/VoiceSettings";
import { Wrapper } from "../components/Wrapper";
interface VoiceSettingsPageProps {}
export const VoiceSettingsPage: React.FC<VoiceSettingsPageProps> = () => {
return (
<Wrapper>
<Backbar actuallyGoBack />
<BodyWrapper>
<VoiceSettings />
</BodyWrapper>
</Wrapper>
);
};
import { useAtom } from "jotai";
import React, { useRef } from "react";
import { useHistory } from "react-router-dom";
import { toast } from "react-toastify";
import { wsend } from "../../createWebsocket";
import { useCurrentRoomStore } from "../../webrtc/stores/useCurrentRoomStore";
import { inviteListAtom } from "../atoms";
import { Avatar } from "../components/Avatar";
import { Backbar } from "../components/Backbar";
import { BodyWrapper } from "../components/BodyWrapper";
import { Button } from "../components/Button";
import { Input } from "../components/Input";
import { InviteButton } from "../components/InviteButton";
import { Wrapper } from "../components/Wrapper";
import { useTypeSafeTranslation } from "../utils/useTypeSafeTranslation";
interface InviteListProps {}
export const InviteList: React.FC<InviteListProps> = () => {
const history = useHistory();
const [{ nextCursor, users }] = useAtom(inviteListAtom);
const { currentRoom: room } = useCurrentRoomStore();
const path = `/room/${room?.id}`;
const url = window.location.origin + path;
const inputRef = useRef<HTMLInputElement>(null);
const { t } = useTypeSafeTranslation();
if (!room) {
return (
<Wrapper>
<Backbar />
<BodyWrapper>
<Button onClick={() => history.push("/")}>
{t("pages.inviteList.roomGone")}
</Button>
</BodyWrapper>
</Wrapper>
);
}
return (
<Wrapper>
<Backbar actuallyGoBack />
<BodyWrapper>
{room.isPrivate ? null : (
<>
{!navigator.share ? (
<div className={`text-2xl mb-2`}>
{t("pages.inviteList.shareRoomLink")}
</div>
) : null}
<div className={`mb-8 flex`}>
<Input readOnly ref={inputRef} value={url} />
<Button
variant="small"
onClick={() => {
if (navigator.share) {
navigator.share({ url });
} else {
inputRef.current?.select();
document.execCommand("copy");
toast("copied to clipboard", { type: "success" });
}
}}
>
{!navigator.share ? "copy" : "share link to room"}
</Button>
</div>
</>
)}
{users.length ? (
<div className={`my-4 text-2xl`}>
{t("pages.inviteList.inviteFollowers")}
</div>
) : (
<div className={`my-4 text-2xl`}>
{t("pages.inviteList.whenFollowersOnline")}
</div>
)}
{users.map((u) => (
<div
className={`border-b border-solid border-simple-gray-3c flex py-4 px-2 items-center`}
key={u.id}
>
<button onClick={() => history.push(`/user`, u)}>
<Avatar src={u.avatarUrl} />
</button>
<button
onClick={() => {
history.push(`/user`, u);
}}
className={`ml-4`}
>
<div className={`text-lg`}>{u.displayName}</div>
<div>@{u.username}</div>
</button>
<div className={`ml-auto`}>
<InviteButton
onClick={() => {
wsend({
op: "invite_to_room",
d: {
userId: u.id,
},
});
}}
/>
</div>
</div>
))}
{nextCursor ? (
<div className={`flex justify-center my-10`}>
<Button
variant="small"
onClick={() =>
wsend({
op: "fetch_invite_list",
d: { cursor: nextCursor },
})
}
>
{t("common.loadMore")}
</Button>
</div>
) : null}
</BodyWrapper>
</Wrapper>
);
};
import React, { useState } from "react";
import { Calendar } from "react-feather";
import { useQuery, useQueryClient } from "react-query";
import { useHistory } from "react-router-dom";
import { wsend, wsFetch } from "../../createWebsocket";
import { useCurrentRoomStore } from "../../webrtc/stores/useCurrentRoomStore";
import { useSocketStatus } from "../../webrtc/stores/useSocketStatus";
import { BodyWrapper } from "../components/BodyWrapper";
import { BottomVoiceControl } from "../components/BottomVoiceControl";
import { Button } from "../components/Button";
import { CircleButton } from "../components/CircleButton";
import { CreateRoomModal } from "../components/CreateRoomModal";
import { ProfileButton } from "../components/ProfileButton";
import { RoomCard } from "../components/RoomCard";
import { Spinner } from "../components/Spinner";
import { Wrapper } from "../components/Wrapper";
import { EditScheduleRoomModalController } from "../modules/scheduled-rooms/EditScheduleRoomModalController";
import { ScheduledRoomCard } from "../modules/scheduled-rooms/ScheduledRoomCard";
import { GET_SCHEDULED_ROOMS } from "../modules/scheduled-rooms/ScheduledRoomsPage";
import { Logo } from "../svgs/Logo";
import { PeopleIcon } from "../svgs/PeopleIcon";
import { CurrentRoom, PublicRoomsQuery, ScheduledRoom } from "../types";
import { useTypeSafeTranslation } from "../utils/useTypeSafeTranslation";
interface HomeProps {}
const get_top_public_rooms = "get_top_public_rooms";
const Page = ({
currentRoom,
cursor,
isLastPage,
isOnlyPage,
}: {
currentRoom: CurrentRoom | null;
cursor: number;
isLastPage: boolean;
isOnlyPage: boolean;
onLoadMore: (o: number) => void;
}) => {
const { t } = useTypeSafeTranslation();
const history = useHistory();
const { status } = useSocketStatus();
const { isLoading, data } = useQuery<PublicRoomsQuery>(
[get_top_public_rooms, cursor],
() =>
wsFetch<any>({
op: get_top_public_rooms,
d: { cursor },
}),
{
staleTime: Infinity,
enabled: status === "auth-good",
refetchOnMount: "always",
}
);
if (isLoading) {
return <Spinner centered={true} />;
}
if (!data) {
return null;
}
if (isOnlyPage && data.rooms.length === 0) {
return null;
}
return (
<>
{data.rooms.map((r) =>
r.id === currentRoom?.id ? null : (
<div className={`mt-4`} key={r.id}>
<RoomCard
onClick={() => {
wsend({ op: "join_room", d: { roomId: r.id } });
history.push("/room/" + r.id);
}}
room={r}
currentRoomId={currentRoom?.id}
/>
</div>
)
)}
{isLastPage && data.nextCursor ? (
<div className={`flex justify-center my-10`}>
<Button
variant="small"
onClick={() =>
wsend({
op: "get_top_public_rooms",
d: { cursor: data.nextCursor },
})
}
>
{t("common.loadMore")}
</Button>
</div>
) : null}
</>
);
};
const get_my_scheduled_rooms_about_to_start =
"get_my_scheduled_rooms_about_to_start";
export type GetMyScheduledRoomsAboutToStartQuery = {
scheduledRooms: ScheduledRoom[];
};
export const Home: React.FC<HomeProps> = () => {
const { t } = useTypeSafeTranslation();
const history = useHistory();
const { currentRoom } = useCurrentRoomStore();
const [cursors, setCursors] = useState([0]);
const [showCreateRoomModal, setShowCreateRoomModal] = useState(false);
const queryClient = useQueryClient();
const { status } = useSocketStatus();
const { data } = useQuery<GetMyScheduledRoomsAboutToStartQuery>(
get_my_scheduled_rooms_about_to_start,
() => wsFetch<any>({ op: get_my_scheduled_rooms_about_to_start, d: {} }),
{
staleTime: Infinity,
enabled: status === "auth-good",
refetchOnMount: "always",
}
);
return (
<div className={`flex flex-col flex-1`}>
<Wrapper>
<BodyWrapper>
<div className={`mb-10 mt-8`}>
<Logo />
</div>
<div className={`mb-6 flex justify-center`}>
<div className={`mr-4`}>
<CircleButton
onClick={() => {
wsend({ op: "fetch_following_online", d: { cursor: 0 } });
history.push("/following-online");
}}
>
<PeopleIcon width={30} height={30} fill="#fff" />
</CircleButton>
</div>
<div className={`ml-2`}>
<CircleButton
onClick={() => {
queryClient.prefetchQuery(
[GET_SCHEDULED_ROOMS, "", false],
() =>
wsFetch({
op: GET_SCHEDULED_ROOMS,
d: {
cursor: "",
getOnlyMyScheduledRooms: false,
},
}),
{ staleTime: 0 }
);
history.push("/scheduled-rooms");
}}
>
<Calendar width={30} height={30} color="#fff" />
</CircleButton>
</div>
<div className={`ml-2`}>
<ProfileButton circle size={60} />
</div>
</div>
<EditScheduleRoomModalController
onScheduledRoom={(editInfo, data, _resp) => {
queryClient.setQueryData<GetMyScheduledRoomsAboutToStartQuery>(
get_my_scheduled_rooms_about_to_start,
(d) => {
return {
scheduledRooms: (d?.scheduledRooms || []).map((x) =>
x.id === editInfo.scheduleRoomToEdit.id
? {
...x,
name: data.name,
description: data.description,
scheduledFor: data.scheduledFor.toISOString(),
}
: x
),
};
}
);
}}
>
{({ onEdit }) =>
data?.scheduledRooms.map((sr) => (
<ScheduledRoomCard
key={sr.id}
info={sr}
onEdit={() => onEdit({ scheduleRoomToEdit: sr, cursor: "" })}
onDeleteComplete={() => {
queryClient.setQueryData<GetMyScheduledRoomsAboutToStartQuery>(
get_my_scheduled_rooms_about_to_start,
(d) => {
return {
scheduledRooms: d?.scheduledRooms.filter(
(x) => x.id !== sr.id
) as ScheduledRoom[],
};
}
);
}}
/>
))
}
</EditScheduleRoomModalController>
{currentRoom ? (
<div className={`my-8`}>
<RoomCard
active
onClick={() => history.push("/room/" + currentRoom.id)}
room={currentRoom}
currentRoomId={currentRoom.id}
/>
</div>
) : null}
{cursors.map((cursor, i) => (
<Page
key={cursor}
currentRoom={currentRoom}
cursor={cursor}
isOnlyPage={cursors.length === 1}
onLoadMore={(c) => setCursors([...cursors, c])}
isLastPage={i === cursors.length - 1}
/>
))}
<div style={{ height: 40 }} />
</BodyWrapper>
</Wrapper>
<BottomVoiceControl>
<div className={`mb-8 flex px-5`}>
<Button
variant="slim"
dogeProbability={0.01}
onClick={() => {
setShowCreateRoomModal(true);
}}
>
<h3 className={`text-2xl`}>{t("pages.home.createRoom")}</h3>
</Button>
</div>
</BottomVoiceControl>
{showCreateRoomModal ? (
<CreateRoomModal onRequestClose={() => setShowCreateRoomModal(false)} />
) : null}
</div>
);
};
import React from "react";
import { Backbar } from "../components/Backbar";
import { BodyWrapper } from "../components/BodyWrapper";
import { SoundEffectSettings } from "../components/SoundEffectSettings";
import { Wrapper } from "../components/Wrapper";
export const SoundEffectSettingsPage: React.FC = () => {
return (
<Wrapper>
<Backbar actuallyGoBack />
<BodyWrapper>
<SoundEffectSettings />
</BodyWrapper>
</Wrapper>
);
};
import React, { useState } from "react";
import { Redirect } from "react-router-dom";
import { wsend } from "../../createWebsocket";
import { Backbar } from "../components/Backbar";
import { BodyWrapper } from "../components/BodyWrapper";
import { Button } from "../components/Button";
import { Wrapper } from "../components/Wrapper";
import { useMeQuery } from "../utils/useMeQuery";
import { useTypeSafeTranslation } from "../utils/useTypeSafeTranslation";
interface SearchUsersProps {}
export const BanUsersPage: React.FC<SearchUsersProps> = ({}) => {
const { me } = useMeQuery();
const [username, setUsername] = useState("");
const [reason, setReason] = useState("");
const { t } = useTypeSafeTranslation();
if (!me) {
return null;
}
if (me.username !== "benawad") {
return <Redirect to="/" />;
}
return (
<Wrapper>
<Backbar />
<BodyWrapper>
<input
className={`mb-8`}
autoFocus
placeholder="username to ban..."
value={username}
onChange={(e) => setUsername(e.target.value)}
/>
<input
className={`mb-16`}
autoFocus
placeholder="reason"
value={reason}
onChange={(e) => setReason(e.target.value)}
/>
<Button
onClick={() => {
if (username && reason) {
wsend({
op: "ban",
d: {
username,
reason,
},
});
}
}}
>
{t("pages.banUser.ban")}
</Button>
</BodyWrapper>
</Wrapper>
);
};
import React, { useState } from "react";
import { useAtom } from "jotai";
import { userSearchAtom } from "../atoms";
import { Backbar } from "../components/Backbar";
import { Button } from "../components/Button";
import { Wrapper } from "../components/Wrapper";
import { Codicon } from "../svgs/Codicon";
import { BodyWrapper } from "../components/BodyWrapper";
import { useTypeSafeTranslation } from "../utils/useTypeSafeTranslation";
interface SearchUsersProps {}
export const SearchUsersPage: React.FC<SearchUsersProps> = ({}) => {
const [{ loading }] = useAtom(userSearchAtom);
const [query, setQuery] = useState("");
const { t } = useTypeSafeTranslation();
return (
<Wrapper>
<Backbar />
<BodyWrapper>
<form
onSubmit={(e) => {
e.preventDefault();
if (query) {
}
}}
className={`flex`}
>
<input
autoFocus
placeholder={t("pages.searchUser.search")}
value={query}
onChange={(e) => setQuery(e.target.value)}
/>
<Button type="submit" variant="small">
<Codicon name="search" />
</Button>
</form>
{loading ? <div className={`my-8`}>{t("common.loading")}</div> : null}
</BodyWrapper>
</Wrapper>
);
};
import { useAtom } from "jotai";
import React from "react";
import { useTypeSafeTranslation } from "../utils/useTypeSafeTranslation";
import { useHistory, useLocation, useRouteMatch } from "react-router-dom";
import { wsend } from "../../createWebsocket";
import { useCurrentRoomStore } from "../../webrtc/stores/useCurrentRoomStore";
import { followerMapAtom, followingMapAtom } from "../atoms";
import { Avatar } from "../components/Avatar";
import { Backbar } from "../components/Backbar";
import { BodyWrapper } from "../components/BodyWrapper";
import { Button } from "../components/Button";
import { Wrapper } from "../components/Wrapper";
import { onFollowUpdater } from "../utils/onFollowUpdater";
import { useMeQuery } from "../utils/useMeQuery";
interface FollowListPageProps {}
export const FollowListPage: React.FC<FollowListPageProps> = () => {
const { pathname } = useLocation();
const {
params: { userId },
} = useRouteMatch<{ userId: string }>();
const [followerMap, setFollowerMap] = useAtom(followerMapAtom);
const [followingMap, setFollowingMap] = useAtom(followingMapAtom);
const { me } = useMeQuery();
const { setCurrentRoom } = useCurrentRoomStore();
const history = useHistory();
const { t } = useTypeSafeTranslation();
const isFollowing = pathname.startsWith("/following");
const users = isFollowing
? followingMap[userId]?.users || []
: followerMap[userId]?.users || [];
const nextCursor = isFollowing
? followingMap[userId]?.nextCursor
: followerMap[userId]?.nextCursor;
return (
<Wrapper>
<Backbar actuallyGoBack />
<BodyWrapper>
{!users.length ? <div>{t("common.noUsersFound")}</div> : null}
{users.map((profile) => (
<div
className={`border-b border-solid border-simple-gray-3c flex py-4 px-2 items-center`}
key={profile.id}
>
<button onClick={() => history.push(`/user`, profile)}>
<Avatar src={profile.avatarUrl} />
</button>
<button
onClick={() => history.push(`/user`, profile)}
className={`ml-8`}
>
<div className={`text-lg`}>{profile.displayName}</div>
<div style={{ color: "" }}>@{profile.username}</div>
</button>
{me?.id === profile.id ||
profile.youAreFollowing === undefined ||
profile.youAreFollowing === null ? null : (
<div className={`ml-auto`}>
<Button
onClick={() => {
wsend({
op: "follow",
d: {
userId: profile.id,
value: !profile.youAreFollowing,
},
});
onFollowUpdater(setCurrentRoom, me, profile);
const fn = isFollowing ? setFollowingMap : setFollowerMap;
fn((m) => ({
...m,
[userId]: {
users: m[userId].users.map((u) => {
if (profile.id === u.id) {
return {
...u,
youAreFollowing: !profile.youAreFollowing,
};
}
return u;
}),
nextCursor: m[userId].nextCursor,
},
}));
}}
variant="small"
>
{profile.youAreFollowing ? "following" : "follow"}
</Button>
</div>
)}
</div>
))}
{nextCursor ? (
<div className={`flex justify-center my-10`}>
<Button
variant="small"
onClick={() =>
wsend({
op: `fetch_follow_list`,
d: { isFollowing, userId, cursor: nextCursor },
})
}
>
{t("common.loadMore")}
</Button>
</div>
) : null}
</BodyWrapper>
</Wrapper>
);
};
import React from "react";
import { Wrapper } from "../components/Wrapper";
import { Logo } from "../svgs/Logo";
import { Link } from "react-router-dom";
import { BodyWrapper } from "../components/BodyWrapper";
import { useTypeSafeTranslation } from "../utils/useTypeSafeTranslation";
interface NotFoundPageProps {}
export const NotFoundPage: React.FC<NotFoundPageProps> = () => {
const { t } = useTypeSafeTranslation();
return (
<Wrapper>
<BodyWrapper>
<div className={`mb-10 mt-8`}>
<Logo />
</div>
<div className={`text-2xl`}>{t("pages.notFound.whoopsError")}</div>
{t("pages.notFound.goHomeMessage")}
<Link to="/" className={`text-blue-400 ml-2`}>
{t("pages.notFound.goHomeLinkText")}
</Link>
</BodyWrapper>
</Wrapper>
);
};
import React, { useState } from "react";
import { Redirect, useRouteMatch } from "react-router-dom";
import { wsend } from "../../createWebsocket";
import { useCurrentRoomStore } from "../../webrtc/stores/useCurrentRoomStore";
import { useMuteStore } from "../../webrtc/stores/useMuteStore";
import { useCurrentRoomInfo } from "../atoms";
import { Backbar } from "../components/Backbar";
import { BodyWrapper } from "../components/BodyWrapper";
import { BottomVoiceControl } from "../components/BottomVoiceControl";
import { CircleButton } from "../components/CircleButton";
import { modalConfirm } from "../components/ConfirmModal";
import { CreateRoomModal } from "../components/CreateRoomModal";
import { ProfileButton } from "../components/ProfileButton";
import { ProfileModal } from "../components/ProfileModal";
import { RoomUserNode } from "../components/RoomUserNode";
import { Wrapper } from "../components/Wrapper";
import { useShouldFullscreenChat } from "../modules/room-chat/useShouldFullscreenChat";
import { Codicon } from "../svgs/Codicon";
import { BaseUser } from "../types";
import { isUuid } from "../utils/isUuid";
import { useMeQuery } from "../utils/useMeQuery";
import { useTypeSafeTranslation } from "../utils/useTypeSafeTranslation";
interface RoomPageProps {}
export const RoomPage: React.FC<RoomPageProps> = () => {
const {
params: { id },
} = useRouteMatch<{ id: string }>();
const [userProfileId, setUserProfileId] = useState("");
const { currentRoom: room } = useCurrentRoomStore();
const { muted } = useMuteStore();
const { me } = useMeQuery();
const {
isMod: iAmMod,
isCreator: iAmCreator,
canSpeak: iCanSpeak,
} = useCurrentRoomInfo();
const fullscreenChatOpen = useShouldFullscreenChat();
const [showCreateRoomModal, setShowCreateRoomModal] = useState(false);
const { t } = useTypeSafeTranslation();
// useEffect(() => {
// if (room?.users.length) {
// setUserProfileId(room.users[0].id);
// wsend({ op: "follow_info", d: { userId: room.users[0].id } });
// }
// }, []);
if (!isUuid(id)) {
return <Redirect to="/" />;
}
if (!room) {
return (
<Wrapper>
<Backbar />
<BodyWrapper>
<div>{t("common.loading")}</div>
</BodyWrapper>
</Wrapper>
);
}
const profile = room.users.find((x) => x.id === userProfileId);
const speakers: BaseUser[] = [];
const unansweredHands: BaseUser[] = [];
const listeners: BaseUser[] = [];
let canIAskToSpeak = false;
room.users.forEach((u) => {
if (u.id === room.creatorId || u.roomPermissions?.isSpeaker) {
speakers.push(u);
} else if (u.roomPermissions?.askedToSpeak) {
unansweredHands.push(u);
} else {
canIAskToSpeak = true;
listeners.push(u);
}
});
return (
<>
<ProfileModal
iAmCreator={iAmCreator}
iAmMod={iAmMod}
isMe={profile?.id === me?.id}
room={room}
onClose={() => setUserProfileId("")}
profile={profile}
/>
{fullscreenChatOpen ? null : (
<Backbar>
<button
disabled={!iAmCreator}
onClick={() => setShowCreateRoomModal(true)}
className={`font-xl truncate flex-1 text-center flex items-center justify-center text-2xl`}
>
<span className={"px-2 truncate"}>{room.name}</span>
</button>
<ProfileButton />
</Backbar>
)}
<Wrapper>
<BodyWrapper>
<div
style={{
gridTemplateColumns: "repeat(auto-fit, 90px)",
}}
className={`w-full grid gap-5`}
>
<div className={`col-span-full text-xl ml-2.5 text-white`}>
{t("pages.room.speakers")} ({speakers.length})
</div>
{speakers.map((u) => (
<RoomUserNode
key={u.id}
room={room}
u={u}
muted={muted}
setUserProfileId={setUserProfileId}
me={me}
profile={profile}
/>
))}
{!iCanSpeak && me && canIAskToSpeak ? (
<div className={`flex flex-col items-center`}>
<CircleButton
title="Request to speak"
size={70}
onClick={() => {
modalConfirm("Would you like to ask to speak?", () => {
wsend({ op: "ask_to_speak", d: {} });
});
}}
>
<Codicon width={36} height={36} name="megaphone" />
</CircleButton>
</div>
) : null}
{unansweredHands.length ? (
<div className={`col-span-full text-xl ml-2.5 text-white`}>
{t("pages.room.requestingToSpeak")} ({unansweredHands.length})
</div>
) : null}
{unansweredHands.map((u) => (
<RoomUserNode
key={u.id}
room={room}
u={u}
muted={muted}
setUserProfileId={setUserProfileId}
me={me}
profile={profile}
/>
))}
{listeners.length ? (
<div className={`col-span-full text-xl mt-2.5 ml-2.5 text-white`}>
{t("pages.room.listeners")} ({listeners.length})
</div>
) : null}
{listeners.map((u) => (
<RoomUserNode
key={u.id}
room={room}
u={u}
muted={muted}
setUserProfileId={setUserProfileId}
me={me}
profile={profile}
/>
))}
</div>
</BodyWrapper>
</Wrapper>
<BottomVoiceControl />
{/* Edit room */}
{showCreateRoomModal ? (
<CreateRoomModal
onRequestClose={() => setShowCreateRoomModal(false)}
name={room.name}
description={room.description}
isPrivate={room.isPrivate}
edit={true}
/>
) : null}
</>
);
};
import React, { useEffect, useRef } from "react";
import { useHistory } from "react-router-dom";
import { useMicIdStore } from "../app/shared-stores";
import { ActiveSpeakerListener } from "./components/ActiveSpeakerListener";
import { AudioRender } from "./components/AudioRender";
import { useCurrentRoomStore } from "./stores/useCurrentRoomStore";
import { useMuteStore } from "./stores/useMuteStore";
import { useVoiceStore } from "./stores/useVoiceStore";
import { useWsHandlerStore } from "./stores/useWsHandlerStore";
import { consumeAudio } from "./utils/consumeAudio";
import { createTransport } from "./utils/createTransport";
import { joinRoom } from "./utils/joinRoom";
import { receiveVoice } from "./utils/receiveVoice";
import { sendVoice } from "./utils/sendVoice";
interface App2Props {}
function closeVoiceConnections(_roomId: string | null) {
const { roomId, mic, nullify } = useVoiceStore.getState();
if (_roomId === null || _roomId === roomId) {
if (mic) {
console.log("stopping mic");
mic.stop();
}
console.log("nulling transports");
nullify();
}
}
export const WebRtcApp: React.FC<App2Props> = () => {
const addMultipleWsListener = useWsHandlerStore(
(s) => s.addMultipleWsListener
);
const { mic } = useVoiceStore();
const { micId } = useMicIdStore();
const { muted } = useMuteStore();
const { setCurrentRoom } = useCurrentRoomStore();
const initialLoad = useRef(true);
const history = useHistory();
useEffect(() => {
if (micId && !initialLoad.current) {
sendVoice();
}
initialLoad.current = false;
}, [micId]);
const consumerQueue = useRef<{ roomId: string; d: any }[]>([]);
async function flushConsumerQueue(_roomId: string) {
try {
for (const {
roomId,
d: { peerId, consumerParameters },
} of consumerQueue.current) {
if (_roomId === roomId) {
await consumeAudio(consumerParameters, peerId);
}
}
} catch (err) {
console.log(err);
} finally {
consumerQueue.current = [];
}
}
useEffect(() => {
if (mic) {
mic.enabled = !muted;
}
}, [mic, muted]);
useEffect(() => {
return addMultipleWsListener({
you_left_room: (d) => {
// assumes you don't rejoin the same room really quickly before websocket fires
setCurrentRoom((cr) => {
if (cr && cr.id === d.roomId) {
history.replace("/");
return null;
}
return cr;
});
closeVoiceConnections(d.roomId);
},
"new-peer-speaker": async (d) => {
const { roomId, recvTransport } = useVoiceStore.getState();
if (recvTransport && roomId === d.roomId) {
await consumeAudio(d.consumerParameters, d.peerId);
} else {
consumerQueue.current = [...consumerQueue.current, { roomId, d: d }];
}
},
"you-are-now-a-speaker": async (d) => {
if (d.roomId !== useVoiceStore.getState().roomId) {
return;
}
// setStatus("connected-speaker");
try {
await createTransport(d.roomId, "send", d.sendTransportOptions);
} catch (err) {
console.log(err);
return;
}
console.log("sending voice");
try {
await sendVoice();
} catch (err) {
console.log(err);
return;
}
},
"you-joined-as-peer": async (d) => {
closeVoiceConnections(null);
useVoiceStore.getState().set({ roomId: d.roomId });
// setStatus("connected-listener");
consumerQueue.current = [];
console.log("creating a device");
try {
await joinRoom(d.routerRtpCapabilities);
} catch (err) {
console.log("error creating a device | ", err);
return;
}
try {
await createTransport(d.roomId, "recv", d.recvTransportOptions);
} catch (err) {
console.log("error creating recv transport | ", err);
return;
}
receiveVoice(() => flushConsumerQueue(d.roomId));
},
"you-joined-as-speaker": async (d) => {
closeVoiceConnections(null);
useVoiceStore.getState().set({ roomId: d.roomId });
// setStatus("connected-speaker");
consumerQueue.current = [];
console.log("creating a device");
try {
await joinRoom(d.routerRtpCapabilities);
} catch (err) {
console.log("error creating a device | ", err);
return;
}
try {
await createTransport(d.roomId, "send", d.sendTransportOptions);
} catch (err) {
console.log("error creating send transport | ", err);
return;
}
console.log("sending voice");
try {
await sendVoice();
} catch (err) {
console.log("error sending voice | ", err);
return;
}
await createTransport(d.roomId, "recv", d.recvTransportOptions);
receiveVoice(() => flushConsumerQueue(d.roomId));
},
});
// eslint-disable-next-line react-hooks/exhaustive-deps
}, []);
return (
<>
<AudioRender />
<ActiveSpeakerListener />
</>
);
};
import { useAtom } from "jotai";
import React, { useEffect, useRef, useState } from "react";
import { Button } from "../../app/components/Button";
import { volumeAtom } from "../../app/shared-atoms";
import { useConsumerStore } from "../stores/useConsumerStore";
interface AudioRenderProps {}
const MyAudio = ({
volume,
onRef,
...props
}: React.DetailedHTMLProps<
React.AudioHTMLAttributes<HTMLAudioElement>,
HTMLAudioElement
> & {
onRef: (a: HTMLAudioElement) => void;
volume: number;
}) => {
const myRef = useRef<HTMLAudioElement>(null);
useEffect(() => {
if (myRef.current) {
myRef.current.volume = volume;
}
}, [volume]);
return (
<audio
ref={(r) => {
// @todo
if (r && !myRef.current) {
(myRef as any).current = r;
onRef(r);
}
}}
{...props}
/>
);
};
export const AudioRender: React.FC<AudioRenderProps> = () => {
const notAllowedErrorCountRef = useRef(0);
const [showAutoPlayModal, setShowAutoPlayModal] = useState(false);
const [globalVolume] = useAtom(volumeAtom);
const { consumerMap } = useConsumerStore();
const audioRefs = useRef<HTMLAudioElement[]>([]);
return (
<>
<div
className={`absolute w-full h-full flex z-50 bg-simple-gray-80 ${
showAutoPlayModal ? "" : "hidden"
}`}
>
<div className={`p-8 rounded m-auto bg-simple-gray-3c`}>
<div className={`text-center mb-4`}>
Browsers require user interaction before they will play audio. Just
click okay to continue.
</div>
<Button
onClick={() => {
setShowAutoPlayModal(false);
audioRefs.current.forEach((a) => {
a.play().catch((err) => {
console.warn(err);
});
});
}}
>
okay
{Object.keys(consumerMap).map((k) => {
const { consumer, volume: userVolume } = consumerMap[k];
return (
<MyAudio
volume={(userVolume / 200) * (globalVolume / 100)}
// autoPlay
playsInline
controls={false}
key={consumer.id}
onRef={(a) => {
console.log(a.duration, a.paused);
audioRefs.current.push(a);
a.srcObject = new MediaStream([consumer.track]);
// prevent modal from showing up more than once in a single render cycle
const notAllowedErrorCount =
notAllowedErrorCountRef.current;
a.play()
.then((x) => console.log({ x }))
.catch((error) => {
if (
error.name === "NotAllowedError" &&
notAllowedErrorCountRef.current ===
notAllowedErrorCount
) {
notAllowedErrorCountRef.current++;
setShowAutoPlayModal(true);
}
console.warn("audioElem.play() failed:%o", error);
});
}}
/>
);
})}
</Button>
</div>
</div>
</>
);
};
import React, { useEffect, useState } from "react";
import { useMicIdStore } from "../../app/shared-stores";
interface MicPickerProps {}
export const MicPicker: React.FC<MicPickerProps> = () => {
const { micId, setMicId } = useMicIdStore();
const [options, setOptions] = useState<
Array<{ id: string; label: string } | null>
>([]);
useEffect(() => {
navigator.mediaDevices
.enumerateDevices()
.then((x) =>
setOptions(
x.map((y) =>
y.kind !== "audioinput" ? null : { id: y.deviceId, label: y.label }
)
)
);
}, []);
return (
<>
{options.length === 0 ? <div>no mics available</div> : null}
{options.length ? (
<select
value={micId}
onChange={(e) => {
const id = e.target.value;
setMicId(id);
}}
>
{options.map((x) =>
!x ? null : (
<option key={x.id} value={x.id}>
{x.label}
</option>
)
)}
</select>
) : null}
</>
);
};
import hark from "hark";
import React, { useEffect } from "react";
import { wsend } from "../../createWebsocket";
import { useCurrentRoomStore } from "../stores/useCurrentRoomStore";
import { useVoiceStore } from "../stores/useVoiceStore";
interface ActiveSpeakerListenerProps {}
export const ActiveSpeakerListener: React.FC<ActiveSpeakerListenerProps> = ({}) => {
const { micStream } = useVoiceStore();
const { currentRoom: room } = useCurrentRoomStore();
const roomId = room?.id;
useEffect(() => {
if (!roomId || !micStream) {
return;
}
const harker = hark(micStream, { threshold: -65, interval: 75 });
harker.on("speaking", () => {
wsend({ op: "speaking_change", d: { value: true } });
});
harker.on("stopped_speaking", () => {
wsend({ op: "speaking_change", d: { value: false } });
});
return () => {
harker.stop();
};
}, [micStream, roomId]);
return null;
};
import { Provider } from "jotai";
import React from "react";
import { QueryClientProvider } from "react-query";
import { ToastContainer } from "react-toastify";
import { AlertModal } from "./app/components/AlertModal";
import { ConfirmModal } from "./app/components/ConfirmModal";
import { InvitedToJoinRoomModal } from "./app/components/InvitedToJoinRoomModal";
import { MuteTitleUpdater } from "./app/components/MuteTitleUpdater";
import { PromptModal } from "./app/components/PromptModal";
import { SoundEffectPlayer } from "./app/modules/sound-effects/SoundEffectPlayer";
import { queryClient } from "./app/queryClient";
interface ProvidersProps {}
export const Providers: React.FC<ProvidersProps> = ({ children }) => {
return (
<Provider>
<QueryClientProvider client={queryClient}>
{children}
<SoundEffectPlayer />
<ToastContainer />
<MuteTitleUpdater />
<InvitedToJoinRoomModal />
<AlertModal />
<PromptModal />
<ConfirmModal />
</QueryClientProvider>
</Provider>
);
};
import translations from "../public/locales/en/translation.json";
const keys: string[] = [];
type TranslationRecord = {
[P in string]: string | TranslationRecord;
};
const _traverseTranslations = (obj: TranslationRecord, path: string[]) => {
Object.keys(obj).forEach((key) => {
if (key.startsWith("_")) {
return;
}
const objOrString = obj[key];
if (typeof objOrString === "string") {
keys.push([...path, key].join("."));
} else {
_traverseTranslations(objOrString, [...path, key]);
}
});
};
export const traverseTranslations = () => {
_traverseTranslations(translations, []);
return keys;
};
import fs from "fs";
import { join } from "path";
import prettier from "prettier";
import { traverseTranslations } from "./traverseTranslations";
const s = `
// this is autogenerated by running \`npm run gen:i18:keys\`
export type TranslationKeys =
${traverseTranslations()
.map((k) => ` "${k}"`)
.join("|\n")}
`;
fs.writeFileSync(
join(__dirname, "../src/generated/translationKeys.ts"),
prettier.format(s, { parser: "babel", useTabs: true })
);
// @ts-ignore
import config from "../../.prettierrc.js";
import english from "../public/locales/en/translation.json";
import fs from "fs";
import { join } from "path";
import prettier from "prettier";
import { traverseTranslations } from "./traverseTranslations";
import { get, set } from "lodash";
const paths = traverseTranslations();
fs.readdirSync(join(__dirname, "../public/locales")).forEach((locale) => {
if (locale === "en") {
return;
}
const filename = join(
__dirname,
"../public/locales",
locale,
"translation.json"
);
const data = JSON.parse(fs.readFileSync(filename, { encoding: "utf-8" }));
paths.forEach((p) => {
if (get(data, p, null) === null) {
set(data, p, get(english, p));
}
});
fs.writeFileSync(
filename,
prettier.format(JSON.stringify(data), {
parser: "json",
useTabs: true,
...config,
})
);
});
// this is autogenerated by running `npm run gen:i18:keys`
export type TranslationKeys =
| "common.loadMore"
| "common.loading"
| "common.noUsersFound"
| "common.ok"
| "common.yes"
| "common.no"
| "common.cancel"
| "common.save"
| "common.edit"
| "common.delete"
| "common.joinRoom"
| "common.copyLink"
| "common.copied"
| "common.formattedIntlDate"
| "common.formattedIntlTime"
| "header.title"
| "header.mutedTitle"
| "footer.link_1"
| "footer.link_2"
| "footer.link_3"
| "pages.banUser.ban"
| "pages.followingOnlineList.listHeader"
| "pages.followingOnlineList.currentRoom"
| "pages.followingOnlineList.startPrivateRoom"
| "pages.home.createRoom"
| "pages.inviteList.roomGone"
| "pages.inviteList.shareRoomLink"
| "pages.inviteList.inviteFollowers"
| "pages.inviteList.whenFollowersOnline"
| "pages.login.headerText"
| "pages.login.featureText_1"
| "pages.login.featureText_2"
| "pages.login.featureText_3"
| "pages.login.featureText_4"
| "pages.login.featureText_5"
| "pages.login.featureText_6"
| "pages.login.loginGithub"
| "pages.login.loginTwitter"
| "pages.login.createTestUser"
| "pages.myProfile.logout"
| "pages.myProfile.probablyLoading"
| "pages.myProfile.voiceSettings"
| "pages.myProfile.soundSettings"
| "pages.myProfile.deleteAccount"
| "pages.notFound.whoopsError"
| "pages.notFound.goHomeMessage"
| "pages.notFound.goHomeLinkText"
| "pages.room.speakers"
| "pages.room.requestingToSpeak"
| "pages.room.listeners"
| "pages.searchUser.search"
| "pages.soundEffectSettings.header"
| "pages.viewUser.editProfile"
| "pages.viewUser.followsYou"
| "pages.viewUser.followers"
| "pages.viewUser.following"
| "pages.voiceSettings.header"
| "pages.voiceSettings.mic"
| "pages.voiceSettings.permissionError"
| "pages.voiceSettings.refresh"
| "pages.voiceSettings.volume"
| "components.blockedFromRoomUsers.header"
| "components.blockedFromRoomUsers.unban"
| "components.blockedFromRoomUsers.noBans"
| "components.bottomVoiceControl.leaveCurrentRoomBtn"
| "components.bottomVoiceControl.confirmLeaveRoom"
| "components.bottomVoiceControl.leave"
| "components.bottomVoiceControl.inviteUsersToRoomBtn"
| "components.bottomVoiceControl.invite"
| "components.bottomVoiceControl.toggleMuteMicBtn"
| "components.bottomVoiceControl.mute"
| "components.bottomVoiceControl.unmute"
| "components.bottomVoiceControl.makeRoomPublicBtn"
| "components.bottomVoiceControl.settings"
| "components.bottomVoiceControl.speaker"
| "components.bottomVoiceControl.listener"
| "components.deviceNotSupported.notSupported"
| "components.deviceNotSupported.linkText"
| "components.deviceNotSupported.addSupport"
| "components.inviteButton.invited"
| "components.inviteButton.inviteToRoom"
| "components.micPermissionBanner.permissionDenied"
| "components.micPermissionBanner.dismiss"
| "components.micPermissionBanner.tryAgain"
| "components.keyboardShortcuts.setKeybind"
| "components.keyboardShortcuts.listening"
| "components.keyboardShortcuts.toggleMuteKeybind"
| "components.keyboardShortcuts.togglePushToTalkKeybind"
| "components.userVolumeSlider.noAudioMessage"
| "components.addToCalendar.add"
| "components.wsKilled.description"
| "components.wsKilled.reconnect"
| "components.modals.createRoomModal.public"
| "components.modals.createRoomModal.private"
| "components.modals.createRoomModal.roomName"
| "components.modals.createRoomModal.roomDescription"
| "components.modals.createRoomModal.descriptionError"
| "components.modals.createRoomModal.nameError"
| "components.modals.invitedToJoinRoomModal.newRoomCreated"
| "components.modals.invitedToJoinRoomModal.roomInviteFrom"
| "components.modals.invitedToJoinRoomModal.justStarted"
| "components.modals.invitedToJoinRoomModal.likeToJoin"
| "components.modals.invitedToJoinRoomModal.inviteReceived"
| "components.modals.editProfileModal.usernameTaken"
| "components.modals.editProfileModal.avatarUrlError"
| "components.modals.editProfileModal.avatarUrlLabel"
| "components.modals.editProfileModal.displayNameError"
| "components.modals.editProfileModal.displayNameLabel"
| "components.modals.editProfileModal.usernameError"
| "components.modals.editProfileModal.usernameLabel"
| "components.modals.editProfileModal.bioError"
| "components.modals.editProfileModal.bioLabel"
| "components.modals.profileModal.blockUserConfirm"
| "components.modals.profileModal.blockUser"
| "components.modals.profileModal.makeMod"
| "components.modals.profileModal.unmod"
| "components.modals.profileModal.addAsSpeaker"
| "components.modals.profileModal.moveToListener"
| "components.modals.profileModal.banFromChat"
| "components.modals.profileModal.banFromRoom"
| "components.modals.profileModal.goBackToListener"
| "components.modals.profileModal.deleteMessage"
| "components.modals.roomSettingsModal.requirePermission"
| "components.modals.roomSettingsModal.makePublic"
| "components.modals.roomSettingsModal.makePrivate"
| "modules.scheduledRooms.title"
| "modules.scheduledRooms.noneFound"
| "modules.scheduledRooms.allRooms"
| "modules.scheduledRooms.myRooms"
| "modules.scheduledRooms.scheduleRoomHeader"
| "modules.scheduledRooms.startRoom"
| "modules.scheduledRooms.modal.needsFuture"
| "modules.scheduledRooms.modal.roomName"
| "modules.scheduledRooms.modal.minLength"
| "modules.roomChat.title"
| "modules.roomChat.emotesSoon"
| "modules.roomChat.bannedAlert"
| "modules.roomChat.waitAlert"
| "modules.roomChat.search"
| "modules.roomChat.searchResults"
| "modules.roomChat.recent"
| "modules.roomChat.sendMessage"
| "modules.roomChat.whisper"
| "modules.roomChat.welcomeMessage"
| "modules.roomChat.roomDescription";
import { PureComponent } from "react";
declare module "react-datetime-picker";
// declare module "react-datetime-picker" {
// export interface DateTimePickerProps {
// amPmAriaLabel?: string;
// autoFocus?: boolean;
// calendarAriaLabel?: string;
// calendarClassName?: string | string[];
// calendarIcon?: React.ReactNode;
// className?: string | string[];
// clearAriaLabel?: string;
// clearIcon?: React.ReactNode;
// clockClassName?: string | string[];
// closeWidgets?: boolean;
// dayAriaLabel?: string;
// dayPlaceholder?: string;
// disableCalendar?: boolean;
// disableClock?: boolean;
// disabled?: boolean;
// format?: string;
// hourAriaLabel?: string;
// hourPlaceholder?: string;
// isCalendarOpen?: boolean;
// isClockOpen?: boolean;
// locale?: string;
// maxDate?: Date;
// maxDetail?: "hour" | "minute" | "second";
// minDate?: Date;
// minuteAriaLabel?: string;
// minutePlaceholder?: string;
// monthAriaLabel?: string;
// monthPlaceholder?: string;
// name?: string;
// nativeInputAriaLabel?: string;
// onCalendarClose?: () => void;
// onCalendarOpen?: () => void;
// onChange?: (value: Date | null) => void;
// onClockClose?: () => void;
// onClockOpen?: () => void;
// onFocus?: (event: React.FocusEvent<HTMLDivElement>) => void;
// required?: boolean;
// secondAriaLabel?: string;
// secondPlaceholder?: string;
// showLeadingZeros?: boolean;
// value?: string | Date | (string | Date)[];
// yearAriaLabel?: string;
// yearPlaceholder?: string;
// }
// declare class DateTimePicker extends PureComponent<DateTimePickerProps> {
// constructor(props: DateTimePickerProps);
// render(): React.ReactNode;
// }
// export default DateTimePicker;
// }
// export {};
import { atom, WritableAtom } from "jotai";
import { useCurrentRoomStore } from "../webrtc/stores/useCurrentRoomStore";
import { Room, BaseUser, UserWithFollowInfo } from "./types";
import { useMeQuery } from "./utils/useMeQuery";
const createSetter = <T>(a: WritableAtom<T, any>) =>
atom(null, (get, set, fn: (x: T) => T) => {
set(a, typeof fn === "function" ? fn(get(a)) : fn);
});
export const voiceBrowserStatusAtom = atom(-1);
export const setVoiceBrowserStatusAtom = createSetter(voiceBrowserStatusAtom);
export const inviteListAtom = atom<{
users: BaseUser[];
nextCursor: number | null;
}>({ users: [], nextCursor: null });
export const setInviteListAtom = createSetter(inviteListAtom);
export const followingOnlineAtom = atom<{
users: UserWithFollowInfo[];
nextCursor: number | null;
}>({ users: [], nextCursor: null });
export const userSearchAtom = atom<{
loading: boolean;
users: BaseUser[];
nextCursor: number | null;
}>({ users: [], loading: false, nextCursor: null });
export const setFollowingOnlineAtom = createSetter(followingOnlineAtom);
export const followerMapAtom = atom<
Record<
string,
{
users: UserWithFollowInfo[];
nextCursor: number | null;
}
>
>({});
export const followingMapAtom = atom<
Record<
string,
{
users: UserWithFollowInfo[];
nextCursor: number | null;
}
>
>({});
export const setFollowingMapAtom = createSetter(followingMapAtom);
export const setFollowerMapAtom = createSetter(followerMapAtom);
export const publicRoomsAtom = atom<{
publicRooms: Room[];
nextCursor: number | null;
}>({ publicRooms: [], nextCursor: null });
export const setPublicRoomsAtom = createSetter(publicRoomsAtom);
export const useCurrentRoomInfo = () => {
const { currentRoom: room } = useCurrentRoomStore();
const { me } = useMeQuery();
if (!room || !me) {
return {
isMod: false,
isCreator: false,
isSpeaker: false,
canSpeak: false,
};
}
let isMod = false;
let isSpeaker = false;
for (const u of room.users) {
if (u.id === me.id) {
if (u.roomPermissions?.isSpeaker) {
isSpeaker = true;
}
if (u.roomPermissions?.isMod) {
isMod = true;
}
break;
}
}
const isCreator = me.id === room.creatorId;
return {
isCreator,
isMod,
isSpeaker,
canSpeak: isCreator || isSpeaker,
};
};
import { toast } from "react-toastify";
export const showErrorToast = (m: string) => {
toast(m, {
type: "error",
});
};
import { useQuery } from "react-query";
import { auth_query } from "../../createWebsocket";
import { BaseUser } from "../types";
export const useMeQuery = () => {
const { data } = useQuery<{ user: BaseUser }>(auth_query, {
notifyOnChangeProps: ["data"],
enabled: false,
});
return { me: data?.user };
};
import create from "zustand";
import { combine } from "zustand/middleware";
import { __prod__ } from "../constants";
const accessTokenKey = "@toum/token" + (__prod__ ? "" : "dev");
const refreshTokenKey = "@toum/refresh-token" + (__prod__ ? "" : "dev");
const getDefaultValues = () => {
try {
return {
accessToken: localStorage.getItem(accessTokenKey) || "",
refreshToken: localStorage.getItem(refreshTokenKey) || "",
};
} catch {
return {
accessToken: "",
refreshToken: "",
};
}
};
export const useTokenStore = create(
combine(getDefaultValues(), (set) => ({
setTokens: (x: { accessToken: string; refreshToken: string }) => {
try {
localStorage.setItem(accessTokenKey, x.accessToken);
localStorage.setItem(refreshTokenKey, x.refreshToken);
} catch {}
set(x);
},
}))
);
import { format } from "date-fns";
// This auto converts UTC to local
export const dateFormat = (timestamp: string, formatString = "hh:mm aaa") => {
const date = new Date(timestamp);
return format(date, formatString);
};
import { useTranslation } from "react-i18next";
import { TranslationKeys } from "../../generated/translationKeys";
interface DateTranslationType {
time?: Date,
date?: Date
}
export const useTypeSafeTranslation = () => {
const { t } = useTranslation();
return {
t: (s: TranslationKeys, f?: DateTranslationType) => t(s, f)
};
};
import { useEffect } from "react";
import { useTokenStore } from "./useTokenStore";
import queryString from "query-string";
import { createWebSocket } from "../../createWebsocket";
export const useSaveTokensFromQueryParams = () => {
useEffect(() => {
const params = queryString.parse(window.location.search);
if (
typeof params.accessToken === "string" &&
typeof params.refreshToken === "string" &&
params.accessToken &&
params.refreshToken
) {
useTokenStore.getState().setTokens({
accessToken: params.accessToken,
refreshToken: params.refreshToken,
});
createWebSocket();
window.history.replaceState({}, document.title, "/");
}
}, []);
};
import { CurrentRoom, Room } from "../types";
export const roomToCurrentRoom = (r: Room): CurrentRoom =>
r
? {
...r,
muteMap: {},
users: [],
activeSpeakerMap: {},
autoSpeaker: false,
}
: r;
const regex = /^[0-9a-f]{8}-[0-9a-f]{4}-[4][0-9a-f]{3}-[89ab][0-9a-f]{3}-[0-9a-f]{12}$/i;
export const isUuid = (s: string) => regex.test(s);
import { Struct } from "superstruct";
export const validateStruct = <T>(struct: Struct<T>) => (values: T) => {
let errors: Record<string, string> = {};
const [result] = struct.validate(values);
for (const failure of result?.failures() || []) {
errors[failure.path[0]] = failure.message;
}
return errors;
};
// https://stackoverflow.com/questions/9038625/detect-if-device-is-ios
export function isIOS() {
return (
[
"iPad Simulator",
"iPhone Simulator",
"iPod Simulator",
"iPad",
"iPhone",
"iPod",
].includes(navigator.platform) ||
// iPad on iOS 13 detection
(navigator.userAgent.includes("Mac") && "ontouchend" in document)
);
}
import { linkRegex } from "./../constants";
import { BaseUser } from "../types";
// @ts-ignore
import normalizeUrl from "normalize-url";
export const createChatMessage = (
message: string,
mentions: BaseUser[],
roomUsers: BaseUser[] = []
) => {
const tokens = ([] as unknown) as [
{
t: string;
v: string;
}
];
const whisperedToUsernames: string[] = [];
message.split(" ").forEach((item) => {
const isLink = linkRegex.test(item);
const withoutAt = item.replace(/@|#/g, "");
const isMention = mentions.find((m) => withoutAt === m.username);
// whisperedTo users list
!isMention ||
item.indexOf("#@") !== 0 ||
whisperedToUsernames.push(withoutAt);
if (isLink || isMention) {
tokens.push({
t: isLink ? "link" : "mention",
v: isMention ? withoutAt : normalizeUrl(item),
});
} else {
const lastToken = tokens[tokens.length - 1];
if (lastToken && lastToken.t === "text") {
tokens[tokens.length - 1].v = lastToken.v + " " + item;
} else {
tokens.push({
t: "text",
v: item,
});
}
}
});
return {
tokens,
whisperedTo: roomUsers
.filter((u) =>
whisperedToUsernames
.map((u) => u?.toLowerCase())
.includes(u.username?.toLowerCase())
)
.map((u) => u.id),
};
};
import { showErrorToast } from "./showErrorToast";
// https://stackoverflow.com/questions/400212/how-do-i-copy-to-the-clipboard-in-javascript
export function copyTextToClipboard(text: string) {
var textArea = document.createElement("textarea");
textArea.value = text;
// Avoid scrolling to bottom
textArea.style.top = "0";
textArea.style.left = "0";
textArea.style.position = "fixed";
document.body.appendChild(textArea);
textArea.focus();
textArea.select();
let good = true;
try {
good = document.execCommand("copy");
} catch (err) {
console.error(err);
showErrorToast(err);
good = false;
}
document.body.removeChild(textArea);
return good;
}
import { auth_query } from "../../createWebsocket";
import { queryClient } from "../queryClient";
import { CurrentRoom, BaseUser, UserWithFollowInfo } from "../types";
export const onFollowUpdater = (
setRoom: (update: (x: CurrentRoom | null) => CurrentRoom | null) => void,
me: BaseUser | null | undefined,
profile: UserWithFollowInfo
) => {
setRoom((r) =>
!r
? r
: {
...r,
users: r.users.map((x) => {
if (x.id === profile.id) {
return {
...x,
numFollowers:
x.numFollowers + (profile.youAreFollowing ? -1 : 1),
youAreFollowing: !profile.youAreFollowing,
};
} else if (x.id === me?.id) {
return {
...x,
numFollowing:
x.numFollowing + (profile.youAreFollowing ? -1 : 1),
};
}
return x;
}),
}
);
queryClient.setQueryData<{ user: BaseUser } | null | undefined>(
auth_query,
(x) =>
!x
? x
: {
...x,
user: {
...x.user,
numFollowing:
x.user.numFollowing + (profile.youAreFollowing ? -1 : 1),
},
}
);
};
export const truncate = (string: string, max: number = 100) =>
string.length > max ? string.substring(0, max) + "..." : string;
export type Room = {
id: string;
name: string;
description?: string;
isPrivate: boolean;
numPeopleInside: number;
creatorId: string;
peoplePreviewList: Array<{
id: string;
displayName: string;
numFollowers: number;
}>;
};
export type BaseUser = {
username: string;
online: boolean;
lastOnline: Date;
id: string;
bio: string;
displayName: string;
avatarUrl: string;
numFollowing: number;
numFollowers: number;
currentRoom?: Room;
};
export type PaginatedBaseUsers = {
users: BaseUser[];
nextCursor: number | null;
};
export type RoomPermissions = {
askedToSpeak: boolean;
isSpeaker: boolean;
isMod: boolean;
};
export type UserWithFollowInfo = BaseUser & {
followsYou?: boolean;
youAreFollowing?: boolean;
};
export type RoomUser = {
roomPermissions?: RoomPermissions | null;
} & UserWithFollowInfo;
export type CurrentRoom = Room & {
users: RoomUser[];
muteMap: Record<string, boolean>;
activeSpeakerMap: Record<string, boolean>;
autoSpeaker: boolean;
};
export type WsParam = {
op: string;
d: any;
};
export interface ScheduledRoom {
roomId: string | null;
description: string;
scheduledFor: string;
numAttending: number;
name: string;
id: string;
creatorId: string;
creator: BaseUser;
}
export interface ScheduledRoomsInfo {
scheduledRooms: ScheduledRoom[];
nextCursor?: string | null;
}
export interface PublicRoomsQuery {
rooms: Room[];
nextCursor: number | null;
}
export interface CalendarEvent {
name: string;
details: string | null;
location: string | null;
startsAt: string;
endsAt: string;
}
const makeDuration = (event: CalendarEvent) => {
const minutes = Math.floor((+new Date(event.endsAt) - +new Date(event.startsAt)) / 60 / 1000);
return `${`0${Math.floor(minutes / 60)}`.slice(-2)}${`0${minutes % 60}`.slice(-2)}`;
};
const makeTime = (time: string) => new Date(time).toISOString().replace(/[-:]|\.\d{3}/g, "");
type Query = { [key: string]: null | boolean | number | string };
const makeUrl = (base: string, query: Query) => Object.keys(query).reduce(
(accum, key, index) => {
const value = query[key];
if (value !== null) {
return `${accum}${index === 0 ? "?" : "&"}${key}=${encodeURIComponent(value)}`;
}
return accum;
},
base
);
const makeGoogleCalendarUrl = (event: CalendarEvent) => makeUrl("https://calendar.google.com/calendar/render", {
action: "TEMPLATE",
dates: `${makeTime(event.startsAt)}/${makeTime(event.endsAt)}`,
location: event.location,
text: event.name,
details: event.details
});
const makeOutlookCalendarUrl = (event: CalendarEvent) => makeUrl("https://outlook.live.com/owa", {
rru: "addevent",
startdt: event.startsAt,
enddt: event.endsAt,
subject: event.name,
location: event.location,
body: event.details,
allday: false,
uid: new Date().getTime().toString(),
path: "/calendar/view/Month"
});
const makeYahooCalendarUrl = (event: CalendarEvent) => makeUrl("https://calendar.yahoo.com", {
v: 60,
view: "d",
type: 20,
title: event.name,
st: makeTime(event.startsAt),
dur: makeDuration(event),
desc: event.details,
in_loc: event.location
});
const makeICSCalendarUrl = (event: CalendarEvent) => {
const components = [
"BEGIN:VCALENDAR",
"VERSION:2.0",
"BEGIN:VEVENT"
];
// In case of SSR, document won't be defined
if (typeof document !== "undefined") {
components.push(`URL:${document.URL}`);
}
components.push(
`DTSTART:${makeTime(event.startsAt)}`,
`DTEND:${makeTime(event.endsAt)}`,
`SUMMARY:${event.name}`,
`DESCRIPTION:${event.details}`,
`LOCATION:${event.location}`,
"END:VEVENT",
"END:VCALENDAR"
);
return encodeURI(`data:text/calendar;charset=utf8,${components.join("\n")}`);
};
type URLSet = { [key: string]: string };
const makeUrls = (event: CalendarEvent): URLSet => ({
google: makeGoogleCalendarUrl(event),
outlook: makeOutlookCalendarUrl(event),
yahoo: makeYahooCalendarUrl(event),
ics: makeICSCalendarUrl(event)
});
export default makeUrls;
export { MuteKeybind } from './MuteKeybind'
export { ChatKeybind } from './ChatKeybind'
export { PTTKeybind } from './PTTKeybind'
export const __prod__ = process.env.NODE_ENV === "production";
export const __staging__ = process.env.REACT_APP_IS_STAGING === "true";
export const apiBaseUrl =
process.env.REACT_APP_API_BASE_URL ||
(__prod__ ? "https://api.dogehouse.tv" : "http://192.168.1.165:4001");
export const linkRegex = /(https?:\/\/)(www\.)?([-a-z0-9]{1,63}\.)*?[a-z0-9][-a-z0-9]{0,61}[a-z0-9]\.[a-z]{1,6}(\/[-\\w@\\+\\.~#\\?&/=%]*)?[^\s()]+/;
import create from "zustand";
import { combine } from "zustand/middleware";
export const MIC_KEY = "micId";
export const useMicIdStore = create(
combine(
{
micId: localStorage.getItem(MIC_KEY) || "",
},
(set) => ({
setMicId: (id: string) => {
try {
localStorage.setItem(MIC_KEY, id);
} catch {}
set({ micId: id });
},
})
)
);
import { QueryClient } from "react-query";
import { apiBaseUrl } from "./constants";
import { showErrorToast } from "./utils/showErrorToast";
import { useTokenStore } from "./utils/useTokenStore";
export const defaultQueryFn = async ({ queryKey }: { queryKey: string }) => {
const { accessToken, refreshToken } = useTokenStore.getState();
const r = await fetch(`${apiBaseUrl}${queryKey[0]}`, {
headers: {
"X-Access-Token": accessToken,
"X-Refresh-Token": refreshToken,
},
});
if (r.status !== 200) {
throw new Error(await r.text());
}
const _accessToken = r.headers.get("access-token");
const _refreshToken = r.headers.get("refresh-token");
if (_accessToken && _refreshToken) {
useTokenStore.getState().setTokens({
accessToken: _accessToken,
refreshToken: _refreshToken,
});
}
return await r.json();
};
export const queryClient = new QueryClient({
defaultOptions: {
mutations: {
onError: (e) => {
if ("message" in (e as Error)) {
showErrorToast((e as Error).message);
}
},
},
queries: {
retry: false,
staleTime: 60 * 1000 * 5,
onError: (e) => {
if ("message" in (e as Error)) {
showErrorToast((e as Error).message);
}
},
queryFn: defaultQueryFn,
},
},
});
import { useAtom } from "jotai";
import { useRef, useEffect } from "react";
import { useLocation, useHistory } from "react-router-dom";
import { toast } from "react-toastify";
import { closeWebSocket, wsend } from "../createWebsocket";
import { useCurrentRoomStore } from "../webrtc/stores/useCurrentRoomStore";
import { useMuteStore } from "../webrtc/stores/useMuteStore";
import { useWsHandlerStore } from "../webrtc/stores/useWsHandlerStore";
import { mergeRoomPermission } from "../webrtc/utils/mergeRoomPermission";
import {
setPublicRoomsAtom,
setFollowerMapAtom,
setFollowingMapAtom,
setFollowingOnlineAtom,
setInviteListAtom,
} from "./atoms";
import { invitedToRoomConfirm } from "./components/InvitedToJoinRoomModal";
import { useRoomChatMentionStore } from "./modules/room-chat/useRoomChatMentionStore";
import {
useRoomChatStore,
RoomChatMessageToken,
} from "./modules/room-chat/useRoomChatStore";
import { useShouldBeSidebar } from "./modules/room-chat/useShouldFullscreenChat";
import { RoomUser } from "./types";
import { isUuid } from "./utils/isUuid";
import { roomToCurrentRoom } from "./utils/roomToCurrentRoom";
import { showErrorToast } from "./utils/showErrorToast";
import { useMeQuery } from "./utils/useMeQuery";
import { useTokenStore } from "./utils/useTokenStore";
export const useMainWsHandler = () => {
const location = useLocation();
const history = useHistory();
const addMultipleWsListener = useWsHandlerStore(
(s) => s.addMultipleWsListener
);
const [, setPublicRooms] = useAtom(setPublicRoomsAtom);
const [, setFollowerMap] = useAtom(setFollowerMapAtom);
const [, setFollowingMap] = useAtom(setFollowingMapAtom);
const [, setFollowingOnline] = useAtom(setFollowingOnlineAtom);
const [, setInviteList] = useAtom(setInviteListAtom);
const setCurrentRoom = useCurrentRoomStore((x) => x.setCurrentRoom);
const { me } = useMeQuery();
const meRef = useRef(me);
meRef.current = me;
const shouldBeSidebar = useShouldBeSidebar();
const shouldBeSidebarRef = useRef(shouldBeSidebar);
shouldBeSidebarRef.current = shouldBeSidebar;
useEffect(() => {
addMultipleWsListener({
new_room_details: ({ name, description, isPrivate, roomId }) => {
setCurrentRoom((cr) =>
!cr || cr.id !== roomId
? cr
: {
...cr,
name,
description,
isPrivate,
}
);
},
chat_user_banned: ({ userId }) => {
useRoomChatStore.getState().addBannedUser(userId);
},
new_chat_msg: ({ msg }) => {
const { open } = useRoomChatStore.getState();
useRoomChatStore.getState().addMessage(msg);
const { isRoomChatScrolledToTop } = useRoomChatStore.getState();
if (
(!open || !document.hasFocus() || isRoomChatScrolledToTop) &&
!!msg.tokens.filter(
(t: RoomChatMessageToken) =>
t.t === "mention" &&
t.v?.toLowerCase() === meRef?.current?.username?.toLowerCase()
).length
) {
useRoomChatMentionStore.getState().incrementIAmMentioned();
}
},
message_deleted({ messageId, deleterId }) {
const { messages, setMessages } = useRoomChatStore.getState();
setMessages(
messages.map((m) => ({
...m,
deleted: m.id === messageId || !!m.deleted,
deleterId: m.id === messageId ? deleterId : m.deleterId,
}))
);
},
room_privacy_change: ({ roomId, isPrivate, name }) => {
setCurrentRoom((cr) =>
!cr || cr.id !== roomId ? cr : { ...cr, name, isPrivate }
);
toast(`Room is now ${isPrivate ? "private" : "public"}`, {
type: "info",
});
},
banned: () => {
toast("you got banned", { type: "error" });
closeWebSocket();
useTokenStore
.getState()
.setTokens({ accessToken: "", refreshToken: "" });
},
ban_done: ({ worked }) => {
if (worked) {
toast("ban worked", { type: "success" });
} else {
toast("ban failed", { type: "error" });
}
},
someone_you_follow_created_a_room: (value) => {
invitedToRoomConfirm(value, history);
},
invitation_to_room: (value) => {
invitedToRoomConfirm(value, history);
},
fetch_invite_list_done: ({ users, nextCursor, initial }) => {
setInviteList((x) => ({
users: initial ? users : [...x.users, ...users],
nextCursor,
}));
},
fetch_following_online_done: ({ users, nextCursor, initial }) => {
setFollowingOnline((x) => ({
users: initial ? users : [...x.users, ...users],
nextCursor,
}));
},
get_top_public_rooms_done: ({ rooms, nextCursor, initial }) => {
setPublicRooms((r) => ({
publicRooms: initial ? rooms : [...r.publicRooms, ...rooms],
nextCursor,
}));
},
fetch_follow_list_done: ({
userId,
users,
isFollowing,
nextCursor,
initial,
}) => {
const fn = isFollowing ? setFollowingMap : setFollowerMap;
fn((m) => ({
...m,
[userId]: {
users: initial ? users : [...m[userId].users, ...users],
nextCursor,
},
}));
},
follow_info_done: ({ userId, followsYou, youAreFollowing }) => {
setCurrentRoom((c) =>
!c
? c
: {
...c,
users: c.users.map((x) =>
x.id === userId ? { ...x, followsYou, youAreFollowing } : x
),
}
);
},
active_speaker_change: ({ roomId, activeSpeakerMap }) => {
setCurrentRoom((c) =>
!c || c.id !== roomId ? c : { ...c, activeSpeakerMap }
);
},
room_destroyed: ({ roomId }) => {
setCurrentRoom((c) => {
if (c && c.id === roomId) {
history.replace("/");
return null;
}
return c;
});
},
new_room_creator: ({ userId, roomId }) => {
setCurrentRoom((cr) =>
cr && cr.id === roomId ? { ...cr, creatorId: userId } : cr
);
},
speaker_removed: ({ userId, roomId, muteMap }) => {
setCurrentRoom((c) =>
!c || c.id !== roomId
? c
: {
...c,
muteMap,
users: c.users.map((x) =>
userId === x.id
? {
...x,
roomPermissions: mergeRoomPermission(
x.roomPermissions,
{ isSpeaker: false, askedToSpeak: false }
),
}
: x
),
}
);
},
speaker_added: ({ userId, roomId, muteMap }) => {
// Mute user upon added as speaker
if (meRef.current?.id === userId) {
const { setMute } = useMuteStore.getState();
wsend({
op: "mute",
d: { value: true },
});
setMute(true);
}
setCurrentRoom((c) =>
!c || c.id !== roomId
? c
: {
...c,
muteMap,
users: c.users.map((x) =>
userId === x.id
? {
...x,
roomPermissions: mergeRoomPermission(
x.roomPermissions,
{
isSpeaker: true,
}
),
}
: x
),
}
);
},
mod_changed: ({ userId, roomId }) => {
setCurrentRoom((c) =>
!c || c.id !== roomId
? c
: {
...c,
users: c.users.map((x) =>
userId === x.id
? {
...x,
roomPermissions: mergeRoomPermission(
x.roomPermissions,
{ isMod: true }
),
}
: x
),
}
);
},
user_left_room: ({ userId }) => {
setCurrentRoom((cr) => {
if (!cr) {
return cr;
}
const { [userId]: _, ...asm } = cr.activeSpeakerMap;
return {
...cr,
activeSpeakerMap: asm,
peoplePreviewList: cr.peoplePreviewList.filter(
(x) => x.id !== userId
),
numPeopleInside: cr.numPeopleInside - 1,
users: cr.users.filter((x) => x.id !== userId),
};
});
},
new_user_join_room: ({ user, muteMap }) => {
setCurrentRoom((cr) =>
!cr || cr.users.some((u) => u.id === user.id)
? cr
: {
...cr,
muteMap,
peoplePreviewList:
cr.peoplePreviewList.length < 10
? [
...cr.peoplePreviewList,
{
id: user.id,
displayName: user.displayName,
numFollowers: user.numFollowers,
},
]
: cr.peoplePreviewList,
numPeopleInside: cr.numPeopleInside + 1,
users: [...cr.users.filter((x) => x.id !== user.id), user],
}
);
},
hand_raised: ({ roomId, userId }) => {
setCurrentRoom((c) => {
if (!c || c.id !== roomId) {
return c;
}
return {
...c,
users: c.users.map((u) =>
u.id === userId
? {
...u,
roomPermissions: mergeRoomPermission(u.roomPermissions, {
askedToSpeak: true,
}),
}
: u
),
};
});
},
mute_changed: ({ userId, value, roomId }) => {
setCurrentRoom((c) => {
if (!c || c.id !== roomId) {
return c;
}
if (value) {
return {
...c,
muteMap: { ...c.muteMap, [userId]: true },
};
} else {
const { [userId]: _, ...newMm } = c.muteMap;
return {
...c,
muteMap: newMm,
};
}
});
},
get_current_room_users_done: ({
users,
muteMap,
roomId,
activeSpeakerMap,
autoSpeaker,
}) => {
// Mute when rejoin and if speaker
if (
!!users.find(
(u: RoomUser) =>
u.id === meRef.current?.id && u.roomPermissions?.isSpeaker
)
) {
const { setMute } = useMuteStore.getState();
wsend({
op: "mute",
d: { value: true },
});
setMute(true);
}
setCurrentRoom((c) => {
if (!c || c.id !== roomId) {
return c;
}
return {
...c,
activeSpeakerMap,
users,
muteMap,
autoSpeaker,
};
});
},
new_current_room: ({ room }) => {
if (room) {
console.log("new room voice server id: " + room.voiceServerId);
useRoomChatStore.getState().clearChat();
wsend({ op: "get_current_room_users", d: {} });
history.push("/room/" + room.id);
}
setCurrentRoom(() => roomToCurrentRoom(room));
},
join_room_done: (d) => {
// Auto open chat to show description and if mobile
if (shouldBeSidebarRef.current) {
useRoomChatStore.getState().setOpen(true);
}
if (d.error) {
if (window.location.pathname.startsWith("/room")) {
history.push("/");
}
showErrorToast(d.error);
} else if (d.room) {
console.log("join with voice server id: " + d.room.voiceServerId);
useRoomChatStore.getState().clearChat();
setCurrentRoom(() => roomToCurrentRoom(d.room));
wsend({ op: "get_current_room_users", d: {} });
}
},
});
// eslint-disable-next-line react-hooks/exhaustive-deps
}, []);
useEffect(() => {
if (location.pathname.startsWith("/room/")) {
let found = false;
const parts = location.pathname.split("/");
const id = parts.find((x) => {
if (found) {
return true;
}
if (x === "room") {
found = true;
}
return false;
});
if (id && isUuid(id)) {
wsend({ op: "join_room", d: { roomId: id } });
}
}
// eslint-disable-next-line react-hooks/exhaustive-deps
}, []);
};
import create from "zustand";
import { combine } from "zustand/middleware";
import { BaseUser } from "../../types";
import { useSoundEffectStore } from "../sound-effects/useSoundEffectStore";
export const useRoomChatMentionStore = create(
combine(
{
mentions: [] as BaseUser[],
queriedUsernames: [] as BaseUser[],
activeUsername: "",
iAmMentioned: 0,
},
(set) => ({
setMentions: (mentions: BaseUser[]) =>
set({
mentions,
}),
setQueriedUsernames: (queriedUsernames: BaseUser[]) =>
set({
queriedUsernames,
}),
setActiveUsername: (activeUsername: string) => {
return set({
activeUsername,
});
},
resetIAmMentioned: () =>
set({
iAmMentioned: 0,
}),
incrementIAmMentioned: () => {
useSoundEffectStore.getState().playSoundEffect("roomChatMention");
set((x) => ({ iAmMentioned: x.iAmMentioned + 1 }));
},
})
)
);
import create from "zustand";
import { combine } from "zustand/middleware";
import { useRoomChatMentionStore } from "./useRoomChatMentionStore";
interface TextToken {
t: "text";
v: string;
}
interface MentionToken {
t: "mention";
v: string;
}
interface LinkToken {
t: "link";
v: string;
}
export type RoomChatMessageToken = TextToken | MentionToken | LinkToken;
const colors = [
"#ff2366",
"#fd51d9",
"#face15",
"#8d4de8",
"#6859ea",
"#7ed321",
"#56b2ba",
"#00CCFF",
"#FF9900",
"#FFFF66",
];
function generateColorFromString(str: string) {
let sum = 0;
for (let x = 0; x < str.length; x++) sum += x * str.charCodeAt(x);
return colors[sum % colors.length];
}
export interface RoomChatMessage {
id: string;
userId: string;
avatarUrl: string;
color: string;
displayName: string;
tokens: RoomChatMessageToken[];
deleted?: boolean;
deleterId?: string;
sentAt: string;
isWhisper?: boolean;
}
export const useRoomChatStore = create(
combine(
{
open: false,
bannedUserIdMap: {} as Record<string, boolean>,
messages: [] as RoomChatMessage[],
newUnreadMessages: false,
message: "" as string,
isRoomChatScrolledToTop: false,
},
(set) => ({
addBannedUser: (userId: string) =>
set((s) => ({
messages: s.messages.filter((m) => m.userId !== userId),
bannedUserIdMap: { ...s.bannedUserIdMap, [userId]: true },
})),
addMessage: (m: RoomChatMessage) =>
set((s) => ({
newUnreadMessages: !s.open,
messages: [
{ ...m, color: generateColorFromString(m.userId) },
...(s.messages.length > 100
? s.messages.slice(0, 100)
: s.messages),
],
})),
setMessages: (messages: RoomChatMessage[]) =>
set((s) => ({
messages,
})),
clearChat: () =>
set({
messages: [],
newUnreadMessages: false,
bannedUserIdMap: {},
}),
reset: () =>
set({
messages: [],
newUnreadMessages: false,
open: false,
bannedUserIdMap: {},
}),
toggleOpen: () =>
set((s) => {
// Reset mention state
useRoomChatMentionStore.getState().resetIAmMentioned();
if (s.open) {
return {
open: false,
newUnreadMessages: false,
};
} else {
return {
open: true,
newUnreadMessages: false,
};
}
}),
setMessage: (message: string) =>
set({
message,
}),
setOpen: (open: boolean) => set((s) => ({ ...s, open })),
setIsRoomChatScrolledToTop: (isRoomChatScrolledToTop: boolean) =>
set({
isRoomChatScrolledToTop,
}),
})
)
);
import { useMediaQuery } from "react-responsive";
import { roomChatMediaQuery } from "./RoomChat";
import { useRoomChatStore } from "./useRoomChatStore";
export const useShouldFullscreenChat = () => {
const chatShouldBeSidebar = useMediaQuery({ query: roomChatMediaQuery });
const open = useRoomChatStore((s) => s.open);
return !chatShouldBeSidebar && open;
};
export const useShouldBeSidebar = () => {
return useMediaQuery({ query: roomChatMediaQuery });
};
import create from "zustand";
import { combine } from "zustand/middleware";
export const soundEffects = {
roomChatMention: "roomChatMention.ogg",
unmute: "unmute.wav",
mute: "mute.wav",
roomInvite: "roomInvite.wav",
};
export type PossibleSoundEffect = keyof typeof soundEffects;
const keyToLocalStorageKey = (s: string) => `@sound-effect/${s}`;
function getInitialSettings() {
const soundEffectSettings: Record<PossibleSoundEffect, boolean> = {
roomChatMention: true,
unmute: true,
mute: true,
roomInvite: true,
};
try {
Object.keys(soundEffects).forEach((key) => {
const v = localStorage.getItem(keyToLocalStorageKey(key)) || "";
soundEffectSettings[key as PossibleSoundEffect] = !v || v === "true";
});
} catch {}
return soundEffectSettings;
}
export const useSoundEffectStore = create(
combine(
{
audioRefMap: {} as Record<string, HTMLAudioElement>,
settings: getInitialSettings(),
},
(set, get) => ({
setSetting: (key: PossibleSoundEffect, value: boolean) => {
try {
localStorage.setItem(keyToLocalStorageKey(key), value.toString());
} catch {}
set((x) => ({
settings: { ...x.settings, [key]: value },
}));
},
playSoundEffect: (se: keyof typeof soundEffects, force = false) => {
const { audioRefMap, settings } = get();
if (force || settings[se]) {
audioRefMap[se]?.play();
}
},
add: (key: string, audio: HTMLAudioElement) =>
set((s) => ({ audioRefMap: { ...s.audioRefMap, [key]: audio } })),
})
)
);
import { atom } from "jotai";
export const volumeAtom = atom(100);
import i18n from "i18next";
import Backend from "i18next-http-backend";
import LanguageDetector from "i18next-browser-languagedetector";
import { initReactI18next } from "react-i18next";
import { __prod__ } from "./app/constants";
import { isDate } from "lodash";
export const init_i18n = () => {
i18n
// import & load translations from -> /public/locales
.use(Backend)
// https://github.com/i18next/i18next-browser-languageDetector
.use(LanguageDetector)
// pass the i18n instance to react-i18next.
.use(initReactI18next)
// init i18next
// see opts @ https://www.i18next.com/overview/configuration-options
.init({
fallbackLng: "en",
debug: __prod__ ? false : true,
interpolation: {
escapeValue: false,
format: (value, format, lng) => {
return isDate(value) && format ? new Intl.DateTimeFormat(lng,
createDateFormatOptions(format)).format(value).toString()
: value;
}
},
react: {
useSuspense: false, // fixes 'no fallback UI was specified' in react i18next when using hooks
},
});
};
function createDateFormatOptions(format: string): Intl.DateTimeFormatOptions {
switch (format) {
case 'intlDate': {
// EN returns 3/16/2021, 5:45 PM
return {
year: 'numeric', month: 'numeric', day: 'numeric',
hour: 'numeric', minute: 'numeric'
}
}
case 'intlTime': {
// EN returns 05:45 PM
return {
hour: 'numeric', minute: 'numeric'
}
}
default: {
// EN returns Tuesday, March 16, 2021, 5:45 PM
return {
weekday: 'long', year: 'numeric', month: 'long',
day: 'numeric', hour: 'numeric', minute: 'numeric'
}
}
}
}
import ReconnectingWebSocket from "reconnecting-websocket";
import { useTokenStore } from "./app/utils/useTokenStore";
import { showErrorToast } from "./app/utils/showErrorToast";
import { apiBaseUrl } from "./app/constants";
import { useSocketStatus } from "./webrtc/stores/useSocketStatus";
import { useWsHandlerStore } from "./webrtc/stores/useWsHandlerStore";
import { useVoiceStore } from "./webrtc/stores/useVoiceStore";
import { useMuteStore } from "./webrtc/stores/useMuteStore";
import { uuidv4 } from "./webrtc/utils/uuidv4";
import { WsParam } from "./app/types";
import { useCurrentRoomStore } from "./webrtc/stores/useCurrentRoomStore";
import { toast } from "react-toastify";
import { queryClient } from "./app/queryClient";
let ws: ReconnectingWebSocket | null;
let authGood = false;
let lastMsg = "";
export const auth_query = "auth";
window.addEventListener("online", () => {
if (ws && ws.readyState === ws.CLOSED) {
toast("reconnecting...", { type: "info" });
console.log("online triggered, calling ws.reconnect()");
ws.reconnect();
}
});
export const closeWebSocket = () => {
ws?.close();
};
export const createWebSocket = (force?: boolean) => {
console.log("createWebSocket ");
if (!force && ws) {
console.log("ws already connected");
return;
} else {
console.log("new ws instance incoming");
}
const { accessToken, refreshToken } = useTokenStore.getState();
if (!accessToken || !refreshToken) {
return;
}
useSocketStatus.getState().setStatus("connecting");
ws = new ReconnectingWebSocket(
apiBaseUrl.replace("http", "ws") + "/socket",
undefined,
{ connectionTimeout: 15000 }
);
ws.addEventListener("close", ({ code, reason }) => {
const { setStatus } = useSocketStatus.getState();
authGood = false;
if (code === 4001) {
console.log("clearing tokens");
useWsHandlerStore.getState().authHandler?.(null);
useTokenStore.getState().setTokens({ accessToken: "", refreshToken: "" });
ws?.close();
ws = null;
setStatus("closed");
} else if (code === 4003) {
ws?.close();
ws = null;
setStatus("closed-by-server");
} else if (code === 4004) {
ws?.close();
ws = null;
} else {
// @todo do more of a status bar thing
setStatus("closed");
}
console.log("ws closed", code, reason);
});
ws.addEventListener("open", () => {
useSocketStatus.getState().setStatus("open");
const { recvTransport, sendTransport } = useVoiceStore.getState();
const reconnectToVoice = !recvTransport
? true
: recvTransport.connectionState !== "connected" &&
sendTransport?.connectionState !== "connected";
console.log({
reconnectToVoice,
recvState: recvTransport?.connectionState,
sendState: sendTransport?.connectionState,
});
queryClient.prefetchQuery(
auth_query,
() =>
wsAuthFetch({
op: auth_query,
d: {
accessToken,
refreshToken,
reconnectToVoice,
currentRoomId: useCurrentRoomStore.getState().currentRoom?.id,
muted: useMuteStore.getState().muted,
platform: "web",
},
}),
{ staleTime: 0 }
);
// @todo do more of a status bar thing
// toast("connected", { type: "success" });
console.log("ws opened");
const id = setInterval(() => {
if (ws && ws.readyState !== ws.CLOSED) {
ws.send("ping");
} else {
clearInterval(id);
}
}, 8000);
});
ws.addEventListener("message", (e) => {
// console.log(e.data);
const json = JSON.parse(e.data as string);
if (e.data === '"pong"') {
return;
}
switch (json.op) {
case "new-tokens": {
useTokenStore.getState().setTokens({
accessToken: json.d.accessToken,
refreshToken: json.d.refreshToken,
});
break;
}
case "error": {
showErrorToast(json.d);
break;
}
default: {
const {
handlerMap,
fetchResolveMap,
authHandler,
} = useWsHandlerStore.getState();
if (json.op === "auth-good") {
if (lastMsg) {
ws?.send(lastMsg);
lastMsg = "";
}
authGood = true;
useSocketStatus.getState().setStatus("auth-good");
if (authHandler) {
authHandler(json.d);
} else {
console.error("something went wrong, authHandler is null");
}
}
// console.log("ws: ", json.op);
if (json.op in handlerMap) {
handlerMap[json.op](json.d);
} else if (
json.op === "fetch_done" &&
json.fetchId &&
json.fetchId in fetchResolveMap
) {
fetchResolveMap[json.fetchId](json.d);
}
break;
}
}
});
};
export const wsend = (d: { op: string; d: any }) => {
if (!authGood || !ws || ws.readyState !== ws.OPEN) {
console.log("ws not ready");
lastMsg = JSON.stringify(d);
} else {
ws?.send(JSON.stringify(d));
}
};
export const wsAuthFetch = <T>(d: WsParam) => {
return new Promise<T>((res, rej) => {
if (!ws || ws.readyState !== ws.OPEN) {
rej(new Error("can't connect to server"));
} else {
setTimeout(() => {
rej(new Error("request timed out"));
}, 10000); // 10 secs
useWsHandlerStore.getState().addAuthHandler((d) => {
if (d) {
res(d);
}
});
ws?.send(JSON.stringify(d));
}
});
};
export const wsFetch = <T>(d: WsParam) => {
return new Promise<T>((res, rej) => {
if (!authGood || !ws || ws.readyState !== ws.OPEN) {
rej(new Error("can't connect to server"));
} else {
const fetchId = uuidv4();
setTimeout(() => {
useWsHandlerStore.getState().clearFetchListener(fetchId);
rej(new Error("request timed out"));
}, 10000); // 10 secs
useWsHandlerStore.getState().addFetchListener(fetchId, (d) => {
res(d);
});
ws?.send(JSON.stringify({ ...d, fetchId }));
}
});
};
export const wsMutation = (d: WsParam) => wsFetch(d);
export const wsMutationThrowError = (d: WsParam) =>
wsFetch(d).then((x: any) => {
if (x.error) {
throw new Error(x.error);
}
return x;
});
/// <reference types="react-scripts" />
import create from "zustand";
import { combine } from "zustand/middleware";
export const useAudioTracks = create(
combine(
{
tracks: [] as MediaStreamTrack[],
},
(set) => ({
add: (track: MediaStreamTrack) =>
set((s) => ({ tracks: [...s.tracks, track] })),
})
)
);
import { Consumer } from "mediasoup-client/lib/types";
import create from "zustand";
import { combine } from "zustand/middleware";
export const useConsumerStore = create(
combine(
{
consumerMap: {} as Record<string, { consumer: Consumer; volume: number }>,
},
(set) => ({
setVolume: (userId: string, volume: number) => {
set((s) =>
userId in s.consumerMap
? {
consumerMap: {
...s.consumerMap,
[userId]: {
...s.consumerMap[userId],
volume,
},
},
}
: s
);
},
add: (c: Consumer, userId: string) =>
set((s) => {
let volume = 100;
if (userId in s.consumerMap) {
const x = s.consumerMap[userId];
volume = x.volume;
x.consumer.close();
}
return {
consumerMap: {
...s.consumerMap,
[userId]: { consumer: c, volume },
},
};
}),
closeAll: () =>
set((s) => {
Object.values(s.consumerMap).forEach(
({ consumer: c }) => !c.closed && c.close()
);
return {
consumerMap: {},
};
}),
})
)
);
import { Producer } from "mediasoup-client/lib/types";
import create from "zustand";
import { combine } from "zustand/middleware";
export const useProducerStore = create(
combine(
{
producer: null as Producer | null,
},
(set) => ({
add: (p: Producer) =>
set((s) => {
if (s.producer && !s.producer.closed) {
s.producer.close();
}
return { producer: p };
}),
close: () =>
set((s) => {
if (s.producer && !s.producer.closed) {
s.producer.close();
}
return {
producer: null,
};
}),
})
)
);
import create from "zustand";
import { combine } from "zustand/middleware";
import { CurrentRoom } from "../../app/types";
export const useCurrentRoomStore = create(
combine(
{
currentRoom: null as CurrentRoom | null,
},
(set) => ({
set,
setCurrentRoom: (fn: (cr: CurrentRoom | null) => CurrentRoom | null) =>
set((s) => ({ currentRoom: fn(s.currentRoom) })),
})
)
);
import create from "zustand";
import { combine } from "zustand/middleware";
type Handler = (d: any) => void;
export const useWsHandlerStore = create(
combine(
{
handlerMap: {} as Record<string, Handler>,
fetchResolveMap: {} as Record<string, Handler>,
authHandler: null as null | Handler,
},
(set) => ({
addAuthHandler: (authHandler: Handler | null) => set({ authHandler }),
addMultipleWsListener: (x: Record<string, Handler>) => {
set((s) => ({
handlerMap: {
...s.handlerMap,
...x,
},
}));
return () =>
set((s) => {
const newMap = { ...s.handlerMap };
Object.keys(x).forEach((k) => {
delete newMap[k];
});
return {
handlerMap: newMap,
};
});
},
addWsListener: (op: string, fn: (d: any) => void) => {
return set((s) => ({
handlerMap: {
...s.handlerMap,
[op]: fn,
},
}));
},
addWsListenerOnce: (op: string, fn: (d: any) => void) => {
return set((s) => ({
handlerMap: {
...s.handlerMap,
[op]: (dx: any) => {
fn(dx);
set(({ handlerMap: { [op]: _, ...handlerMap } }) => ({
handlerMap,
}));
},
},
}));
},
clearFetchListener: (id: string) => {
return set(({ fetchResolveMap: { [id]: _, ...fetchResolveMap } }) => ({
fetchResolveMap,
}));
},
addFetchListener: (id: string, fn: (d: any) => void) => {
return set((s) => ({
fetchResolveMap: {
...s.fetchResolveMap,
[id]: (dx: any) => {
fn(dx);
set(({ fetchResolveMap: { [id]: _, ...fetchResolveMap } }) => ({
fetchResolveMap,
}));
},
},
}));
},
set,
})
)
);
import { Device } from "mediasoup-client";
import { detectDevice, Transport } from "mediasoup-client/lib/types";
import create from "zustand";
import { combine } from "zustand/middleware";
export const getDevice = () => {
try {
let handlerName = detectDevice();
if (!handlerName) {
console.warn(
"mediasoup does not recognize this device, so ben has defaulted it to Chrome74"
);
handlerName = "Chrome74";
}
return new Device({ handlerName });
} catch {
return null;
}
};
export const useVoiceStore = create(
combine(
{
roomId: "",
micStream: null as MediaStream | null,
mic: null as MediaStreamTrack | null,
recvTransport: null as Transport | null,
sendTransport: null as Transport | null,
device: getDevice(),
},
(set) => ({
nullify: () =>
set({
recvTransport: null,
sendTransport: null,
roomId: "",
mic: null,
micStream: null,
}),
set,
})
)
);
import create from "zustand";
import { combine } from "zustand/middleware";
export const useMicPermErrorStore = create(
combine(
{
error: false,
},
(set) => ({
set,
})
)
);
import { KeyMap } from "react-hotkeys";
import create from "zustand";
import { combine } from "zustand/middleware";
const MUTE_KEY = "@keybind/mute";
const CHAT_KEY = "@keybind/chat";
const PTT_KEY = "@keybind/ptt";
function getMuteKeybind() {
let v = "";
try {
v = localStorage.getItem(MUTE_KEY) || "";
} catch {}
return v || "Control+m";
}
function getChatKeybind() {
let v = "";
try {
v = localStorage.getItem(CHAT_KEY) || "";
} catch {}
return v || "Control+9";
}
function getPTTKeybind() {
let v = "";
try {
v = localStorage.getItem(PTT_KEY) || "";
} catch {}
return v || "Control+0";
}
const keyMap: KeyMap = {
MUTE: getMuteKeybind(),
CHAT: getChatKeybind(),
PTT: [
{ sequence: getPTTKeybind(), action: "keydown" },
{ sequence: getPTTKeybind(), action: "keyup" },
],
};
const keyNames: KeyMap = {
MUTE: getMuteKeybind(),
CHAT: getChatKeybind(),
PTT: getPTTKeybind(),
};
export const useKeyMapStore = create(
combine(
{
keyMap,
keyNames,
},
(set) => ({
setMuteKeybind: (id: string) => {
try {
localStorage.setItem(MUTE_KEY, id);
} catch {}
set((x) => ({
keyMap: { ...x.keyMap, MUTE: id },
keyNames: { ...x.keyNames, MUTE: id },
}));
},
setChatKeybind: (id: string) => {
try {
localStorage.setItem(CHAT_KEY, id);
} catch {}
set((x) => ({
keyMap: { ...x.keyMap, CHAT: id },
keyNames: { ...x.keyNames, CHAT: id },
}));
},
setPTTKeybind: (id: string) => {
try {
localStorage.setItem(PTT_KEY, id);
} catch {}
set((x) => ({
keyMap: {
...x.keyMap,
PTT: [
{ sequence: id, action: "keydown" },
{ sequence: id, action: "keyup" },
],
},
keyNames: { ...x.keyNames, PTT: id },
}));
},
})
)
);
import create from "zustand";
import { combine } from "zustand/middleware";
export const useAskForMicStore = create(
combine(
{
hasAsked: false,
},
(set) => ({
set,
})
)
);
import create from "zustand";
import { combine } from "zustand/middleware";
type State =
| "init"
| "ws-disconnected"
| "voice-server-disconnected"
| "connected-no-room"
| "connected-listener"
| "bad-auth"
| "killed"
| "connected-speaker";
export const useStatus = create(
combine(
{
status: "init" as State,
},
(set) => ({
setStatus: (status: State) => set({ status }),
})
)
);
import create from "zustand";
import { combine } from "zustand/middleware";
type State =
| "auth-good"
| "open"
| "connecting"
| "closed-by-server"
| "closed";
export const useSocketStatus = create(
combine(
{
status: "connecting" as State,
},
(set) => ({
setStatus: (status: State) => set({ status }),
})
)
);
import create from "zustand";
import { combine } from "zustand/middleware";
import { useSoundEffectStore } from "../../app/modules/sound-effects/useSoundEffectStore";
export const useMuteStore = create(
combine(
{
muted: false,
},
(set) => ({
setMute: (muted: boolean) => {
if (muted) {
useSoundEffectStore.getState().playSoundEffect("mute");
} else {
useSoundEffectStore.getState().playSoundEffect("unmute");
}
set({ muted });
},
})
)
);
import { RtpCapabilities } from "mediasoup-client/lib/types";
import { useVoiceStore } from "../stores/useVoiceStore";
export const joinRoom = async (routerRtpCapabilities: RtpCapabilities) => {
const { device } = useVoiceStore.getState();
if (!device!.loaded) {
await device!.load({ routerRtpCapabilities });
}
};
import { wsend } from "../../createWebsocket";
import { modalPrompt } from "../../app/components/PromptModal";
export const renameRoomAndMakePrivate = (currentName: string) => {
modalPrompt(
"Set private room name",
(roomName) => {
if (roomName) {
wsend({ op: "make_room_private", d: { newName: roomName } });
}
},
currentName
);
};
import { useConsumerStore } from "../stores/useConsumerStore";
import { useVoiceStore } from "../stores/useVoiceStore";
export const consumeAudio = async (consumerParameters: any, peerId: string) => {
const { recvTransport } = useVoiceStore.getState();
if (!recvTransport) {
console.log("skipping consumeAudio because recvTransport is null");
return false;
}
const consumer = await recvTransport.consume({
...consumerParameters,
appData: {
peerId,
producerId: consumerParameters.producerId,
mediaTag: "cam-audio",
},
});
useConsumerStore.getState().add(consumer, peerId);
return true;
};
import { wsend } from "../../createWebsocket";
import { modalPrompt } from "../../app/components/PromptModal";
export const renameRoomAndMakePublic = (currentName: string) => {
modalPrompt(
"Set public room name",
(roomName) => {
if (roomName) {
wsend({ op: "make_room_public", d: { newName: roomName } });
}
},
currentName
);
};
import { wsend } from "../../createWebsocket";
import { useVoiceStore } from "../stores/useVoiceStore";
import { useWsHandlerStore } from "../stores/useWsHandlerStore";
import { consumeAudio } from "./consumeAudio";
export const receiveVoice = (flushQueue: () => void) => {
useWsHandlerStore
.getState()
.addWsListenerOnce(
"@get-recv-tracks-done",
async ({ consumerParametersArr }) => {
try {
for (const { peerId, consumerParameters } of consumerParametersArr) {
if (!(await consumeAudio(consumerParameters, peerId))) {
break;
}
}
} catch (err) {
console.log(err);
} finally {
flushQueue();
}
}
);
wsend({
op: "@get-recv-tracks",
d: {
rtpCapabilities: useVoiceStore.getState().device!.rtpCapabilities,
},
});
};
import { useMicIdStore } from "../../app/shared-stores";
import { useMicPermErrorStore } from "../stores/useMicPermErrorStore";
import { useProducerStore } from "../stores/useProducerStore";
import { useVoiceStore } from "../stores/useVoiceStore";
export const sendVoice = async () => {
const { micId } = useMicIdStore.getState();
const { sendTransport, set, mic } = useVoiceStore.getState();
if (!sendTransport) {
console.log("no sendTransport in sendVoice");
return;
}
mic?.stop();
let micStream: MediaStream;
try {
micStream = await navigator.mediaDevices.getUserMedia({
audio: micId ? { deviceId: micId } : true,
});
useMicPermErrorStore.getState().set({ error: false });
} catch (err) {
set({ mic: null, micStream: null });
console.log(err);
useMicPermErrorStore.getState().set({ error: true });
return;
}
const audioTracks = micStream.getAudioTracks();
if (audioTracks.length) {
console.log("creating producer...");
const track = audioTracks[0];
useProducerStore.getState().add(
await sendTransport.produce({
track: track,
appData: { mediaTag: "cam-audio" },
})
);
set({ mic: track, micStream });
return;
}
set({ mic: null, micStream: null });
};
import { TransportOptions } from "mediasoup-client/lib/types";
import { wsend } from "../../createWebsocket";
import { useVoiceStore } from "../stores/useVoiceStore";
import { useWsHandlerStore } from "../stores/useWsHandlerStore";
export async function createTransport(
_roomId: string,
direction: "recv" | "send",
transportOptions: TransportOptions
) {
console.log(`create ${direction} transport`);
const { device, set } = useVoiceStore.getState();
// ask the server to create a server-side transport object and send
// us back the info we need to create a client-side transport
console.log("transport options", transportOptions);
let transport =
direction === "recv"
? await device!.createRecvTransport(transportOptions)
: await device!.createSendTransport(transportOptions);
// mediasoup-client will emit a connect event when media needs to
// start flowing for the first time. send dtlsParameters to the
// server, then call callback() on success or errback() on failure.
transport.on("connect", async ({ dtlsParameters }, callback, errback) => {
useWsHandlerStore
.getState()
.addWsListenerOnce(`@connect-transport-${direction}-done`, (d) => {
if (d.error) {
console.log(`connect-transport ${direction} failed`, d.error);
if (d.error.includes("already called")) {
callback();
} else {
errback();
}
} else {
console.log(`connect-transport ${direction} success`);
callback();
}
});
wsend({
op: "@connect-transport",
d: { transportId: transportOptions.id, dtlsParameters, direction },
});
});
if (direction === "send") {
// sending transports will emit a produce event when a new track
// needs to be set up to start sending. the producer's appData is
// passed as a parameter
transport.on(
"produce",
async ({ kind, rtpParameters, appData }, callback, errback) => {
console.log("transport produce event", appData.mediaTag);
// we may want to start out paused (if the checkboxes in the ui
// aren't checked, for each media type. not very clean code, here
// but, you know, this isn't a real application.)
let paused = false;
// if (appData.mediaTag === "cam-video") {
// paused = getCamPausedState();
// } else if (appData.mediaTag === "cam-audio") {
// paused = getMicPausedState();
// }
// tell the server what it needs to know from us in order to set
// up a server-side producer object, and get back a
// producer.id. call callback() on success or errback() on
// failure.
useWsHandlerStore
.getState()
.addWsListenerOnce(`@send-track-${direction}-done`, (d) => {
if (d.error) {
console.log(`send-track ${direction} failed`, d.error);
errback();
} else {
console.log(`send-track-transport ${direction} success`);
callback({ id: d.id });
}
});
wsend({
op: "@send-track",
d: {
transportId: transportOptions.id,
kind,
rtpParameters,
rtpCapabilities: device!.rtpCapabilities,
paused,
appData,
direction,
},
});
}
);
}
// for this simple demo, any time a transport transitions to closed,
// failed, or disconnected, leave the room and reset
//
transport.on("connectionstatechange", async (state) => {
console.log(
`${direction} transport ${transport.id} connectionstatechange ${state}`
);
});
if (direction === "recv") {
set({ recvTransport: transport });
} else {
set({ sendTransport: transport });
}
}
import { RoomPermissions } from "../../app/types";
export const mergeRoomPermission = (
currentRoomPermission: RoomPermissions | null | undefined,
newRoomPermissions: Partial<RoomPermissions>
) => {
return {
...(currentRoomPermission || {
askedToSpeak: false,
isMod: false,
isSpeaker: false,
}),
...newRoomPermissions,
};
};
// https://stackoverflow.com/questions/105034/how-to-create-a-guid-uuid
export function uuidv4(): string {
// @ts-ignore
return ([1e7] + -1e3 + -4e3 + -8e3 + -1e11).replace(/[018]/g, (c) =>
(
c ^
(crypto.getRandomValues(new Uint8Array(1))[0] & (15 >> (c / 4)))
).toString(16)
);
}
export type AddWsListenerOnce = (op: string, fn: (d: any) => void) => void;
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment