diff --git a/stylesheets/_session.scss b/stylesheets/_session.scss index b7e76c228..c740299fd 100644 --- a/stylesheets/_session.scss +++ b/stylesheets/_session.scss @@ -269,7 +269,7 @@ textarea { /* CONVERSATION AND MESSAGES */ .module-conversation-header { - height: $main-view-header-height; + height: var(--main-view-header-height); } .module-conversation-header__title-flex, @@ -563,37 +563,9 @@ label { } .session-toggle { - width: 51px; - height: 31px; - border: 1.5px solid #e5e5ea; - border-radius: 16px; - position: relative; - - cursor: pointer; - background-color: rgba(0, 0, 0, 0); - - .knob { - position: absolute; - top: 0.5px; - left: 0.5px; - height: 27px; - width: 27px; - border-radius: 28px; - background-color: $session-color-white; - box-shadow: 0 0 3px 1px rgba(0, 0, 0, 0.05), 0 3px 1px 0 rgba(0, 0, 0, 0.05), - 0 2px 2px 0 rgba(0, 0, 0, 0.1), 0 3px 3px 0 rgba(0, 0, 0, 0.05); - - transition: transform var(--default-duration) ease, - background-color var(--default-duration) ease; - } - &.active { background-color: $session-color-green; border-color: $session-color-green; - - .knob { - transform: translateX(20px); - } } } @@ -830,68 +802,6 @@ label { overflow-x: hidden; } - &-header { - display: flex; - flex-direction: row; - justify-content: center; - align-items: center; - height: $main-view-header-height; - background: var(--color-cell-background); - - &-title { - line-height: $main-view-header-height; - font-weight: bold; - font-size: var(--font-size-lg); - text-align: center; - flex-grow: 1; - } - - .session-button, - .session-icon-button { - margin-inline-end: var(--margins-lg); - } - } - - &-item { - font-size: $session-font-md; - padding: var(--margins-lg); - margin-bottom: 20px; - - background: var(--color-cell-background); - color: var(--color-text); - border-bottom: var(--border-session); - - &.inline { - display: flex; - align-items: center; - justify-content: space-between; - } - - &__info { - padding-inline-end: var(--margins-lg); - } - - &__title { - line-height: 1.7; - font-size: var(--font-size-lg); - font-weight: bold; - } - - &__description { - font-family: $session-font-default; - font-size: $session-font-sm; - font-weight: 100; - max-width: 700px; - color: var(--color-text-subtle); - } - - &__selection { - .session-toggle { - transition: var(--default-duration); - } - } - } - &-view { flex-grow: 1; display: flex; @@ -900,24 +810,6 @@ label { overflow: hidden; } - &__version-info { - display: flex; - justify-content: space-between; - - padding: var(--margins-sm) var(--margins-md); - background: none; - font-size: $session-font-xs; - - span { - opacity: 0.4; - transition: var(--default-duration); - - &:hover { - opacity: 1; - } - } - } - &__password-lock { display: flex; align-items: center; @@ -938,27 +830,6 @@ label { border: 1px solid $session-shade-8; border-radius: 5px; - - h3 { - padding: 0px; - margin-bottom: var(--margins-lg); - } - - input { - width: 100%; - background: var(--color-input-background); - color: var(--color-text); - - padding: var(--margins-xs) var(--margins-md); - margin-bottom: var(--margins-lg); - outline: none; - border: none; - border-radius: 2px; - text-align: center; - font-size: 24px; - letter-spacing: 5px; - font-family: $session-font-default; - } } } } diff --git a/stylesheets/_session_constants.scss b/stylesheets/_session_constants.scss index f45bf3803..c38c47f7a 100644 --- a/stylesheets/_session_constants.scss +++ b/stylesheets/_session_constants.scss @@ -148,7 +148,6 @@ $session-subtle-factor: 0.6; // Default Components $session-search-input-height: 34px; -$main-view-header-height: 63px; $session-left-pane-width: 300px; // Various Components diff --git a/stylesheets/_session_conversation.scss b/stylesheets/_session_conversation.scss index fe997a2cf..4ca66d6b7 100644 --- a/stylesheets/_session_conversation.scss +++ b/stylesheets/_session_conversation.scss @@ -67,7 +67,7 @@ padding: 0px var(--margins-md); align-items: center; justify-content: space-between; - height: $main-view-header-height; + height: var(--main-view-header-height); background: var(--color-cell-background); .close-button { diff --git a/stylesheets/_session_left_pane.scss b/stylesheets/_session_left_pane.scss index b2d91aa3a..31d6a4792 100644 --- a/stylesheets/_session_left_pane.scss +++ b/stylesheets/_session_left_pane.scss @@ -86,7 +86,7 @@ $session-compose-margin: 20px; flex-direction: row; justify-content: space-between; align-items: center; - height: $main-view-header-height; + height: var(--main-view-header-height); padding-inline-end: 7px; transition: var(--default-duration); } diff --git a/ts/components/basic/SessionToggle.tsx b/ts/components/basic/SessionToggle.tsx index e26125e4d..49aa51220 100644 --- a/ts/components/basic/SessionToggle.tsx +++ b/ts/components/basic/SessionToggle.tsx @@ -1,7 +1,7 @@ import React from 'react'; -import classNames from 'classnames'; import { updateConfirmModal } from '../../state/ducks/modalDialog'; import { useDispatch } from 'react-redux'; +import styled from 'styled-components'; type Props = { active: boolean; @@ -12,7 +12,7 @@ type Props = { export const SessionToggle = (props: Props) => { const dispatch = useDispatch(); - const clickHandler = (event: any) => { + const clickHandler = (event: React.MouseEvent) => { const stateManager = (e: any) => { e.stopPropagation(); props.onClick(); @@ -45,14 +45,39 @@ export const SessionToggle = (props: Props) => { }; return ( -
-
-
-
-
+ + + ); }; + +const StyledKnob = styled.div<{ active: boolean }>` + position: absolute; + top: 0.5px; + left: 0.5px; + height: 27px; + width: 27px; + border-radius: 28px; + background-color: white; + box-shadow: 0 0 3px 1px rgba(0, 0, 0, 0.05), 0 3px 1px 0 rgba(0, 0, 0, 0.05), + 0 2px 2px 0 rgba(0, 0, 0, 0.1), 0 3px 3px 0 rgba(0, 0, 0, 0.05); + + transition: transform var(--default-duration) ease, background-color var(--default-duration) ease; + + transform: ${props => (props.active ? 'translateX(20px)' : '')}; +`; + +const StyledSessionToggle = styled.div<{ active: boolean }>` + width: 51px; + height: 31px; + border: 1.5px solid #e5e5ea; + border-radius: 16px; + position: relative; + + cursor: pointer; + background-color: rgba(0, 0, 0, 0); + transition: var(--default-duration); + + background-color: ${props => (props.active ? 'var(--color-accent)' : 'unset')}; + border-color: ${props => (props.active ? 'var(--color-accent)' : 'unset')}; +`; diff --git a/ts/components/settings/SessionSettingListItem.tsx b/ts/components/settings/SessionSettingListItem.tsx index 52a8ac79e..15a752394 100644 --- a/ts/components/settings/SessionSettingListItem.tsx +++ b/ts/components/settings/SessionSettingListItem.tsx @@ -1,8 +1,8 @@ import React from 'react'; -import classNames from 'classnames'; import { SessionButton, SessionButtonColor } from '../basic/SessionButton'; import { SessionToggle } from '../basic/SessionToggle'; import { SessionConfirmDialogProps } from '../dialog/SessionConfirm'; +import styled from 'styled-components'; type ButtonSettingsProps = { title?: string; @@ -13,36 +13,65 @@ type ButtonSettingsProps = { onClick: () => void; }; +const StyledDescription = styled.div` + font-family: var(--font-default); + font-size: var(--font-size-sm); + font-weight: 100; + max-width: 700px; + color: var(--color-text-subtle); +`; + +const StyledTitle = styled.div` + line-height: 1.7; + font-size: var(--font-size-lg); + font-weight: bold; +`; + +const StyledInfo = styled.div` + padding-inline-end: var(--margins-lg); +`; + const SettingsTitleAndDescription = (props: { title?: string; description?: string }) => { return ( -
-
{props.title}
+ + {props.title} - {props.description && ( -
{props.description}
- )} -
+ {props.description && {props.description}} + ); }; -const SessionSettingsContent = (props: { children: React.ReactNode }) => { - return
{props.children}
; -}; - export const SessionSettingsItemWrapper = (props: { inline: boolean; title?: string; description?: string; children?: React.ReactNode; }) => { + const ComponentToRender = props.inline ? StyledSettingItemInline : StyledSettingItem; return ( -
+ - {props.children} -
+
{props.children}
+ ); }; +const StyledSettingItem = styled.div` + font-size: var(--font-size-md); + padding: var(--margins-lg); + margin-bottom: 20px; + + background: var(--color-cell-background); + color: var(--color-text); + border-bottom: var(--border-session); +`; + +const StyledSettingItemInline = styled(StyledSettingItem)` + display: flex; + align-items: center; + justify-content: space-between; +`; + export const SessionToggleWithDescription = (props: { title?: string; description?: string; diff --git a/ts/components/settings/SessionSettings.tsx b/ts/components/settings/SessionSettings.tsx index 1cf4e65fc..2b03c4644 100644 --- a/ts/components/settings/SessionSettings.tsx +++ b/ts/components/settings/SessionSettings.tsx @@ -11,10 +11,10 @@ import { SettingsCategoryPrivacy } from './section/CategoryPrivacy'; import { SettingsCategoryAppearance } from './section/CategoryAppearance'; import { SessionButton, SessionButtonColor, SessionButtonType } from '../basic/SessionButton'; import { Data } from '../../data/data'; -import { LocalizerKeys } from '../../types/LocalizerKeys'; import { matchesHash } from '../../util/passwordUtils'; import { SettingsCategoryPermissions } from './section/CategoryPermissions'; import { SettingsCategoryHelp } from './section/CategoryHelp'; +import styled from 'styled-components'; export function getMediaPermissionsSettings() { return window.getSettingValue('media-permissions'); @@ -50,22 +50,62 @@ interface State { shouldLockSettings: boolean | null; } +const StyledVersionInfo = styled.div` + display: flex; + justify-content: space-between; + + padding: var(--margins-sm) var(--margins-md); + background: none; + font-size: var(--font-size-xs); +`; + +const StyledSpanSessionInfo = styled.span` + opacity: 0.4; + transition: var(--default-duration); + user-select: text; + + :hover { + opacity: 1; + } +`; + const SessionInfo = () => { const openOxenWebsite = () => { void shell.openExternal('https://oxen.io/'); }; return ( -
- v{window.versionInfo.version} - + + v{window.versionInfo.version} + - - {window.versionInfo.commitHash} -
+ + {window.versionInfo.commitHash} + ); }; -export const PasswordLock = ({ +const StyledPasswordInput = styled.input` + width: 100%; + background: var(--color-input-background); + color: var(--color-text); + + padding: var(--margins-xs) var(--margins-md); + margin-bottom: var(--margins-lg); + outline: none; + border: none; + border-radius: 2px; + text-align: center; + font-size: 24px; + letter-spacing: 5px; + font-family: var(--font-default); +`; + +const StyledH3 = styled.h3` + padding: 0px; + margin-bottom: var(--margins-lg); +`; + +const PasswordLock = ({ pwdLockError, validatePasswordLock, }: { @@ -75,13 +115,14 @@ export const PasswordLock = ({ return (
-

{window.i18n('password')}

- {window.i18n('password')} + {pwdLockError &&
{pwdLockError}
} @@ -97,6 +138,42 @@ export const PasswordLock = ({ ); }; +const SettingInCategory = (props: { + category: SessionSettingCategory; + hasPassword: boolean; + onPasswordUpdated: (action: string) => void; +}) => { + const { category, hasPassword, onPasswordUpdated } = props; + + if (hasPassword === null) { + return null; + } + switch (category) { + // special case for blocked user + case SessionSettingCategory.Conversations: + return ; + case SessionSettingCategory.Appearance: + return ; + case SessionSettingCategory.Notifications: + return ; + case SessionSettingCategory.Privacy: + return ( + + ); + case SessionSettingCategory.Help: + return ; + case SessionSettingCategory.Permissions: + return ; + + // those three down there have no options, they are just a button + case SessionSettingCategory.ClearData: + case SessionSettingCategory.MessageRequests: + case SessionSettingCategory.RecoveryPhrase: + default: + return null; + } +}; + export class SessionSettingsView extends React.Component { public settingsViewRef: React.RefObject; @@ -114,7 +191,11 @@ export class SessionSettingsView extends React.Component { + this.setState({ + hasPassword: !!hash, + }); + }); } public componentDidMount() { @@ -123,51 +204,12 @@ export class SessionSettingsView extends React.Component document.getElementById('password-lock-input')?.focus(), 100); } public componentWillUnmount() { window.removeEventListener('keyup', this.onKeyUp); } - /* tslint:disable-next-line:max-func-body-length */ - public renderSettingInCategory() { - const { category } = this.props; - - if (this.state.hasPassword === null) { - return null; - } - const hasPassword = this.state.hasPassword; - switch (category) { - // special case for blocked user - case SessionSettingCategory.Conversations: - return ; - case SessionSettingCategory.Appearance: - return ; - case SessionSettingCategory.Notifications: - return ; - case SessionSettingCategory.Privacy: - return ( - - ); - case SessionSettingCategory.Help: - return ; - case SessionSettingCategory.Permissions: - return ; - - // those three down there have no options, they are just a button - case SessionSettingCategory.ClearData: - case SessionSettingCategory.MessageRequests: - case SessionSettingCategory.RecoveryPhrase: - default: - return null; - } - } - public async validatePasswordLock() { const enteredPassword = String( (document.getElementById('password-lock-input') as HTMLInputElement)?.value @@ -203,18 +245,10 @@ export class SessionSettingsView extends React.Component - +
{shouldRenderPasswordLock ? ( @@ -224,7 +258,11 @@ export class SessionSettingsView extends React.Component ) : (
- {this.renderSettingInCategory()} +
)} @@ -233,14 +271,6 @@ export class SessionSettingsView extends React.Component & { - categoryTitle: string; -}; +type Props = Pick; export const SettingsHeader = (props: Props) => { - const { categoryTitle } = props; + const { category } = props; + + let categoryLocalized: LocalizerKeys | null = null; + switch (category) { + case SessionSettingCategory.Appearance: + categoryLocalized = 'appearanceSettingsTitle'; + break; + case SessionSettingCategory.Conversations: + categoryLocalized = 'blockedSettingsTitle'; + break; + case SessionSettingCategory.Notifications: + categoryLocalized = 'notificationsSettingsTitle'; + break; + case SessionSettingCategory.Help: + categoryLocalized = 'helpSettingsTitle'; + break; + case SessionSettingCategory.Permissions: + categoryLocalized = 'permissionsSettingsTitle'; + break; + case SessionSettingCategory.Privacy: + categoryLocalized = 'privacySettingsTitle'; + break; + default: + throw missingCaseError('SettingsHeader' as never); + } + + const categoryTitle = window.i18n(categoryLocalized); + return ( -
-
{categoryTitle}
-
+ + {categoryTitle} + ); }; + +const StyledSettingsHeader = styled.div` + display: flex; + flex-direction: row; + justify-content: center; + align-items: center; + height: var(--main-view-header-height); + background: var(--color-cell-background); +`; + +const StyledHeaderTittle = styled.div` + line-height: var(--main-view-header-height); + font-weight: bold; + font-size: var(--font-size-lg); + text-align: center; + flex-grow: 1; +`; diff --git a/ts/state/ducks/SessionTheme.tsx b/ts/state/ducks/SessionTheme.tsx index 518c56f61..170a33187 100644 --- a/ts/state/ducks/SessionTheme.tsx +++ b/ts/state/ducks/SessionTheme.tsx @@ -331,6 +331,9 @@ export const SessionGlobalStyles = createGlobalStyle` --margins-md: 15px; --margins-lg: 20px; + /* SIZES */ + --main-view-header-height: 63px; + /* ANIMATIONS */ --default-duration: 0.25s; /* FILTERS */