From ff257cfc6f37ec7f1fac3ce77c4fd6b58abba44c Mon Sep 17 00:00:00 2001 From: Jake Laderman Date: Thu, 19 Dec 2024 16:34:59 -0500 Subject: [PATCH] feat: add header option to cards (#671) --- src/components/Card.tsx | 163 ++++++++++++++------ src/components/Chip.tsx | 27 +++- src/components/IconFrame.tsx | 2 +- src/components/icons/CollapseListIcon.tsx | 55 +++++++ src/components/icons/CostManagementIcon.tsx | 28 ++++ src/components/icons/ExpandListIcon.tsx | 55 +++++++ src/components/table/Table.tsx | 1 + src/icons.ts | 5 +- src/index.ts | 7 +- src/stories/Card.stories.tsx | 53 ++++++- src/stories/Chip.stories.tsx | 13 +- src/types/react-table.d.ts | 3 +- 12 files changed, 338 insertions(+), 74 deletions(-) create mode 100644 src/components/icons/CollapseListIcon.tsx create mode 100644 src/components/icons/CostManagementIcon.tsx create mode 100644 src/components/icons/ExpandListIcon.tsx diff --git a/src/components/Card.tsx b/src/components/Card.tsx index ff4be8d5d..26546ac6b 100644 --- a/src/components/Card.tsx +++ b/src/components/Card.tsx @@ -1,20 +1,18 @@ import chroma from 'chroma-js' import { Div, type DivProps } from 'honorable' -import { forwardRef } from 'react' -import styled, { type DefaultTheme } from 'styled-components' import { memoize } from 'lodash-es' +import { type ComponentProps, type ReactNode, forwardRef } from 'react' +import styled, { type DefaultTheme } from 'styled-components' import { type Severity, type SeverityExt, sanitizeSeverity } from '../types' import { type FillLevel, FillLevelProvider, - isFillLevel, toFillLevel, useFillLevel, } from './contexts/FillLevelContext' - -const HUES = ['default', 'lighter', 'lightest'] as const +import WrapWithIf from './WrapWithIf' const CARD_SEVERITIES = [ 'info', @@ -27,12 +25,9 @@ const CARD_SEVERITIES = [ type CornerSize = 'medium' | 'large' type CardFillLevel = Exclude -type CardHue = (typeof HUES)[number] type CardSeverity = Extract type BaseCardProps = { - /** @deprecated Colors set by `FillLevelContext`. If you need to override context, use `fillLevel` */ - hue?: CardHue /** Used to override a fill level set by `FillLevelContext` */ fillLevel?: FillLevel cornerSize?: CornerSize @@ -40,6 +35,12 @@ type BaseCardProps = { disabled?: boolean selected?: boolean severity?: SeverityExt + header?: { + size?: 'medium' | 'large' + content?: ReactNode + headerProps?: ComponentProps<'div'> + outerProps?: ComponentProps<'div'> + } } type CardProps = DivProps & BaseCardProps @@ -65,12 +66,6 @@ const fillToNeutralHoverBgC = { 3: 'fill-three-hover', } as const satisfies Record -const hueToFill = { - default: 1, - lighter: 2, - lightest: 3, -} as const satisfies Record - const fillToNeutralSelectedBgC = { 0: 'fill-one-selected', 1: 'fill-one-selected', @@ -78,22 +73,14 @@ const fillToNeutralSelectedBgC = { 3: 'fill-three-selected', } as const satisfies Record -export function useDecideFillLevel({ - hue, - fillLevel, -}: { - hue?: CardHue - fillLevel?: number -}) { +export function useDecideFillLevel({ fillLevel }: { fillLevel?: number }) { const parentFillLevel = useFillLevel() - if (isFillLevel(fillLevel)) { - return toFillLevel(Math.max(1, fillLevel)) as CardFillLevel - } - - return isFillLevel(hueToFill[hue]) - ? hueToFill[hue] - : (toFillLevel(parentFillLevel + 1) as CardFillLevel) + return ( + typeof fillLevel === 'number' + ? toFillLevel(Math.max(1, fillLevel)) + : toFillLevel(parentFillLevel + 1) + ) as CardFillLevel } export const getFillToLightBgC = memoize( @@ -151,7 +138,38 @@ const getBgColor = ({ return fillToLightBgC[severity][fillLevel] } +const HeaderSC = styled.div<{ + $fillLevel: CardFillLevel + $selected: boolean + $size: 'medium' | 'large' + $cornerSize: CornerSize +}>( + ({ + theme, + $fillLevel: fillLevel, + $selected: selected, + $size: size, + $cornerSize: cornerSize, + }) => ({ + ...theme.partials.text.overline, + flexShrink: 0, + display: 'flex', + alignItems: 'center', + color: theme.colors['text-xlight'], + border: `1px solid ${theme.colors[fillToNeutralBorderC[fillLevel]]}`, + borderBottom: 'none', + borderRadius: `${theme.borderRadiuses[cornerSize]}px ${theme.borderRadiuses[cornerSize]}px 0 0`, + backgroundColor: selected + ? theme.colors[fillToNeutralSelectedBgC[fillLevel]] + : getBgColor({ theme, fillLevel }), + height: size === 'large' ? 48 : 40, + padding: `0 ${theme.spacing.medium}px`, + overflow: 'hidden', + }) +) + const CardSC = styled(Div)<{ + $hasHeader: boolean $fillLevel: CardFillLevel $cornerSize: CornerSize $severity: Severity @@ -161,6 +179,7 @@ const CardSC = styled(Div)<{ }>( ({ theme, + $hasHeader, $fillLevel: fillLevel, $cornerSize: cornerSize, $severity: severity, @@ -169,8 +188,16 @@ const CardSC = styled(Div)<{ $disabled: disabled, }) => ({ ...theme.partials.reset.button, - border: `1px solid ${theme.colors[fillToNeutralBorderC[fillLevel]]}`, - borderRadius: theme.borderRadiuses[cornerSize], + border: `1px solid ${ + theme.colors[ + fillToNeutralBorderC[ + $hasHeader ? toFillLevel(fillLevel + 1) : fillLevel + ] + ] + }`, + borderRadius: $hasHeader + ? `0 0 ${theme.borderRadiuses[cornerSize]}px ${theme.borderRadiuses[cornerSize]}px` + : theme.borderRadiuses[cornerSize], backgroundColor: selected ? theme.colors[fillToNeutralSelectedBgC[fillLevel]] : getBgColor({ theme, fillLevel }), @@ -196,47 +223,85 @@ const CardSC = styled(Div)<{ }) ) +const OuterWrapSC = styled.div({ + display: 'flex', + flexDirection: 'column', + overflow: 'hidden', + width: '100%', + height: '100%', +}) + const Card = forwardRef( ( { + header, cornerSize = 'large', - hue, // Deprecated, prefer fillLevel severity = 'neutral', fillLevel, selected = false, clickable = false, disabled = false, + children, ...props }: CardProps, ref ) => { - fillLevel = useDecideFillLevel({ hue, fillLevel }) + const hasHeader = !!header + const { + size, + content: headerContent, + headerProps, + outerProps, + } = header ?? {} + + const mainFillLevel = useDecideFillLevel({ fillLevel }) + const headerFillLevel = useDecideFillLevel({ fillLevel: mainFillLevel + 1 }) + const cardSeverity = sanitizeSeverity(severity, { allowList: CARD_SEVERITIES, default: 'neutral', }) return ( - - + + } + > + {header && ( + + {headerContent} + + )} + + {children} + + ) } ) export default Card -export type { BaseCardProps, CardProps, CornerSize, CardHue, CardFillLevel } +export type { BaseCardProps, CardFillLevel, CardProps, CornerSize } diff --git a/src/components/Chip.tsx b/src/components/Chip.tsx index 0ffe712f5..b346f515d 100644 --- a/src/components/Chip.tsx +++ b/src/components/Chip.tsx @@ -22,14 +22,15 @@ import Tooltip from './Tooltip' export const CHIP_CLOSE_ATTR_KEY = 'data-close-button' as const const SIZES = ['small', 'medium', 'large'] as const -type ChipSize = (typeof SIZES)[number] -type ChipSeverity = (typeof SEVERITIES)[number] +export type ChipSize = (typeof SIZES)[number] +export type ChipSeverity = (typeof SEVERITIES)[number] export type ChipProps = Omit & BaseCardProps & { size?: ChipSize condensed?: boolean severity?: ChipSeverity + inactive?: boolean icon?: ReactElement loading?: boolean closeButton?: boolean @@ -73,15 +74,26 @@ const sizeToCloseHeight = { const ChipCardSC = styled(Card)<{ $size: ChipSize $severity: ChipSeverity + $inactive: boolean $truncateWidth?: number $truncateEdge?: 'start' | 'end' $condensed?: boolean -}>(({ $size, $severity, $truncateWidth, $truncateEdge, $condensed, theme }) => { - const textColor = - theme.colors[severityToColor[$severity]] || theme.colors.text +}>(({ + $size, + $severity, + $inactive, + $truncateWidth, + $truncateEdge, + $condensed, + theme, +}) => { + const textColor = $inactive + ? theme.colors['text-xlight'] + : theme.colors[severityToColor[$severity]] ?? theme.colors.text return { '&&': { + backgroundColor: $inactive ? 'transparent' : undefined, padding: `${$size === 'large' ? 6 : theme.spacing.xxxsmall}px ${ $size === 'large' && $condensed ? 6 @@ -164,9 +176,9 @@ function ChipRef( size = 'medium', condensed = false, severity = 'neutral', + inactive = false, truncateWidth, truncateEdge, - hue, fillLevel, loading = false, icon, @@ -180,7 +192,7 @@ function ChipRef( }: ChipProps, ref: Ref ) { - fillLevel = useDecideFillLevel({ hue, fillLevel }) + fillLevel = useDecideFillLevel({ fillLevel }) const theme = useTheme() const iconCol = severityToIconColor[severity] || 'icon-default' @@ -193,6 +205,7 @@ function ChipRef( fillLevel={fillLevel} clickable={clickable} disabled={clickable && disabled} + $inactive={inactive} $size={size} $condensed={condensed} $severity={severity} diff --git a/src/components/IconFrame.tsx b/src/components/IconFrame.tsx index ee6684919..25fd93cb1 100644 --- a/src/components/IconFrame.tsx +++ b/src/components/IconFrame.tsx @@ -74,7 +74,7 @@ const sizeToIconSize: Record = { xsmall: 8, small: 16, medium: 16, - large: 24, + large: 16, xlarge: 24, } diff --git a/src/components/icons/CollapseListIcon.tsx b/src/components/icons/CollapseListIcon.tsx new file mode 100644 index 000000000..a94d85403 --- /dev/null +++ b/src/components/icons/CollapseListIcon.tsx @@ -0,0 +1,55 @@ +import createIcon from './createIcon' + +export default createIcon(({ size, color }) => ( + + + + + + + + + +)) diff --git a/src/components/icons/CostManagementIcon.tsx b/src/components/icons/CostManagementIcon.tsx new file mode 100644 index 000000000..abbaf68ef --- /dev/null +++ b/src/components/icons/CostManagementIcon.tsx @@ -0,0 +1,28 @@ +import createIcon from './createIcon' + +export default createIcon(({ size, color }) => ( + + + + + +)) diff --git a/src/components/icons/ExpandListIcon.tsx b/src/components/icons/ExpandListIcon.tsx new file mode 100644 index 000000000..54dcebd5a --- /dev/null +++ b/src/components/icons/ExpandListIcon.tsx @@ -0,0 +1,55 @@ +import createIcon from './createIcon' + +export default createIcon(({ size, color }) => ( + + + + + + + + + +)) diff --git a/src/components/table/Table.tsx b/src/components/table/Table.tsx index c15b8d916..d36223aa5 100644 --- a/src/components/table/Table.tsx +++ b/src/components/table/Table.tsx @@ -324,6 +324,7 @@ function TableRef( setScrollTop(target?.scrollTop) } width="100%" + height="100%" {...props} > diff --git a/src/icons.ts b/src/icons.ts index f626a48d4..23a3c443d 100644 --- a/src/icons.ts +++ b/src/icons.ts @@ -46,6 +46,7 @@ export { default as CloseRoundedIcon } from './components/icons/CloseRoundedIcon export { default as CloudIcon } from './components/icons/CloudIcon' export { default as ClusterIcon } from './components/icons/ClusterIcon' export { default as CollapseIcon } from './components/icons/CollapseIcon' +export { default as CollapseListIcon } from './components/icons/CollapseListIcon' export { default as CommandIcon } from './components/icons/CommandIcon' export { default as CompassIcon } from './components/icons/CompassIcon' export { default as CompatibilityIcon } from './components/icons/CompatibilityIcon' @@ -56,6 +57,7 @@ export { default as ConfettiIcon } from './components/icons/ConfettiIcon' export { default as ConsoleIcon } from './components/icons/ConsoleIcon' export { default as CookieIcon } from './components/icons/CookieIcon' export { default as CopyIcon } from './components/icons/CopyIcon' +export { default as CostManagementIcon } from './components/icons/CostManagementIcon' export { default as CpuIcon } from './components/icons/CpuIcon' export { default as CraneIcon } from './components/icons/CraneIcon' export { default as CreditCardIcon } from './components/icons/CreditCardIcon' @@ -78,6 +80,7 @@ export { default as EmojiHoverIcon } from './components/icons/EmojiHoverIcon' export { default as EmojiIcon } from './components/icons/EmojiIcon' export { default as ErrorIcon } from './components/icons/ErrorIcon' export { default as ExpandIcon } from './components/icons/ExpandIcon' +export { default as ExpandListIcon } from './components/icons/ExpandListIcon' export { default as EyeClosedIcon } from './components/icons/EyeClosedIcon' export { default as EyeIcon } from './components/icons/EyeIcon' export { default as FastForwardIcon } from './components/icons/FastForwardIcon' @@ -144,7 +147,6 @@ export { default as PadlockIcon } from './components/icons/PadlockIcon' export { default as PadlockLockedIcon } from './components/icons/PadlockLockedIcon' export { default as PaperclipIcon } from './components/icons/PaperclipIcon' export { default as PauseIcon } from './components/icons/PauseIcon' -export { default as RamIcon } from './components/icons/RamIcon' export { default as PencilIcon } from './components/icons/PencilIcon' export { default as PeopleIcon } from './components/icons/PeopleIcon' export { default as PeoplePlusIcon } from './components/icons/PeoplePlusIcon' @@ -164,6 +166,7 @@ export { default as ProtectedManagementClusterIcon } from './components/icons/Pr export { default as PrQueueIcon } from './components/icons/PrQueueIcon' export { default as PushPinFilledIcon } from './components/icons/PushPinFilledIcon' export { default as PushPinOutlineIcon } from './components/icons/PushPinOutlineIcon' +export { default as RamIcon } from './components/icons/RamIcon' export { default as ReloadIcon } from './components/icons/ReloadIcon' export { default as RestoreIcon } from './components/icons/RestoreIcon' export { default as ReturnIcon } from './components/icons/ReturnIcon' diff --git a/src/index.ts b/src/index.ts index fd61fb751..809c2dda9 100644 --- a/src/index.ts +++ b/src/index.ts @@ -19,7 +19,12 @@ export type { CalloutProps } from './components/Callout' export { default as Callout } from './components/Callout' export { default as CatalogCard } from './components/CatalogCard' export { default as Checkbox } from './components/Checkbox' -export { default as Chip, type ChipProps } from './components/Chip' +export { + default as Chip, + type ChipProps, + type ChipSize, + type ChipSeverity, +} from './components/Chip' export { default as ChipList } from './components/ChipList' export { default as Code } from './components/Code' export { default as CodeEditor } from './components/CodeEditor' diff --git a/src/stories/Card.stories.tsx b/src/stories/Card.stories.tsx index b2a96b335..32aacdf47 100644 --- a/src/stories/Card.stories.tsx +++ b/src/stories/Card.stories.tsx @@ -1,12 +1,11 @@ -import { Flex } from 'honorable' -import { type ComponentProps } from 'react' +import { type ComponentProps, type ReactNode } from 'react' import { useTheme } from 'styled-components' import { type FillLevel } from '../components/contexts/FillLevelContext' -import { Card } from '../index' import type { CardProps } from '../components/Card' +import { Card, Flex, InfoOutlineIcon, Tooltip } from '../index' export default { title: 'Card', @@ -16,6 +15,13 @@ export default { options: ['neutral', 'info', 'success', 'warning', 'danger', 'critical'], control: { type: 'select' }, }, + headerSize: { + options: ['medium', 'large'], + control: { type: 'select' }, + }, + headerContent: { + control: { type: 'text' }, + }, }, } @@ -32,7 +38,14 @@ function Template({ width, height, severity, -}: { width: number; height: number } & CardProps) { + headerSize, + headerContent, +}: { + width: number + height: number + headerSize: ComponentProps['header']['size'] + headerContent: ReactNode +} & CardProps) { return ( - {fillLevels.map((fillLevel) => ( + {fillLevels.map((fillLevel, index) => ( fillLevel= {fillLevel === undefined ? 'undefined' : `"${fillLevel}"`} @@ -135,6 +156,15 @@ Default.args = { width: 150, height: 150, severity: 'neutral', + headerSize: 'medium', + headerContent: ( + +

Header

+ + + +
+ ), } export const Clickable = Template.bind({}) @@ -151,4 +181,13 @@ WithFillLevelContext.args = { clickable: false, disabled: false, width: 400, + headerSize: 'medium', + headerContent: ( + +

Header

+ + + +
+ ), } diff --git a/src/stories/Chip.stories.tsx b/src/stories/Chip.stories.tsx index 7bd0353c5..7573fc19b 100644 --- a/src/stories/Chip.stories.tsx +++ b/src/stories/Chip.stories.tsx @@ -13,12 +13,6 @@ export default { title: 'Chip', component: Chip, argTypes: { - hue: { - options: [undefined, 'default', 'lighter', 'lightest'], - control: { - type: 'select', - }, - }, onFillLevel: { options: [0, 1, 2, 3], control: { @@ -42,7 +36,12 @@ const sizes: ComponentProps['size'][] = [ const severities = SEVERITIES -const versionsArgs = [{}, { loading: true }, { icon: }] +const versionsArgs = [ + {}, + { loading: true }, + { icon: }, + { inactive: true }, +] function Template({ onFillLevel, asLink, ...args }: any) { if (asLink) { diff --git a/src/types/react-table.d.ts b/src/types/react-table.d.ts index 713f24faa..28c5d461a 100644 --- a/src/types/react-table.d.ts +++ b/src/types/react-table.d.ts @@ -1,4 +1,5 @@ import '@tanstack/react-table' +import { type ReactNode } from 'react' declare module '@tanstack/table-core' { // eslint-disable-next-line @typescript-eslint/no-unused-vars @@ -6,7 +7,7 @@ declare module '@tanstack/table-core' { truncate?: boolean gridTemplate?: string center?: boolean - tooltip?: string + tooltip?: ReactNode highlight?: boolean } }