feat: Button and Input primitives
This commit is contained in:
47
src/design-system/primitives/Button/Button.module.css
Normal file
47
src/design-system/primitives/Button/Button.module.css
Normal file
@@ -0,0 +1,47 @@
|
|||||||
|
.btn {
|
||||||
|
display: inline-flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
gap: 6px;
|
||||||
|
font-family: var(--font-body);
|
||||||
|
font-weight: 500;
|
||||||
|
border-radius: var(--radius-sm);
|
||||||
|
cursor: pointer;
|
||||||
|
transition: all 0.15s;
|
||||||
|
white-space: nowrap;
|
||||||
|
}
|
||||||
|
|
||||||
|
.btn:disabled { opacity: 0.6; cursor: not-allowed; }
|
||||||
|
|
||||||
|
.sm { padding: 4px 10px; font-size: 11px; }
|
||||||
|
.md { padding: 6px 14px; font-size: 12px; }
|
||||||
|
|
||||||
|
.primary {
|
||||||
|
background: var(--amber);
|
||||||
|
color: white;
|
||||||
|
border: none;
|
||||||
|
}
|
||||||
|
.primary:hover:not(:disabled) { background: var(--amber-deep); }
|
||||||
|
|
||||||
|
.secondary {
|
||||||
|
background: none;
|
||||||
|
border: 1px solid var(--border);
|
||||||
|
color: var(--text-secondary);
|
||||||
|
}
|
||||||
|
.secondary:hover:not(:disabled) { border-color: var(--text-faint); color: var(--text-primary); }
|
||||||
|
|
||||||
|
.danger {
|
||||||
|
background: none;
|
||||||
|
border: 1px solid var(--error-border);
|
||||||
|
color: var(--error);
|
||||||
|
}
|
||||||
|
.danger:hover:not(:disabled) { background: var(--error-bg); }
|
||||||
|
|
||||||
|
.ghost {
|
||||||
|
background: none;
|
||||||
|
border: none;
|
||||||
|
color: var(--text-secondary);
|
||||||
|
}
|
||||||
|
.ghost:hover:not(:disabled) { background: var(--bg-hover); color: var(--text-primary); }
|
||||||
|
|
||||||
|
.hiddenText { visibility: hidden; }
|
||||||
29
src/design-system/primitives/Button/Button.test.tsx
Normal file
29
src/design-system/primitives/Button/Button.test.tsx
Normal file
@@ -0,0 +1,29 @@
|
|||||||
|
import { describe, it, expect, vi } from 'vitest'
|
||||||
|
import { render, screen } from '@testing-library/react'
|
||||||
|
import userEvent from '@testing-library/user-event'
|
||||||
|
import { Button } from './Button'
|
||||||
|
|
||||||
|
describe('Button', () => {
|
||||||
|
it('renders children text', () => {
|
||||||
|
render(<Button>Click me</Button>)
|
||||||
|
expect(screen.getByRole('button', { name: 'Click me' })).toBeInTheDocument()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('fires onClick', async () => {
|
||||||
|
const onClick = vi.fn()
|
||||||
|
const user = userEvent.setup()
|
||||||
|
render(<Button onClick={onClick}>Click</Button>)
|
||||||
|
await user.click(screen.getByRole('button'))
|
||||||
|
expect(onClick).toHaveBeenCalledOnce()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('shows spinner when loading', () => {
|
||||||
|
render(<Button loading>Save</Button>)
|
||||||
|
expect(screen.getByRole('status')).toBeInTheDocument()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('is disabled when loading', () => {
|
||||||
|
render(<Button loading>Save</Button>)
|
||||||
|
expect(screen.getByRole('button')).toBeDisabled()
|
||||||
|
})
|
||||||
|
})
|
||||||
31
src/design-system/primitives/Button/Button.tsx
Normal file
31
src/design-system/primitives/Button/Button.tsx
Normal file
@@ -0,0 +1,31 @@
|
|||||||
|
import styles from './Button.module.css'
|
||||||
|
import { Spinner } from '../Spinner/Spinner'
|
||||||
|
import type { ButtonHTMLAttributes, ReactNode } from 'react'
|
||||||
|
|
||||||
|
interface ButtonProps extends ButtonHTMLAttributes<HTMLButtonElement> {
|
||||||
|
variant?: 'primary' | 'secondary' | 'danger' | 'ghost'
|
||||||
|
size?: 'sm' | 'md'
|
||||||
|
loading?: boolean
|
||||||
|
children: ReactNode
|
||||||
|
}
|
||||||
|
|
||||||
|
export function Button({
|
||||||
|
variant = 'secondary',
|
||||||
|
size = 'md',
|
||||||
|
loading = false,
|
||||||
|
children,
|
||||||
|
className,
|
||||||
|
disabled,
|
||||||
|
...rest
|
||||||
|
}: ButtonProps) {
|
||||||
|
return (
|
||||||
|
<button
|
||||||
|
className={`${styles.btn} ${styles[variant]} ${styles[size]} ${className ?? ''}`}
|
||||||
|
disabled={disabled || loading}
|
||||||
|
{...rest}
|
||||||
|
>
|
||||||
|
{loading && <Spinner size="sm" />}
|
||||||
|
<span className={loading ? styles.hiddenText : ''}>{children}</span>
|
||||||
|
</button>
|
||||||
|
)
|
||||||
|
}
|
||||||
29
src/design-system/primitives/Input/Input.module.css
Normal file
29
src/design-system/primitives/Input/Input.module.css
Normal file
@@ -0,0 +1,29 @@
|
|||||||
|
.wrap { position: relative; }
|
||||||
|
|
||||||
|
.icon {
|
||||||
|
position: absolute;
|
||||||
|
left: 10px;
|
||||||
|
top: 50%;
|
||||||
|
transform: translateY(-50%);
|
||||||
|
color: var(--text-faint);
|
||||||
|
font-size: 13px;
|
||||||
|
pointer-events: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.input {
|
||||||
|
width: 100%;
|
||||||
|
padding: 6px 12px;
|
||||||
|
border: 1px solid var(--border);
|
||||||
|
border-radius: var(--radius-sm);
|
||||||
|
background: var(--bg-raised);
|
||||||
|
color: var(--text-primary);
|
||||||
|
font-family: var(--font-body);
|
||||||
|
font-size: 12px;
|
||||||
|
outline: none;
|
||||||
|
transition: border-color 0.15s, box-shadow 0.15s;
|
||||||
|
}
|
||||||
|
|
||||||
|
.input::placeholder { color: var(--text-faint); }
|
||||||
|
.input:focus { border-color: var(--amber); box-shadow: 0 0 0 3px var(--amber-bg); }
|
||||||
|
|
||||||
|
.hasIcon { padding-left: 30px; }
|
||||||
23
src/design-system/primitives/Input/Input.tsx
Normal file
23
src/design-system/primitives/Input/Input.tsx
Normal file
@@ -0,0 +1,23 @@
|
|||||||
|
import styles from './Input.module.css'
|
||||||
|
import { forwardRef, type InputHTMLAttributes, type ReactNode } from 'react'
|
||||||
|
|
||||||
|
interface InputProps extends InputHTMLAttributes<HTMLInputElement> {
|
||||||
|
icon?: ReactNode
|
||||||
|
}
|
||||||
|
|
||||||
|
export const Input = forwardRef<HTMLInputElement, InputProps>(
|
||||||
|
({ icon, className, ...rest }, ref) => {
|
||||||
|
return (
|
||||||
|
<div className={`${styles.wrap} ${className ?? ''}`}>
|
||||||
|
{icon && <span className={styles.icon}>{icon}</span>}
|
||||||
|
<input
|
||||||
|
ref={ref}
|
||||||
|
className={`${styles.input} ${icon ? styles.hasIcon : ''}`}
|
||||||
|
{...rest}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
Input.displayName = 'Input'
|
||||||
Reference in New Issue
Block a user