import * as Slot from '~/components/primitives/slot';
import { ComponentPropsWithAsChild, SlottableViewProps, ViewRef } from '~/components/primitives/types';
import * as React from 'react';
type ImageErrorEventData,
type ImageSourcePropType,
type NativeSyntheticEvent,
import { AvatarImageProps, AvatarRootProps } from './types';
type AvatarState = 'loading' | 'error' | 'loaded';
interface IRootContext extends AvatarRootProps {
setStatus: (status: AvatarState) => void;
const RootContext = React.createContext<IRootContext | null>(null);
const Root = React.forwardRef<ViewRef, SlottableViewProps & AvatarRootProps>(
({ asChild, alt, ...viewProps }, ref) => {
const [status, setStatus] = React.useState<AvatarState>('error');
const Component = asChild ? Slot.View : View;
<RootContext.Provider value={{ alt, status, setStatus }}>
<Component ref={ref} {...viewProps} />
Root.displayName = 'RootAvatar';
function useRootContext() {
const context = React.useContext(RootContext);
throw new Error('Avatar compound components cannot be rendered outside the Avatar component');
const Image = React.forwardRef<
React.ElementRef<typeof RNImage>,
Omit<ComponentPropsWithAsChild<typeof RNImage>, 'alt'> & AvatarImageProps
{ asChild, onLoad: onLoadProps, onError: onErrorProps, onLoadingStatusChange, ...props },
const { alt, setStatus, status } = useRootContext();
React.useLayoutEffect(() => {
if (isValidSource(props?.source)) {
const onLoad = React.useCallback(
(e: NativeSyntheticEvent<ImageLoadEventData>) => {
onLoadingStatusChange?.('loaded');
const onError = React.useCallback(
(e: NativeSyntheticEvent<ImageErrorEventData>) => {
onLoadingStatusChange?.('error');
if (status === 'error') {
const Component = asChild ? Slot.Image : RNImage;
return <Component ref={ref} alt={alt} onLoad={onLoad} onError={onError} {...props} />;
Image.displayName = 'ImageAvatar';
const Fallback = React.forwardRef<ViewRef, SlottableViewProps>(({ asChild, ...props }, ref) => {
const { alt, status } = useRootContext();
if (status !== 'error') {
const Component = asChild ? Slot.View : View;
return <Component ref={ref} role={'img'} aria-label={alt} {...props} />;
Fallback.displayName = 'FallbackAvatar';
export { Fallback, Image, Root };
function isValidSource(source?: ImageSourcePropType) {
// Using require() for the source returns a number
if (typeof source === 'number') {
if (Array.isArray(source)) {
return source.some((source) => !!source.uri);