64 lines
1.7 KiB
TypeScript
64 lines
1.7 KiB
TypeScript
![]() |
// Copyright 2025 Signal Messenger, LLC
|
||
|
// SPDX-License-Identifier: AGPL-3.0-only
|
||
|
import type { FormEvent } from 'react';
|
||
|
import React, { memo, useCallback, useRef } from 'react';
|
||
|
import { CC_NUMBER_FORMATTER, useInputMask } from '../../../hooks/useInputMask';
|
||
|
import type { LocalizerType } from '../../../types/I18N';
|
||
|
import { CardNumberError } from '../../../types/DonationsCardForm';
|
||
|
import { missingCaseError } from '../../../util/missingCaseError';
|
||
|
|
||
|
export function getCardNumberErrorMessage(
|
||
|
_i18n: LocalizerType,
|
||
|
error: CardNumberError
|
||
|
): string {
|
||
|
switch (error) {
|
||
|
case CardNumberError.EMPTY:
|
||
|
return 'EMPTY';
|
||
|
case CardNumberError.INVALID_CHARS:
|
||
|
case CardNumberError.INVALID_OR_INCOMPLETE_NUMBER:
|
||
|
case CardNumberError.INVALID_NUMBER:
|
||
|
return 'INVALID';
|
||
|
default:
|
||
|
throw missingCaseError(error);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
export type DonateInputCardNumberProps = Readonly<{
|
||
|
id: string;
|
||
|
value: string;
|
||
|
onValueChange: (newValue: string) => void;
|
||
|
maxInputLength: number;
|
||
|
onBlur?: () => void;
|
||
|
}>;
|
||
|
|
||
|
export const DonateInputCardNumber = memo(function DonateInputCardNumber(
|
||
|
props: DonateInputCardNumberProps
|
||
|
) {
|
||
|
const { onValueChange } = props;
|
||
|
const inputRef = useRef<HTMLInputElement>(null);
|
||
|
|
||
|
useInputMask(inputRef, CC_NUMBER_FORMATTER);
|
||
|
|
||
|
const handleInput = useCallback(
|
||
|
(event: FormEvent<HTMLInputElement>) => {
|
||
|
onValueChange(event.currentTarget.value);
|
||
|
},
|
||
|
[onValueChange]
|
||
|
);
|
||
|
|
||
|
return (
|
||
|
<input
|
||
|
ref={inputRef}
|
||
|
id={props.id}
|
||
|
placeholder="0000 0000 0000 0000"
|
||
|
type="text"
|
||
|
inputMode="numeric"
|
||
|
autoComplete="cc-number"
|
||
|
maxLength={props.maxInputLength}
|
||
|
value={props.value}
|
||
|
onInput={handleInput}
|
||
|
onBlur={props.onBlur}
|
||
|
/>
|
||
|
);
|
||
|
});
|