Skip to content

Commit c0319d0

Browse files
committed
feat: add a demo using NativeWind with RN Reusables
1 parent 0414d8b commit c0319d0

File tree

9 files changed

+212
-31
lines changed

9 files changed

+212
-31
lines changed

apps/tester-app/package.json

+5-1
Original file line numberDiff line numberDiff line change
@@ -18,13 +18,17 @@
1818
},
1919
"dependencies": {
2020
"@react-native-async-storage/async-storage": "^1.23.1",
21+
"@rn-primitives/slot": "^1.1.0",
22+
"@rn-primitives/types": "^1.1.0",
23+
"class-variance-authority": "^0.7.1",
2124
"nativewind": "^4.1.23",
2225
"react": "18.3.1",
2326
"react-native": "0.76.3",
2427
"react-native-css-interop": "0.1.22",
2528
"react-native-reanimated": "^3.16.3",
2629
"react-native-safe-area-context": "^4.14.0",
27-
"react-native-svg": "15.8.0"
30+
"react-native-svg": "15.8.0",
31+
"tailwind-merge": "^2.6.0"
2832
},
2933
"devDependencies": {
3034
"@babel/core": "^7.25.2",

apps/tester-app/src/nativewind/Input.tsx

-27
This file was deleted.

apps/tester-app/src/nativewind/NativeWindView.tsx

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
import { StyleSheet, Text, View } from 'react-native';
22
import ComponentWithVariants from './ComponentWithVariants.tsx';
33
import CustomComponent from './CustomComponent.tsx';
4-
import { Input } from './Input.tsx';
4+
import Reusables from './Reusables.tsx';
55

66
export function NativeWindView() {
77
return (
@@ -14,7 +14,7 @@ export function NativeWindView() {
1414
</View>
1515
<CustomComponent className="bg-green-600" />
1616
<ComponentWithVariants variant="primary" className="bg-yellow-500" />
17-
<Input />
17+
<Reusables />
1818
</View>
1919
);
2020
}

apps/tester-app/src/nativewind/README.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -8,8 +8,8 @@ This component demonstrates compatibility with different NativeWind features:
88
* general styling with className and Tailwind classes
99
* custom components styled with className
1010
* components using the variants pattern
11+
* using component libraries (React Native Reusables)
1112
* TODO: themes https://www.nativewind.dev/guides/themes
12-
* using component libraries (NativeWindUI)
1313
* responsive design
1414
* dark mode
1515
* functions & directives
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,10 @@
1+
import { Button } from './ui/Button';
2+
import { Text } from './ui/Text';
3+
4+
export default function Reusables() {
5+
return (
6+
<Button variant="primary" className="bg-purple-700">
7+
<Text className="text-white">I am a button from RN Reusables</Text>
8+
</Button>
9+
);
10+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,92 @@
1+
import { type VariantProps, cva } from 'class-variance-authority';
2+
import * as React from 'react';
3+
import { Pressable } from 'react-native';
4+
import { TextClassContext } from './Text';
5+
import { cn } from './utils';
6+
7+
const buttonVariants = cva(
8+
'group flex items-center justify-center rounded-md web:ring-offset-background web:transition-colors web:focus-visible:outline-none web:focus-visible:ring-2 web:focus-visible:ring-ring web:focus-visible:ring-offset-2',
9+
{
10+
variants: {
11+
variant: {
12+
default: 'bg-primary web:hover:opacity-90 active:opacity-90',
13+
destructive: 'bg-destructive web:hover:opacity-90 active:opacity-90',
14+
outline:
15+
'border border-input bg-background web:hover:bg-accent web:hover:text-accent-foreground active:bg-accent',
16+
secondary: 'bg-secondary web:hover:opacity-80 active:opacity-80',
17+
ghost:
18+
'web:hover:bg-accent web:hover:text-accent-foreground active:bg-accent',
19+
link: 'web:underline-offset-4 web:hover:underline web:focus:underline',
20+
},
21+
size: {
22+
default: 'h-10 px-4 py-2 native:h-12 native:px-5 native:py-3',
23+
sm: 'h-9 rounded-md px-3',
24+
lg: 'h-11 rounded-md px-8 native:h-14',
25+
icon: 'h-10 w-10',
26+
},
27+
},
28+
defaultVariants: {
29+
variant: 'default',
30+
size: 'default',
31+
},
32+
}
33+
);
34+
35+
const buttonTextVariants = cva(
36+
'web:whitespace-nowrap text-sm native:text-base font-medium text-foreground web:transition-colors',
37+
{
38+
variants: {
39+
variant: {
40+
default: 'text-primary-foreground',
41+
destructive: 'text-destructive-foreground',
42+
outline: 'group-active:text-accent-foreground',
43+
secondary:
44+
'text-secondary-foreground group-active:text-secondary-foreground',
45+
ghost: 'group-active:text-accent-foreground',
46+
link: 'text-primary group-active:underline',
47+
},
48+
size: {
49+
default: '',
50+
sm: '',
51+
lg: 'native:text-lg',
52+
icon: '',
53+
},
54+
},
55+
defaultVariants: {
56+
variant: 'default',
57+
size: 'default',
58+
},
59+
}
60+
);
61+
62+
type ButtonProps = React.ComponentPropsWithoutRef<typeof Pressable> &
63+
VariantProps<typeof buttonVariants>;
64+
65+
const Button = React.forwardRef<
66+
React.ElementRef<typeof Pressable>,
67+
ButtonProps
68+
>(({ className, variant, size, ...props }, ref) => {
69+
return (
70+
<TextClassContext.Provider
71+
value={buttonTextVariants({
72+
variant,
73+
size,
74+
className: 'web:pointer-events-none',
75+
})}
76+
>
77+
<Pressable
78+
className={cn(
79+
props.disabled && 'opacity-50 web:pointer-events-none',
80+
buttonVariants({ variant, size, className })
81+
)}
82+
ref={ref}
83+
role="button"
84+
{...props}
85+
/>
86+
</TextClassContext.Provider>
87+
);
88+
});
89+
Button.displayName = 'Button';
90+
91+
export { Button, buttonTextVariants, buttonVariants };
92+
export type { ButtonProps };
+28
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,28 @@
1+
import * as Slot from '@rn-primitives/slot';
2+
import type { SlottableTextProps, TextRef } from '@rn-primitives/types';
3+
import * as React from 'react';
4+
import { Text as RNText } from 'react-native';
5+
import { cn } from './utils';
6+
7+
const TextClassContext = React.createContext<string | undefined>(undefined);
8+
9+
const Text = React.forwardRef<TextRef, SlottableTextProps>(
10+
({ className, asChild = false, ...props }, ref) => {
11+
const textClass = React.useContext(TextClassContext);
12+
const Component = asChild ? Slot.Text : RNText;
13+
return (
14+
<Component
15+
className={cn(
16+
'text-base text-foreground web:select-text',
17+
textClass,
18+
className
19+
)}
20+
ref={ref}
21+
{...props}
22+
/>
23+
);
24+
}
25+
);
26+
Text.displayName = 'Text';
27+
28+
export { Text, TextClassContext };
+14
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,14 @@
1+
import { type ClassValue, clsx } from 'clsx';
2+
import { twMerge } from 'tailwind-merge';
3+
4+
export function cn(...inputs: ClassValue[]) {
5+
return twMerge(clsx(inputs));
6+
}
7+
8+
export function fixImports(rawfile: string) {
9+
return rawfile
10+
.replace('./typography', '~/components/ui/typography')
11+
.replace('./text', '~/components/ui/text')
12+
.replaceAll('../../components', '~/components')
13+
.replaceAll('../../lib', '~/lib');
14+
}

pnpm-lock.yaml

+60
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

0 commit comments

Comments
 (0)