Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Enhancement : Badge Component #66555

Merged
merged 24 commits into from
Dec 13, 2024
Merged
Show file tree
Hide file tree
Changes from 4 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
6 changes: 6 additions & 0 deletions docs/manifest.json
Original file line number Diff line number Diff line change
Expand Up @@ -683,6 +683,12 @@
"markdown_source": "../packages/components/src/autocomplete/README.md",
"parent": "components"
},
{
"title": "Badge",
"slug": "badge",
"markdown_source": "../packages/components/src/badge/README.md",
"parent": "components"
},
Vrishabhsk marked this conversation as resolved.
Show resolved Hide resolved
{
"title": "BaseControl",
"slug": "base-control",
Expand Down
59 changes: 59 additions & 0 deletions packages/components/src/badge/README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,59 @@
# Badge
Vrishabhsk marked this conversation as resolved.
Show resolved Hide resolved

`Badge` is a `reusable component` to display important information in (not limited to) `data-views`.

## Usage

```jsx
import { Badge } from '@wordpress/components';

const ExampleBadge = () => {
return (
<Badge className="my-badge" as="span">
Code is Poetry
</Badge>
);
};
```

## Props

### `className`: `string`

Additional classes for the badge component.

- Required: No

### `icon`: `IconType`

Icon to be displayed within the badge component.

- Required: No

### `as`: `ElementType`

Component type that will be used to render the badge component.

- Required: No
- Default: `div`

### `variant`: `'generic' | 'info' | 'success' | 'warning' | 'error'`

Variant of the badge component.

- Required: No
- Default: `'generic'`
-

### `showContext`: `boolean`

Whether to display the badge with a contextual message when variant is set other than `'generic'`.

- Required: No
- Default: `true`

### `children`: `ReactNode`

The content to be displayed within the component.

- Required: Yes
54 changes: 54 additions & 0 deletions packages/components/src/badge/index.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,54 @@
/**
* External dependencies
*/
import clsx from 'clsx';

/**
* Internal dependencies
*/
import type { BadgeProps } from './types';
import Icon from '../icon';

function Badge( {
className,
icon,
as: Component = 'div',
variant = 'generic',
showContext = true,
children,
...props
}: BadgeProps ) {
Vrishabhsk marked this conversation as resolved.
Show resolved Hide resolved
/**
* Formats the variant string to be displayed when showContext is true.
*
* @param {string} str
*
* @return {string} Formatted variant string.
*/
function formatVariant( str: string ): string {
return (
str.charAt( 0 ).toUpperCase() + str.slice( 1 ).toLowerCase() + ': '
);
}

return (
<Component
className={ clsx(
'components-badge',
`components-badge--${ variant }`,
className
) }
{ ...props }
>
{ icon && <Icon icon={ icon } /> }
<span>
{ showContext &&
variant !== 'generic' &&
formatVariant( variant ) }
</span>
{ children }
</Component>
);
}

export default Badge;
74 changes: 74 additions & 0 deletions packages/components/src/badge/stories/index.story.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,74 @@
/**
* External dependencies
*/
import type { Meta, StoryFn } from '@storybook/react';

/**
* Internal dependencies
*/
import Badge from '..';

/**
* WordPress dependencies
*/
import { info, bug, help, published } from '@wordpress/icons';

const meta: Meta< typeof Badge > = {
component: Badge,
title: 'Components/Containers/Badge',
argTypes: {
className: {
control: { type: 'text' },
},
icon: {
control: { type: 'select' },
options: [ '-', 'info', 'bug', 'help', 'published' ],
mapping: {
'-': undefined,
info,
bug,
help,
published,
},
},
as: {
control: { type: 'select' },
options: [ 'div', 'span' ],
},
children: {
control: { type: null },
},
},
};

export default meta;

const Template: StoryFn< typeof Badge > = ( args ) => {
Vrishabhsk marked this conversation as resolved.
Show resolved Hide resolved
return <Badge { ...args } />;
};

export const Default = Template.bind( {} );
Default.args = {
children: 'Code is Poetry',
};

export const WithIcon = Template.bind( {} );
WithIcon.args = {
children: 'Code is Poetry',
icon: bug,
variant: 'error',
};

export const WithVariant = Template.bind( {} );
WithVariant.args = {
children: 'Code is Poetry',
variant: 'success',
};

export const WithoutContext = Template.bind( {} );
WithoutContext.args = {
children: 'Code is Poetry',
icon: help,
variant: 'warning',
showContext: false,
};
34 changes: 34 additions & 0 deletions packages/components/src/badge/styles.scss
Original file line number Diff line number Diff line change
@@ -0,0 +1,34 @@
@mixin badge-color-variant( $base-color ) {
background-color: mix($white, $base-color, 90%);
color: mix($black, $base-color, 50%);
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@WordPress/gutenberg-design Have we considered what we want to happen when a Badge is placed on a dark background? (This can be a follow-up)

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

What's the concern?

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Just thinking forward about theming. The default badge is subtle on a white background, but eye catching on a dark one. Color differences are also harder to perceive, I think, especially in isolation.

Default badge Success badge

FWIW, the default badge looks like this when rendered in the current theming system:

Default badge in dark theming

(To be clear, these issues aren't blockers for this PR to land.)

}

$badge-colors: (
"info": #3858e9,
"warning": $alert-yellow,
"error": $alert-red,
"success": $alert-green,
);

.components-badge {
background: $gray-100;
color: $gray-800;
padding: $grid-unit-05 $grid-unit-10;
min-height: $grid-unit-30;
border-radius: $radius-small;
font-size: $font-size-small;
font-weight: 400;
flex-shrink: 0;
line-height: $font-line-height-small;
width: fit-content;
display: flex;
align-items: center;
gap: 2px;

// Generate color variants
@each $type, $color in $badge-colors {
&.components-badge--#{$type} {
@include badge-color-variant( $color );
}
}
Vrishabhsk marked this conversation as resolved.
Show resolved Hide resolved
}
66 changes: 66 additions & 0 deletions packages/components/src/badge/test/index.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,66 @@
/**
* External dependencies
*/
import { render, screen } from '@testing-library/react';

/**
* Internal dependencies
*/
import Badge from '..';

describe( 'Badge', () => {
it( 'should render correctly with default props', () => {
render( <Badge>Code is Poetry</Badge> );
const badge = screen.getByText( 'Code is Poetry' );
expect( badge ).toBeInTheDocument();
expect( badge.tagName ).toBe( 'DIV' ); // Default element should be a div
expect( badge ).toHaveClass( 'components-badge' );
} );

it( 'should render as a span when specified', () => {
render( <Badge as="span">Code is Poetry</Badge> );
const badge = screen.getByText( 'Code is Poetry' );
expect( badge.tagName ).toBe( 'SPAN' );
expect( badge ).toHaveClass( 'components-badge' );
} );

it( 'should render as a custom element when specified', () => {
render( <Badge as="article">Code is Poetry</Badge> );
const badge = screen.getByText( 'Code is Poetry' );
expect( badge.tagName ).toBe( 'ARTICLE' );
expect( badge ).toHaveClass( 'components-badge' );
} );

it( 'should combine custom className with default class', () => {
render( <Badge className="custom-class">Code is Poetry</Badge> );
const badge = screen.getByText( 'Code is Poetry' );
expect( badge ).toHaveClass( 'components-badge' );
expect( badge ).toHaveClass( 'custom-class' );
} );

it( 'should render children correctly', () => {
render(
<Badge>
<span>Nested</span> Content
</Badge>
);

const badge = screen.getByText( ( content, element ) => {
return element?.classList?.contains( 'components-badge' ) ?? false;
} );

expect( badge ).toBeInTheDocument();
expect( badge ).toHaveClass( 'components-badge' );
expect( badge ).toHaveTextContent( 'Nested Content' );

const nestedSpan = screen.getByText( 'Nested' );
expect( nestedSpan.tagName ).toBe( 'SPAN' );
} );

it( 'should pass through additional props', () => {
render( <Badge data-testid="custom-badge">Code is Poetry</Badge> );
const badge = screen.getByTestId( 'custom-badge' );
expect( badge ).toHaveTextContent( 'Code is Poetry' );
expect( badge ).toHaveClass( 'components-badge' );
} );
} );
42 changes: 42 additions & 0 deletions packages/components/src/badge/types.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,42 @@
/**
* External dependencies
*/
import type { ElementType, ReactNode } from 'react';

/**
* Internal dependencies
*/
import type { IconType } from '../icon';

export type BadgeProps = {
/**
* Element to display inside the badge.
*/
children: ReactNode;
/**
* Additional classes for the badge component.
*/
className?: string;
/**
* Optional Icon to display inside the badge.
*/
icon?: IconType;
/**
* Component type that will be used to render the badge component.
*
* @default 'div'
*/
as?: ElementType;
Vrishabhsk marked this conversation as resolved.
Show resolved Hide resolved
/**
* Badge variant.
*
* @default 'generic'
Vrishabhsk marked this conversation as resolved.
Show resolved Hide resolved
*/
variant?: 'generic' | 'info' | 'success' | 'warning' | 'error';
/**
* Show context for the type of badge.
*
* @default true
*/
showContext?: boolean;
};
1 change: 1 addition & 0 deletions packages/components/src/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -241,6 +241,7 @@ export {
} from './higher-order/with-focus-return';
export { default as withNotices } from './higher-order/with-notices';
export { default as withSpokenMessages } from './higher-order/with-spoken-messages';
export { default as Badge } from './badge';
Vrishabhsk marked this conversation as resolved.
Show resolved Hide resolved

// Private APIs.
export { privateApis } from './private-apis';
1 change: 1 addition & 0 deletions packages/components/src/style.scss
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,7 @@
// Components
@import "./animate/style.scss";
@import "./autocomplete/style.scss";
@import "./badge/styles.scss";
@import "./button-group/style.scss";
@import "./button/style.scss";
@import "./checkbox-control/style.scss";
Expand Down
Loading