import { Meta } from '@storybook/addon-docs/blocks';
This guide explains how to create components using the tasty style helper and React Aria Hooks. Components in the Cube UI Kit follow consistent patterns for styling, accessibility, and functionality.
Creating a component involves several optional but recommended elements:
- Base properties handling using
filterBaseProps - Style properties for direct styling capabilities
- Modifiers (
mods) for state-based styling - Additional style properties for styling inner elements
- Sub-elements for targeted styling within components
- React Aria integration for accessibility
Everything is optional - you can create simplified components without all features, but exported UI Kit components should be feature-rich and consistent.
The filterBaseProps helper ensures only valid DOM properties are passed to elements, filtering out design system properties and unsupported event handlers.
import { filterBaseProps, AllBaseProps } from '@tenphi/tasty';
interface MyComponentProps extends AllBaseProps {
customProp?: string;
}
function MyComponent(props: MyComponentProps) {
return (
<div {...filterBaseProps(props, { eventProps: true })}>
{props.children}
</div>
);
}Filter options:
eventProps: true- Include standard DOM event handlers (onClick, onFocus, etc.)labelable: true- Include ARIA labeling propertiespropNames: Set<string>- Include specific additional properties
Style properties allow consumers to apply layout and styling directly as component props. There are two approaches:
Approach A — styleProps on tasty() (simpler, when rest props go directly to the tasty element):
styleProps tells tasty which CSS properties the component accepts as typed React props. The component still needs to declare the matching TypeScript interfaces so consumers get autocomplete. This works when rest props are spread directly onto the tasty element.
import { tasty, BaseProps, OUTER_STYLES, CONTAINER_STYLES } from '@tenphi/tasty';
import type { OuterStyleProps, ContainerStyleProps } from '@tenphi/tasty';
interface CubeMyComponentProps extends BaseProps, OuterStyleProps, ContainerStyleProps {
label?: string;
isDisabled?: boolean;
}
const MyElement = tasty({
qa: 'MyComponent',
styles: {
display: 'flex',
flow: 'column',
padding: '2x',
fill: '#surface',
radius: '1r',
border: {
'': '#border',
disabled: '#border.5',
},
},
styleProps: [...OUTER_STYLES, ...CONTAINER_STYLES],
});
function MyComponent({ label, isDisabled, ...props }: CubeMyComponentProps) {
return (
<MyElement {...props} mods={{ disabled: isDisabled }}>
{label}
</MyElement>
);
}Approach B — extractStyles + filterBaseProps (when you need to split props between multiple elements or add custom logic):
import {
CONTAINER_STYLES,
ContainerStyleProps,
TEXT_STYLES,
TextStyleProps,
extractStyles,
filterBaseProps,
} from '@tenphi/tasty';
interface MyComponentProps
extends AllBaseProps,
ContainerStyleProps,
TextStyleProps {
}
function MyComponent(props: MyComponentProps) {
const styles = extractStyles(props, [...CONTAINER_STYLES, ...TEXT_STYLES]);
return (
<Element
{...filterBaseProps(props, { eventProps: true })}
styles={styles}
/>
);
}Available style property sets:
BASE_STYLES-display,font,preset,hide,opacity,whiteSpacePOSITION_STYLES-gridArea,order,margin,inset,position,zIndex, etc.BLOCK_STYLES-padding,border,radius,shadow,overflow, etc.COLOR_STYLES-color,fill,fadeTEXT_STYLES-textTransform(for other text styling, usepreset)DIMENSION_STYLES-width,height,flex,flexBasis, etc.FLOW_STYLES-flow,gap,align,justify,gridColumns, etc.CONTAINER_STYLES- Combination of all above stylesOUTER_STYLES- Position and dimension styles only
Modifiers enable state-based styling through the mods prop, which creates data-* attributes:
interface MyComponentProps extends AllBaseProps {
isLoading?: boolean;
}
function MyComponent({ isLoading, mods, ...props }: MyComponentProps) {
return (
<Element
mods={{
loading: isLoading,
...mods, // Allow external mods to override
}}
styles={{
color: {
'': '#dark',
loading: '#gray',
'loading & hovered': '#light-gray',
}
}}
/>
);
}Usage:
<MyComponent mods={{ custom: true, highlighted: isHighlighted }} />
// Renders: <div data-custom data-highlighted class="...">Sub-elements allow styling inner parts through the main styles prop. They automatically inherit the parent's state and modifiers, making them ideal for tightly coupled UI elements.
const MyElement = tasty({
styles: {
display: 'flex',
gap: '1x',
// Sub-element styles
Icon: {
preset: 'h3',
color: '#purple',
},
Label: {
preset: 'default / bold',
},
},
});
function MyComponent(props) {
return (
<MyElement {...props}>
<span data-element="Icon">🎉</span>
<span data-element="Label">{props.children}</span>
</MyElement>
);
}Usage with style overrides:
<MyComponent
styles={{
Icon: { color: '#red' }, // Override icon color
Label: { preset: 'h4' }, // Override label size
}}
>
Custom styled content
</MyComponent>When to use sub-elements:
- Elements that should share the parent's state (hover, focus, disabled, etc.)
- Tightly coupled UI parts (icon + label, title + subtitle)
- When you want a single
stylesprop to control everything - Simpler components with predictable inner structure
Create dedicated style properties for styling inner elements independently. This approach gives more control and allows inner elements to have their own state.
import { Styles } from '@tenphi/tasty';
interface MyComponentProps extends AllBaseProps {
styles?: Styles;
iconStyles?: Styles; // Additional style property
labelStyles?: Styles; // Another additional style property
}
function MyComponent({ iconStyles, labelStyles, ...props }: MyComponentProps) {
const styles = extractStyles(props, CONTAINER_STYLES);
return (
<Element styles={styles}>
<IconElement styles={iconStyles} />
<LabelElement styles={labelStyles} />
</Element>
);
}When to use additional style properties:
- Inner elements need independent state management (separate hover, focus behaviors)
- Complex components with multiple logical sections
- Components that need fine-grained styling control
- When different parts of the component might have different interaction patterns
Use React Aria hooks for accessibility and behavior:
import { useButton } from 'react-aria';
import { useRef } from 'react';
function MyButton(props) {
const ref = useRef();
const { buttonProps } = useButton(props, ref);
return (
<Element
{...buttonProps}
ref={ref}
styles={extractStyles(props, CONTAINER_STYLES)}
/>
);
}This demonstrates a basic component with:
- Base properties support
- All container style properties
- Clean, minimal implementation
import { forwardRef } from 'react';
import {
AllBaseProps,
CONTAINER_STYLES,
ContainerStyleProps,
extractStyles,
filterBaseProps,
tasty,
} from '@tenphi/tasty';
const BlockElement = tasty({
styles: {
display: 'block',
},
});
export interface CubeBlockProps
extends Omit<AllBaseProps, keyof ContainerStyleProps | 'as'>,
ContainerStyleProps {}
export const Block = forwardRef(function Block(props: CubeBlockProps, ref) {
const styles = extractStyles(props, CONTAINER_STYLES);
return (
<BlockElement
{...filterBaseProps(props, { eventProps: true })}
ref={ref}
styles={styles}
/>
);
});This demonstrates advanced features:
- React Aria integration (
useButton) - Complex modifier handling
- Multiple style property sets
- Custom behavior logic
import { FocusableRef } from '@react-types/shared';
import { forwardRef } from 'react';
import { AriaButtonProps, useButton, useHover } from 'react-aria';
import {
AllBaseProps,
BaseStyleProps,
CONTAINER_STYLES,
ContainerStyleProps,
extractStyles,
Styles,
TagName,
tasty,
TEXT_STYLES,
TextStyleProps,
} from '@tenphi/tasty';
export interface CubeActionProps<T extends TagName = 'button'>
extends Omit<AllBaseProps<T>, 'htmlType'>,
BaseStyleProps,
ContainerStyleProps,
TextStyleProps,
Omit<AriaButtonProps, 'type'> {
to?: string;
label?: string;
htmlType?: 'button' | 'submit' | 'reset';
}
const DEFAULT_ACTION_STYLES: Styles = {
recipe: 'reset button',
position: 'relative',
preset: 'inherit',
outline: {
'': '#purple-03.0',
focused: '#purple-03',
},
transition: 'theme',
cursor: '$pointer',
} as const;
const ActionElement = tasty({
as: 'button',
styles: DEFAULT_ACTION_STYLES,
});
const STYLE_PROPS = [...CONTAINER_STYLES, ...TEXT_STYLES];
export const Action = forwardRef(function Action(
{
to,
as = 'button',
htmlType,
label,
theme,
mods,
onPress,
isDisabled,
...props
}: CubeActionProps,
ref: FocusableRef<HTMLElement>,
) {
const domRef = useRef();
const { buttonProps, isPressed } = useButton(
{
'aria-label': label,
onPress,
isDisabled,
...props,
},
domRef,
);
const { hoverProps, isHovered } = useHover({ isDisabled });
const styles = extractStyles(props, STYLE_PROPS);
return (
<ActionElement
data-theme={theme}
{...buttonProps}
{...hoverProps}
mods={{
hovered: isHovered && !isDisabled,
pressed: isPressed && !isDisabled,
disabled: isDisabled,
...mods, // Allow external mods to override
}}
ref={domRef}
type={htmlType || 'button'}
as={as}
styles={styles}
/>
);
});For input components that need to integrate with forms, see the Field documentation for comprehensive guidance on:
- Form state management
- Validation integration
- Label and help text handling
- Error state management
- Accessibility requirements for form controls
-
Always support base properties: Use
styleProps(recommended) orfilterBaseProps, and extendAllBaseProps -
Provide appropriate style properties: Choose the right combination of style property sets for your component's needs
-
Design for extensibility: Allow
modsoverrides and provide additional style properties for inner elements -
Use React Aria: Integrate appropriate React Aria hooks for accessibility and standard behavior
-
Follow naming conventions: Use
Cube[ComponentName]Propsfor interfaces and descriptive names for sub-elements -
Document sub-elements: Clearly document what sub-elements are available for styling
-
Keep it simple when appropriate: Not every component needs all features - use what makes sense for your use case
-
Use named React imports: Import
useCallback,useState,forwardRefetc. directly from'react'instead of usingReact.useCallback(...). -
Use
useEventfor stable callbacks: When a callback doesn't need to trigger re-renders, preferuseEventfromsrc/_internal/hooks/use-eventfor a stable identity that always calls the latest implementation:
import { useEvent } from '../_internal/hooks/use-event';
const handleClick = useEvent((e) => {
// always has latest closure, but stable identity
});- Use
variantsfor theming, not sizing: Thevariantsoption intasty()defines mutually exclusive visual variations (e.g.default,danger,outline). CSS is only generated for variants actually rendered at runtime. Use modifiers for dimensions likesize.
When creating a new component, consider:
- [ ] Base properties support (
AllBaseProps,filterBaseProps) - [ ] Appropriate style property sets
- [ ] Modifier (
mods) support for state-based styling - [ ] Additional style properties for inner elements
- [ ] Sub-element definitions for targeted styling
- [ ] React Aria integration for accessibility
- [ ] TypeScript interfaces following naming conventions
- [ ] forwardRef for ref passing
- [ ] Default styles and theme support
- [ ] Documentation following UI Kit standards
Remember: Start simple and add features as needed. A basic, working component is better than an over-engineered one!