2023-02-01 20:41:47 +00:00
|
|
|
// @TODO: turn this into a more generic component that has the same Props API as MUI https://mui.com/material-ui/react-card/
|
2023-06-20 17:54:08 +00:00
|
|
|
import type { VariantProps } from "class-variance-authority";
|
|
|
|
import { cva } from "class-variance-authority";
|
2022-08-24 20:18:42 +00:00
|
|
|
import Link from "next/link";
|
2023-02-16 22:39:57 +00:00
|
|
|
import type { ReactNode } from "react";
|
2022-08-24 20:18:42 +00:00
|
|
|
import React from "react";
|
|
|
|
|
|
|
|
import classNames from "@calcom/lib/classNames";
|
2023-04-12 15:26:31 +00:00
|
|
|
import { ArrowRight } from "@calcom/ui/components/icon";
|
2022-08-24 20:18:42 +00:00
|
|
|
|
2023-01-10 02:22:04 +00:00
|
|
|
import { Button } from "../button";
|
2022-08-24 20:18:42 +00:00
|
|
|
|
2023-06-20 17:54:08 +00:00
|
|
|
const cvaCardTypeByVariant = cva("", {
|
|
|
|
// Variants won't have any style by default. Style will only be applied if the variants are combined.
|
|
|
|
// So, style is defined in compoundVariants.
|
|
|
|
variants: {
|
|
|
|
variant: {
|
|
|
|
basic: "",
|
|
|
|
ProfileCard: "",
|
|
|
|
SidebarCard: "",
|
|
|
|
},
|
|
|
|
structure: {
|
|
|
|
image: "",
|
|
|
|
card: "",
|
|
|
|
title: "",
|
|
|
|
description: "",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
compoundVariants: [
|
|
|
|
// Style for Basic Variants types
|
|
|
|
{
|
|
|
|
variant: "basic",
|
|
|
|
structure: "image",
|
|
|
|
className: "w-10 h-auto",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
variant: "basic",
|
|
|
|
structure: "card",
|
|
|
|
className: "p-5",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
variant: "basic",
|
|
|
|
structure: "title",
|
|
|
|
className: "text-base mt-4",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
variant: "basic",
|
|
|
|
structure: "description",
|
|
|
|
className: "text-sm leading-[18px] text-subtle font-normal",
|
|
|
|
},
|
|
|
|
|
|
|
|
// Style for ProfileCard Variant Types
|
|
|
|
{
|
|
|
|
variant: "ProfileCard",
|
|
|
|
structure: "image",
|
|
|
|
className: "w-9 h-auto rounded-full mb-4s",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
variant: "ProfileCard",
|
|
|
|
structure: "card",
|
|
|
|
className: "w-80 p-4 hover:bg-subtle",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
variant: "ProfileCard",
|
|
|
|
structure: "title",
|
|
|
|
className: "text-base",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
variant: "ProfileCard",
|
|
|
|
structure: "description",
|
|
|
|
className: "text-sm leading-[18px] text-subtle font-normal",
|
|
|
|
},
|
|
|
|
|
|
|
|
// Style for SidebarCard Variant Types
|
|
|
|
{
|
|
|
|
variant: "SidebarCard",
|
|
|
|
structure: "image",
|
|
|
|
className: "w-9 h-auto rounded-full mb-4s",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
variant: "SidebarCard",
|
|
|
|
structure: "card",
|
|
|
|
className: "w-full p-3 border border-subtle",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
variant: "SidebarCard",
|
|
|
|
structure: "title",
|
|
|
|
className: "text-sm font-cal",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
variant: "SidebarCard",
|
|
|
|
structure: "description",
|
|
|
|
className: "text-xs text-default line-clamp-2",
|
|
|
|
},
|
|
|
|
],
|
|
|
|
});
|
|
|
|
|
|
|
|
type CVACardType = Required<Pick<VariantProps<typeof cvaCardTypeByVariant>, "variant">>;
|
|
|
|
|
|
|
|
export interface BaseCardProps extends CVACardType {
|
2022-08-24 20:18:42 +00:00
|
|
|
image?: string;
|
2023-02-01 20:41:47 +00:00
|
|
|
icon?: ReactNode;
|
2022-08-24 20:18:42 +00:00
|
|
|
imageProps?: JSX.IntrinsicElements["img"];
|
|
|
|
title: string;
|
|
|
|
description: ReactNode;
|
|
|
|
containerProps?: JSX.IntrinsicElements["div"];
|
|
|
|
actionButton?: {
|
|
|
|
href?: string;
|
|
|
|
child: ReactNode;
|
|
|
|
onClick?: (event: React.MouseEvent<HTMLElement, MouseEvent>) => void;
|
|
|
|
};
|
|
|
|
learnMore?: {
|
|
|
|
href: string;
|
|
|
|
text: string;
|
|
|
|
};
|
|
|
|
mediaLink?: string;
|
|
|
|
thumbnailUrl?: string;
|
2023-07-20 05:06:15 +00:00
|
|
|
structure?: string;
|
2023-06-20 17:54:08 +00:00
|
|
|
}
|
2022-08-24 20:18:42 +00:00
|
|
|
|
|
|
|
export function Card({
|
|
|
|
image,
|
|
|
|
title,
|
2023-02-01 20:41:47 +00:00
|
|
|
icon,
|
2022-08-24 20:18:42 +00:00
|
|
|
description,
|
|
|
|
variant,
|
|
|
|
actionButton,
|
|
|
|
containerProps,
|
|
|
|
imageProps,
|
|
|
|
mediaLink,
|
|
|
|
thumbnailUrl,
|
|
|
|
learnMore,
|
|
|
|
}: BaseCardProps) {
|
2023-04-03 17:25:45 +00:00
|
|
|
const LinkComponent = learnMore && learnMore.href.startsWith("https") ? "a" : Link;
|
2022-08-24 20:18:42 +00:00
|
|
|
return (
|
|
|
|
<div
|
|
|
|
className={classNames(
|
|
|
|
containerProps?.className,
|
2023-06-20 17:54:08 +00:00
|
|
|
cvaCardTypeByVariant({ variant, structure: "card" }),
|
2023-04-05 18:14:46 +00:00
|
|
|
"bg-default border-subtle text-default flex flex-col justify-between rounded-md border"
|
2022-08-24 20:18:42 +00:00
|
|
|
)}
|
2023-07-20 05:06:15 +00:00
|
|
|
data-testid="card-container"
|
2022-08-24 20:18:42 +00:00
|
|
|
{...containerProps}>
|
2023-02-01 20:41:47 +00:00
|
|
|
<div>
|
|
|
|
{icon && icon}
|
|
|
|
{image && (
|
|
|
|
<img
|
|
|
|
src={image}
|
|
|
|
// Stops eslint complaining - not smart enough to realise it comes from ...imageProps
|
|
|
|
alt={imageProps?.alt}
|
2023-06-20 17:54:08 +00:00
|
|
|
className={classNames(
|
|
|
|
imageProps?.className,
|
|
|
|
cvaCardTypeByVariant({ variant, structure: "image" })
|
|
|
|
)}
|
2023-02-01 20:41:47 +00:00
|
|
|
{...imageProps}
|
|
|
|
/>
|
|
|
|
)}
|
|
|
|
<h5
|
|
|
|
title={title}
|
|
|
|
className={classNames(
|
2023-06-20 17:54:08 +00:00
|
|
|
cvaCardTypeByVariant({ variant, structure: "title" }),
|
2023-06-22 22:25:37 +00:00
|
|
|
"text-emphasis line-clamp-1 font-bold leading-5"
|
2023-02-01 20:41:47 +00:00
|
|
|
)}>
|
|
|
|
{title}
|
|
|
|
</h5>
|
|
|
|
{description && (
|
|
|
|
<p
|
|
|
|
title={description.toString()}
|
2023-06-20 17:54:08 +00:00
|
|
|
className={classNames(cvaCardTypeByVariant({ variant, structure: "description" }), "pt-1")}>
|
2023-02-01 20:41:47 +00:00
|
|
|
{description}
|
|
|
|
</p>
|
|
|
|
)}
|
|
|
|
</div>
|
2022-08-24 20:18:42 +00:00
|
|
|
{variant === "SidebarCard" && (
|
|
|
|
<a
|
2022-10-31 08:11:24 +00:00
|
|
|
onClick={actionButton?.onClick}
|
2022-08-24 20:18:42 +00:00
|
|
|
target="_blank"
|
|
|
|
rel="noreferrer"
|
|
|
|
href={mediaLink}
|
|
|
|
className="group relative my-3 flex aspect-video items-center overflow-hidden rounded">
|
|
|
|
<div className="absolute inset-0 bg-black bg-opacity-50 transition-opacity group-hover:bg-opacity-40" />
|
|
|
|
<svg
|
2023-06-22 22:25:37 +00:00
|
|
|
className="text-inverted absolute left-1/2 top-1/2 h-8 w-8 -translate-x-1/2 -translate-y-1/2 transform rounded-full shadow-lg hover:-mt-px"
|
2022-08-24 20:18:42 +00:00
|
|
|
viewBox="0 0 32 32"
|
|
|
|
fill="none"
|
|
|
|
xmlns="http://www.w3.org/2000/svg">
|
|
|
|
<path
|
|
|
|
d="M16 32C24.8366 32 32 24.8366 32 16C32 7.16344 24.8366 0 16 0C7.16344 0 0 7.16344 0 16C0 24.8366 7.16344 32 16 32Z"
|
|
|
|
fill="white"
|
|
|
|
/>
|
|
|
|
<path
|
|
|
|
d="M12.1667 8.5L23.8334 16L12.1667 23.5V8.5Z"
|
|
|
|
fill="#111827"
|
|
|
|
stroke="#111827"
|
|
|
|
strokeWidth="1.5"
|
|
|
|
strokeLinecap="round"
|
|
|
|
strokeLinejoin="round"
|
|
|
|
/>
|
|
|
|
</svg>
|
|
|
|
<img alt="play feature video" src={thumbnailUrl} />
|
|
|
|
</a>
|
|
|
|
)}
|
2023-02-01 20:41:47 +00:00
|
|
|
|
|
|
|
{/* TODO: this should be CardActions https://mui.com/material-ui/api/card-actions/ */}
|
|
|
|
<div>
|
|
|
|
{variant === "basic" && (
|
2023-04-12 15:26:31 +00:00
|
|
|
<Button color="secondary" href={actionButton?.href} className="mt-10" EndIcon={ArrowRight}>
|
2023-02-01 20:41:47 +00:00
|
|
|
{actionButton?.child}
|
|
|
|
</Button>
|
|
|
|
)}
|
|
|
|
</div>
|
|
|
|
|
2022-08-24 20:18:42 +00:00
|
|
|
{variant === "SidebarCard" && (
|
|
|
|
<div className="mt-2 flex items-center justify-between">
|
|
|
|
{learnMore && (
|
2023-04-03 17:25:45 +00:00
|
|
|
<LinkComponent
|
2023-01-06 12:13:56 +00:00
|
|
|
href={learnMore.href}
|
|
|
|
onClick={actionButton?.onClick}
|
|
|
|
target="_blank"
|
|
|
|
rel="noreferrer"
|
2023-04-05 18:14:46 +00:00
|
|
|
className="text-default text-xs font-medium">
|
2023-01-06 12:13:56 +00:00
|
|
|
{learnMore.text}
|
2023-04-03 17:25:45 +00:00
|
|
|
</LinkComponent>
|
2022-08-24 20:18:42 +00:00
|
|
|
)}
|
2023-05-25 14:58:44 +00:00
|
|
|
{actionButton?.child && (
|
|
|
|
<button
|
|
|
|
className="text-default hover:text-emphasis p-0 text-xs font-normal"
|
|
|
|
color="minimal"
|
|
|
|
onClick={actionButton?.onClick}>
|
|
|
|
{actionButton?.child}
|
|
|
|
</button>
|
|
|
|
)}
|
2022-08-24 20:18:42 +00:00
|
|
|
</div>
|
|
|
|
)}
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
export default Card;
|