From 98341b440aefb9cca8fcf766a4de793019821308 Mon Sep 17 00:00:00 2001 From: Joel Jeremy Marquez <joeljeremy.marquez@gmail.com> Date: Thu, 6 Jun 2024 17:12:59 -0700 Subject: [PATCH] [Mobile] Use AmountInput on mobile transfer and hold buffer modals (#2837) * Use AmountInput on mobile transfer and hold buffer modals * Release notes * VRT + typecheck error fixes * VRT * VRT --- .../mobile/transactions/TransactionEdit.jsx | 2 +- .../src/components/modals/HoldBufferModal.tsx | 30 +++++++++++-------- .../src/components/modals/TransferModal.tsx | 29 ++++++++++-------- .../src/components/util/AmountInput.tsx | 10 +++++-- upcoming-release-notes/2837.md | 6 ++++ 5 files changed, 47 insertions(+), 30 deletions(-) create mode 100644 upcoming-release-notes/2837.md diff --git a/packages/desktop-client/src/components/mobile/transactions/TransactionEdit.jsx b/packages/desktop-client/src/components/mobile/transactions/TransactionEdit.jsx index e09e5c256..62593477e 100644 --- a/packages/desktop-client/src/components/mobile/transactions/TransactionEdit.jsx +++ b/packages/desktop-client/src/components/mobile/transactions/TransactionEdit.jsx @@ -342,7 +342,7 @@ const ChildTransactionEdit = forwardRef( value={amountToInteger(transaction.amount)} zeroSign={amountSign} style={{ marginRight: 8 }} - textStyle={{ ...styles.smallText, textAlign: 'right' }} + inputStyle={{ ...styles.smallText, textAlign: 'right' }} onFocus={() => onRequestActiveEdit(getFieldName(transaction.id, 'amount')) } diff --git a/packages/desktop-client/src/components/modals/HoldBufferModal.tsx b/packages/desktop-client/src/components/modals/HoldBufferModal.tsx index 93304923a..976f1948e 100644 --- a/packages/desktop-client/src/components/modals/HoldBufferModal.tsx +++ b/packages/desktop-client/src/components/modals/HoldBufferModal.tsx @@ -1,17 +1,16 @@ import React, { useState } from 'react'; import { rolloverBudget } from 'loot-core/client/queries'; -import { evalArithmetic } from 'loot-core/shared/arithmetic'; -import { amountToInteger, integerToCurrency } from 'loot-core/shared/util'; import { styles } from '../../style'; import { Button } from '../common/Button'; import { InitialFocus } from '../common/InitialFocus'; import { Modal } from '../common/Modal'; import { View } from '../common/View'; -import { FieldLabel, InputField } from '../mobile/MobileForms'; +import { FieldLabel } from '../mobile/MobileForms'; import { type CommonModalProps } from '../Modals'; import { useSheetValue } from '../spreadsheet/useSheetValue'; +import { AmountInput } from '../util/AmountInput'; type HoldBufferModalProps = { modalProps: CommonModalProps; @@ -24,13 +23,11 @@ export function HoldBufferModal({ onSubmit, }: HoldBufferModalProps) { const available = useSheetValue(rolloverBudget.toBudget); - const initialAmount = integerToCurrency(Math.max(available, 0)); - const [amount, setAmount] = useState<string | null>(null); + const [amount, setAmount] = useState<number>(0); - const _onSubmit = (newAmount: string | null) => { - const parsedAmount = evalArithmetic(newAmount || ''); - if (parsedAmount) { - onSubmit?.(amountToInteger(parsedAmount)); + const _onSubmit = (newAmount: number) => { + if (newAmount) { + onSubmit?.(newAmount); } modalProps.onClose(); @@ -46,10 +43,17 @@ export function HoldBufferModal({ <View> <FieldLabel title="Hold this amount:" /> <InitialFocus> - <InputField - inputMode="decimal" - defaultValue={initialAmount} - onUpdate={value => setAmount(value)} + <AmountInput + value={available} + autoDecimals={true} + style={{ + marginLeft: styles.mobileEditingPadding, + marginRight: styles.mobileEditingPadding, + }} + inputStyle={{ + height: styles.mobileMinHeight, + }} + onUpdate={setAmount} onEnter={() => _onSubmit(amount)} /> </InitialFocus> diff --git a/packages/desktop-client/src/components/modals/TransferModal.tsx b/packages/desktop-client/src/components/modals/TransferModal.tsx index 8b19a5dd5..5af37bb8b 100644 --- a/packages/desktop-client/src/components/modals/TransferModal.tsx +++ b/packages/desktop-client/src/components/modals/TransferModal.tsx @@ -2,8 +2,6 @@ import React, { useMemo, useState } from 'react'; import { useDispatch } from 'react-redux'; import { pushModal } from 'loot-core/client/actions'; -import { evalArithmetic } from 'loot-core/shared/arithmetic'; -import { amountToInteger, integerToCurrency } from 'loot-core/shared/util'; import { useCategories } from '../../hooks/useCategories'; import { styles } from '../../style'; @@ -12,8 +10,9 @@ import { Button } from '../common/Button'; import { InitialFocus } from '../common/InitialFocus'; import { Modal } from '../common/Modal'; import { View } from '../common/View'; -import { FieldLabel, InputField, TapField } from '../mobile/MobileForms'; +import { FieldLabel, TapField } from '../mobile/MobileForms'; import { type CommonModalProps } from '../Modals'; +import { AmountInput } from '../util/AmountInput'; type TransferModalProps = { modalProps: CommonModalProps; @@ -42,8 +41,7 @@ export function TransferModal({ return [expenseGroups, expenseCategories]; }, [originalCategoryGroups, showToBeBudgeted]); - const _initialAmount = integerToCurrency(Math.max(initialAmount, 0)); - const [amount, setAmount] = useState<string | null>(null); + const [amount, setAmount] = useState<number>(0); const [toCategoryId, setToCategoryId] = useState<string | null>(null); const dispatch = useDispatch(); @@ -60,10 +58,9 @@ export function TransferModal({ ); }; - const _onSubmit = (newAmount: string | null, categoryId: string | null) => { - const parsedAmount = evalArithmetic(newAmount || ''); - if (parsedAmount && categoryId) { - onSubmit?.(amountToInteger(parsedAmount), categoryId); + const _onSubmit = (newAmount: number, categoryId: string | null) => { + if (newAmount && categoryId) { + onSubmit?.(newAmount, categoryId); } modalProps.onClose(); @@ -77,10 +74,16 @@ export function TransferModal({ <View> <FieldLabel title="Transfer this amount:" /> <InitialFocus> - <InputField - inputMode="decimal" - tabIndex={0} - defaultValue={_initialAmount} + <AmountInput + value={initialAmount} + autoDecimals={true} + style={{ + marginLeft: styles.mobileEditingPadding, + marginRight: styles.mobileEditingPadding, + }} + inputStyle={{ + height: styles.mobileMinHeight, + }} onUpdate={setAmount} onEnter={() => { if (!toCategoryId) { diff --git a/packages/desktop-client/src/components/util/AmountInput.tsx b/packages/desktop-client/src/components/util/AmountInput.tsx index cde149e0c..ae385f28f 100644 --- a/packages/desktop-client/src/components/util/AmountInput.tsx +++ b/packages/desktop-client/src/components/util/AmountInput.tsx @@ -5,6 +5,7 @@ import React, { useState, useEffect, type FocusEventHandler, + type KeyboardEventHandler, } from 'react'; import { evalArithmetic } from 'loot-core/src/shared/arithmetic'; @@ -27,9 +28,10 @@ type AmountInputProps = { onChangeValue?: (value: string) => void; onFocus?: FocusEventHandler<HTMLInputElement>; onBlur?: FocusEventHandler<HTMLInputElement>; + onEnter?: KeyboardEventHandler<HTMLInputElement>; onUpdate?: (amount: number) => void; style?: CSSProperties; - textStyle?: CSSProperties; + inputStyle?: CSSProperties; focused?: boolean; disabled?: boolean; autoDecimals?: boolean; @@ -44,8 +46,9 @@ export function AmountInput({ onBlur, onChangeValue, onUpdate, + onEnter, style, - textStyle, + inputStyle, focused, disabled = false, autoDecimals = false, @@ -120,7 +123,7 @@ export function AmountInput({ disabled={disabled} focused={focused} style={{ flex: 1, alignItems: 'stretch', ...style }} - inputStyle={{ paddingLeft: 0, ...textStyle }} + inputStyle={inputStyle} onKeyUp={e => { if (e.key === 'Enter') { fireUpdate(negative); @@ -129,6 +132,7 @@ export function AmountInput({ onChangeValue={onInputTextChange} onBlur={onInputAmountBlur} onFocus={onFocus} + onEnter={onEnter} /> ); } diff --git a/upcoming-release-notes/2837.md b/upcoming-release-notes/2837.md new file mode 100644 index 000000000..1b50a2727 --- /dev/null +++ b/upcoming-release-notes/2837.md @@ -0,0 +1,6 @@ +--- +category: Enhancements +authors: [joel-jeremy] +--- + +Use AmountInput on mobile balance transfer and hold buffer modals to allow auto insertion of decimals in their amounts. -- GitLab