[WEB-4737] dev: propel pill component (#7743)

* dev: pill component added to propel

* dev: pill story added

* chore: propel config updated

* chore: code refactor
This commit is contained in:
Anmol Singh Bhatia 2025-09-10 00:15:37 +05:30 committed by GitHub
parent 3b8bb1effc
commit 1c8ac3d247
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
5 changed files with 219 additions and 0 deletions

View File

@ -29,6 +29,7 @@
"./emoji-icon-picker": "./dist/emoji-icon-picker/index.js",
"./icons": "./dist/icons/index.js",
"./menu": "./dist/menu/index.js",
"./pill": "./dist/pill/index.js",
"./popover": "./dist/popover/index.js",
"./scrollarea": "./dist/scrollarea/index.js",
"./skeleton": "./dist/skeleton/index.js",

View File

@ -0,0 +1,2 @@
export { Pill } from "./pill";
export type { PillProps } from "./pill";

View File

@ -0,0 +1,143 @@
import type { Meta, StoryObj } from "@storybook/react-vite";
import { Pill, EPillVariant, EPillSize } from "./pill";
const meta: Meta<typeof Pill> = {
title: "Components/Pill",
component: Pill,
parameters: {
layout: "centered",
},
tags: ["autodocs"],
argTypes: {
variant: {
control: "select",
options: Object.values(EPillVariant),
},
size: {
control: "select",
options: Object.values(EPillSize),
},
},
};
export default meta;
type Story = StoryObj<typeof Pill>;
export const Default: Story = {
args: {
children: "Default",
},
};
export const Primary: Story = {
args: {
variant: EPillVariant.PRIMARY,
children: "Primary",
},
};
export const Success: Story = {
args: {
variant: EPillVariant.SUCCESS,
children: "Success",
},
};
export const Warning: Story = {
args: {
variant: EPillVariant.WARNING,
children: "Warning",
},
};
export const Error: Story = {
args: {
variant: EPillVariant.ERROR,
children: "Error",
},
};
export const Info: Story = {
args: {
variant: EPillVariant.INFO,
children: "Info",
},
};
export const Small: Story = {
args: {
size: EPillSize.SM,
children: "Small",
},
};
export const Medium: Story = {
args: {
size: EPillSize.MD,
children: "Medium",
},
};
export const Large: Story = {
args: {
size: EPillSize.LG,
children: "Large",
},
};
export const AllVariants: Story = {
render: () => (
<div className="space-y-4">
<div className="flex flex-wrap gap-2">
<Pill variant={EPillVariant.DEFAULT}>Default</Pill>
<Pill variant={EPillVariant.PRIMARY}>Primary</Pill>
<Pill variant={EPillVariant.SUCCESS}>Success</Pill>
<Pill variant={EPillVariant.WARNING}>Warning</Pill>
<Pill variant={EPillVariant.ERROR}>Error</Pill>
<Pill variant={EPillVariant.INFO}>Info</Pill>
</div>
</div>
),
};
export const AllSizes: Story = {
render: () => (
<div className="space-y-4">
<div className="flex items-center gap-2">
<Pill size={EPillSize.SM}>Small</Pill>
<Pill size={EPillSize.MD}>Medium</Pill>
<Pill size={EPillSize.LG}>Large</Pill>
</div>
</div>
),
};
export const WithNumbers: Story = {
render: () => (
<div className="space-y-4">
<div className="flex flex-wrap gap-2">
<Pill variant={EPillVariant.PRIMARY}>3</Pill>
<Pill variant={EPillVariant.SUCCESS}>12</Pill>
<Pill variant={EPillVariant.WARNING}>99+</Pill>
<Pill variant={EPillVariant.ERROR}>!</Pill>
</div>
</div>
),
};
export const StatusExamples: Story = {
render: () => (
<div className="space-y-4">
<div className="space-y-2">
<h3 className="text-sm font-medium">Task Status</h3>
<div className="flex flex-wrap gap-2">
<Pill variant={EPillVariant.DEFAULT}>Draft</Pill>
<Pill variant={EPillVariant.WARNING}>In Progress</Pill>
<Pill variant={EPillVariant.INFO}>In Review</Pill>
<Pill variant={EPillVariant.SUCCESS}>Completed</Pill>
<Pill variant={EPillVariant.ERROR}>Blocked</Pill>
</div>
</div>
</div>
),
};

View File

@ -0,0 +1,72 @@
import * as React from "react";
import { cn } from "../utils";
export enum EPillVariant {
DEFAULT = "default",
PRIMARY = "primary",
SUCCESS = "success",
WARNING = "warning",
ERROR = "error",
INFO = "info",
}
export enum EPillSize {
SM = "sm",
MD = "md",
LG = "lg",
}
export type TPillVariant =
| EPillVariant.DEFAULT
| EPillVariant.PRIMARY
| EPillVariant.SUCCESS
| EPillVariant.WARNING
| EPillVariant.ERROR
| EPillVariant.INFO;
export type TPillSize = EPillSize.SM | EPillSize.MD | EPillSize.LG;
export interface PillProps extends React.HTMLAttributes<HTMLSpanElement> {
variant?: TPillVariant;
size?: TPillSize;
className?: string;
children: React.ReactNode;
}
const pillVariants = {
[EPillVariant.DEFAULT]: "bg-custom-background-90 text-custom-text-200 border border-custom-border-200",
[EPillVariant.PRIMARY]: "bg-custom-primary-100/10 text-custom-primary-100 border border-custom-primary-100/20",
[EPillVariant.SUCCESS]: "bg-green-50 text-green-700 border border-green-200",
[EPillVariant.WARNING]: "bg-amber-50 text-amber-700 border border-amber-200",
[EPillVariant.ERROR]: "bg-red-50 text-red-700 border border-red-200",
[EPillVariant.INFO]: "bg-blue-50 text-blue-700 border border-blue-200",
};
const pillSizes = {
[EPillSize.SM]: "px-2 py-0.5 text-xs",
[EPillSize.MD]: "px-2.5 py-1 text-sm",
[EPillSize.LG]: "px-3 py-1.5 text-base",
};
const Pill = React.forwardRef<HTMLSpanElement, PillProps>(
({ variant = EPillVariant.DEFAULT, size = EPillSize.MD, className, children, ...props }, ref) => (
<span
ref={ref}
className={cn(
// Base styles
"inline-flex items-center justify-center rounded-full font-medium whitespace-nowrap",
// Variant styles
pillVariants[variant],
// Size styles
pillSizes[size],
className
)}
{...props}
>
{children}
</span>
)
);
Pill.displayName = "Pill";
export { Pill };

View File

@ -15,6 +15,7 @@ export default defineConfig({
"src/emoji-icon-picker/index.ts",
"src/icons/index.ts",
"src/menu/index.ts",
"src/pill/index.ts",
"src/popover/index.ts",
"src/scrollarea/index.ts",
"src/skeleton/index.ts",