* fix intersection & resize observer * add binary search util * add scroll info util * add virtual paginator hook - WIP * render timeline using paginator hook * add continuous pagination to fill timeline * add doc comments in virtual paginator hook * add scroll to element func in virtual paginator * extract timeline pagination login into hook * add sliding name for timeline messages - testing * scroll with live event * change message rending style * make message timestamp smaller * remove unused imports * add random number between util * add compact message component * add sanitize html types * fix sending alias in room mention * get room member display name util * add get room with canonical alias util * add sanitize html util * render custom html with new styles * fix linkifying link text * add reaction component * display message reactions in timeline * Change mention color * show edited message * add event sent by function factory * add functions to get emoji shortcode * add component for reaction msg * add tooltip for who has reacted * add message layouts & placeholder * fix reaction size * fix dark theme colors * add code highlight with prismjs * add options to configure spacing in msgs * render message reply * fix trim reply from body regex * fix crash when loading reply * fix reply hover style * decrypt event on timeline paginate * update custom html code style * remove console logs * fix virtual paginator scroll to func * fix virtual paginator scroll to types * add stop scroll for in view item options * fix virtual paginator out of range scroll to index * scroll to and highlight reply on click * fix reply hover style * make message avatar clickable * fix scrollTo issue in virtual paginator * load reply from fetch * import virtual paginator restore scroll * load timeline for specific event * Fix back pagination recalibration * fix reply min height * revert code block colors to secondary * stop sanitizing text in code block * add decrypt file util * add image media component * update folds * fix code block font style * add msg event type * add scale dimension util * strict msg layout type * add image renderer component * add message content fallback components * add message matrix event renderer components * render matrix event using hooks * add attachment component * add attachment content types * handle error when rendering image in timeline * add video component * render video * include blurhash in thumbnails * generate thumbnails for image message * fix reactToDom spoiler opts * add hooks for HTMLMediaElement * render audio file in timeline * add msg image content component * fix image content props * add video content component * render new image/video component in timeline * remove console.log * convert seconds to milliseconds in video info * add load thumbnail prop to video content component * add file saver types * add file header component * add file content component * render file in timeline * add media control component * render audio message in room timeline * remove moved components * safely load message reply * add media loading hook * update media control layout * add loading indication in audio component * fill audio play icon when playing audio * fix media expanding * add image viewer - WIP * add pan and zoom control to image viewer * add text based file viewer * add pdf viewer * add error handling in pdf viewer * add download btn to pdf viewer * fix file button spinner fill * fix file opens on re-render * add range slider in audio content player * render location in timeline * update folds * display membership event in timeline * make reactions toggle * render sticker messages in timeline * render room name, topic, avatar change and event * fix typos * update render state event type style * add room intro in start of timeline * add power levels context * fix wrong param passing in RoomView * fix sending typing notification in wrong room Slate onChange callback was not updating with react re-renders. * send typing status on key up * add typing indicator component * add typing member atom * display typing status in member drawer * add room view typing member component * display typing members in room view * remove old roomTimeline uses * add event readers hook * add latest event hook * display following members in room view * fetch event instead of event context for reply * fix typo in virtual paginator hook * add scroll to latest btn in timeline * change scroll to latest chip variant * destructure paginator object to improve perf * restore forward dir scroll in virtual paginator * run scroll to bottom in layout effect * display unread message indicator in timeline * make component for room timeline float * add timeline divider component * add day divider and format message time * apply message spacing to dividers * format date in room intro * send read receipt on message arrive * add event readers component * add reply, read receipt, source delete opt * bug fixes * update timeline on delete & show reason * fix empty reaction container style * show msg selection effect on msg option open * add report message options * add options to send quick reactions * add emoji board in message options * add reaction viewer * fix styles * show view reaction in msg options menu * fix spacing between two msg by same person * add option menu in other rendered event * handle m.room.encrypted messages * fix italic reply text overflow cut * handle encrypted sticker messages * remove console log * prevent message context menu with alt key pressed * make mentions clickable in messages * add options to show and hidden events in timeline * add option to disable media autoload * remove old emojiboard opener * add options to use system emoji * refresh timeline on reset * fix stuck typing member in member drawer
441 lines
16 KiB
JavaScript
441 lines
16 KiB
JavaScript
import React, { useState, useEffect } from 'react';
|
|
import './Settings.scss';
|
|
|
|
import initMatrix from '../../../client/initMatrix';
|
|
import cons from '../../../client/state/cons';
|
|
import settings from '../../../client/state/settings';
|
|
import navigation from '../../../client/state/navigation';
|
|
import {
|
|
toggleSystemTheme, toggleMarkdown,
|
|
toggleNotifications, toggleNotificationSounds,
|
|
} from '../../../client/action/settings';
|
|
import { usePermission } from '../../hooks/usePermission';
|
|
|
|
import Text from '../../atoms/text/Text';
|
|
import IconButton from '../../atoms/button/IconButton';
|
|
import Button from '../../atoms/button/Button';
|
|
import Toggle from '../../atoms/button/Toggle';
|
|
import Tabs from '../../atoms/tabs/Tabs';
|
|
import { MenuHeader } from '../../atoms/context-menu/ContextMenu';
|
|
import SegmentedControls from '../../atoms/segmented-controls/SegmentedControls';
|
|
|
|
import PopupWindow from '../../molecules/popup-window/PopupWindow';
|
|
import SettingTile from '../../molecules/setting-tile/SettingTile';
|
|
import ImportE2ERoomKeys from '../../molecules/import-export-e2e-room-keys/ImportE2ERoomKeys';
|
|
import ExportE2ERoomKeys from '../../molecules/import-export-e2e-room-keys/ExportE2ERoomKeys';
|
|
import { ImagePackUser, ImagePackGlobal } from '../../molecules/image-pack/ImagePack';
|
|
import GlobalNotification from '../../molecules/global-notification/GlobalNotification';
|
|
import KeywordNotification from '../../molecules/global-notification/KeywordNotification';
|
|
import IgnoreUserList from '../../molecules/global-notification/IgnoreUserList';
|
|
|
|
import ProfileEditor from '../profile-editor/ProfileEditor';
|
|
import CrossSigning from './CrossSigning';
|
|
import KeyBackup from './KeyBackup';
|
|
import DeviceManage from './DeviceManage';
|
|
|
|
import SunIC from '../../../../public/res/ic/outlined/sun.svg';
|
|
import EmojiIC from '../../../../public/res/ic/outlined/emoji.svg';
|
|
import LockIC from '../../../../public/res/ic/outlined/lock.svg';
|
|
import BellIC from '../../../../public/res/ic/outlined/bell.svg';
|
|
import InfoIC from '../../../../public/res/ic/outlined/info.svg';
|
|
import PowerIC from '../../../../public/res/ic/outlined/power.svg';
|
|
import CrossIC from '../../../../public/res/ic/outlined/cross.svg';
|
|
|
|
import CinnySVG from '../../../../public/res/svg/cinny.svg';
|
|
import { confirmDialog } from '../../molecules/confirm-dialog/ConfirmDialog';
|
|
import { useSetting } from '../../state/hooks/settings';
|
|
import { settingsAtom } from '../../state/settings';
|
|
|
|
function AppearanceSection() {
|
|
const [, updateState] = useState({});
|
|
|
|
const [messageLayout, setMessageLayout] = useSetting(settingsAtom, 'messageLayout');
|
|
const [messageSpacing, setMessageSpacing] = useSetting(settingsAtom, 'messageSpacing');
|
|
const [useSystemEmoji, setUseSystemEmoji] = useSetting(settingsAtom, 'useSystemEmoji');
|
|
const [hideMembershipEvents, setHideMembershipEvents] = useSetting(settingsAtom, 'hideMembershipEvents');
|
|
const [hideNickAvatarEvents, setHideNickAvatarEvents] = useSetting(settingsAtom, 'hideNickAvatarEvents');
|
|
const [mediaAutoLoad, setMediaAutoLoad] = useSetting(settingsAtom, 'mediaAutoLoad');
|
|
const [showHiddenEvents, setShowHiddenEvents] = useSetting(settingsAtom, 'showHiddenEvents');
|
|
const spacings = ['0', '100', '200', '300', '400', '500']
|
|
|
|
return (
|
|
<div className="settings-appearance">
|
|
<div className="settings-appearance__card">
|
|
<MenuHeader>Theme</MenuHeader>
|
|
<SettingTile
|
|
title="Follow system theme"
|
|
options={(
|
|
<Toggle
|
|
isActive={settings.useSystemTheme}
|
|
onToggle={() => { toggleSystemTheme(); updateState({}); }}
|
|
/>
|
|
)}
|
|
content={<Text variant="b3">Use light or dark mode based on the system settings.</Text>}
|
|
/>
|
|
<SettingTile
|
|
title="Theme"
|
|
content={(
|
|
<SegmentedControls
|
|
selected={settings.useSystemTheme ? -1 : settings.getThemeIndex()}
|
|
segments={[
|
|
{ text: 'Light' },
|
|
{ text: 'Silver' },
|
|
{ text: 'Dark' },
|
|
{ text: 'Butter' },
|
|
]}
|
|
onSelect={(index) => {
|
|
if (settings.useSystemTheme) toggleSystemTheme();
|
|
settings.setTheme(index);
|
|
updateState({});
|
|
}}
|
|
/>
|
|
)}
|
|
/>
|
|
<SettingTile
|
|
title="Use System Emoji"
|
|
options={(
|
|
<Toggle
|
|
isActive={useSystemEmoji}
|
|
onToggle={() => setUseSystemEmoji(!useSystemEmoji)}
|
|
/>
|
|
)}
|
|
content={<Text variant="b3">Use system emoji instead of Twitter emojis.</Text>}
|
|
/>
|
|
</div>
|
|
<div className="settings-appearance__card">
|
|
<MenuHeader>Room messages</MenuHeader>
|
|
<SettingTile
|
|
title="Message Layout"
|
|
content={
|
|
<SegmentedControls
|
|
selected={messageLayout}
|
|
segments={[
|
|
{ text: 'Modern' },
|
|
{ text: 'Compact' },
|
|
{ text: 'Bubble' },
|
|
]}
|
|
onSelect={(index) => setMessageLayout(index)}
|
|
/>
|
|
}
|
|
/>
|
|
<SettingTile
|
|
title="Message Spacing"
|
|
content={
|
|
<SegmentedControls
|
|
selected={spacings.findIndex((s) => s === messageSpacing)}
|
|
segments={[
|
|
{ text: 'No' },
|
|
{ text: 'XXS' },
|
|
{ text: 'XS' },
|
|
{ text: 'S' },
|
|
{ text: 'M' },
|
|
{ text: 'L' },
|
|
]}
|
|
onSelect={(index) => {
|
|
setMessageSpacing(spacings[index])
|
|
}}
|
|
/>
|
|
}
|
|
/>
|
|
<SettingTile
|
|
title="Markdown formatting"
|
|
options={(
|
|
<Toggle
|
|
isActive={settings.isMarkdown}
|
|
onToggle={() => { toggleMarkdown(); updateState({}); }}
|
|
/>
|
|
)}
|
|
content={<Text variant="b3">Format messages with markdown syntax before sending.</Text>}
|
|
/>
|
|
<SettingTile
|
|
title="Hide membership events"
|
|
options={(
|
|
<Toggle
|
|
isActive={hideMembershipEvents}
|
|
onToggle={() => setHideMembershipEvents(!hideMembershipEvents)}
|
|
/>
|
|
)}
|
|
content={<Text variant="b3">Hide membership change messages from room timeline. (Join, Leave, Invite, Kick and Ban)</Text>}
|
|
/>
|
|
<SettingTile
|
|
title="Hide nick/avatar events"
|
|
options={(
|
|
<Toggle
|
|
isActive={hideNickAvatarEvents}
|
|
onToggle={() => setHideNickAvatarEvents(!hideNickAvatarEvents)}
|
|
/>
|
|
)}
|
|
content={<Text variant="b3">Hide nick and avatar change messages from room timeline.</Text>}
|
|
/>
|
|
<SettingTile
|
|
title="Disable media auto load"
|
|
options={(
|
|
<Toggle
|
|
isActive={!mediaAutoLoad}
|
|
onToggle={() => setMediaAutoLoad(!mediaAutoLoad)}
|
|
/>
|
|
)}
|
|
content={<Text variant="b3">Prevent images and videos from auto loading to save bandwidth.</Text>}
|
|
/>
|
|
<SettingTile
|
|
title="Show hidden events"
|
|
options={(
|
|
<Toggle
|
|
isActive={showHiddenEvents}
|
|
onToggle={() => setShowHiddenEvents(!showHiddenEvents)}
|
|
/>
|
|
)}
|
|
content={<Text variant="b3">Show hidden state and message events.</Text>}
|
|
/>
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
function NotificationsSection() {
|
|
const [permission, setPermission] = usePermission('notifications', window.Notification?.permission);
|
|
|
|
const [, updateState] = useState({});
|
|
|
|
const renderOptions = () => {
|
|
if (window.Notification === undefined) {
|
|
return <Text className="settings-notifications__not-supported">Not supported in this browser.</Text>;
|
|
}
|
|
|
|
if (permission === 'granted') {
|
|
return (
|
|
<Toggle
|
|
isActive={settings._showNotifications}
|
|
onToggle={() => {
|
|
toggleNotifications();
|
|
setPermission(window.Notification?.permission);
|
|
updateState({});
|
|
}}
|
|
/>
|
|
);
|
|
}
|
|
|
|
return (
|
|
<Button
|
|
variant="primary"
|
|
onClick={() => window.Notification.requestPermission().then(setPermission)}
|
|
>
|
|
Request permission
|
|
</Button>
|
|
);
|
|
};
|
|
|
|
return (
|
|
<>
|
|
<div className="settings-notifications">
|
|
<MenuHeader>Notification & Sound</MenuHeader>
|
|
<SettingTile
|
|
title="Desktop notification"
|
|
options={renderOptions()}
|
|
content={<Text variant="b3">Show desktop notification when new messages arrive.</Text>}
|
|
/>
|
|
<SettingTile
|
|
title="Notification Sound"
|
|
options={(
|
|
<Toggle
|
|
isActive={settings.isNotificationSounds}
|
|
onToggle={() => { toggleNotificationSounds(); updateState({}); }}
|
|
/>
|
|
)}
|
|
content={<Text variant="b3">Play sound when new messages arrive.</Text>}
|
|
/>
|
|
</div>
|
|
<GlobalNotification />
|
|
<KeywordNotification />
|
|
<IgnoreUserList />
|
|
</>
|
|
);
|
|
}
|
|
|
|
function EmojiSection() {
|
|
return (
|
|
<>
|
|
<div className="settings-emoji__card"><ImagePackUser /></div>
|
|
<div className="settings-emoji__card"><ImagePackGlobal /></div>
|
|
</>
|
|
);
|
|
}
|
|
|
|
function SecuritySection() {
|
|
return (
|
|
<div className="settings-security">
|
|
<div className="settings-security__card">
|
|
<MenuHeader>Cross signing and backup</MenuHeader>
|
|
<CrossSigning />
|
|
<KeyBackup />
|
|
</div>
|
|
<DeviceManage />
|
|
<div className="settings-security__card">
|
|
<MenuHeader>Export/Import encryption keys</MenuHeader>
|
|
<SettingTile
|
|
title="Export E2E room keys"
|
|
content={(
|
|
<>
|
|
<Text variant="b3">Export end-to-end encryption room keys to decrypt old messages in other session. In order to encrypt keys you need to set a password, which will be used while importing.</Text>
|
|
<ExportE2ERoomKeys />
|
|
</>
|
|
)}
|
|
/>
|
|
<SettingTile
|
|
title="Import E2E room keys"
|
|
content={(
|
|
<>
|
|
<Text variant="b3">{'To decrypt older messages, Export E2EE room keys from Element (Settings > Security & Privacy > Encryption > Cryptography) and import them here. Imported keys are encrypted so you\'ll have to enter the password you set in order to decrypt it.'}</Text>
|
|
<ImportE2ERoomKeys />
|
|
</>
|
|
)}
|
|
/>
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
function AboutSection() {
|
|
return (
|
|
<div className="settings-about">
|
|
<div className="settings-about__card">
|
|
<MenuHeader>Application</MenuHeader>
|
|
<div className="settings-about__branding">
|
|
<img width="60" height="60" src={CinnySVG} alt="Cinny logo" />
|
|
<div>
|
|
<Text variant="h2" weight="medium">
|
|
Cinny
|
|
<span className="text text-b3" style={{ margin: '0 var(--sp-extra-tight)' }}>{`v${cons.version}`}</span>
|
|
</Text>
|
|
<Text>Yet another matrix client</Text>
|
|
|
|
<div className="settings-about__btns">
|
|
<Button onClick={() => window.open('https://github.com/ajbura/cinny')}>Source code</Button>
|
|
<Button onClick={() => window.open('https://cinny.in/#sponsor')}>Support</Button>
|
|
<Button onClick={() => initMatrix.clearCacheAndReload()} variant="danger">Clear cache & reload</Button>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<div className="settings-about__card">
|
|
<MenuHeader>Credits</MenuHeader>
|
|
<div className="settings-about__credits">
|
|
<ul>
|
|
<li>
|
|
{/* eslint-disable-next-line react/jsx-one-expression-per-line */ }
|
|
<Text>The <a href="https://github.com/matrix-org/matrix-js-sdk" rel="noreferrer noopener" target="_blank">matrix-js-sdk</a> is © <a href="https://matrix.org/foundation" rel="noreferrer noopener" target="_blank">The Matrix.org Foundation C.I.C</a> used under the terms of <a href="http://www.apache.org/licenses/LICENSE-2.0" rel="noreferrer noopener" target="_blank">Apache 2.0</a>.</Text>
|
|
</li>
|
|
<li>
|
|
{/* eslint-disable-next-line react/jsx-one-expression-per-line */ }
|
|
<Text>The <a href="https://twemoji.twitter.com" target="_blank" rel="noreferrer noopener">Twemoji</a> emoji art is © <a href="https://twemoji.twitter.com" target="_blank" rel="noreferrer noopener">Twitter, Inc and other contributors</a> used under the terms of <a href="https://creativecommons.org/licenses/by/4.0/" target="_blank" rel="noreferrer noopener">CC-BY 4.0</a>.</Text>
|
|
</li>
|
|
<li>
|
|
{/* eslint-disable-next-line react/jsx-one-expression-per-line */ }
|
|
<Text>The <a href="https://material.io/design/sound/sound-resources.html" target="_blank" rel="noreferrer noopener">Material sound resources</a> are © <a href="https://google.com" target="_blank" rel="noreferrer noopener">Google</a> used under the terms of <a href="https://creativecommons.org/licenses/by/4.0/" target="_blank" rel="noreferrer noopener">CC-BY 4.0</a>.</Text>
|
|
</li>
|
|
</ul>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
export const tabText = {
|
|
APPEARANCE: 'Appearance',
|
|
NOTIFICATIONS: 'Notifications',
|
|
EMOJI: 'Emoji',
|
|
SECURITY: 'Security',
|
|
ABOUT: 'About',
|
|
};
|
|
const tabItems = [{
|
|
text: tabText.APPEARANCE,
|
|
iconSrc: SunIC,
|
|
disabled: false,
|
|
render: () => <AppearanceSection />,
|
|
}, {
|
|
text: tabText.NOTIFICATIONS,
|
|
iconSrc: BellIC,
|
|
disabled: false,
|
|
render: () => <NotificationsSection />,
|
|
}, {
|
|
text: tabText.EMOJI,
|
|
iconSrc: EmojiIC,
|
|
disabled: false,
|
|
render: () => <EmojiSection />,
|
|
}, {
|
|
text: tabText.SECURITY,
|
|
iconSrc: LockIC,
|
|
disabled: false,
|
|
render: () => <SecuritySection />,
|
|
}, {
|
|
text: tabText.ABOUT,
|
|
iconSrc: InfoIC,
|
|
disabled: false,
|
|
render: () => <AboutSection />,
|
|
}];
|
|
|
|
function useWindowToggle(setSelectedTab) {
|
|
const [isOpen, setIsOpen] = useState(false);
|
|
|
|
useEffect(() => {
|
|
const openSettings = (tab) => {
|
|
const tabItem = tabItems.find((item) => item.text === tab);
|
|
if (tabItem) setSelectedTab(tabItem);
|
|
setIsOpen(true);
|
|
};
|
|
navigation.on(cons.events.navigation.SETTINGS_OPENED, openSettings);
|
|
return () => {
|
|
navigation.removeListener(cons.events.navigation.SETTINGS_OPENED, openSettings);
|
|
};
|
|
}, []);
|
|
|
|
const requestClose = () => setIsOpen(false);
|
|
|
|
return [isOpen, requestClose];
|
|
}
|
|
|
|
function Settings() {
|
|
const [selectedTab, setSelectedTab] = useState(tabItems[0]);
|
|
const [isOpen, requestClose] = useWindowToggle(setSelectedTab);
|
|
|
|
const handleTabChange = (tabItem) => setSelectedTab(tabItem);
|
|
const handleLogout = async () => {
|
|
if (await confirmDialog('Logout', 'Are you sure that you want to logout your session?', 'Logout', 'danger')) {
|
|
initMatrix.logout();
|
|
}
|
|
};
|
|
|
|
return (
|
|
<PopupWindow
|
|
isOpen={isOpen}
|
|
className="settings-window"
|
|
title={<Text variant="s1" weight="medium" primary>Settings</Text>}
|
|
contentOptions={(
|
|
<>
|
|
<Button variant="danger" iconSrc={PowerIC} onClick={handleLogout}>
|
|
Logout
|
|
</Button>
|
|
<IconButton src={CrossIC} onClick={requestClose} tooltip="Close" />
|
|
</>
|
|
)}
|
|
onRequestClose={requestClose}
|
|
>
|
|
{isOpen && (
|
|
<div className="settings-window__content">
|
|
<ProfileEditor userId={initMatrix.matrixClient.getUserId()} />
|
|
<Tabs
|
|
items={tabItems}
|
|
defaultSelected={tabItems.findIndex((tab) => tab.text === selectedTab.text)}
|
|
onSelect={handleTabChange}
|
|
/>
|
|
<div className="settings-window__cards-wrapper">
|
|
{ selectedTab.render() }
|
|
</div>
|
|
</div>
|
|
)}
|
|
</PopupWindow>
|
|
);
|
|
}
|
|
|
|
export default Settings;
|