Skip to content

Commit

Permalink
Merge branch 'main' into settings-tabs
Browse files Browse the repository at this point in the history
  • Loading branch information
sidvishnoi committed Oct 16, 2024
2 parents 008da6a + bff27af commit 70e17ba
Show file tree
Hide file tree
Showing 10 changed files with 284 additions and 250 deletions.
1 change: 1 addition & 0 deletions src/assets/images/bg-tile.svg
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
103 changes: 22 additions & 81 deletions src/popup/components/ConnectWalletForm.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -5,12 +5,8 @@ import { Switch } from '@/popup/components/ui/Switch';
import { Code } from '@/popup/components/ui/Code';
import { ErrorMessage } from '@/popup/components/ErrorMessage';
import { LoadingSpinner } from '@/popup/components/LoadingSpinner';
import {
charIsNumber,
formatNumber,
getCurrencySymbol,
toWalletAddressUrl,
} from '@/popup/lib/utils';
import { InputAmount, validateAmount } from '@/popup/components/InputAmount';
import { toWalletAddressUrl } from '@/popup/lib/utils';
import { useTranslation } from '@/popup/lib/context';
import {
cn,
Expand Down Expand Up @@ -107,11 +103,6 @@ export const ConnectWalletForm = ({
state?.status?.startsWith('connecting') || false,
);

const [currencySymbol, setCurrencySymbol] = React.useState<{
symbol: string;
scale: number;
}>({ symbol: '$', scale: 2 });

const getWalletInformation = React.useCallback(
async (walletAddressUrl: string): Promise<boolean> => {
setErrors((prev) => ({ ...prev, walletAddressUrl: null }));
Expand Down Expand Up @@ -153,25 +144,19 @@ export const ConnectWalletForm = ({
);

const handleAmountChange = React.useCallback(
(value: string, input: HTMLInputElement) => {
const error = validateAmount(value, currencySymbol.symbol);
setErrors((prev) => ({ ...prev, amount: toErrorInfo(error) }));

const amountValue = formatNumber(+value, currencySymbol.scale);
if (!error) {
setAmount(amountValue);
input.value = amountValue;
}
saveValue('amount', error ? value : amountValue);
(amountValue: string) => {
setErrors((prev) => ({ ...prev, amount: null }));
setAmount(amountValue);
saveValue('amount', amountValue);
},
[saveValue, currencySymbol, toErrorInfo],
[saveValue],
);

const handleSubmit = async (ev?: React.FormEvent<HTMLFormElement>) => {
ev?.preventDefault();

const errWalletAddressUrl = validateWalletAddressUrl(walletAddressUrl);
const errAmount = validateAmount(amount, currencySymbol.symbol);
const errAmount = validateAmount(amount, walletAddressInfo!);
if (errAmount || errWalletAddressUrl) {
setErrors((prev) => ({
...prev,
Expand Down Expand Up @@ -221,14 +206,6 @@ export const ConnectWalletForm = ({
}
};

React.useEffect(() => {
if (!walletAddressInfo) return;
setCurrencySymbol({
symbol: getCurrencySymbol(walletAddressInfo.assetCode),
scale: walletAddressInfo.assetScale,
});
}, [walletAddressInfo]);

React.useEffect(() => {
if (defaultValues.walletAddressUrl) {
handleWalletAddressUrlChange(defaultValues.walletAddressUrl);
Expand Down Expand Up @@ -332,27 +309,23 @@ export const ConnectWalletForm = ({
{t('connectWallet_labelGroup_amount')}
</legend>
<div className="flex items-center gap-6">
<Input
<InputAmount
id="connectAmount"
type="text"
inputMode="numeric"
aria-label={t('connectWallet_label_amount')}
placeholder="5.00"
className="max-w-32"
defaultValue={amount}
label={t('connectWallet_label_amount')}
labelHidden={true}
amount={amount}
walletAddress={
walletAddressInfo || { assetCode: 'USD', assetScale: 2 }
}
errorMessage={errors.amount?.message}
errorHidden={true}
readOnly={!walletAddressInfo?.assetCode || isSubmitting}
addOn={<span className="text-weak">{currencySymbol.symbol}</span>}
aria-invalid={!!errors.amount}
aria-describedby={errors.amount?.message}
required={true}
onKeyDown={allowOnlyNumericInput}
onBlur={(ev) => {
const value = ev.currentTarget.value;
if (value === amount && !ev.currentTarget.required) {
return;
}
handleAmountChange(value, ev.currentTarget);
onError={(err) => {
setErrors((prev) => ({ ...prev, amount: toErrorInfo(err) }));
}}
onChange={handleAmountChange}
className="max-w-32"
placeholder="5.00"
/>

<Switch
Expand Down Expand Up @@ -547,35 +520,3 @@ function validateWalletAddressUrl(value: string): null | ErrorWithKeyLike {

return null;
}

function validateAmount(
value: string,
currencySymbol: string,
): null | ErrorWithKeyLike {
if (!value) {
return errorWithKey('connectWallet_error_amountRequired');
}
const val = Number(value);
if (Number.isNaN(val)) {
return errorWithKey('connectWallet_error_amountInvalidNumber', [
`${currencySymbol}${value}`,
]);
}
if (val <= 0) {
return errorWithKey('connectWallet_error_amountMinimum');
}
return null;
}

function allowOnlyNumericInput(ev: React.KeyboardEvent<HTMLInputElement>) {
if (
(!charIsNumber(ev.key) &&
ev.key !== 'Backspace' &&
ev.key !== 'Delete' &&
ev.key !== 'Enter' &&
ev.key !== 'Tab') ||
(ev.key === '.' && ev.currentTarget.value.includes('.'))
) {
ev.preventDefault();
}
}
2 changes: 1 addition & 1 deletion src/popup/components/Icons.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -72,7 +72,7 @@ export const Settings = (props: React.SVGProps<SVGSVGElement>) => {
<g mask="url(#mask0_140_3136)">
<path
d="M10.0002 22.5L9.6002 19.3C9.38353 19.2167 9.17953 19.1167 8.9882 19C8.7962 18.8833 8.60853 18.7583 8.4252 18.625L5.4502 19.875L2.7002 15.125L5.2752 13.175C5.25853 13.0583 5.2502 12.9457 5.2502 12.837C5.2502 12.729 5.2502 12.6167 5.2502 12.5C5.2502 12.3833 5.2502 12.2707 5.2502 12.162C5.2502 12.054 5.25853 11.9417 5.2752 11.825L2.7002 9.875L5.4502 5.125L8.4252 6.375C8.60853 6.24167 8.8002 6.11667 9.0002 6C9.2002 5.88333 9.4002 5.78333 9.6002 5.7L10.0002 2.5H15.5002L15.9002 5.7C16.1169 5.78333 16.3212 5.88333 16.5132 6C16.7045 6.11667 16.8919 6.24167 17.0752 6.375L20.0502 5.125L22.8002 9.875L20.2252 11.825C20.2419 11.9417 20.2502 12.054 20.2502 12.162C20.2502 12.2707 20.2502 12.3833 20.2502 12.5C20.2502 12.6167 20.2502 12.729 20.2502 12.837C20.2502 12.9457 20.2335 13.0583 20.2002 13.175L22.7752 15.125L20.0252 19.875L17.0752 18.625C16.8919 18.7583 16.7002 18.8833 16.5002 19C16.3002 19.1167 16.1002 19.2167 15.9002 19.3L15.5002 22.5H10.0002ZM12.8002 16C13.7669 16 14.5919 15.6583 15.2752 14.975C15.9585 14.2917 16.3002 13.4667 16.3002 12.5C16.3002 11.5333 15.9585 10.7083 15.2752 10.025C14.5919 9.34167 13.7669 9 12.8002 9C11.8169 9 10.9875 9.34167 10.3122 10.025C9.63753 10.7083 9.3002 11.5333 9.3002 12.5C9.3002 13.4667 9.63753 14.2917 10.3122 14.975C10.9875 15.6583 11.8169 16 12.8002 16ZM12.8002 14C12.3835 14 12.0295 13.854 11.7382 13.562C11.4462 13.2707 11.3002 12.9167 11.3002 12.5C11.3002 12.0833 11.4462 11.7293 11.7382 11.438C12.0295 11.146 12.3835 11 12.8002 11C13.2169 11 13.5712 11.146 13.8632 11.438C14.1545 11.7293 14.3002 12.0833 14.3002 12.5C14.3002 12.9167 14.1545 13.2707 13.8632 13.562C13.5712 13.854 13.2169 14 12.8002 14ZM11.7502 20.5H13.7252L14.0752 17.85C14.5919 17.7167 15.0712 17.5207 15.5132 17.262C15.9545 17.004 16.3585 16.6917 16.7252 16.325L19.2002 17.35L20.1752 15.65L18.0252 14.025C18.1085 13.7917 18.1669 13.5457 18.2002 13.287C18.2335 13.029 18.2502 12.7667 18.2502 12.5C18.2502 12.2333 18.2335 11.9707 18.2002 11.712C18.1669 11.454 18.1085 11.2083 18.0252 10.975L20.1752 9.35L19.2002 7.65L16.7252 8.7C16.3585 8.31667 15.9545 7.99567 15.5132 7.737C15.0712 7.479 14.5919 7.28333 14.0752 7.15L13.7502 4.5H11.7752L11.4252 7.15C10.9085 7.28333 10.4295 7.479 9.9882 7.737C9.5462 7.99567 9.14186 8.30833 8.7752 8.675L6.3002 7.65L5.3252 9.35L7.47519 10.95C7.39186 11.2 7.33353 11.45 7.3002 11.7C7.26686 11.95 7.2502 12.2167 7.2502 12.5C7.2502 12.7667 7.26686 13.025 7.3002 13.275C7.33353 13.525 7.39186 13.775 7.47519 14.025L5.3252 15.65L6.3002 17.35L8.7752 16.3C9.14186 16.6833 9.5462 17.004 9.9882 17.262C10.4295 17.5207 10.9085 17.7167 11.4252 17.85L11.7502 20.5Z"
fill="#475569"
fill="currentColor"
/>
</g>
</svg>
Expand Down
113 changes: 113 additions & 0 deletions src/popup/components/InputAmount.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,113 @@
import React from 'react';
import { Input } from './ui/Input';
import type { WalletAddress } from '@interledger/open-payments';
import { charIsNumber, formatNumber, getCurrencySymbol } from '../lib/utils';
import {
errorWithKey,
ErrorWithKeyLike,
formatCurrency,
} from '@/shared/helpers';

interface Props {
id: string;
label: string | React.ReactNode;
walletAddress: Pick<WalletAddress, 'assetCode' | 'assetScale'>;
amount: string;
onChange: (amount: string, inputEl: HTMLInputElement) => void;
onError: (error: ErrorWithKeyLike) => void;
className?: string;
placeholder?: string;
errorMessage?: string;
readOnly?: boolean;
labelHidden?: boolean;
errorHidden?: boolean;
min?: number;
max?: number;
controls?: boolean;
}

export const InputAmount = ({
label,
id,
walletAddress,
amount,
className,
placeholder,
errorMessage,
onChange,
onError,
labelHidden,
errorHidden,
min = 0,
max,
readOnly,
}: Props) => {
const currencySymbol = getCurrencySymbol(walletAddress.assetCode);
return (
<Input
id={id}
type="text"
inputMode="numeric"
label={labelHidden ? null : label}
aria-label={labelHidden && typeof label === 'string' ? label : undefined}
placeholder={placeholder}
className={className}
defaultValue={amount}
readOnly={readOnly}
addOn={<span className="text-weak">{currencySymbol}</span>}
errorMessage={errorHidden ? '' : errorMessage}
aria-invalid={errorHidden ? !!errorMessage : false}
required={true}
onKeyDown={allowOnlyNumericInput}
onBlur={(ev) => {
const input = ev.currentTarget;
const value = input.value;
if (value === amount && !input.required) {
return;
}
const error = validateAmount(value, walletAddress, min, max);
if (error) {
onError(error);
} else {
const amountValue = formatNumber(+value, walletAddress.assetScale);
input.value = amountValue;
onChange(amountValue, input);
}
}}
/>
);
};

export function validateAmount(
value: string,
walletAddress: Pick<WalletAddress, 'assetCode' | 'assetScale'>,
min: number = 0,
_max?: number,
): null | ErrorWithKeyLike {
if (!value) {
return errorWithKey('connectWallet_error_amountRequired');
}
const val = Number(value);
if (Number.isNaN(val)) {
return errorWithKey('connectWallet_error_amountInvalidNumber');
}
if (val <= min) {
return errorWithKey('connectWallet_error_amountMinimum', [
formatCurrency(min, walletAddress.assetCode, walletAddress.assetScale),
]);
}
return null;
}

function allowOnlyNumericInput(ev: React.KeyboardEvent<HTMLInputElement>) {
if (
(!charIsNumber(ev.key) &&
ev.key !== 'Backspace' &&
ev.key !== 'Delete' &&
ev.key !== 'Enter' &&
ev.key !== 'Tab') ||
(ev.key === '.' && ev.currentTarget.value.includes('.'))
) {
ev.preventDefault();
}
}
Loading

0 comments on commit 70e17ba

Please sign in to comment.