Files
hive/frontend/src/components/ui/input.stories.tsx
anthonyrawlins 268214d971 Major WHOOSH system refactoring and feature enhancements
- Migrated from HIVE branding to WHOOSH across all components
- Enhanced backend API with new services: AI models, BZZZ integration, templates, members
- Added comprehensive testing suite with security, performance, and integration tests
- Improved frontend with new components for project setup, AI models, and team management
- Updated MCP server implementation with WHOOSH-specific tools and resources
- Enhanced deployment configurations with production-ready Docker setups
- Added comprehensive documentation and setup guides
- Implemented age encryption service and UCXL integration

🤖 Generated with [Claude Code](https://claude.ai/code)

Co-Authored-By: Claude <noreply@anthropic.com>
2025-08-27 08:34:48 +10:00

423 lines
9.5 KiB
TypeScript

import type { Meta, StoryObj } from '@storybook/react';
import { Input } from './input';
import { Label } from './label';
import { Button } from './button';
/**
* Input component for WHOOSH UI
*
* A versatile input component for forms and user input throughout the WHOOSH application.
* Supports various input types with consistent styling and behavior.
*/
const meta = {
title: 'UI Components/Input',
component: Input,
parameters: {
layout: 'centered',
docs: {
description: {
component: `
The Input component provides consistent styling and behavior for form inputs across the WHOOSH application.
It supports all standard HTML input types with enhanced styling and focus states.
## Features
- Consistent styling across all input types
- Built-in focus and disabled states
- File upload support with custom styling
- Form validation integration
- Responsive design
- Accessibility support
## Usage
\`\`\`tsx
import { Input } from '@/components/ui/input';
<Input
type="text"
placeholder="Enter agent name"
value={agentName}
onChange={(e) => setAgentName(e.target.value)}
required
/>
\`\`\`
## Input Types
- **text**: General text input
- **email**: Email address input with validation
- **password**: Password input with hidden text
- **number**: Numeric input with step controls
- **search**: Search input with enhanced styling
- **url**: URL input with validation
- **tel**: Telephone number input
- **file**: File upload input
`,
},
},
},
tags: ['autodocs'],
argTypes: {
type: {
control: 'select',
options: ['text', 'email', 'password', 'number', 'search', 'url', 'tel', 'file'],
description: 'HTML input type',
},
placeholder: {
control: 'text',
description: 'Placeholder text',
},
value: {
control: 'text',
description: 'Input value',
},
disabled: {
control: 'boolean',
description: 'Whether the input is disabled',
},
required: {
control: 'boolean',
description: 'Whether the input is required',
},
className: {
control: 'text',
description: 'Additional CSS classes',
},
onChange: {
action: 'changed',
description: 'Change event handler',
},
},
args: {
onChange: (e: any) => console.log('Input changed:', e.target.value),
},
} satisfies Meta<typeof Input>;
export default meta;
type Story = StoryObj<typeof meta>;
/**
* Default text input
*/
export const Default: Story = {
args: {
type: 'text',
placeholder: 'Enter text...',
},
};
/**
* Email input with validation
*/
export const Email: Story = {
args: {
type: 'email',
placeholder: 'Enter email address',
},
};
/**
* Password input
*/
export const Password: Story = {
args: {
type: 'password',
placeholder: 'Enter password',
},
};
/**
* Number input
*/
export const Number: Story = {
args: {
type: 'number',
placeholder: 'Enter number',
},
};
/**
* Search input
*/
export const Search: Story = {
args: {
type: 'search',
placeholder: 'Search agents...',
},
};
/**
* File input
*/
export const File: Story = {
args: {
type: 'file',
},
};
/**
* Disabled state
*/
export const Disabled: Story = {
args: {
type: 'text',
placeholder: 'Disabled input',
disabled: true,
value: 'Cannot edit this value',
},
};
/**
* Required input
*/
export const Required: Story = {
args: {
type: 'text',
placeholder: 'Required field',
required: true,
},
};
/**
* Input with label (form example)
*/
export const WithLabel: Story = {
render: () => (
<div className="space-y-2">
<Label htmlFor="agent-name">Agent Name</Label>
<Input
id="agent-name"
name="agentName"
type="text"
placeholder="e.g., walnut-codellama"
required
/>
</div>
),
parameters: {
docs: {
description: {
story: 'Input component used with a label in a form context',
},
},
},
};
/**
* Form example with multiple inputs
*/
export const FormExample: Story = {
render: () => (
<div className="space-y-4 w-80">
<div className="space-y-2">
<Label htmlFor="agent-id">Agent ID</Label>
<Input
id="agent-id"
name="agentId"
type="text"
placeholder="unique-agent-id"
required
/>
</div>
<div className="space-y-2">
<Label htmlFor="endpoint">Endpoint URL</Label>
<Input
id="endpoint"
name="endpoint"
type="url"
placeholder="http://hostname:port"
required
/>
</div>
<div className="space-y-2">
<Label htmlFor="model">Model Name</Label>
<Input
id="model"
name="model"
type="text"
placeholder="codellama:34b"
required
/>
</div>
<div className="space-y-2">
<Label htmlFor="max-concurrent">Max Concurrent Tasks</Label>
<Input
id="max-concurrent"
name="maxConcurrent"
type="number"
placeholder="4"
min="1"
max="10"
required
/>
</div>
<div className="flex gap-2">
<Button variant="default" className="flex-1">Register Agent</Button>
<Button variant="outline" className="flex-1">Cancel</Button>
</div>
</div>
),
parameters: {
docs: {
description: {
story: 'Complete form example showing agent registration with multiple input types',
},
},
},
};
/**
* Search and filter inputs
*/
export const SearchAndFilter: Story = {
render: () => (
<div className="space-y-4 w-96">
<div className="space-y-2">
<Label htmlFor="search-agents">Search Agents</Label>
<Input
id="search-agents"
type="search"
placeholder="Search by name, model, or status..."
/>
</div>
<div className="grid grid-cols-2 gap-4">
<div className="space-y-2">
<Label htmlFor="min-tasks">Min Tasks</Label>
<Input
id="min-tasks"
type="number"
placeholder="0"
min="0"
/>
</div>
<div className="space-y-2">
<Label htmlFor="max-tasks">Max Tasks</Label>
<Input
id="max-tasks"
type="number"
placeholder="10"
min="0"
/>
</div>
</div>
<Button variant="default" className="w-full">Apply Filters</Button>
</div>
),
parameters: {
docs: {
description: {
story: 'Input components used for search and filtering functionality',
},
},
},
};
/**
* All input types showcase
*/
export const AllTypes: Story = {
render: () => (
<div className="grid grid-cols-1 md:grid-cols-2 gap-4 w-full max-w-4xl">
<div className="space-y-2">
<Label>Text Input</Label>
<Input type="text" placeholder="Text input" />
</div>
<div className="space-y-2">
<Label>Email Input</Label>
<Input type="email" placeholder="email@example.com" />
</div>
<div className="space-y-2">
<Label>Password Input</Label>
<Input type="password" placeholder="Password" />
</div>
<div className="space-y-2">
<Label>Number Input</Label>
<Input type="number" placeholder="123" />
</div>
<div className="space-y-2">
<Label>Search Input</Label>
<Input type="search" placeholder="Search..." />
</div>
<div className="space-y-2">
<Label>URL Input</Label>
<Input type="url" placeholder="https://example.com" />
</div>
<div className="space-y-2">
<Label>Tel Input</Label>
<Input type="tel" placeholder="+1 (555) 123-4567" />
</div>
<div className="space-y-2">
<Label>File Input</Label>
<Input type="file" />
</div>
</div>
),
parameters: {
docs: {
description: {
story: 'Showcase of all supported input types',
},
},
},
};
/**
* Validation states example
*/
export const ValidationStates: Story = {
render: () => (
<div className="space-y-4 w-80">
<div className="space-y-2">
<Label htmlFor="valid-input">Valid Input</Label>
<Input
id="valid-input"
type="text"
value="valid-agent-name"
className="border-green-500 focus-visible:ring-green-500"
/>
<p className="text-sm text-green-600"> Agent name is available</p>
</div>
<div className="space-y-2">
<Label htmlFor="error-input">Error Input</Label>
<Input
id="error-input"
type="text"
value="invalid name!"
className="border-red-500 focus-visible:ring-red-500"
/>
<p className="text-sm text-red-600"> Agent name contains invalid characters</p>
</div>
<div className="space-y-2">
<Label htmlFor="warning-input">Warning Input</Label>
<Input
id="warning-input"
type="text"
value="existing-agent"
className="border-yellow-500 focus-visible:ring-yellow-500"
/>
<p className="text-sm text-yellow-600"> Similar agent name already exists</p>
</div>
</div>
),
parameters: {
docs: {
description: {
story: 'Examples of input validation states with custom styling',
},
},
},
};