import {AbstractModal} from "tc-shared/ui/react-elements/modal/Definitions"; import {createIpcUiVariableConsumer, IpcVariableDescriptor} from "tc-shared/ui/utils/IpcVariable"; import { BookmarkConnectInfo, BookmarkListEntry, ModalBookmarkEvents, ModalBookmarkVariables } from "tc-shared/ui/modal/bookmarks/Definitions"; import {IpcRegistryDescription, Registry} from "tc-events"; import {UiVariableConsumer} from "tc-shared/ui/utils/Variable"; import {Translatable, VariadicTranslatable} from "tc-shared/ui/react-elements/i18n"; import {useContext, useEffect, useRef} from "react"; import {ContextDivider} from "tc-shared/ui/react-elements/ContextDivider"; import {joinClassList, useDependentState, useTr} from "tc-shared/ui/react-elements/Helper"; import {Button} from "tc-shared/ui/react-elements/Button"; import {LoadingDots} from "tc-shared/ui/react-elements/LoadingDots"; import {IconRenderer, RemoteIconRenderer} from "tc-shared/ui/react-elements/Icon"; import {getIconManager} from "tc-shared/file/Icons"; import {ClientIcon} from "svg-sprites/client-icons"; import {ClientIconRenderer} from "tc-shared/ui/react-elements/Icons"; import {spawnContextMenu} from "tc-shared/ui/ContextMenu"; import {spawnYesNo} from "tc-shared/ui/modal/ModalYesNo"; import {formatMessage} from "tc-shared/ui/frames/chat"; import {createErrorModal, createInfoModal, createInputModal} from "tc-shared/ui/elements/Modal"; import {HostBannerRenderer} from "tc-shared/ui/frames/HostBannerRenderer"; import {ControlledBoxedInputField, ControlledSelect} from "tc-shared/ui/react-elements/InputField"; import {Checkbox} from "tc-shared/ui/react-elements/Checkbox"; import * as React from "react"; import DefaultHeaderImage from "./header_background.png"; import ServerInfoImage from "./serverinfo.png"; import {IconTooltip} from "tc-shared/ui/react-elements/Tooltip"; import {CountryIcon} from "tc-shared/ui/react-elements/CountryIcon"; import {downloadTextAsFile, requestFileAsText} from "tc-shared/file/Utils"; const EventContext = React.createContext>(undefined); const VariableContext = React.createContext>(undefined); const SelectedBookmarkIdContext = React.createContext<{ type: "empty" | "bookmark" | "directory", id: string | undefined }>({ type: "empty", id: undefined }); const SelectedBookmarkInfoContext = React.createContext(undefined); const cssStyle = require("./Renderer.scss"); const Link = (props: { connected: boolean }) => (
); const BookmarkListEntryRenderer = React.memo((props: { entry: BookmarkListEntry }) => { const variables = useContext(VariableContext); const events = useContext(EventContext); const selectedItem = variables.useVariable("bookmarkSelected", undefined, undefined); const tryDelete = () => { if(props.entry.type === "directory" && props.entry.childCount > 0) { spawnYesNo(tr("Are you sure?"), formatMessage( tr("Do you really want to delete the directory \"{0}\"?{:br:}The directory contains {1} entries."), props.entry.displayName, props.entry.childCount ), result => { if(result) { events.fire("action_delete_bookmark", { uniqueId: props.entry.uniqueId }); } }).open(); } else { events.fire("action_delete_bookmark", { uniqueId: props.entry.uniqueId }); } }; let icon; if(props.entry.icon) { icon = ; } else if(props.entry.type === "directory") { icon = ; } else { icon = ; } let links = []; for(let i = 0; i < props.entry.depth; i++) { links.push(); } let buttons = []; if(props.entry.type === "bookmark") { buttons.push(
events.fire("action_duplicate_bookmark", { uniqueId: props.entry.uniqueId, displayName: undefined, originalName: props.entry.displayName })} >
); } buttons.push(
); return (
0 ? cssStyle.linkStart : undefined, selectedItem.remoteValue?.id === props.entry.uniqueId ? cssStyle.selected : undefined, )} onClick={() => { if(selectedItem.remoteValue?.id === props.entry.uniqueId) { return; } selectedItem.setValue({id: props.entry.uniqueId}); }} onDoubleClick={() => { if(props.entry.type !== "bookmark") { return; } events.fire("action_connect", { uniqueId: props.entry.uniqueId, newTab: false, closeModal: true }); }} onContextMenu={event => { event.preventDefault(); if(selectedItem.remoteValue?.id !== props.entry.uniqueId) { selectedItem.setValue({ id: props.entry.uniqueId }); } spawnContextMenu({ pageX: event.pageX, pageY: event.pageY }, [ { type: "normal", label: tr("Connect to server"), visible: props.entry.type === "bookmark", icon: ClientIcon.Connect, click: () => events.fire("action_connect", { uniqueId: props.entry.uniqueId, newTab: false, closeModal: true }) }, { type: "normal", label: tr("Connect in a new tab"), visible: props.entry.type === "bookmark", icon: ClientIcon.Connect, click: () => events.fire("action_connect", { uniqueId: props.entry.uniqueId, newTab: true, closeModal: true }) }, { type: "separator", visible: props.entry.type === "bookmark", }, { type: "normal", label: tr("Duplicate Bookmark"), visible: props.entry.type === "bookmark", icon: ClientIcon.BookmarkDuplicate, click: () => events.fire("action_duplicate_bookmark", { uniqueId: props.entry.uniqueId, displayName: undefined, originalName: props.entry.displayName }) }, { type: "normal", label: tr("Add bookmark"), icon: ClientIcon.BookmarkAdd, click: () => events.fire("action_create_bookmark", { entryType: "bookmark", order: { type: "parent", entry: props.entry.uniqueId }, displayName: undefined }) }, { type: "normal", label: tr("Add directory"), icon: ClientIcon.BookmarkAddFolder, click: () => events.fire("action_create_bookmark", { entryType: "directory", order: { type: "previous", entry: props.entry.uniqueId }, displayName: undefined }) }, { type: "normal", label: tr("Add sub directory"), visible: props.entry.type === "directory", icon: ClientIcon.BookmarkAddFolder, click: () => events.fire("action_create_bookmark", { entryType: "directory", order: { type: "parent", entry: props.entry.uniqueId }, displayName: undefined }) }, { type: "separator", }, { type: "normal", label: props.entry.type === "bookmark" ? tr("Delete bookmark") : tr("Delete directory"), icon: ClientIcon.BookmarkRemove, click: tryDelete } ]); }} > {...links} {icon}
{props.entry.displayName}
{...buttons}
); }); const BookmarkList = React.memo(() => { const events = useContext(EventContext); const variables = useContext(VariableContext); const bookmarksInfo = variables.useReadOnly("bookmarks"); const bookmarks = bookmarksInfo.status === "loaded" ? bookmarksInfo.value : []; return (
{bookmarks.map(entry => )}
loading
You don't have any bookmarks
); }); const BookmarkListContainer = React.memo(() => { const events = useContext(EventContext); return (
Your bookmarks
events.fire("action_create_bookmark", { entryType: "bookmark", order: { type: "selected" }, displayName: undefined })} >
); }); const SelectedBookmarkBanner = React.memo(() => { const bookmarkInfo = useContext(SelectedBookmarkInfoContext); if(!bookmarkInfo?.hostBannerUrl) { return ( {""} ); } return (
); }); const SelectedBookmarkName = React.memo(() => { const refEditPanel = useRef(); const selectedBookmarkId = useContext(SelectedBookmarkIdContext); const variables = useContext(VariableContext); const nameVariable = variables.useVariable("bookmarkName", selectedBookmarkId.id); let [ editMode, setEditMode ] = useDependentState(() => false, [ selectedBookmarkId.id ]); if(selectedBookmarkId.type === "empty") { editMode = false; } useEffect(() => { if(refEditPanel.current) { refEditPanel.current.textContent = nameVariable.localValue; refEditPanel.current.focus(); } }, [ editMode ]); if(nameVariable.status === "loading") { return (
loading
); } else if(editMode) { return (
{ if(event.key === "Enter") { event.preventDefault(); refEditPanel.current?.blur(); } else if(event.key === "Backspace" || event.key === "Delete") { /* never prevent these */ } else if(event.ctrlKey) { /* don't prevent this */ } else if(event.currentTarget.textContent?.length >= 32) { event.preventDefault(); } }} onInput={event => { const value = event.currentTarget.textContent; const valid = typeof value === "string" && value.length > 0 && value.length <= 32; refEditPanel.current?.classList.toggle(cssStyle.invalid, !valid); }} onBlur={() => { const value = refEditPanel.current?.textContent; setEditMode(false); if(!value || value.length > 32) { return; } nameVariable.setValue(value); }} >
); } else { return (
{nameVariable.status === "applying" ? tr("applying") : nameVariable.localValue}
setEditMode(true)}>
); } }) const SelectedBookmarkHeader = React.memo(() => { const selectedBookmarkId = useContext(SelectedBookmarkIdContext); const variables = useContext(VariableContext); const addressVariable = variables.useReadOnly("bookmarkServerAddress", selectedBookmarkId.id); let address; if(selectedBookmarkId.type === "bookmark") { if(addressVariable.status === "loading") { address = loading } else { address = {addressVariable.value}; } } return (
{address}
) }); const BookmarkSettingsGroup = React.memo((props: { children, className?: string }) => { return (
{props.children}
) }); const BookmarkSetting = React.memo((props: { children: [React.ReactNode, React.ReactNode] }) => { return (
{props.children[0]}
{props.children[1]}
) }); const BookmarkSettingConnectProfile = () => { const selectedBookmark = useContext(SelectedBookmarkIdContext); const variables = useContext(VariableContext); const selectedProfile = variables.useVariable("bookmarkConnectProfile", selectedBookmark.id); const availableProfiles = variables.useReadOnly("connectProfiles"); let value; const profiles = []; let invalid = false; if(selectedBookmark.type !== "bookmark") { value = "empty"; } else if(availableProfiles.status !== "loaded") { value = "loading"; } else if(selectedProfile.status === "loading") { value = "loading"; } else { value = selectedProfile.localValue; profiles.push(...availableProfiles.value.map(entry => ( ))); if(availableProfiles.value.findIndex(entry => entry.id === selectedProfile.localValue) === -1) { invalid = true; profiles.push( ); } } return ( selectedProfile.setValue(event.target.value)} invalid={invalid} > {profiles as any} ); }; const BookmarkSettingAutoConnect = () => { const selectedBookmark = useContext(SelectedBookmarkIdContext); const variables = useContext(VariableContext); const value = variables.useVariable("bookmarkConnectOnStartup", selectedBookmark.id, false); return ( value.setValue(newValue)} value={value.localValue} disabled={value.status !== "loaded" || selectedBookmark.type !== "bookmark"} label={Automatically connect to server on client start} /> ); }; const BookmarkSettingServerAddress = React.memo(() => { const selectedBookmark = useContext(SelectedBookmarkIdContext); const variables = useContext(VariableContext); const value = variables.useVariable("bookmarkServerAddress", selectedBookmark.id); return ( value.setValue(newValue, true)} onBlur={() => value.setValue(value.localValue)} finishOnEnter={true} /> ); }); const BookmarkSettingPassword = React.memo((props: { field: "bookmarkServerPassword" | "bookmarkDefaultChannelPassword", disabled: boolean }) => { const selectedBookmark = useContext(SelectedBookmarkIdContext); const variables = useContext(VariableContext); const value = variables.useVariable(props.field, selectedBookmark.id); let placeholder = "", inputValue = ""; if(props.disabled) { /* disabled, show nothing */ } else if(value.status === "loaded") { if(value.localValue && value.localValue === value.remoteValue) { placeholder = tr("password hidden"); } else { inputValue = value.localValue; } } else if(value.status === "applying") { if(value.localValue) { placeholder = tr("hashing password"); } else { /* we've resetted the password. Don't show "hashing password" */ } } const disabled = props.disabled || selectedBookmark.type !== "bookmark" || value.status !== "loaded"; return ( value.setValue(newValue, true)} onBlur={() => value.setValue(value.localValue)} rightIcon={() => (
!disabled && value.setValue("")}>
)} finishOnEnter={true} /> ); }); const BookmarkSettingChannel = React.memo(() => { const selectedBookmark = useContext(SelectedBookmarkIdContext); const variables = useContext(VariableContext); const defaultChannel = variables.useVariable("bookmarkDefaultChannel", selectedBookmark.id); const currentClientChannel = variables.useReadOnly("currentClientChannel", selectedBookmark.id); const inputDisabled = selectedBookmark.type !== "bookmark" || defaultChannel.status !== "loaded"; const channelSelectDisabled = inputDisabled || currentClientChannel.status !== "loaded" || !currentClientChannel.value; let selectCurrentTitle; if(channelSelectDisabled) { selectCurrentTitle = tr("Select current channel.\nYou're not connected to the target server."); } else { selectCurrentTitle = tr("Select current channel") + ":\n" + currentClientChannel.value?.name; selectCurrentTitle += "\n\n" + tr("Shift click to use the channel name path."); } return ( defaultChannel.setValue(newValue, true)} onBlur={() => defaultChannel.setValue(defaultChannel.localValue)} rightIcon={() => (
{ if(currentClientChannel.status !== "loaded") { return; } if(!currentClientChannel.value) { return; } if(event.shiftKey) { defaultChannel.setValue(currentClientChannel.value.path); } else { defaultChannel.setValue("/" + currentClientChannel.value.channelId); } variables.setVariable("bookmarkDefaultChannelPassword", selectedBookmark.id, currentClientChannel.value.passwordHash); }} >
)} finishOnEnter={true} /> ); }); const BookmarkSettingChannelPassword = () => { const selectedBookmark = useContext(SelectedBookmarkIdContext); const variables = useContext(VariableContext); const value = variables.useReadOnly("bookmarkDefaultChannel", selectedBookmark.id, undefined); return ; } const BookmarkInfoRenderer = React.memo(() => { const bookmarkInfo = useContext(SelectedBookmarkInfoContext); let connectCount = bookmarkInfo ? Math.max(bookmarkInfo.connectCountUniqueId, bookmarkInfo.connectCountAddress) : -1; return (
{""}
{useTr("Server name")}
{bookmarkInfo?.serverName}
{useTr("Server region")}
{useTr("Last ping")}
{useTr("Not yet supported")}
{useTr("Last client count")}
{bookmarkInfo?.clientsOnline} / {bookmarkInfo?.clientsMax}
{useTr("Connection count")}
{connectCount === -1 ? tr("fetch error") : connectCount}
{bookmarkInfo?.connectCountUniqueId}
{bookmarkInfo?.connectCountAddress}
You never connected to that server.
); }); const BookmarkInfoContainerInner = React.memo(() => (
Connect profile Server Address Server Password Default Channel Channel password
)); const BookmarkInfoContainer = React.memo(() => { const variables = useContext(VariableContext); const selectedBookmark = variables.useReadOnly("bookmarkSelected", undefined, { type: "empty", id: undefined }); const selectedBookmarkInfo = variables.useReadOnly("bookmarkInfo", selectedBookmark.id, undefined); return ( ) }); class ModalBookmarks extends AbstractModal { readonly events: Registry; readonly variables: UiVariableConsumer; constructor(events: IpcRegistryDescription, variables: IpcVariableDescriptor) { super(); this.events = Registry.fromIpcDescription(events); this.variables = createIpcUiVariableConsumer(variables); this.events.on("action_create_bookmark", event => { if(event.displayName) { return; } createInputModal(tr("Please enter a name"), tr("Please enter the bookmark name"), input => input.length > 0, value => { if(typeof value !== "string" || !value) { return; } this.events.fire("action_create_bookmark", { entryType: event.entryType, order: event.order, displayName: value }); }).open(); }); this.events.on("action_duplicate_bookmark", event => { if(event.displayName) { return; } createInputModal(tr("Please enter a name"), tr("Please enter the new bookmark name"), input => input.length > 0, value => { if(typeof value !== "string" || !value) { return; } this.events.fire("action_duplicate_bookmark", { displayName: value, uniqueId: event.uniqueId, originalName: event.originalName }); }, { defaultValue: event.originalName + " (Copy)" }).open(); }); this.events.on("notify_export_data", event => { downloadTextAsFile(event.payload, "bookmarks.json"); }); this.events.on("action_import", event => { if(event.payload) { return; } requestFileAsText().then(payload => { if(payload.length === 0) { this.events.fire("notify_import_result", { status: "error", message: tr("File payload is empty") }); return; } this.events.fire("action_import", { payload: payload }); }); }) this.events.on("notify_import_result", event => { switch (event.status) { case "error": createErrorModal(tr("Failed to import bookmarks"), tr("Failed to import bookmarks:") + "\n" + event.message).open(); break; case "success": createInfoModal(tr("Successfully imported"), formatMessage(tr("Successfully imported {0} bookmarks."), event.importedBookmarks)).open(); break; } }); } protected onDestroy() { super.onDestroy(); this.events.destroy(); this.variables.destroy(); } renderBody(): React.ReactElement { return (
); } renderTitle(): string | React.ReactElement { return Manage bookmarks; } } export = ModalBookmarks;