From gluestack
Build accessible UIs with gluestack-ui components for React and React Native. Covers CLI installation, composition, variants, sizes, states, accessibility props, and platform specifics.
npx claudepluginhub thebushidocollective/han --plugin gluestackThis skill is limited to using the following tools:
Expert knowledge of gluestack-ui's universal component library for building accessible, performant UI across React and React Native platforms.
Ensures accessible gluestack-ui implementations in React Native and web apps. Covers WAI-ARIA patterns, screen readers, keyboard navigation, focus management, and WCAG 2.1 AA compliance.
Installs, configures, and implements shadcn/ui React components with Tailwind CSS, React Hook Form, Zod for accessible UIs like buttons, forms, dialogs, tables.
Provides shadcn/ui installation, configuration, and implementation patterns for accessible React components with Tailwind CSS, Radix UI, React Hook Form, and Zod. Use for buttons, dialogs, tables, forms, and themes.
Share bugs, ideas, or general feedback.
Expert knowledge of gluestack-ui's universal component library for building accessible, performant UI across React and React Native platforms.
gluestack-ui provides 50+ unstyled, accessible components that work seamlessly on web and mobile. Components are copy-pasteable into your project and styled with NativeWind (Tailwind CSS for React Native).
Add components using the CLI:
# Initialize gluestack-ui in your project
npx gluestack-ui init
# Add individual components
npx gluestack-ui add button
npx gluestack-ui add input
npx gluestack-ui add modal
# Add multiple components
npx gluestack-ui add button input select modal
# Add all components
npx gluestack-ui add --all
Every gluestack-ui component follows a consistent pattern:
import { Button, ButtonText, ButtonSpinner, ButtonIcon } from '@/components/ui/button';
// Components are composable with sub-components
<Button size="md" variant="solid" action="primary">
<ButtonIcon as={PlusIcon} />
<ButtonText>Add Item</ButtonText>
<ButtonSpinner />
</Button>
Components support consistent prop APIs:
// Button variants
<Button variant="solid">Solid</Button>
<Button variant="outline">Outline</Button>
<Button variant="link">Link</Button>
// Button sizes
<Button size="xs">Extra Small</Button>
<Button size="sm">Small</Button>
<Button size="md">Medium</Button>
<Button size="lg">Large</Button>
<Button size="xl">Extra Large</Button>
// Button actions (semantic colors)
<Button action="primary">Primary</Button>
<Button action="secondary">Secondary</Button>
<Button action="positive">Positive</Button>
<Button action="negative">Negative</Button>
Interactive button with loading states and icons:
import { Button, ButtonText, ButtonSpinner, ButtonIcon } from '@/components/ui/button';
import { SaveIcon } from 'lucide-react-native';
function SaveButton({ isLoading, onPress }: { isLoading: boolean; onPress: () => void }) {
return (
<Button
size="md"
variant="solid"
action="primary"
isDisabled={isLoading}
onPress={onPress}
>
{isLoading ? (
<ButtonSpinner />
) : (
<ButtonIcon as={SaveIcon} />
)}
<ButtonText>{isLoading ? 'Saving...' : 'Save'}</ButtonText>
</Button>
);
}
Text input with labels and validation:
import {
FormControl,
FormControlLabel,
FormControlLabelText,
FormControlHelper,
FormControlHelperText,
FormControlError,
FormControlErrorIcon,
FormControlErrorText,
} from '@/components/ui/form-control';
import { Input, InputField, InputSlot, InputIcon } from '@/components/ui/input';
import { AlertCircleIcon, MailIcon } from 'lucide-react-native';
function EmailInput({ value, onChange, error }: {
value: string;
onChange: (text: string) => void;
error?: string;
}) {
return (
<FormControl isInvalid={!!error}>
<FormControlLabel>
<FormControlLabelText>Email</FormControlLabelText>
</FormControlLabel>
<Input variant="outline" size="md">
<InputSlot pl="$3">
<InputIcon as={MailIcon} />
</InputSlot>
<InputField
placeholder="Enter your email"
value={value}
onChangeText={onChange}
keyboardType="email-address"
autoCapitalize="none"
/>
</Input>
{error ? (
<FormControlError>
<FormControlErrorIcon as={AlertCircleIcon} />
<FormControlErrorText>{error}</FormControlErrorText>
</FormControlError>
) : (
<FormControlHelper>
<FormControlHelperText>We'll never share your email</FormControlHelperText>
</FormControlHelper>
)}
</FormControl>
);
}
Dropdown selection component:
import {
Select,
SelectTrigger,
SelectInput,
SelectIcon,
SelectPortal,
SelectBackdrop,
SelectContent,
SelectDragIndicatorWrapper,
SelectDragIndicator,
SelectItem,
} from '@/components/ui/select';
import { ChevronDownIcon } from 'lucide-react-native';
function CountrySelect({ value, onValueChange }: {
value: string;
onValueChange: (value: string) => void;
}) {
return (
<Select selectedValue={value} onValueChange={onValueChange}>
<SelectTrigger variant="outline" size="md">
<SelectInput placeholder="Select country" />
<SelectIcon as={ChevronDownIcon} mr="$3" />
</SelectTrigger>
<SelectPortal>
<SelectBackdrop />
<SelectContent>
<SelectDragIndicatorWrapper>
<SelectDragIndicator />
</SelectDragIndicatorWrapper>
<SelectItem label="United States" value="us" />
<SelectItem label="Canada" value="ca" />
<SelectItem label="United Kingdom" value="uk" />
<SelectItem label="Australia" value="au" />
</SelectContent>
</SelectPortal>
</Select>
);
}
Dialog overlay component:
import {
Modal,
ModalBackdrop,
ModalContent,
ModalHeader,
ModalCloseButton,
ModalBody,
ModalFooter,
} from '@/components/ui/modal';
import { Heading } from '@/components/ui/heading';
import { Text } from '@/components/ui/text';
import { Button, ButtonText } from '@/components/ui/button';
import { CloseIcon, Icon } from '@/components/ui/icon';
function ConfirmModal({ isOpen, onClose, onConfirm, title, message }: {
isOpen: boolean;
onClose: () => void;
onConfirm: () => void;
title: string;
message: string;
}) {
return (
<Modal isOpen={isOpen} onClose={onClose} size="md">
<ModalBackdrop />
<ModalContent>
<ModalHeader>
<Heading size="lg">{title}</Heading>
<ModalCloseButton>
<Icon as={CloseIcon} />
</ModalCloseButton>
</ModalHeader>
<ModalBody>
<Text>{message}</Text>
</ModalBody>
<ModalFooter>
<Button variant="outline" action="secondary" onPress={onClose}>
<ButtonText>Cancel</ButtonText>
</Button>
<Button action="negative" onPress={onConfirm}>
<ButtonText>Delete</ButtonText>
</Button>
</ModalFooter>
</ModalContent>
</Modal>
);
}
Notification component:
import {
Toast,
ToastTitle,
ToastDescription,
useToast,
} from '@/components/ui/toast';
function NotificationExample() {
const toast = useToast();
const showToast = () => {
toast.show({
placement: 'top',
render: ({ id }) => (
<Toast nativeID={`toast-${id}`} action="success" variant="solid">
<ToastTitle>Success!</ToastTitle>
<ToastDescription>Your changes have been saved.</ToastDescription>
</Toast>
),
});
};
return (
<Button onPress={showToast}>
<ButtonText>Show Toast</ButtonText>
</Button>
);
}
Expandable content sections:
import {
Accordion,
AccordionItem,
AccordionHeader,
AccordionTrigger,
AccordionTitleText,
AccordionIcon,
AccordionContent,
AccordionContentText,
} from '@/components/ui/accordion';
import { ChevronDownIcon, ChevronUpIcon } from 'lucide-react-native';
function FAQAccordion({ items }: { items: { question: string; answer: string }[] }) {
return (
<Accordion type="multiple" defaultValue={['item-0']}>
{items.map((item, index) => (
<AccordionItem key={index} value={`item-${index}`}>
<AccordionHeader>
<AccordionTrigger>
{({ isExpanded }) => (
<>
<AccordionTitleText>{item.question}</AccordionTitleText>
<AccordionIcon as={isExpanded ? ChevronUpIcon : ChevronDownIcon} />
</>
)}
</AccordionTrigger>
</AccordionHeader>
<AccordionContent>
<AccordionContentText>{item.answer}</AccordionContentText>
</AccordionContent>
</AccordionItem>
))}
</Accordion>
);
}
Selection controls:
import {
Checkbox,
CheckboxIndicator,
CheckboxIcon,
CheckboxLabel,
} from '@/components/ui/checkbox';
import {
RadioGroup,
Radio,
RadioIndicator,
RadioIcon,
RadioLabel,
} from '@/components/ui/radio';
import { CheckIcon, CircleIcon } from 'lucide-react-native';
function TermsCheckbox({ isChecked, onChange }: {
isChecked: boolean;
onChange: (checked: boolean) => void;
}) {
return (
<Checkbox value="terms" isChecked={isChecked} onChange={onChange}>
<CheckboxIndicator>
<CheckboxIcon as={CheckIcon} />
</CheckboxIndicator>
<CheckboxLabel>I agree to the terms and conditions</CheckboxLabel>
</Checkbox>
);
}
function ShippingOptions({ value, onChange }: {
value: string;
onChange: (value: string) => void;
}) {
return (
<RadioGroup value={value} onChange={onChange}>
<Radio value="standard">
<RadioIndicator>
<RadioIcon as={CircleIcon} />
</RadioIndicator>
<RadioLabel>Standard Shipping (5-7 days)</RadioLabel>
</Radio>
<Radio value="express">
<RadioIndicator>
<RadioIcon as={CircleIcon} />
</RadioIndicator>
<RadioLabel>Express Shipping (2-3 days)</RadioLabel>
</Radio>
<Radio value="overnight">
<RadioIndicator>
<RadioIcon as={CircleIcon} />
</RadioIndicator>
<RadioLabel>Overnight Shipping (1 day)</RadioLabel>
</Radio>
</RadioGroup>
);
}
Compose components with sub-components for flexibility:
// Good: Composable structure
<Button>
<ButtonIcon as={PlusIcon} />
<ButtonText>Add</ButtonText>
</Button>
// Avoid: Prop-heavy configuration
<Button icon={PlusIcon} text="Add" iconPosition="left" />
Wrap inputs in FormControl for consistent validation:
<FormControl isRequired isInvalid={!!error}>
<FormControlLabel>
<FormControlLabelText>Password</FormControlLabelText>
</FormControlLabel>
<Input>
<InputField type="password" />
</Input>
<FormControlError>
<FormControlErrorText>{error}</FormControlErrorText>
</FormControlError>
</FormControl>
Use platform-specific logic when needed:
import { Platform } from 'react-native';
function ResponsiveModal({ children }: { children: React.ReactNode }) {
return (
<Modal size={Platform.OS === 'web' ? 'lg' : 'full'}>
<ModalContent>
{children}
</ModalContent>
</Modal>
);
}
Show loading feedback for async operations:
function SubmitButton({ isLoading, onPress }: {
isLoading: boolean;
onPress: () => void;
}) {
return (
<Button isDisabled={isLoading} onPress={onPress}>
{isLoading && <ButtonSpinner mr="$2" />}
<ButtonText>{isLoading ? 'Submitting...' : 'Submit'}</ButtonText>
</Button>
);
}
Build app-specific components on top of gluestack-ui:
// components/app/PrimaryButton.tsx
import { Button, ButtonText, ButtonSpinner } from '@/components/ui/button';
interface PrimaryButtonProps {
children: string;
isLoading?: boolean;
onPress: () => void;
}
export function PrimaryButton({ children, isLoading, onPress }: PrimaryButtonProps) {
return (
<Button
size="lg"
variant="solid"
action="primary"
isDisabled={isLoading}
onPress={onPress}
className="rounded-full"
>
{isLoading && <ButtonSpinner mr="$2" />}
<ButtonText>{children}</ButtonText>
</Button>
);
}
import { useState } from 'react';
import { VStack } from '@/components/ui/vstack';
import { Button, ButtonText } from '@/components/ui/button';
import { FormControl, FormControlError, FormControlErrorText } from '@/components/ui/form-control';
import { Input, InputField } from '@/components/ui/input';
interface FormData {
email: string;
password: string;
}
interface FormErrors {
email?: string;
password?: string;
}
function LoginForm({ onSubmit }: { onSubmit: (data: FormData) => void }) {
const [formData, setFormData] = useState<FormData>({ email: '', password: '' });
const [errors, setErrors] = useState<FormErrors>({});
const validate = (): boolean => {
const newErrors: FormErrors = {};
if (!formData.email) {
newErrors.email = 'Email is required';
} else if (!/\S+@\S+\.\S+/.test(formData.email)) {
newErrors.email = 'Invalid email format';
}
if (!formData.password) {
newErrors.password = 'Password is required';
} else if (formData.password.length < 8) {
newErrors.password = 'Password must be at least 8 characters';
}
setErrors(newErrors);
return Object.keys(newErrors).length === 0;
};
const handleSubmit = () => {
if (validate()) {
onSubmit(formData);
}
};
return (
<VStack space="md">
<FormControl isInvalid={!!errors.email}>
<Input>
<InputField
placeholder="Email"
value={formData.email}
onChangeText={(text) => setFormData({ ...formData, email: text })}
/>
</Input>
<FormControlError>
<FormControlErrorText>{errors.email}</FormControlErrorText>
</FormControlError>
</FormControl>
<FormControl isInvalid={!!errors.password}>
<Input>
<InputField
placeholder="Password"
type="password"
value={formData.password}
onChangeText={(text) => setFormData({ ...formData, password: text })}
/>
</Input>
<FormControlError>
<FormControlErrorText>{errors.password}</FormControlErrorText>
</FormControlError>
</FormControl>
<Button onPress={handleSubmit}>
<ButtonText>Login</ButtonText>
</Button>
</VStack>
);
}
import { HStack } from '@/components/ui/hstack';
import { VStack } from '@/components/ui/vstack';
import { Box } from '@/components/ui/box';
import { Text } from '@/components/ui/text';
import { Heading } from '@/components/ui/heading';
import { Button, ButtonIcon } from '@/components/ui/button';
import { Pressable } from '@/components/ui/pressable';
import { TrashIcon, EditIcon } from 'lucide-react-native';
interface Item {
id: string;
title: string;
description: string;
}
function ItemList({ items, onEdit, onDelete }: {
items: Item[];
onEdit: (id: string) => void;
onDelete: (id: string) => void;
}) {
return (
<VStack space="sm">
{items.map((item) => (
<Box
key={item.id}
className="p-4 bg-background-50 rounded-lg border border-outline-200"
>
<HStack justifyContent="space-between" alignItems="center">
<VStack space="xs" flex={1}>
<Heading size="sm">{item.title}</Heading>
<Text size="sm" className="text-typography-500">
{item.description}
</Text>
</VStack>
<HStack space="xs">
<Button
size="sm"
variant="outline"
action="secondary"
onPress={() => onEdit(item.id)}
>
<ButtonIcon as={EditIcon} />
</Button>
<Button
size="sm"
variant="outline"
action="negative"
onPress={() => onDelete(item.id)}
>
<ButtonIcon as={TrashIcon} />
</Button>
</HStack>
</HStack>
</Box>
))}
</VStack>
);
}
// Bad: Mixing inline styles with NativeWind
<Button style={{ backgroundColor: 'blue' }} className="p-4">
<ButtonText>Click</ButtonText>
</Button>
// Good: Use NativeWind classes consistently
<Button className="bg-blue-500 p-4">
<ButtonText>Click</ButtonText>
</Button>
// Bad: Missing accessibility information
<Pressable onPress={handlePress}>
<Icon as={MenuIcon} />
</Pressable>
// Good: Include accessibility props
<Pressable
onPress={handlePress}
accessibilityLabel="Open menu"
accessibilityRole="button"
>
<Icon as={MenuIcon} />
</Pressable>
// Bad: Using web-only APIs on native
<Modal>
<ModalContent onClick={handleClick}> {/* onClick doesn't work on native */}
...
</ModalContent>
</Modal>
// Good: Use cross-platform events
<Modal>
<ModalContent>
<Pressable onPress={handlePress}>
...
</Pressable>
</ModalContent>
</Modal>
// Bad: Hardcoded colors
<Box className="bg-[#3B82F6]">
<Text className="text-[#FFFFFF]">Hello</Text>
</Box>
// Good: Use theme tokens
<Box className="bg-primary-500">
<Text className="text-typography-0">Hello</Text>
</Box>