feat(ui): replace native inputs with Mantine components in forms and settings

- Updated front-ends to use maintine component
- Update input such ass PasswordInput etc to use mantine

Signed-off-by: Balázs Szücs <bszucs1209@gmail.com>
This commit is contained in:
Balázs Szücs 2025-11-13 09:22:03 +01:00
parent fb3ec99f0a
commit e3486f45d0
5 changed files with 67 additions and 104 deletions

View File

@ -1,5 +1,4 @@
import { useState } from "react"; import { Stack, Text, NumberInput, Select, Divider, Checkbox, Slider } from "@mantine/core";
import { Stack, Text, NumberInput, Select, Divider, Checkbox } from "@mantine/core";
import { useTranslation } from "react-i18next"; import { useTranslation } from "react-i18next";
import { CompressParameters } from "@app/hooks/tools/compress/useCompressParameters"; import { CompressParameters } from "@app/hooks/tools/compress/useCompressParameters";
import ButtonSelector from "@app/components/shared/ButtonSelector"; import ButtonSelector from "@app/components/shared/ButtonSelector";
@ -12,7 +11,6 @@ interface CompressSettingsProps {
const CompressSettings = ({ parameters, onParameterChange, disabled = false }: CompressSettingsProps) => { const CompressSettings = ({ parameters, onParameterChange, disabled = false }: CompressSettingsProps) => {
const { t } = useTranslation(); const { t } = useTranslation();
const [isSliding, setIsSliding] = useState(false);
return ( return (
<Stack gap="md"> <Stack gap="md">
@ -35,50 +33,19 @@ const CompressSettings = ({ parameters, onParameterChange, disabled = false }: C
<Stack gap="sm"> <Stack gap="sm">
<Divider /> <Divider />
<Text size="sm" fw={500}>Compression Level</Text> <Text size="sm" fw={500}>Compression Level</Text>
<div style={{ position: 'relative' }}> <Slider
<input min={1}
type="range" max={9}
min="1" step={1}
max="9" value={parameters.compressionLevel}
step="1" onChange={(value) => onParameterChange('compressionLevel', value)}
value={parameters.compressionLevel} disabled={disabled}
onChange={(e) => onParameterChange('compressionLevel', parseInt(e.target.value))} marks={[
onMouseDown={() => setIsSliding(true)} { value: 1, label: 'Min 1' },
onMouseUp={() => setIsSliding(false)} { value: 9, label: 'Max 9' },
onTouchStart={() => setIsSliding(true)} ]}
onTouchEnd={() => setIsSliding(false)} label={(value) => `${value}`}
disabled={disabled} />
style={{
width: '100%',
height: '6px',
borderRadius: '3px',
background: `linear-gradient(to right, #228be6 0%, #228be6 ${(parameters.compressionLevel - 1) / 8 * 100}%, #e9ecef ${(parameters.compressionLevel - 1) / 8 * 100}%, #e9ecef 100%)`,
outline: 'none',
WebkitAppearance: 'none'
}}
/>
{isSliding && (
<div style={{
position: 'absolute',
top: '-25px',
left: `${(parameters.compressionLevel - 1) / 8 * 100}%`,
transform: 'translateX(-50%)',
background: '#f8f9fa',
border: '1px solid #dee2e6',
borderRadius: '4px',
padding: '2px 6px',
fontSize: '12px',
color: '#228be6',
whiteSpace: 'nowrap'
}}>
{parameters.compressionLevel}
</div>
)}
</div>
<div style={{ display: 'flex', justifyContent: 'space-between', fontSize: '12px', color: '#6c757d' }}>
<span>Min 1</span>
<span>Max 9</span>
</div>
<Text size="xs" c="dimmed" style={{ marginTop: '8px' }}> <Text size="xs" c="dimmed" style={{ marginTop: '8px' }}>
{parameters.compressionLevel <= 3 && "1-3 PDF compression"} {parameters.compressionLevel <= 3 && "1-3 PDF compression"}
{parameters.compressionLevel >= 4 && parameters.compressionLevel <= 6 && "4-6 lite image compression"} {parameters.compressionLevel >= 4 && parameters.compressionLevel <= 6 && "4-6 lite image compression"}

View File

@ -1,5 +1,6 @@
import { useTranslation } from 'react-i18next'; import { useTranslation } from 'react-i18next';
import '@app/routes/authShared/auth.css'; import '@app/routes/authShared/auth.css';
import { TextInput, PasswordInput, Button } from '@mantine/core';
interface EmailPasswordFormProps { interface EmailPasswordFormProps {
email: string email: string
@ -38,49 +39,46 @@ export default function EmailPasswordForm({
<form onSubmit={handleSubmit}> <form onSubmit={handleSubmit}>
<div className="auth-fields"> <div className="auth-fields">
<div className="auth-field"> <div className="auth-field">
<label htmlFor="email" className="auth-label">{t('login.username', 'Username')}</label> <TextInput
<input
id="email" id="email"
label={t('login.username', 'Username')}
type="text" type="text"
name="username" name="username"
autoComplete="username" autoComplete="username"
placeholder={t('login.enterUsername', 'Enter username')} placeholder={t('login.enterUsername', 'Enter username')}
value={email} value={email}
onChange={(e) => setEmail(e.target.value)} onChange={(e) => setEmail(e.target.value)}
className={`auth-input ${fieldErrors.email ? 'auth-input-error' : ''}`} error={fieldErrors.email}
classNames={{ label: 'auth-label' }}
/> />
{fieldErrors.email && (
<div className="auth-field-error">{fieldErrors.email}</div>
)}
</div> </div>
{showPasswordField && ( {showPasswordField && (
<div className="auth-field"> <div className="auth-field">
<label htmlFor="password" className="auth-label">{t('login.password')}</label> <PasswordInput
<input
id="password" id="password"
type="password" label={t('login.password')}
name="current-password" name="current-password"
autoComplete="current-password" autoComplete="current-password"
placeholder={t('login.enterPassword')} placeholder={t('login.enterPassword')}
value={password} value={password}
onChange={(e) => setPassword(e.target.value)} onChange={(e) => setPassword(e.target.value)}
className={`auth-input ${fieldErrors.password ? 'auth-input-error' : ''}`} error={fieldErrors.password}
classNames={{ label: 'auth-label' }}
/> />
{fieldErrors.password && (
<div className="auth-field-error">{fieldErrors.password}</div>
)}
</div> </div>
)} )}
</div> </div>
<button <Button
type="submit" type="submit"
disabled={isSubmitting || !email || (showPasswordField && !password)} disabled={isSubmitting || !email || (showPasswordField && !password)}
className="auth-button" className="auth-button"
fullWidth
loading={isSubmitting}
> >
{submitButtonText} {submitButtonText}
</button> </Button>
</form> </form>
); );
} }

View File

@ -1,3 +1,5 @@
import { Button } from '@mantine/core';
interface NavigationLinkProps { interface NavigationLinkProps {
onClick: () => void onClick: () => void
text: string text: string
@ -7,13 +9,14 @@ interface NavigationLinkProps {
export default function NavigationLink({ onClick, text, isDisabled = false }: NavigationLinkProps) { export default function NavigationLink({ onClick, text, isDisabled = false }: NavigationLinkProps) {
return ( return (
<div className="navigation-link-container"> <div className="navigation-link-container">
<button <Button
onClick={onClick} onClick={onClick}
disabled={isDisabled} disabled={isDisabled}
className="navigation-link-button" className="navigation-link-button"
variant="subtle"
> >
{text} {text}
</button> </Button>
</div> </div>
); );
} }

View File

@ -1,5 +1,6 @@
import { useTranslation } from 'react-i18next'; import { useTranslation } from 'react-i18next';
import { BASE_PATH } from '@app/constants/app'; import { BASE_PATH } from '@app/constants/app';
import { Button } from '@mantine/core';
// OAuth provider configuration // OAuth provider configuration
const oauthProviders = [ const oauthProviders = [
@ -26,14 +27,15 @@ export default function OAuthButtons({ onProviderClick, isSubmitting, layout = '
<div className="oauth-container-icons"> <div className="oauth-container-icons">
{enabledProviders.map((p) => ( {enabledProviders.map((p) => (
<div key={p.id} title={`${t('login.signInWith', 'Sign in with')} ${p.label}`}> <div key={p.id} title={`${t('login.signInWith', 'Sign in with')} ${p.label}`}>
<button <Button
onClick={() => onProviderClick(p.id as any)} onClick={() => onProviderClick(p.id as any)}
disabled={isSubmitting} disabled={isSubmitting}
className="oauth-button-icon" className="oauth-button-icon"
aria-label={`${t('login.signInWith', 'Sign in with')} ${p.label}`} aria-label={`${t('login.signInWith', 'Sign in with')} ${p.label}`}
variant="default"
> >
<img src={`${BASE_PATH}/Login/${p.file}`} alt={p.label} className="oauth-icon-small"/> <img src={`${BASE_PATH}/Login/${p.file}`} alt={p.label} className="oauth-icon-small"/>
</button> </Button>
</div> </div>
))} ))}
</div> </div>
@ -45,14 +47,15 @@ export default function OAuthButtons({ onProviderClick, isSubmitting, layout = '
<div className="oauth-container-grid"> <div className="oauth-container-grid">
{enabledProviders.map((p) => ( {enabledProviders.map((p) => (
<div key={p.id} title={`${t('login.signInWith', 'Sign in with')} ${p.label}`}> <div key={p.id} title={`${t('login.signInWith', 'Sign in with')} ${p.label}`}>
<button <Button
onClick={() => onProviderClick(p.id as any)} onClick={() => onProviderClick(p.id as any)}
disabled={isSubmitting} disabled={isSubmitting}
className="oauth-button-grid" className="oauth-button-grid"
aria-label={`${t('login.signInWith', 'Sign in with')} ${p.label}`} aria-label={`${t('login.signInWith', 'Sign in with')} ${p.label}`}
variant="default"
> >
<img src={`${BASE_PATH}/Login/${p.file}`} alt={p.label} className="oauth-icon-medium"/> <img src={`${BASE_PATH}/Login/${p.file}`} alt={p.label} className="oauth-icon-medium"/>
</button> </Button>
</div> </div>
))} ))}
</div> </div>
@ -62,16 +65,17 @@ export default function OAuthButtons({ onProviderClick, isSubmitting, layout = '
return ( return (
<div className="oauth-container-vertical"> <div className="oauth-container-vertical">
{enabledProviders.map((p) => ( {enabledProviders.map((p) => (
<button <Button
key={p.id} key={p.id}
onClick={() => onProviderClick(p.id as any)} onClick={() => onProviderClick(p.id as any)}
disabled={isSubmitting} disabled={isSubmitting}
className="oauth-button-vertical" className="oauth-button-vertical"
title={p.label} title={p.label}
variant="default"
leftSection={<img src={`${BASE_PATH}/Login/${p.file}`} alt={p.label} className="oauth-icon-tiny" />}
> >
<img src={`${BASE_PATH}/Login/${p.file}`} alt={p.label} className="oauth-icon-tiny" />
{p.label} {p.label}
</button> </Button>
))} ))}
</div> </div>
); );

View File

@ -1,7 +1,7 @@
import { useEffect } from 'react'; import { useEffect } from 'react';
import '@app/routes/authShared/auth.css'; import '@app/routes/authShared/auth.css';
import { useTranslation } from 'react-i18next'; import { useTranslation } from 'react-i18next';
import { Checkbox } from '@mantine/core'; import { Checkbox, TextInput, PasswordInput, Button } from '@mantine/core';
import { SignupFieldErrors } from '@app/routes/signup/SignupFormValidation'; import { SignupFieldErrors } from '@app/routes/signup/SignupFormValidation';
interface SignupFormProps { interface SignupFormProps {
@ -53,57 +53,49 @@ export default function SignupForm({
<div className="auth-fields"> <div className="auth-fields">
{showName && ( {showName && (
<div className="auth-field"> <div className="auth-field">
<label htmlFor="name" className="auth-label">{t('signup.name')}</label> <TextInput
<input
id="name" id="name"
type="text" label={t('signup.name')}
name="name" name="name"
autoComplete="name" autoComplete="name"
placeholder={t('signup.enterName')} placeholder={t('signup.enterName')}
value={name} value={name}
onChange={(e) => setName?.(e.target.value)} onChange={(e) => setName?.(e.target.value)}
className={`auth-input ${fieldErrors.name ? 'auth-input-error' : ''}`} error={fieldErrors.name}
classNames={{ label: 'auth-label' }}
/> />
{fieldErrors.name && (
<div className="auth-field-error">{fieldErrors.name}</div>
)}
</div> </div>
)} )}
<div className="auth-field"> <div className="auth-field">
<label htmlFor="email" className="auth-label">{t('signup.email')}</label> <TextInput
<input
id="email" id="email"
label={t('signup.email')}
type="email" type="email"
name="email" name="email"
autoComplete="email" autoComplete="email"
placeholder={t('signup.enterEmail')} placeholder={t('signup.enterEmail')}
value={email} value={email}
onChange={(e) => setEmail(e.target.value)} onChange={(e) => setEmail(e.target.value)}
onKeyPress={(e) => e.key === 'Enter' && !isSubmitting && onSubmit()} onKeyDown={(e) => e.key === 'Enter' && !isSubmitting && onSubmit()}
className={`auth-input ${fieldErrors.email ? 'auth-input-error' : ''}`} error={fieldErrors.email}
classNames={{ label: 'auth-label' }}
/> />
{fieldErrors.email && (
<div className="auth-field-error">{fieldErrors.email}</div>
)}
</div> </div>
<div className="auth-field"> <div className="auth-field">
<label htmlFor="password" className="auth-label">{t('signup.password')}</label> <PasswordInput
<input
id="password" id="password"
type="password" label={t('signup.password')}
name="new-password" name="new-password"
autoComplete="new-password" autoComplete="new-password"
placeholder={t('signup.enterPassword')} placeholder={t('signup.enterPassword')}
value={password} value={password}
onChange={(e) => setPassword(e.target.value)} onChange={(e) => setPassword(e.target.value)}
onKeyPress={(e) => e.key === 'Enter' && !isSubmitting && onSubmit()} onKeyDown={(e) => e.key === 'Enter' && !isSubmitting && onSubmit()}
className={`auth-input ${fieldErrors.password ? 'auth-input-error' : ''}`} error={fieldErrors.password}
classNames={{ label: 'auth-label' }}
/> />
{fieldErrors.password && (
<div className="auth-field-error">{fieldErrors.password}</div>
)}
</div> </div>
<div <div
@ -112,21 +104,18 @@ export default function SignupForm({
style={{ maxHeight: showConfirm ? 96 : 0, opacity: showConfirm ? 1 : 0 }} style={{ maxHeight: showConfirm ? 96 : 0, opacity: showConfirm ? 1 : 0 }}
> >
<div className="auth-field"> <div className="auth-field">
<label htmlFor="confirmPassword" className="auth-label">{t('signup.confirmPassword')}</label> <PasswordInput
<input
id="confirmPassword" id="confirmPassword"
type="password" label={t('signup.confirmPassword')}
name="new-password" name="new-password"
autoComplete="new-password" autoComplete="new-password"
placeholder={t('signup.confirmPasswordPlaceholder')} placeholder={t('signup.confirmPasswordPlaceholder')}
value={confirmPassword} value={confirmPassword}
onChange={(e) => setConfirmPassword(e.target.value)} onChange={(e) => setConfirmPassword(e.target.value)}
onKeyPress={(e) => e.key === 'Enter' && !isSubmitting && onSubmit()} onKeyDown={(e) => e.key === 'Enter' && !isSubmitting && onSubmit()}
className={`auth-input ${fieldErrors.confirmPassword ? 'auth-input-error' : ''}`} error={fieldErrors.confirmPassword}
classNames={{ label: 'auth-label' }}
/> />
{fieldErrors.confirmPassword && (
<div className="auth-field-error">{fieldErrors.confirmPassword}</div>
)}
</div> </div>
</div> </div>
</div> </div>
@ -152,13 +141,15 @@ export default function SignupForm({
)} )}
{/* Sign Up Button */} {/* Sign Up Button */}
<button <Button
onClick={onSubmit} onClick={onSubmit}
disabled={isSubmitting || !email || !password || !confirmPassword || (showTerms && !agree)} disabled={isSubmitting || !email || !password || !confirmPassword || (showTerms && !agree)}
className="auth-button" className="auth-button"
fullWidth
loading={isSubmitting}
> >
{isSubmitting ? t('signup.creatingAccount') : t('signup.signUp')} {isSubmitting ? t('signup.creatingAccount') : t('signup.signUp')}
</button> </Button>
</> </>
); );
} }