Files
actual/packages/desktop-client/src/components/util/AmountInput.tsx
2025-03-10 17:21:00 +00:00

222 lines
5.6 KiB
TypeScript

// @ts-strict-ignore
import React, {
type Ref,
useRef,
useState,
useEffect,
type FocusEventHandler,
type KeyboardEventHandler,
type CSSProperties,
} from 'react';
import { useTranslation } from 'react-i18next';
import { Button } from '@actual-app/components/button';
import { SvgAdd, SvgSubtract } from '@actual-app/components/icons/v1';
import { defaultInputStyle, Input } from '@actual-app/components/input';
import { theme } from '@actual-app/components/theme';
import { View } from '@actual-app/components/view';
import { evalArithmetic } from 'loot-core/shared/arithmetic';
import { amountToInteger, appendDecimals } from 'loot-core/shared/util';
import { useMergedRefs } from '../../hooks/useMergedRefs';
import { useSyncedPref } from '../../hooks/useSyncedPref';
import { useFormat } from '../spreadsheet/useFormat';
type AmountInputProps = {
id?: string;
inputRef?: Ref<HTMLInputElement>;
value: number;
zeroSign?: '-' | '+';
onChangeValue?: (value: string) => void;
onFocus?: FocusEventHandler<HTMLInputElement>;
onBlur?: FocusEventHandler<HTMLInputElement>;
onEnter?: KeyboardEventHandler<HTMLInputElement>;
onUpdate?: (amount: number) => void;
style?: CSSProperties;
inputStyle?: CSSProperties;
focused?: boolean;
disabled?: boolean;
autoDecimals?: boolean;
};
export function AmountInput({
id,
inputRef,
value: initialValue,
zeroSign = '-', // + or -
onFocus,
onBlur,
onChangeValue,
onUpdate,
onEnter,
style,
inputStyle,
focused,
disabled = false,
autoDecimals = false,
}: AmountInputProps) {
const format = useFormat();
const [symbol, setSymbol] = useState<'+' | '-'>(
initialValue === 0 ? zeroSign : initialValue > 0 ? '+' : '-',
);
const [isFocused, setIsFocused] = useState(focused ?? false);
const initialValueAbsolute = format(Math.abs(initialValue || 0), 'financial');
const [value, setValue] = useState(initialValueAbsolute);
useEffect(() => setValue(initialValueAbsolute), [initialValueAbsolute]);
const buttonRef = useRef();
const ref = useRef<HTMLInputElement>(null);
const mergedRef = useMergedRefs<HTMLInputElement>(inputRef, ref);
const [hideFraction] = useSyncedPref('hideFraction');
useEffect(() => {
if (focused) {
ref.current?.focus();
}
}, [focused]);
function onSwitch() {
const amount = getAmount();
if (amount === 0) {
setSymbol(symbol === '+' ? '-' : '+');
}
fireUpdate(amount * -1);
}
function getAmount() {
const signedValued = symbol === '-' ? symbol + value : value;
return amountToInteger(evalArithmetic(signedValued));
}
function onInputTextChange(val) {
val = autoDecimals
? appendDecimals(val, String(hideFraction) === 'true')
: val;
setValue(val ? val : '');
onChangeValue?.(val);
}
function fireUpdate(amount) {
onUpdate?.(amount);
if (amount > 0) {
setSymbol('+');
} else if (amount < 0) {
setSymbol('-');
}
}
function onInputAmountBlur(e) {
if (!ref.current?.contains(e.relatedTarget)) {
const amount = getAmount();
fireUpdate(amount);
}
onBlur?.(e);
}
return (
<View
style={{
...defaultInputStyle,
padding: 0,
flexDirection: 'row',
flex: 1,
alignItems: 'stretch',
...style,
...(isFocused && {
boxShadow: '0 0 0 1px ' + theme.formInputShadowSelected,
}),
}}
>
<Button
variant="bare"
isDisabled={disabled}
aria-label={`Make ${symbol === '-' ? 'positive' : 'negative'}`}
style={{ padding: '0 7px' }}
onPress={onSwitch}
ref={buttonRef}
>
{symbol === '-' && (
<SvgSubtract style={{ width: 8, height: 8, color: 'inherit' }} />
)}
{symbol === '+' && (
<SvgAdd style={{ width: 8, height: 8, color: 'inherit' }} />
)}
</Button>
<Input
id={id}
inputRef={mergedRef}
inputMode="decimal"
value={value}
disabled={disabled}
style={{
width: '100%',
...inputStyle,
flex: 1,
'&, &:focus, &:hover': {
border: 0,
backgroundColor: 'transparent',
boxShadow: 'none',
color: 'inherit',
},
}}
onFocus={e => {
setIsFocused(true);
onFocus?.(e);
}}
onBlur={e => {
setIsFocused(false);
onInputAmountBlur(e);
}}
onKeyUp={e => {
if (e.key === 'Enter') {
const amount = getAmount();
fireUpdate(amount);
}
}}
onEnter={onEnter}
onChangeValue={onInputTextChange}
/>
</View>
);
}
type BetweenAmountInputProps = {
defaultValue: { num1: number; num2: number };
onChange: (newValue: { num1: number; num2: number }) => void;
};
export function BetweenAmountInput({
defaultValue,
onChange,
}: BetweenAmountInputProps) {
const { t } = useTranslation();
const [num1, setNum1] = useState(defaultValue.num1);
const [num2, setNum2] = useState(defaultValue.num2);
return (
<View style={{ flexDirection: 'row', alignItems: 'center' }}>
<AmountInput
value={num1}
onUpdate={value => {
setNum1(value);
onChange({ num1: value, num2 });
}}
style={{ color: theme.formInputText }}
/>
<View style={{ margin: '0 5px' }}>{t('and')}</View>
<AmountInput
value={num2}
onUpdate={value => {
setNum2(value);
onChange({ num1, num2: value });
}}
style={{ color: theme.formInputText }}
/>
</View>
);
}