Merged in feat/LOY-311-New-Avatar-Component (pull request #2694)
Feat(LOY-311) Create avatar design system component * feat(LOY-311): Creat & use New Avatar Design System Component * refactor(LOY-311): replace avatar used in app header with design system component * fix(LOY-311): use correct space vars Approved-by: Erik Tiekstra
This commit is contained in:
@@ -54,6 +54,7 @@ GOOGLE_DYNAMIC_MAP_ID=""
|
|||||||
|
|
||||||
ENABLE_SURPRISES="true"
|
ENABLE_SURPRISES="true"
|
||||||
ENABLE_DTMC="true"
|
ENABLE_DTMC="true"
|
||||||
|
ENABLE_NEW_OVERVIEW_SECTION="true"
|
||||||
|
|
||||||
SHOW_SITE_WIDE_ALERT="false"
|
SHOW_SITE_WIDE_ALERT="false"
|
||||||
|
|
||||||
|
|||||||
@@ -0,0 +1,58 @@
|
|||||||
|
import { Avatar } from "@scandic-hotels/design-system/Avatar"
|
||||||
|
import { Typography } from "@scandic-hotels/design-system/Typography"
|
||||||
|
|
||||||
|
import CopyMembershipIdButton from "@/components/MyPages/CopyMembershipIdButton"
|
||||||
|
import { getIntl } from "@/i18n"
|
||||||
|
import { getInitials } from "@/utils/user"
|
||||||
|
|
||||||
|
import styles from "./userBaseInfo.module.css"
|
||||||
|
|
||||||
|
import type { User } from "@scandic-hotels/trpc/types/user"
|
||||||
|
|
||||||
|
interface UserBaseInfoProps {
|
||||||
|
user: User
|
||||||
|
}
|
||||||
|
|
||||||
|
export default async function UserBaseInfo({ user }: UserBaseInfoProps) {
|
||||||
|
const intl = await getIntl()
|
||||||
|
const initials = getInitials(user.firstName, user.lastName)
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className={styles.container}>
|
||||||
|
<Avatar
|
||||||
|
alt={`${user.firstName} ${user.lastName}`}
|
||||||
|
initials={initials}
|
||||||
|
size="lg"
|
||||||
|
/>
|
||||||
|
<div>
|
||||||
|
<Typography variant="Title/smLowCase">
|
||||||
|
<h3 className={styles.fullName}>
|
||||||
|
{user.firstName} {user.lastName}
|
||||||
|
</h3>
|
||||||
|
</Typography>
|
||||||
|
<div className={styles.membershipInfo}>
|
||||||
|
<Typography variant="Body/Supporting text (caption)/smBold">
|
||||||
|
<span>
|
||||||
|
{intl.formatMessage({
|
||||||
|
defaultMessage: "Membership ID:",
|
||||||
|
})}
|
||||||
|
</span>
|
||||||
|
</Typography>
|
||||||
|
{user.membership?.membershipNumber ? (
|
||||||
|
<CopyMembershipIdButton
|
||||||
|
membershipNumber={user.membership.membershipNumber}
|
||||||
|
/>
|
||||||
|
) : (
|
||||||
|
<Typography variant="Body/Supporting text (caption)/smBold">
|
||||||
|
<span className={styles.noMembership}>
|
||||||
|
{intl.formatMessage({
|
||||||
|
defaultMessage: "N/A",
|
||||||
|
})}
|
||||||
|
</span>
|
||||||
|
</Typography>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,18 @@
|
|||||||
|
.container {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: var(--Space-x15);
|
||||||
|
}
|
||||||
|
|
||||||
|
.fullName {
|
||||||
|
color: var(--Text-Heading);
|
||||||
|
text-transform: capitalize;
|
||||||
|
}
|
||||||
|
|
||||||
|
.membershipInfo {
|
||||||
|
display: flex;
|
||||||
|
flex-wrap: wrap;
|
||||||
|
align-items: center;
|
||||||
|
gap: var(--Space-x05);
|
||||||
|
color: var(--Scandic-Red-100);
|
||||||
|
}
|
||||||
@@ -16,6 +16,7 @@ import Hero from "./Friend/Hero"
|
|||||||
import MembershipNumber from "./Friend/MembershipNumber"
|
import MembershipNumber from "./Friend/MembershipNumber"
|
||||||
import Friend from "./Friend"
|
import Friend from "./Friend"
|
||||||
import Stats from "./Stats"
|
import Stats from "./Stats"
|
||||||
|
import UserBaseInfo from "./UserBaseInfo"
|
||||||
|
|
||||||
import styles from "./overview.module.css"
|
import styles from "./overview.module.css"
|
||||||
|
|
||||||
@@ -57,6 +58,8 @@ export default async function Overview({
|
|||||||
</>
|
</>
|
||||||
</TeamMemberCardTrigger>
|
</TeamMemberCardTrigger>
|
||||||
</DigitalTeamMemberCard>
|
</DigitalTeamMemberCard>
|
||||||
|
{env.ENABLE_NEW_OVERVIEW_SECTION ? <UserBaseInfo user={user} /> : null}
|
||||||
|
{/*TODO: Replace Hero Section Cards with New ones. */}
|
||||||
<Hero color="red">
|
<Hero color="red">
|
||||||
<Friend membership={user.membership} name={user.name}>
|
<Friend membership={user.membership} name={user.name}>
|
||||||
<MembershipNumber color="burgundy" membership={user.membership} />
|
<MembershipNumber color="burgundy" membership={user.membership} />
|
||||||
|
|||||||
@@ -1,14 +0,0 @@
|
|||||||
.avatar {
|
|
||||||
display: flex;
|
|
||||||
justify-content: center;
|
|
||||||
align-items: center;
|
|
||||||
overflow: hidden;
|
|
||||||
border-radius: var(--Corner-radius-rounded);
|
|
||||||
width: 2rem;
|
|
||||||
height: 2rem;
|
|
||||||
background-color: var(--UI-Grey-40);
|
|
||||||
}
|
|
||||||
|
|
||||||
.initials {
|
|
||||||
background-color: var(--Base-Icon-Low-contrast);
|
|
||||||
}
|
|
||||||
@@ -1,28 +0,0 @@
|
|||||||
import Footnote from "@scandic-hotels/design-system/Footnote"
|
|
||||||
import { MaterialIcon } from "@scandic-hotels/design-system/Icons/MaterialIcon"
|
|
||||||
import Image from "@scandic-hotels/design-system/Image"
|
|
||||||
|
|
||||||
import styles from "./avatar.module.css"
|
|
||||||
|
|
||||||
import type { AvatarProps } from "@/types/components/header/avatar"
|
|
||||||
|
|
||||||
export default function Avatar({ image, initials }: AvatarProps) {
|
|
||||||
let classNames = [styles.avatar]
|
|
||||||
let element = <MaterialIcon icon="person" color="Icon/Inverted" />
|
|
||||||
if (image) {
|
|
||||||
classNames.push(styles.image)
|
|
||||||
element = <Image src={image.src} alt={image.alt} width={28} height={28} />
|
|
||||||
} else if (initials) {
|
|
||||||
classNames.push(styles.initials)
|
|
||||||
element = (
|
|
||||||
<Footnote type="label" color="white" textTransform="uppercase" asChild>
|
|
||||||
<span>{initials}</span>
|
|
||||||
</Footnote>
|
|
||||||
)
|
|
||||||
}
|
|
||||||
return (
|
|
||||||
<span data-hj-suppress className={classNames.join(" ")}>
|
|
||||||
{element}
|
|
||||||
</span>
|
|
||||||
)
|
|
||||||
}
|
|
||||||
@@ -3,6 +3,7 @@
|
|||||||
import { useRef } from "react"
|
import { useRef } from "react"
|
||||||
import { useIntl } from "react-intl"
|
import { useIntl } from "react-intl"
|
||||||
|
|
||||||
|
import { Avatar } from "@scandic-hotels/design-system/Avatar"
|
||||||
import Body from "@scandic-hotels/design-system/Body"
|
import Body from "@scandic-hotels/design-system/Body"
|
||||||
import { MaterialIcon } from "@scandic-hotels/design-system/Icons/MaterialIcon"
|
import { MaterialIcon } from "@scandic-hotels/design-system/Icons/MaterialIcon"
|
||||||
import SkeletonShimmer from "@scandic-hotels/design-system/SkeletonShimmer"
|
import SkeletonShimmer from "@scandic-hotels/design-system/SkeletonShimmer"
|
||||||
@@ -13,7 +14,6 @@ import useClickOutside from "@/hooks/useClickOutside"
|
|||||||
import { useHandleKeyUp } from "@/hooks/useHandleKeyUp"
|
import { useHandleKeyUp } from "@/hooks/useHandleKeyUp"
|
||||||
import { getInitials } from "@/utils/user"
|
import { getInitials } from "@/utils/user"
|
||||||
|
|
||||||
import Avatar from "../Avatar"
|
|
||||||
import MainMenuButton from "../MainMenuButton"
|
import MainMenuButton from "../MainMenuButton"
|
||||||
import MyPagesMenuContent, { useMyPagesNavigation } from "../MyPagesMenuContent"
|
import MyPagesMenuContent, { useMyPagesNavigation } from "../MyPagesMenuContent"
|
||||||
|
|
||||||
|
|||||||
@@ -4,13 +4,13 @@ import { useSession } from "next-auth/react"
|
|||||||
import { useIntl } from "react-intl"
|
import { useIntl } from "react-intl"
|
||||||
|
|
||||||
import { MembershipLevelEnum } from "@scandic-hotels/common/constants/membershipLevels"
|
import { MembershipLevelEnum } from "@scandic-hotels/common/constants/membershipLevels"
|
||||||
|
import { Avatar } from "@scandic-hotels/design-system/Avatar"
|
||||||
import { trpc } from "@scandic-hotels/trpc/client"
|
import { trpc } from "@scandic-hotels/trpc/client"
|
||||||
|
|
||||||
import LoginButton from "@/components/LoginButton"
|
import LoginButton from "@/components/LoginButton"
|
||||||
import useLang from "@/hooks/useLang"
|
import useLang from "@/hooks/useLang"
|
||||||
import { isValidClientSession } from "@/utils/clientSession"
|
import { isValidClientSession } from "@/utils/clientSession"
|
||||||
|
|
||||||
import Avatar from "../Avatar"
|
|
||||||
import MyPagesMenu, { MyPagesMenuSkeleton } from "../MyPagesMenu"
|
import MyPagesMenu, { MyPagesMenuSkeleton } from "../MyPagesMenu"
|
||||||
import MyPagesMobileMenu, {
|
import MyPagesMobileMenu, {
|
||||||
MyPagesMobileMenuSkeleton,
|
MyPagesMobileMenuSkeleton,
|
||||||
|
|||||||
@@ -5,12 +5,13 @@ import { Dialog, Modal } from "react-aria-components"
|
|||||||
import { useIntl } from "react-intl"
|
import { useIntl } from "react-intl"
|
||||||
import { useMediaQuery } from "usehooks-ts"
|
import { useMediaQuery } from "usehooks-ts"
|
||||||
|
|
||||||
|
import { Avatar } from "@scandic-hotels/design-system/Avatar"
|
||||||
|
|
||||||
import useDropdownStore from "@/stores/main-menu"
|
import useDropdownStore from "@/stores/main-menu"
|
||||||
|
|
||||||
import { useHandleKeyUp } from "@/hooks/useHandleKeyUp"
|
import { useHandleKeyUp } from "@/hooks/useHandleKeyUp"
|
||||||
import { getInitials } from "@/utils/user"
|
import { getInitials } from "@/utils/user"
|
||||||
|
|
||||||
import Avatar from "../Avatar"
|
|
||||||
import MainMenuButton from "../MainMenuButton"
|
import MainMenuButton from "../MainMenuButton"
|
||||||
import MyPagesMenuContent from "../MyPagesMenuContent"
|
import MyPagesMenuContent from "../MyPagesMenuContent"
|
||||||
|
|
||||||
|
|||||||
@@ -0,0 +1,11 @@
|
|||||||
|
.container {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: var(--Space-x1);
|
||||||
|
color: var(--Scandic-Red-100);
|
||||||
|
}
|
||||||
|
|
||||||
|
.copyButton {
|
||||||
|
min-width: auto;
|
||||||
|
padding: var(--Space-x05);
|
||||||
|
}
|
||||||
@@ -0,0 +1,56 @@
|
|||||||
|
"use client"
|
||||||
|
|
||||||
|
import { useIntl } from "react-intl"
|
||||||
|
|
||||||
|
import { Button } from "@scandic-hotels/design-system/Button"
|
||||||
|
import { MaterialIcon } from "@scandic-hotels/design-system/Icons/MaterialIcon"
|
||||||
|
import { Typography } from "@scandic-hotels/design-system/Typography"
|
||||||
|
|
||||||
|
import { toast } from "@/components/TempDesignSystem/Toasts"
|
||||||
|
|
||||||
|
import styles from "./copyMembershipIdButton.module.css"
|
||||||
|
|
||||||
|
interface CopyMembershipIdButtonProps {
|
||||||
|
membershipNumber: string
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function CopyMembershipIdButton({
|
||||||
|
membershipNumber,
|
||||||
|
}: CopyMembershipIdButtonProps) {
|
||||||
|
const intl = useIntl()
|
||||||
|
|
||||||
|
function handleCopy() {
|
||||||
|
try {
|
||||||
|
navigator.clipboard.writeText(membershipNumber)
|
||||||
|
toast.success(
|
||||||
|
intl.formatMessage({
|
||||||
|
defaultMessage: "Membership ID copied to clipboard",
|
||||||
|
})
|
||||||
|
)
|
||||||
|
} catch {
|
||||||
|
toast.error(
|
||||||
|
intl.formatMessage({
|
||||||
|
defaultMessage: "Failed to copy",
|
||||||
|
})
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className={styles.container}>
|
||||||
|
<Typography variant="Body/Supporting text (caption)/smBold">
|
||||||
|
<code data-hj-suppress className={styles.membershipCode}>
|
||||||
|
{membershipNumber}
|
||||||
|
</code>
|
||||||
|
</Typography>
|
||||||
|
<Button
|
||||||
|
onClick={handleCopy}
|
||||||
|
className={styles.copyButton}
|
||||||
|
variant="Text"
|
||||||
|
size="Small"
|
||||||
|
>
|
||||||
|
<MaterialIcon icon="content_copy" color="CurrentColor" />
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
6
apps/scandic-web/env/server.ts
vendored
6
apps/scandic-web/env/server.ts
vendored
@@ -160,6 +160,11 @@ export const env = createEnv({
|
|||||||
.refine((s) => s === "1" || s === "0")
|
.refine((s) => s === "1" || s === "0")
|
||||||
.transform((s) => s === "1")
|
.transform((s) => s === "1")
|
||||||
.default("1"),
|
.default("1"),
|
||||||
|
ENABLE_NEW_OVERVIEW_SECTION: z
|
||||||
|
.string()
|
||||||
|
.refine((s) => s === "true" || s === "false")
|
||||||
|
.transform((s) => s === "true")
|
||||||
|
.default("false"),
|
||||||
},
|
},
|
||||||
emptyStringAsUndefined: true,
|
emptyStringAsUndefined: true,
|
||||||
runtimeEnv: {
|
runtimeEnv: {
|
||||||
@@ -243,6 +248,7 @@ export const env = createEnv({
|
|||||||
DTMC_ENTRA_ID_SECRET: process.env.DTMC_ENTRA_ID_SECRET,
|
DTMC_ENTRA_ID_SECRET: process.env.DTMC_ENTRA_ID_SECRET,
|
||||||
CAMPAIGN_PAGES_ENABLED: process.env.CAMPAIGN_PAGES_ENABLED,
|
CAMPAIGN_PAGES_ENABLED: process.env.CAMPAIGN_PAGES_ENABLED,
|
||||||
WEBVIEW_SHOW_OVERVIEW: process.env.WEBVIEW_SHOW_OVERVIEW,
|
WEBVIEW_SHOW_OVERVIEW: process.env.WEBVIEW_SHOW_OVERVIEW,
|
||||||
|
ENABLE_NEW_OVERVIEW_SECTION: process.env.ENABLE_NEW_OVERVIEW_SECTION,
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
|
|
||||||
|
|||||||
125
packages/design-system/lib/components/Avatar/Avatar.stories.tsx
Normal file
125
packages/design-system/lib/components/Avatar/Avatar.stories.tsx
Normal file
@@ -0,0 +1,125 @@
|
|||||||
|
import type { Meta, StoryObj } from '@storybook/nextjs-vite'
|
||||||
|
|
||||||
|
import { Avatar } from '.'
|
||||||
|
import { config } from './variants'
|
||||||
|
|
||||||
|
const meta: Meta<typeof Avatar> = {
|
||||||
|
title: 'Components/Avatar',
|
||||||
|
component: Avatar,
|
||||||
|
parameters: {
|
||||||
|
layout: 'centered',
|
||||||
|
},
|
||||||
|
argTypes: {
|
||||||
|
size: {
|
||||||
|
control: { type: 'select' },
|
||||||
|
options: Object.keys(config.variants.size),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
export default meta
|
||||||
|
|
||||||
|
type Story = StoryObj<typeof Avatar>
|
||||||
|
|
||||||
|
export const WithImage: Story = {
|
||||||
|
args: {
|
||||||
|
src: `../../../public/img/profile-picture.png`,
|
||||||
|
alt: 'Profile photo',
|
||||||
|
size: 'md',
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
export const WithInitials: Story = {
|
||||||
|
args: {
|
||||||
|
initials: 'FR',
|
||||||
|
size: 'md',
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
export const Fallback: Story = {
|
||||||
|
args: {
|
||||||
|
size: 'md',
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
export const SmallSize: Story = {
|
||||||
|
render: () => (
|
||||||
|
<div style={{ display: 'flex', gap: '16px', alignItems: 'center' }}>
|
||||||
|
<Avatar
|
||||||
|
src={`../../../public/img/profile-picture.png`}
|
||||||
|
alt="Profile photo"
|
||||||
|
size="sm"
|
||||||
|
/>
|
||||||
|
<Avatar initials="FR" size="sm" />
|
||||||
|
<Avatar size="sm" />
|
||||||
|
</div>
|
||||||
|
),
|
||||||
|
}
|
||||||
|
|
||||||
|
export const MediumSize: Story = {
|
||||||
|
render: () => (
|
||||||
|
<div style={{ display: 'flex', gap: '16px', alignItems: 'center' }}>
|
||||||
|
<Avatar
|
||||||
|
src={`../../../public/img/profile-picture.png`}
|
||||||
|
alt="Profile photo"
|
||||||
|
size="md"
|
||||||
|
/>
|
||||||
|
<Avatar initials="FR" size="md" />
|
||||||
|
<Avatar size="md" />
|
||||||
|
</div>
|
||||||
|
),
|
||||||
|
}
|
||||||
|
|
||||||
|
export const LargeSize: Story = {
|
||||||
|
render: () => (
|
||||||
|
<div style={{ display: 'flex', gap: '16px', alignItems: 'center' }}>
|
||||||
|
<Avatar
|
||||||
|
src={`../../../public/img/profile-picture.png`}
|
||||||
|
alt="Profile photo"
|
||||||
|
size="lg"
|
||||||
|
/>
|
||||||
|
<Avatar initials="FR" size="lg" />
|
||||||
|
<Avatar size="lg" />
|
||||||
|
</div>
|
||||||
|
),
|
||||||
|
}
|
||||||
|
|
||||||
|
export const AllSizes: Story = {
|
||||||
|
render: () => (
|
||||||
|
<div style={{ display: 'flex', gap: '24px', alignItems: 'center' }}>
|
||||||
|
<div
|
||||||
|
style={{
|
||||||
|
display: 'flex',
|
||||||
|
flexDirection: 'column',
|
||||||
|
gap: '8px',
|
||||||
|
alignItems: 'center',
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<Avatar initials="FR" size="sm" />
|
||||||
|
<span style={{ fontSize: '12px' }}>Small (20px)</span>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
style={{
|
||||||
|
display: 'flex',
|
||||||
|
flexDirection: 'column',
|
||||||
|
gap: '8px',
|
||||||
|
alignItems: 'center',
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<Avatar initials="FR" size="md" />
|
||||||
|
<span style={{ fontSize: '12px' }}>Medium (32px)</span>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
style={{
|
||||||
|
display: 'flex',
|
||||||
|
flexDirection: 'column',
|
||||||
|
gap: '8px',
|
||||||
|
alignItems: 'center',
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<Avatar initials="FR" size="lg" />
|
||||||
|
<span style={{ fontSize: '12px' }}>Large (55px)</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
),
|
||||||
|
}
|
||||||
@@ -0,0 +1,38 @@
|
|||||||
|
.avatar {
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
align-items: center;
|
||||||
|
overflow: hidden;
|
||||||
|
border-radius: var(--Corner-radius-rounded);
|
||||||
|
background-color: var(--Overlay-40);
|
||||||
|
}
|
||||||
|
|
||||||
|
.avatar:has(.initials) {
|
||||||
|
background-color: var(--Icon-Accent);
|
||||||
|
}
|
||||||
|
|
||||||
|
.avatar img {
|
||||||
|
object-fit: cover;
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.size-sm {
|
||||||
|
width: 24px;
|
||||||
|
height: 24px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.size-md {
|
||||||
|
width: 32px;
|
||||||
|
height: 32px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.size-lg {
|
||||||
|
width: 55px;
|
||||||
|
height: 55px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.initials {
|
||||||
|
color: white;
|
||||||
|
text-transform: uppercase;
|
||||||
|
}
|
||||||
35
packages/design-system/lib/components/Avatar/index.tsx
Normal file
35
packages/design-system/lib/components/Avatar/index.tsx
Normal file
@@ -0,0 +1,35 @@
|
|||||||
|
import { MaterialIcon } from '../Icons/MaterialIcon'
|
||||||
|
import { Typography } from '../Typography'
|
||||||
|
import Image from '../Image'
|
||||||
|
|
||||||
|
import { variants } from './variants'
|
||||||
|
import type { AvatarProps } from './types'
|
||||||
|
|
||||||
|
export function Avatar({
|
||||||
|
src,
|
||||||
|
alt,
|
||||||
|
initials,
|
||||||
|
size = 'md',
|
||||||
|
className,
|
||||||
|
}: AvatarProps) {
|
||||||
|
const classNames = variants({ size, className })
|
||||||
|
const pixelSize = size === 'sm' ? 24 : size === 'md' ? 32 : 55
|
||||||
|
const iconSize = size === 'sm' ? 16 : 24
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className={classNames}>
|
||||||
|
{src ? (
|
||||||
|
<Image src={src} alt={alt || ''} width={pixelSize} height={pixelSize} />
|
||||||
|
) : initials ? (
|
||||||
|
<Typography
|
||||||
|
variant={size === 'lg' ? 'Title/Overline/sm' : 'Tag/sm'}
|
||||||
|
className={variants.initials}
|
||||||
|
>
|
||||||
|
<span>{initials}</span>
|
||||||
|
</Typography>
|
||||||
|
) : (
|
||||||
|
<MaterialIcon icon="person" color="Icon/Inverted" size={iconSize} />
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
23
packages/design-system/lib/components/Avatar/types.ts
Normal file
23
packages/design-system/lib/components/Avatar/types.ts
Normal file
@@ -0,0 +1,23 @@
|
|||||||
|
export interface AvatarProps {
|
||||||
|
/**
|
||||||
|
* The URL of the image to display
|
||||||
|
*/
|
||||||
|
src?: string
|
||||||
|
/**
|
||||||
|
* Alt text for the image (for accessibility)
|
||||||
|
*/
|
||||||
|
alt?: string
|
||||||
|
/**
|
||||||
|
* Initials to display when no image is provided
|
||||||
|
*/
|
||||||
|
initials?: string | null
|
||||||
|
/**
|
||||||
|
* Size of the avatar
|
||||||
|
* @default 'md'
|
||||||
|
*/
|
||||||
|
size?: 'sm' | 'md' | 'lg'
|
||||||
|
/**
|
||||||
|
* Additional CSS class names
|
||||||
|
*/
|
||||||
|
className?: string
|
||||||
|
}
|
||||||
20
packages/design-system/lib/components/Avatar/variants.ts
Normal file
20
packages/design-system/lib/components/Avatar/variants.ts
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
import { cva } from 'class-variance-authority'
|
||||||
|
|
||||||
|
import styles from './avatar.module.css'
|
||||||
|
|
||||||
|
export const config = {
|
||||||
|
variants: {
|
||||||
|
size: {
|
||||||
|
sm: styles['size-sm'],
|
||||||
|
md: styles['size-md'],
|
||||||
|
lg: styles['size-lg'],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
defaultVariants: {
|
||||||
|
size: 'md',
|
||||||
|
},
|
||||||
|
} as const
|
||||||
|
|
||||||
|
export const variants = Object.assign(cva(styles.avatar, config), {
|
||||||
|
initials: styles.initials,
|
||||||
|
})
|
||||||
@@ -6,6 +6,7 @@
|
|||||||
"exports": {
|
"exports": {
|
||||||
"./Accordion": "./lib/components/Accordion/index.tsx",
|
"./Accordion": "./lib/components/Accordion/index.tsx",
|
||||||
"./Accordion/AccordionItem": "./lib/components/Accordion/AccordionItem/index.tsx",
|
"./Accordion/AccordionItem": "./lib/components/Accordion/AccordionItem/index.tsx",
|
||||||
|
"./Avatar": "./lib/components/Avatar/index.tsx",
|
||||||
"./BackToTopButton": "./lib/components/BackToTopButton/index.tsx",
|
"./BackToTopButton": "./lib/components/BackToTopButton/index.tsx",
|
||||||
"./Body": "./lib/components/Body/index.tsx",
|
"./Body": "./lib/components/Body/index.tsx",
|
||||||
"./BookingCodeChip": "./lib/components/BookingCodeChip/index.tsx",
|
"./BookingCodeChip": "./lib/components/BookingCodeChip/index.tsx",
|
||||||
|
|||||||
BIN
packages/design-system/public/img/profile-picture.png
Normal file
BIN
packages/design-system/public/img/profile-picture.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 2.6 KiB |
Reference in New Issue
Block a user