Files
metabuilder/fakemui/react/components/inputs/InputBase.tsx
johndoe6345789 54a819ed71 chore(fakemui): reorganize folder structure by implementation type
ORGANIZED INTO 4 MAIN CATEGORIES:
- react/              React TypeScript components (145 components + Python bindings)
- qml/               QML desktop components (104+ QML components)
- python/            Python package implementations
- legacy/            Utilities, contexts, and migration-in-progress code

SUPPORTING FOLDERS (kept as-is):
- icons/             421 SVG icons
- theming/           Material Design 3 theme system
- styles/            SCSS modules and utilities
- scss/              SCSS preprocessor files
- docs/              Documentation files

STRUCTURE IMPROVEMENTS:
 All code preserved (nothing deleted)
 Clear separation by implementation type
 Better navigation and discoverability
 Easy to find what you need
 Professional organization

DOCUMENTATION:
- Added STRUCTURE.md explaining the new layout
- Updated folder organization with clear purpose
- Maintained all original functionality

All files reorganized while keeping full functionality intact.

Co-Authored-By: Claude Haiku 4.5 <noreply@anthropic.com>
2026-01-23 17:09:48 +00:00

190 lines
5.1 KiB
TypeScript

import React, { forwardRef } from 'react'
import { classNames } from '../utils/classNames'
export interface ButtonBaseProps extends React.HTMLAttributes<HTMLElement> {
component?: React.ElementType
children?: React.ReactNode
disabled?: boolean
onClick?: React.MouseEventHandler
onFocus?: React.FocusEventHandler
onBlur?: React.FocusEventHandler
tabIndex?: number
type?: 'button' | 'submit' | 'reset'
}
export const ButtonBase = forwardRef<HTMLElement, ButtonBaseProps>(function ButtonBase(
{
component: Component = 'button',
className,
disabled = false,
children,
onClick,
onFocus,
onBlur,
tabIndex = 0,
type = 'button',
...props
},
ref
) {
return (
<Component
ref={ref}
className={classNames('fakemui-button-base', className, {
'fakemui-button-base-disabled': disabled,
})}
disabled={disabled}
onClick={disabled ? undefined : onClick}
onFocus={onFocus}
onBlur={onBlur}
tabIndex={disabled ? -1 : tabIndex}
type={Component === 'button' ? type : undefined}
{...props}
>
{children}
</Component>
)
})
export interface InputBaseProps extends Omit<React.HTMLAttributes<HTMLDivElement>, 'onChange' | 'defaultValue' | 'onFocus' | 'onBlur'> {
disabled?: boolean
error?: boolean
fullWidth?: boolean
multiline?: boolean
rows?: number
minRows?: number
maxRows?: number
startAdornment?: React.ReactNode
endAdornment?: React.ReactNode
inputComponent?: React.ElementType
inputProps?: React.InputHTMLAttributes<HTMLInputElement> | React.TextareaHTMLAttributes<HTMLTextAreaElement>
inputRef?: React.Ref<HTMLInputElement | HTMLTextAreaElement>
value?: string | number
defaultValue?: string | number
onChange?: React.ChangeEventHandler<HTMLInputElement | HTMLTextAreaElement>
onFocus?: React.FocusEventHandler<HTMLInputElement | HTMLTextAreaElement>
onBlur?: React.FocusEventHandler<HTMLInputElement | HTMLTextAreaElement>
placeholder?: string
readOnly?: boolean
required?: boolean
type?: string
}
export const InputBase = forwardRef<HTMLDivElement, InputBaseProps>(function InputBase(
{
className,
disabled = false,
error = false,
fullWidth = false,
multiline = false,
rows,
minRows,
maxRows,
startAdornment,
endAdornment,
inputComponent: InputComponent = 'input',
inputProps = {},
inputRef,
value,
defaultValue,
onChange,
onFocus,
onBlur,
placeholder,
readOnly = false,
required = false,
type = 'text',
...props
},
ref
) {
const inputElement = multiline ? (
<textarea
ref={inputRef as React.Ref<HTMLTextAreaElement>}
className="fakemui-input-base-input"
disabled={disabled}
readOnly={readOnly}
required={required}
value={value}
defaultValue={defaultValue}
onChange={onChange as React.ChangeEventHandler<HTMLTextAreaElement>}
onFocus={onFocus as React.FocusEventHandler<HTMLTextAreaElement>}
onBlur={onBlur as React.FocusEventHandler<HTMLTextAreaElement>}
placeholder={placeholder}
rows={rows}
{...(inputProps as React.TextareaHTMLAttributes<HTMLTextAreaElement>)}
/>
) : (
<InputComponent
ref={inputRef as React.Ref<HTMLInputElement>}
className="fakemui-input-base-input"
type={type}
disabled={disabled}
readOnly={readOnly}
required={required}
value={value}
defaultValue={defaultValue}
onChange={onChange as React.ChangeEventHandler<HTMLInputElement>}
onFocus={onFocus as React.FocusEventHandler<HTMLInputElement>}
onBlur={onBlur as React.FocusEventHandler<HTMLInputElement>}
placeholder={placeholder}
{...(inputProps as React.InputHTMLAttributes<HTMLInputElement>)}
/>
)
return (
<div
ref={ref}
className={classNames('fakemui-input-base', className, {
'fakemui-input-base-disabled': disabled,
'fakemui-input-base-error': error,
'fakemui-input-base-fullwidth': fullWidth,
'fakemui-input-base-multiline': multiline,
})}
{...props}
>
{startAdornment && (
<span className="fakemui-input-base-adornment-start">{startAdornment}</span>
)}
{inputElement}
{endAdornment && (
<span className="fakemui-input-base-adornment-end">{endAdornment}</span>
)}
</div>
)
})
export interface FilledInputProps extends InputBaseProps {}
export const FilledInput = forwardRef<HTMLDivElement, FilledInputProps>(function FilledInput(props, ref) {
return (
<InputBase
ref={ref}
{...props}
className={classNames('fakemui-filled-input', props.className)}
/>
)
})
export interface OutlinedInputProps extends InputBaseProps {
label?: React.ReactNode
notched?: boolean
}
export const OutlinedInput = forwardRef<HTMLDivElement, OutlinedInputProps>(function OutlinedInput(
{ label, notched, ...props },
ref
) {
return (
<InputBase
ref={ref}
{...props}
className={classNames('fakemui-outlined-input', props.className, {
'fakemui-outlined-input-notched': notched,
})}
/>
)
})
export default InputBase