mirror of
https://github.com/emsesp/EMS-ESP32.git
synced 2025-12-08 16:59:50 +03:00
258 lines
7.3 KiB
TypeScript
258 lines
7.3 KiB
TypeScript
import { useCallback, useMemo, useState } from 'react';
|
|
|
|
import CancelIcon from '@mui/icons-material/Cancel';
|
|
import WarningIcon from '@mui/icons-material/Warning';
|
|
import { Button, Checkbox, MenuItem } from '@mui/material';
|
|
|
|
import * as APApi from 'api/ap';
|
|
|
|
import type { ValidateFieldsError } from 'async-validator';
|
|
import {
|
|
BlockFormControlLabel,
|
|
BlockNavigation,
|
|
ButtonRow,
|
|
FormLoader,
|
|
SectionContent,
|
|
ValidatedPasswordField,
|
|
ValidatedTextField,
|
|
useLayoutTitle
|
|
} from 'components';
|
|
import { useI18nContext } from 'i18n/i18n-react';
|
|
import type { APSettingsType } from 'types';
|
|
import { APProvisionMode } from 'types';
|
|
import { numberValue, updateValueDirty, useRest } from 'utils';
|
|
import { createAPSettingsValidator, validate } from 'validators';
|
|
|
|
export const isAPEnabled = ({ provision_mode }: APSettingsType) =>
|
|
provision_mode === APProvisionMode.AP_MODE_ALWAYS ||
|
|
provision_mode === APProvisionMode.AP_MODE_DISCONNECTED;
|
|
|
|
// Efficient range function without recursion
|
|
const createRange = (start: number, end: number): number[] => {
|
|
const result: number[] = [];
|
|
for (let i = start; i <= end; i++) {
|
|
result.push(i);
|
|
}
|
|
return result;
|
|
};
|
|
|
|
// Pre-computed ranges for better performance
|
|
const CHANNEL_RANGE = createRange(1, 14);
|
|
const MAX_CLIENTS_RANGE = createRange(1, 9);
|
|
|
|
const APSettings = () => {
|
|
const {
|
|
loadData,
|
|
saving,
|
|
data,
|
|
updateDataValue,
|
|
origData,
|
|
dirtyFlags,
|
|
setDirtyFlags,
|
|
blocker,
|
|
saveData,
|
|
errorMessage
|
|
} = useRest<APSettingsType>({
|
|
read: APApi.readAPSettings,
|
|
update: APApi.updateAPSettings
|
|
});
|
|
|
|
const { LL } = useI18nContext();
|
|
|
|
useLayoutTitle(LL.ACCESS_POINT(0));
|
|
|
|
const [fieldErrors, setFieldErrors] = useState<ValidateFieldsError>();
|
|
|
|
const updateFormValue = useMemo(
|
|
() =>
|
|
updateValueDirty(
|
|
origData as unknown as Record<string, unknown>,
|
|
dirtyFlags,
|
|
setDirtyFlags,
|
|
updateDataValue as (value: unknown) => void
|
|
),
|
|
[origData, dirtyFlags, setDirtyFlags, updateDataValue]
|
|
);
|
|
|
|
// Memoize AP enabled state
|
|
const apEnabled = useMemo(() => (data ? isAPEnabled(data) : false), [data]);
|
|
|
|
// Memoize validation and submit handler
|
|
const validateAndSubmit = useCallback(async () => {
|
|
if (!data) return;
|
|
|
|
try {
|
|
setFieldErrors(undefined);
|
|
await validate(createAPSettingsValidator(data), data);
|
|
await saveData();
|
|
} catch (error) {
|
|
setFieldErrors(error as ValidateFieldsError);
|
|
}
|
|
}, [data, saveData]);
|
|
|
|
const content = () => {
|
|
if (!data) {
|
|
return <FormLoader onRetry={loadData} errorMessage={errorMessage || ''} />;
|
|
}
|
|
|
|
return (
|
|
<>
|
|
<ValidatedTextField
|
|
fieldErrors={fieldErrors || {}}
|
|
name="provision_mode"
|
|
label={LL.AP_PROVIDE() + '...'}
|
|
value={data.provision_mode}
|
|
fullWidth
|
|
select
|
|
variant="outlined"
|
|
onChange={updateFormValue}
|
|
margin="normal"
|
|
>
|
|
<MenuItem value={APProvisionMode.AP_MODE_ALWAYS}>
|
|
{LL.AP_PROVIDE_TEXT_1()}
|
|
</MenuItem>
|
|
<MenuItem value={APProvisionMode.AP_MODE_DISCONNECTED}>
|
|
{LL.AP_PROVIDE_TEXT_2()}
|
|
</MenuItem>
|
|
<MenuItem value={APProvisionMode.AP_NEVER}>
|
|
{LL.AP_PROVIDE_TEXT_3()}
|
|
</MenuItem>
|
|
</ValidatedTextField>
|
|
{apEnabled && (
|
|
<>
|
|
<ValidatedTextField
|
|
fieldErrors={fieldErrors || {}}
|
|
name="ssid"
|
|
label={LL.ACCESS_POINT(2) + ' SSID'}
|
|
fullWidth
|
|
variant="outlined"
|
|
value={data.ssid}
|
|
onChange={updateFormValue}
|
|
margin="normal"
|
|
/>
|
|
<ValidatedPasswordField
|
|
fieldErrors={fieldErrors || {}}
|
|
name="password"
|
|
label={LL.ACCESS_POINT(2) + ' ' + LL.PASSWORD()}
|
|
fullWidth
|
|
variant="outlined"
|
|
value={data.password}
|
|
onChange={updateFormValue}
|
|
margin="normal"
|
|
/>
|
|
<ValidatedTextField
|
|
fieldErrors={fieldErrors || {}}
|
|
name="channel"
|
|
label={LL.AP_PREFERRED_CHANNEL()}
|
|
value={numberValue(data.channel)}
|
|
fullWidth
|
|
select
|
|
type="number"
|
|
variant="outlined"
|
|
onChange={updateFormValue}
|
|
margin="normal"
|
|
>
|
|
{CHANNEL_RANGE.map((i) => (
|
|
<MenuItem key={i} value={i}>
|
|
{i}
|
|
</MenuItem>
|
|
))}
|
|
</ValidatedTextField>
|
|
<BlockFormControlLabel
|
|
control={
|
|
<Checkbox
|
|
name="ssid_hidden"
|
|
checked={data.ssid_hidden}
|
|
onChange={updateFormValue}
|
|
/>
|
|
}
|
|
label={LL.AP_HIDE_SSID()}
|
|
/>
|
|
<ValidatedTextField
|
|
fieldErrors={fieldErrors || {}}
|
|
name="max_clients"
|
|
label={LL.AP_MAX_CLIENTS()}
|
|
value={numberValue(data.max_clients)}
|
|
fullWidth
|
|
select
|
|
type="number"
|
|
variant="outlined"
|
|
onChange={updateFormValue}
|
|
margin="normal"
|
|
>
|
|
{MAX_CLIENTS_RANGE.map((i) => (
|
|
<MenuItem key={i} value={i}>
|
|
{i}
|
|
</MenuItem>
|
|
))}
|
|
</ValidatedTextField>
|
|
<ValidatedTextField
|
|
fieldErrors={fieldErrors || {}}
|
|
name="local_ip"
|
|
label={LL.AP_LOCAL_IP()}
|
|
fullWidth
|
|
variant="outlined"
|
|
value={data.local_ip}
|
|
onChange={updateFormValue}
|
|
margin="normal"
|
|
/>
|
|
<ValidatedTextField
|
|
fieldErrors={fieldErrors || {}}
|
|
name="gateway_ip"
|
|
label={LL.NETWORK_GATEWAY()}
|
|
fullWidth
|
|
variant="outlined"
|
|
value={data.gateway_ip}
|
|
onChange={updateFormValue}
|
|
margin="normal"
|
|
/>
|
|
<ValidatedTextField
|
|
fieldErrors={fieldErrors || {}}
|
|
name="subnet_mask"
|
|
label={LL.NETWORK_SUBNET()}
|
|
fullWidth
|
|
variant="outlined"
|
|
value={data.subnet_mask}
|
|
onChange={updateFormValue}
|
|
margin="normal"
|
|
/>
|
|
</>
|
|
)}
|
|
{dirtyFlags && dirtyFlags.length !== 0 && (
|
|
<ButtonRow>
|
|
<Button
|
|
startIcon={<CancelIcon />}
|
|
disabled={saving}
|
|
variant="outlined"
|
|
color="secondary"
|
|
type="submit"
|
|
onClick={loadData}
|
|
>
|
|
{LL.CANCEL()}
|
|
</Button>
|
|
<Button
|
|
startIcon={<WarningIcon color="warning" />}
|
|
disabled={saving}
|
|
variant="contained"
|
|
color="info"
|
|
type="submit"
|
|
onClick={validateAndSubmit}
|
|
>
|
|
{LL.APPLY_CHANGES(dirtyFlags.length)}
|
|
</Button>
|
|
</ButtonRow>
|
|
)}
|
|
</>
|
|
);
|
|
};
|
|
|
|
return (
|
|
<SectionContent>
|
|
{blocker ? <BlockNavigation blocker={blocker} /> : null}
|
|
{content()}
|
|
</SectionContent>
|
|
);
|
|
};
|
|
|
|
export default APSettings;
|