From eaeabaf813dfbdb5cc42da4e94bf2fa28c32a489 Mon Sep 17 00:00:00 2001 From: yougotwill Date: Wed, 31 Jul 2024 15:11:50 +1000 Subject: [PATCH 01/23] fix: dont trim display name when sanitizing trim on submission --- ts/components/dialog/edit-profile/EditProfileDialog.tsx | 1 + ts/components/registration/utils/index.tsx | 5 ++--- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/ts/components/dialog/edit-profile/EditProfileDialog.tsx b/ts/components/dialog/edit-profile/EditProfileDialog.tsx index b4ffdf7d9..d679b672f 100644 --- a/ts/components/dialog/edit-profile/EditProfileDialog.tsx +++ b/ts/components/dialog/edit-profile/EditProfileDialog.tsx @@ -216,6 +216,7 @@ export const EditProfileDialog = () => { setLoading(true); const validName = await ProfileManager.updateOurProfileDisplayName(profileName); setUpdateProfileName(validName); + setProfileName(validName); setMode('default'); } catch (err) { window.log.error('Profile update error', err); diff --git a/ts/components/registration/utils/index.tsx b/ts/components/registration/utils/index.tsx index 3ea00dfc6..4220c06d2 100644 --- a/ts/components/registration/utils/index.tsx +++ b/ts/components/registration/utils/index.tsx @@ -9,15 +9,14 @@ export function sanitizeDisplayNameOrToast( ) { try { const sanitizedName = sanitizeSessionUsername(displayName); - const trimName = sanitizedName.trim(); - const errorString = !trimName ? window.i18n('displayNameEmpty') : undefined; + const errorString = !sanitizedName ? window.i18n('displayNameEmpty') : undefined; if (dispatch) { dispatch(onDisplayNameError(errorString)); } else { onDisplayNameError(errorString); // this is is either calling dispatch in the caller or just `setDisplayNameError` } - return trimName; + return sanitizedName; } catch (e) { if (dispatch) { dispatch(onDisplayNameError(window.i18n('displayNameErrorDescriptionShorter'))); From e66d443a97bcd15d70e30a9d968fc09bff037672 Mon Sep 17 00:00:00 2001 From: yougotwill Date: Wed, 31 Jul 2024 15:32:06 +1000 Subject: [PATCH 02/23] fix: reset input color after error if the value changes --- ts/components/inputs/SessionInput.tsx | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/ts/components/inputs/SessionInput.tsx b/ts/components/inputs/SessionInput.tsx index f52ec154f..699435cd4 100644 --- a/ts/components/inputs/SessionInput.tsx +++ b/ts/components/inputs/SessionInput.tsx @@ -277,6 +277,7 @@ export const SessionInput = (props: Props) => { } = props; const [inputValue, setInputValue] = useState(''); const [errorString, setErrorString] = useState(''); + const [textErrorStyle, setTextErrorStyle] = useState(false); const [forceShow, setForceShow] = useState(false); const correctType = forceShow ? 'text' : type; @@ -288,6 +289,7 @@ export const SessionInput = (props: Props) => { e.preventDefault(); const val = e.target.value; setInputValue(val); + setTextErrorStyle(false); if (onValueChanged) { onValueChanged(val); } @@ -332,7 +334,7 @@ export const SessionInput = (props: Props) => { const containerProps = { noValue: isEmpty(value), - error: Boolean(error), + error: textErrorStyle, centerText, textSize, monospaced, @@ -342,6 +344,7 @@ export const SessionInput = (props: Props) => { useEffect(() => { if (error && !isEmpty(error) && !isEqual(error, errorString)) { setErrorString(error); + setTextErrorStyle(!!error); } }, [error, errorString]); From d52461aecc1f6b5127f047c4d6bbb818f658a997 Mon Sep 17 00:00:00 2001 From: yougotwill Date: Wed, 31 Jul 2024 17:20:15 +1000 Subject: [PATCH 03/23] fix: don't use name length for the recovery password input and trim it --- ts/components/registration/stages/RestoreAccount.tsx | 6 ++++-- 1 file changed, 4 insertions(+), 2 deletions(-) diff --git a/ts/components/registration/stages/RestoreAccount.tsx b/ts/components/registration/stages/RestoreAccount.tsx index af3e7850b..cbfc3b320 100644 --- a/ts/components/registration/stages/RestoreAccount.tsx +++ b/ts/components/registration/stages/RestoreAccount.tsx @@ -133,13 +133,15 @@ export const RestoreAccount = () => { if (!(!!recoveryPassword && !recoveryPasswordError)) { return; } + const trimmedPassword = recoveryPassword.trim(); + setRecoveryPassword(trimmedPassword); const abortController = new AbortController(); try { dispatch(setProgress(0)); dispatch(setAccountRestorationStep(AccountRestoration.Loading)); await signInAndFetchDisplayName({ - recoveryPassword, + recoveryPassword: trimmedPassword, dispatch, abortSignal: abortController.signal, }); @@ -248,7 +250,6 @@ export const RestoreAccount = () => { }} onEnterPressed={recoverAndFetchDisplayName} error={recoveryPasswordError} - maxLength={LIBSESSION_CONSTANTS.CONTACT_MAX_NAME_LENGTH} enableShowHideButton={true} showHideButtonAriaLabels={{ hide: 'Hide recovery password toggle', @@ -289,6 +290,7 @@ export const RestoreAccount = () => { }} onEnterPressed={recoverAndEnterDisplayName} error={displayNameError} + maxLength={LIBSESSION_CONSTANTS.CONTACT_MAX_NAME_LENGTH} inputDataTestId="display-name-input" /> From 8260f0481b9394c320627717f4208516910d4faa Mon Sep 17 00:00:00 2001 From: yougotwill Date: Thu, 1 Aug 2024 16:25:49 +1000 Subject: [PATCH 04/23] fix: session input with textarea now centers content correctly --- ts/components/dialog/UserDetailsDialog.tsx | 3 +- ts/components/inputs/SessionInput.tsx | 53 +++++++++++++------ .../leftpane/overlay/OverlayInvite.tsx | 11 ++-- 3 files changed, 46 insertions(+), 21 deletions(-) diff --git a/ts/components/dialog/UserDetailsDialog.tsx b/ts/components/dialog/UserDetailsDialog.tsx index 36f447c91..baa388e20 100644 --- a/ts/components/dialog/UserDetailsDialog.tsx +++ b/ts/components/dialog/UserDetailsDialog.tsx @@ -16,8 +16,9 @@ import { ConversationTypeEnum } from '../../models/types'; const StyledInputContainer = styled(Flex)` textarea { + cursor: default; overflow: hidden; - padding-top: var(--margins-xs); + top: 14px; } `; diff --git a/ts/components/inputs/SessionInput.tsx b/ts/components/inputs/SessionInput.tsx index 699435cd4..2f30ce0e3 100644 --- a/ts/components/inputs/SessionInput.tsx +++ b/ts/components/inputs/SessionInput.tsx @@ -1,4 +1,4 @@ -import { ChangeEvent, ReactNode, RefObject, useEffect, useState } from 'react'; +import { ChangeEvent, ReactNode, RefObject, useEffect, useRef, useState } from 'react'; import { motion } from 'framer-motion'; import { isEmpty, isEqual } from 'lodash'; @@ -13,7 +13,7 @@ type TextSizes = 'xs' | 'sm' | 'md' | 'lg' | 'xl'; const StyledSessionInput = styled(Flex)<{ error: boolean; - textSize?: TextSizes; + textSize: TextSizes; }>` position: relative; width: 100%; @@ -68,6 +68,7 @@ const StyledBorder = styled(AnimatedFlex)` const StyledInput = styled(motion.input)<{ error: boolean; + textSize: TextSizes; centerText?: boolean; monospaced?: boolean; }>` @@ -79,9 +80,9 @@ const StyledInput = styled(motion.input)<{ color: ${props => (props.error ? 'var(--danger-color)' : 'var(--input-text-color)')}; font-family: ${props => (props.monospaced ? 'var(--font-mono)' : 'var(--font-default)')}; - font-size: 12px; line-height: 1.4; ${props => props.centerText && 'text-align: center;'} + ${props => `font-size: var(--font-size-${props.textSize});`} &::placeholder { color: var(--input-text-placeholder-color); @@ -89,13 +90,15 @@ const StyledInput = styled(motion.input)<{ } `; -const StyledTextAreaContainer = styled(motion.div)<{ +export const StyledTextAreaContainer = styled(motion.div)<{ noValue: boolean; error: boolean; + textSize: TextSizes; centerText?: boolean; - textSize?: TextSizes; monospaced?: boolean; }>` + display: flex; + align-items: center; overflow: hidden; position: relative; height: ${props => (props.textSize ? `calc(var(--font-size-${props.textSize}) * 4)` : '48px')}; @@ -107,8 +110,8 @@ const StyledTextAreaContainer = styled(motion.div)<{ outline: 0; font-family: ${props => (props.monospaced ? 'var(--font-mono)' : 'var(--font-default)')}; - font-size: 12px; - line-height: 1.4; + ${props => `font-size: var(--font-size-${props.textSize});`} + line-height: 1; ${props => props.centerText && 'text-align: center;'} @@ -121,21 +124,20 @@ const StyledTextAreaContainer = styled(motion.div)<{ border: none; background: transparent; - ${props => - props.noValue && - `position: absolute; - top: ${props.textSize ? `calc(var(--font-size-${props.textSize}) + 5px)` : 'calc(12px + 5px)'};`} + position: absolute; + top: ${props => + `calc(var(--font-size-${props.textSize}) + ${props.textSize === 'xl' ? '8px' : '5px'})`}; resize: none; - overflow-wrap: break-word; + word-break: break-all; user-select: all; ${props => props.centerText && 'text-align: center;'} &:placeholder-shown { font-family: ${props => (props.monospaced ? 'var(--font-mono)' : 'var(--font-default)')}; - font-size: 12px; - line-height: 1.4; + ${props => `font-size: var(--font-size-${props.textSize});`} + line-height: 1; } &::placeholder { @@ -267,7 +269,7 @@ export const SessionInput = (props: Props) => { showHideButtonDataTestIds, ctaButton, monospaced, - textSize, + textSize = 'sm', centerText, editable = true, isTextArea, @@ -280,6 +282,8 @@ export const SessionInput = (props: Props) => { const [textErrorStyle, setTextErrorStyle] = useState(false); const [forceShow, setForceShow] = useState(false); + const textAreaRef = useRef(inputRef?.current || null); + const correctType = forceShow ? 'text' : type; const updateInputValue = (e: ChangeEvent) => { @@ -290,6 +294,16 @@ export const SessionInput = (props: Props) => { const val = e.target.value; setInputValue(val); setTextErrorStyle(false); + if (isTextArea && textAreaRef && textAreaRef.current !== null) { + const scrollHeight = `${textAreaRef.current.scrollHeight}px`; + if (isEmpty(val)) { + // resets the height of the text area so it's centered if we clear the text + textAreaRef.current.style.height = 'unset'; + } + if (scrollHeight !== textAreaRef.current.style.height) { + textAreaRef.current.style.height = `${textAreaRef.current.scrollHeight}px`; + } + } if (onValueChanged) { onValueChanged(val); } @@ -301,6 +315,7 @@ export const SessionInput = (props: Props) => { type: correctType, placeholder, value, + textSize, disabled: !editable, maxLength, autoFocus, @@ -308,7 +323,6 @@ export const SessionInput = (props: Props) => { required, 'aria-required': required, tabIndex, - ref: inputRef, onChange: updateInputValue, style: { paddingInlineEnd: enableShowHideButton ? '48px' : undefined }, // just in case onChange isn't triggered @@ -372,12 +386,17 @@ export const SessionInput = (props: Props) => { > {isTextArea ? ( -