You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
187 lines
4.3 KiB
187 lines
4.3 KiB
"use client";
|
|
import React from "react";
|
|
import { createAvatar } from "@gluestack-ui/avatar";
|
|
|
|
import { View, Text, Image, Platform } from "react-native";
|
|
|
|
import { tva } from "@gluestack-ui/nativewind-utils/tva";
|
|
import {
|
|
withStyleContext,
|
|
useStyleContext,
|
|
} from "@gluestack-ui/nativewind-utils/withStyleContext";
|
|
import type { VariantProps } from "@gluestack-ui/nativewind-utils";
|
|
const SCOPE = "AVATAR";
|
|
|
|
const UIAvatar = createAvatar({
|
|
Root: withStyleContext(View, SCOPE),
|
|
Badge: View,
|
|
Group: View,
|
|
Image: Image,
|
|
FallbackText: Text,
|
|
});
|
|
|
|
const avatarStyle = tva({
|
|
base: "rounded-full justify-center items-center relative bg-primary-600 group-[.avatar-group]/avatar-group:-ml-2.5",
|
|
variants: {
|
|
size: {
|
|
xs: "w-6 h-6",
|
|
sm: "w-8 h-8",
|
|
md: "w-12 h-12",
|
|
lg: "w-16 h-16",
|
|
xl: "w-24 h-24",
|
|
"2xl": "w-32 h-32",
|
|
},
|
|
},
|
|
});
|
|
|
|
const avatarFallbackTextStyle = tva({
|
|
base: "text-typography-0 font-semibold overflow-hidden text-transform:uppercase web:cursor-default",
|
|
|
|
parentVariants: {
|
|
size: {
|
|
xs: "text-2xs",
|
|
sm: "text-xs",
|
|
md: "text-base",
|
|
lg: "text-xl",
|
|
xl: "text-3xl",
|
|
"2xl": "text-5xl",
|
|
},
|
|
},
|
|
});
|
|
|
|
const avatarGroupStyle = tva({
|
|
base: "group/avatar-group flex-row-reverse relative avatar-group",
|
|
});
|
|
|
|
const avatarBadgeStyle = tva({
|
|
base: "w-5 h-5 bg-success-500 rounded-full absolute right-0 bottom-0 border-background-0 border-2",
|
|
parentVariants: {
|
|
size: {
|
|
xs: "w-2 h-2",
|
|
sm: "w-2 h-2",
|
|
md: "w-3 h-3",
|
|
lg: "w-4 h-4",
|
|
xl: "w-6 h-6",
|
|
"2xl": "w-8 h-8",
|
|
},
|
|
},
|
|
});
|
|
|
|
const avatarImageStyle = tva({
|
|
base: "h-full w-full rounded-full absolute",
|
|
});
|
|
|
|
type IAvatarProps = Omit<
|
|
React.ComponentPropsWithoutRef<typeof UIAvatar>,
|
|
"context"
|
|
> &
|
|
VariantProps<typeof avatarStyle>;
|
|
|
|
const Avatar = React.forwardRef<
|
|
React.ElementRef<typeof UIAvatar>,
|
|
IAvatarProps
|
|
>(({ className, size = "md", ...props }, ref) => {
|
|
return (
|
|
<UIAvatar
|
|
ref={ref}
|
|
{...props}
|
|
className={avatarStyle({ size, class: className })}
|
|
context={{ size }}
|
|
/>
|
|
);
|
|
});
|
|
|
|
type IAvatarBadgeProps = React.ComponentPropsWithoutRef<typeof UIAvatar.Badge> &
|
|
VariantProps<typeof avatarBadgeStyle>;
|
|
|
|
const AvatarBadge = React.forwardRef<
|
|
React.ElementRef<typeof UIAvatar.Badge>,
|
|
IAvatarBadgeProps
|
|
>(({ className, size, ...props }, ref) => {
|
|
const { size: parentSize } = useStyleContext(SCOPE);
|
|
|
|
return (
|
|
<UIAvatar.Badge
|
|
ref={ref}
|
|
{...props}
|
|
className={avatarBadgeStyle({
|
|
parentVariants: {
|
|
size: parentSize,
|
|
},
|
|
size,
|
|
class: className,
|
|
})}
|
|
/>
|
|
);
|
|
});
|
|
|
|
type IAvatarFallbackTextProps = React.ComponentPropsWithoutRef<
|
|
typeof UIAvatar.FallbackText
|
|
> &
|
|
VariantProps<typeof avatarFallbackTextStyle>;
|
|
const AvatarFallbackText = React.forwardRef<
|
|
React.ElementRef<typeof UIAvatar.FallbackText>,
|
|
IAvatarFallbackTextProps
|
|
>(({ className, size, ...props }, ref) => {
|
|
const { size: parentSize } = useStyleContext(SCOPE);
|
|
|
|
return (
|
|
<UIAvatar.FallbackText
|
|
ref={ref}
|
|
{...props}
|
|
className={avatarFallbackTextStyle({
|
|
parentVariants: {
|
|
size: parentSize,
|
|
},
|
|
size,
|
|
class: className,
|
|
})}
|
|
/>
|
|
);
|
|
});
|
|
|
|
type IAvatarImageProps = React.ComponentPropsWithoutRef<typeof UIAvatar.Image> &
|
|
VariantProps<typeof avatarImageStyle>;
|
|
|
|
const AvatarImage = React.forwardRef<
|
|
React.ElementRef<typeof UIAvatar.Image>,
|
|
IAvatarImageProps
|
|
>(({ className, ...props }, ref) => {
|
|
return (
|
|
<UIAvatar.Image
|
|
ref={ref}
|
|
{...props}
|
|
className={avatarImageStyle({
|
|
class: className,
|
|
})}
|
|
// @ts-expect-error
|
|
style={
|
|
Platform.OS === "web"
|
|
? // eslint-disable-next-line react-native/no-inline-styles
|
|
{ height: "revert-layer", width: "revert-layer" }
|
|
: undefined
|
|
}
|
|
/>
|
|
);
|
|
});
|
|
|
|
type IAvatarGroupProps = React.ComponentPropsWithoutRef<typeof UIAvatar.Group> &
|
|
VariantProps<typeof avatarGroupStyle>;
|
|
|
|
const AvatarGroup = React.forwardRef<
|
|
React.ElementRef<typeof UIAvatar.Group>,
|
|
IAvatarGroupProps
|
|
>(({ className, ...props }, ref) => {
|
|
return (
|
|
<UIAvatar.Group
|
|
ref={ref}
|
|
{...props}
|
|
className={avatarGroupStyle({
|
|
class: className,
|
|
})}
|
|
/>
|
|
);
|
|
});
|
|
|
|
export { Avatar, AvatarBadge, AvatarFallbackText, AvatarImage, AvatarGroup };
|