gomuks/web/src/ui/MainScreen.tsx

235 lines
7.9 KiB
TypeScript

// gomuks - A Matrix client written in Go.
// Copyright (C) 2024 Tulir Asokan
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU Affero General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU Affero General Public License for more details.
//
// You should have received a copy of the GNU Affero General Public License
// along with this program. If not, see <https://www.gnu.org/licenses/>.
import { JSX, use, useEffect, useInsertionEffect, useLayoutEffect, useMemo, useState } from "react"
import { SyncLoader } from "react-spinners"
import Client from "@/api/client.ts"
import { RoomStateStore } from "@/api/statestore"
import type { RoomID } from "@/api/types"
import { useEventAsState } from "@/util/eventdispatcher.ts"
import ClientContext from "./ClientContext.ts"
import MainScreenContext, { MainScreenContextFields } from "./MainScreenContext.ts"
import StylePreferences from "./StylePreferences.tsx"
import Keybindings from "./keybindings.ts"
import { ModalWrapper } from "./modal/Modal.tsx"
import RightPanel, { RightPanelProps } from "./rightpanel/RightPanel.tsx"
import RoomList from "./roomlist/RoomList.tsx"
import RoomView from "./roomview/RoomView.tsx"
import { useResizeHandle } from "./util/useResizeHandle.tsx"
import "./MainScreen.css"
function objectIsEqual(a: RightPanelProps | null, b: RightPanelProps | null): boolean {
if (a === null || b === null) {
return a === null && b === null
}
for (const key of Object.keys(a)) {
// @ts-expect-error 3:<
if (a[key] !== b[key]) {
return false
}
}
return true
}
class ContextFields implements MainScreenContextFields {
public keybindings: Keybindings
private rightPanelStack: RightPanelProps[] = []
constructor(
private directSetRightPanel: (props: RightPanelProps | null) => void,
private directSetActiveRoom: (room: RoomStateStore | null) => void,
private client: Client,
) {
this.keybindings = new Keybindings(client.store, this)
client.store.switchRoom = this.setActiveRoom
}
get currentRightPanel(): RightPanelProps | null {
return this.rightPanelStack.length ? this.rightPanelStack[this.rightPanelStack.length-1] : null
}
setRightPanel = (props: RightPanelProps | null, pushState = true) => {
const isEqual = objectIsEqual(this.currentRightPanel, props)
if (isEqual && !pushState) {
return
}
if (isEqual || props === null) {
const length = this.rightPanelStack.length
this.rightPanelStack = []
this.directSetRightPanel(null)
if (length && pushState) {
history.go(-length)
}
} else {
this.directSetRightPanel(props)
for (let i = this.rightPanelStack.length - 1; i >= 0; i--) {
if (objectIsEqual(this.rightPanelStack[i], props)) {
this.rightPanelStack = this.rightPanelStack.slice(0, i + 1)
if (pushState) {
history.go(i - this.rightPanelStack.length)
}
return
}
} // else:
this.rightPanelStack.push(props)
if (pushState) {
history.pushState({ ...(history.state ?? {}), right_panel: props }, "")
}
}
}
setActiveRoom = (roomID: RoomID | null, pushState = true) => {
console.log("Switching to room", roomID)
const room = (roomID && this.client.store.rooms.get(roomID)) || null
window.activeRoom = room
this.directSetActiveRoom(room)
this.directSetRightPanel(null)
this.rightPanelStack = []
this.client.store.activeRoomID = room?.roomID ?? null
this.keybindings.activeRoom = room
if (room) {
room.lastOpened = Date.now()
if (!room.stateLoaded) {
this.client.loadRoomState(room.roomID)
.catch(err => console.error("Failed to load room state", err))
}
document
.querySelector(`div.room-entry[data-room-id="${CSS.escape(room.roomID)}"]`)
?.scrollIntoView({ block: "nearest" })
}
if (pushState) {
history.pushState({ room_id: roomID }, "")
}
}
clickRoom = (evt: React.MouseEvent) => {
const roomID = evt.currentTarget.getAttribute("data-room-id")
if (roomID) {
this.setActiveRoom(roomID)
} else {
console.warn("No room ID :(", evt.currentTarget)
}
}
clickRightPanelOpener = (evt: React.MouseEvent) => {
const type = evt.currentTarget.getAttribute("data-target-panel")
if (type === "pinned-messages" || type === "members") {
this.setRightPanel({ type })
} else if (type === "user") {
this.setRightPanel({ type, userID: evt.currentTarget.getAttribute("data-target-user")! })
} else {
throw new Error(`Invalid right panel type ${type}`)
}
}
clearActiveRoom = () => this.setActiveRoom(null)
closeRightPanel = () => this.setRightPanel(null)
}
const SYNC_ERROR_HIDE_DELAY = 30 * 1000
const MainScreen = () => {
const [activeRoom, directSetActiveRoom] = useState<RoomStateStore | null>(null)
const [rightPanel, directSetRightPanel] = useState<RightPanelProps | null>(null)
const client = use(ClientContext)!
const syncStatus = useEventAsState(client.syncStatus)
const context = useMemo(
() => new ContextFields(directSetRightPanel, directSetActiveRoom, client),
[client],
)
useLayoutEffect(() => {
window.mainScreenContext = context
}, [context])
useEffect(() => {
const listener = (evt: PopStateEvent) => {
const roomID = evt.state?.room_id ?? null
if (roomID !== client.store.activeRoomID) {
context.setActiveRoom(roomID, false)
}
context.setRightPanel(evt.state?.right_panel ?? null, false)
}
window.addEventListener("popstate", listener)
return () => window.removeEventListener("popstate", listener)
}, [context, client])
useEffect(() => context.keybindings.listen(), [context])
useInsertionEffect(() => {
const styleTags = document.createElement("style")
styleTags.textContent = `
div.html-body > a.hicli-matrix-uri-user[href="matrix:u/${client.userID.slice(1).replaceAll(`"`, `\\"`)}"] {
background-color: var(--highlight-pill-background-color);
color: var(--highlight-pill-text-color);
}
`
document.head.appendChild(styleTags)
return () => {
document.head.removeChild(styleTags)
}
}, [client.userID])
const [roomListWidth, resizeHandle1] = useResizeHandle(
300, 48, 900, "roomListWidth", { className: "room-list-resizer" },
)
const [rightPanelWidth, resizeHandle2] = useResizeHandle(
300, 100, 900, "rightPanelWidth", { className: "right-panel-resizer", inverted: true },
)
const extraStyle = {
["--room-list-width" as string]: `${roomListWidth}px`,
["--right-panel-width" as string]: `${rightPanelWidth}px`,
}
const classNames = ["matrix-main"]
if (activeRoom) {
classNames.push("room-selected")
}
if (rightPanel) {
classNames.push("right-panel-open")
}
let syncLoader: JSX.Element | null = null
if (syncStatus.type === "waiting") {
syncLoader = <div className="sync-status waiting">
<SyncLoader color="var(--primary-color)"/>
Waiting for first sync...
</div>
} else if (
syncStatus.type === "errored"
&& (syncStatus.error_count > 2 || (syncStatus.last_sync ?? 0) + SYNC_ERROR_HIDE_DELAY < Date.now())
) {
syncLoader = <div className="sync-status errored" title={syncStatus.error}>
<SyncLoader color="var(--error-color)"/>
Sync is failing
</div>
}
return <MainScreenContext value={context}>
<ModalWrapper>
<StylePreferences client={client} activeRoom={activeRoom}/>
<main className={classNames.join(" ")} style={extraStyle}>
<RoomList activeRoomID={activeRoom?.roomID ?? null}/>
{resizeHandle1}
{activeRoom
? <RoomView
key={activeRoom.roomID}
room={activeRoom}
rightPanel={rightPanel}
rightPanelResizeHandle={resizeHandle2}
/>
: rightPanel && <>
{resizeHandle2}
{rightPanel && <RightPanel {...rightPanel}/>}
</>}
</main>
{syncLoader}
</ModalWrapper>
</MainScreenContext>
}
export default MainScreen