Skip to content

Commit

Permalink
Merge remote-tracking branch 'refs/remotes/origin/feat/TET-851-dialog…
Browse files Browse the repository at this point in the history
…' into feat/TET-851-dialog
  • Loading branch information
karolinaszarek committed Sep 17, 2024
2 parents 28cfb41 + e86efa3 commit 1fbe42e
Show file tree
Hide file tree
Showing 73 changed files with 3,828 additions and 22 deletions.
9 changes: 9 additions & 0 deletions src/components/ButtonGroup/ButtonGroup.props.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
import { HTMLAttributes } from 'react';

import { ButtonGroupConfig } from './ButtonGroup.styles';

export type ButtonGroupSize = 'medium' | 'small';
export type ButtonGroupProps = {
size?: ButtonGroupSize;
custom?: ButtonGroupConfig;
} & Omit<HTMLAttributes<HTMLSpanElement>, 'color'>;
47 changes: 47 additions & 0 deletions src/components/ButtonGroup/ButtonGroup.stories.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,47 @@
import type { Meta, StoryObj } from '@storybook/react';

import { ButtonGroup } from './ButtonGroup';
import { Button } from '../Button/Button';

import { ButtonGroupDocs } from '@/docs-components/ButtonGroupDocs';
import { TetDocs } from '@/docs-components/TetDocs';

const meta = {
title: 'ButtonGroup',
component: ButtonGroup,
tags: ['autodocs'],
argTypes: {
size: {
options: ['small', 'medium'],
defaultValue: 'medium',
control: { type: 'radio' },
},
},
parameters: {
docs: {
description: {
component:
'A set of related buttons that are visually and functionally grouped. Button Group acts as a cohesive unit, providing users with clear options for actions or navigation.',
},
page: () => (
<TetDocs docs="https://docs.tetrisly.com/components/list/buttongroup">
<ButtonGroupDocs />
</TetDocs>
),
},
},
} satisfies Meta<typeof ButtonGroup>;

export default meta;
type Story = StoryObj<typeof meta>;

export const Default: Story = {
args: {
children: [
<Button label="button" />,
<Button label="button" />,
<Button label="button" />,
<Button label="button" />,
],
},
};
50 changes: 50 additions & 0 deletions src/components/ButtonGroup/ButtonGroup.styles.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,50 @@
import { BaseProps } from '@/types';

export type ButtonGroupConfig = {
button: {
size: {
medium: BaseProps;
small: BaseProps;
};
} & BaseProps;
} & BaseProps;

export const defaultConfig = {
display: 'inline-flex',
justifyContent: 'center',
alignItems: 'center',
button: {
color: {
_: '$color-action-neutral-normal',
hover: '$color-action-neutral-hover',
},
backgroundColor: {
_: '$color-action-inverted-normal',
active: '$color-action-ghost-active',
hover: '$color-action-ghost-hover',
selected: '$color-action-ghost-selected',
},
ringColor: '$color-action-outline-normal',
size: {
// TODO think if it can be done by passing size prop to a button component
medium: {
h: '$size-medium',
px: '$space-component-padding-large',
},
small: {
h: '$size-small',
px: '$space-component-padding-medium',
},
},
borderRadius: {
first: `$border-radius-large 0px 0px $border-radius-large`,
last: `0px $border-radius-large $border-radius-large 0px`,
},
transition: true,
transitionDuration: 200,
},
} as const satisfies ButtonGroupConfig;

export const buttonGroupStyles = {
defaultConfig,
};
41 changes: 41 additions & 0 deletions src/components/ButtonGroup/ButtonGroup.test.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,41 @@
import { render } from '../../tests/render';

import { ButtonGroup } from '.';

import { customPropTester } from '@/tests/customPropTester';

// TODO

const getButtonGroup = (jsx: JSX.Element) => {
const { getByTestId } = render(jsx);
return {
buttonGroup: getByTestId('button-group'),
};
};
describe('ButtonGroup', () => {
it('should render the ButtonGroup ', () => {
const { buttonGroup } = getButtonGroup(<ButtonGroup />);
expect(buttonGroup).toBeInTheDocument();
});

it('should render correct number of children', () => {
const { buttonGroup } = getButtonGroup(
<ButtonGroup>
<ButtonGroup.Item label="label" />
<ButtonGroup.Item label="label" />
<ButtonGroup.Item label="label" />
<ButtonGroup.Item label="label" />
<ButtonGroup.Item label="label" />
</ButtonGroup>,
);

expect(buttonGroup?.children.length).toEqual(5);
});

customPropTester(<ButtonGroup />, {
containerId: 'button-group',
props: {
options: ['small', 'medium'],
},
});
});
58 changes: 58 additions & 0 deletions src/components/ButtonGroup/ButtonGroup.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,58 @@
import { MarginProps } from '@xstyled/styled-components';
import {
Children,
cloneElement,
isValidElement,
PropsWithChildren,
useMemo,
type FC,
} from 'react';

import { ButtonGroupProps } from './ButtonGroup.props';
import { stylesBuilder } from './stylesBuilder';
import { Button, ButtonProps } from '../Button';

import { tet } from '@/tetrisly';

type Props = FC<PropsWithChildren<ButtonGroupProps & MarginProps>> & {
Item: FC<ButtonProps & MarginProps>;
};

export const ButtonGroup: Props = ({
size = 'medium',
children,
custom,
...rest
}) => {
const styles = useMemo(
() =>
stylesBuilder({
size,
custom,
}),
[custom, size],
);

Children.forEach(children, (child) => {
if (isValidElement(child) && child?.type !== ButtonGroup.Item) {
console.error(
'You should use only ButtonGroup.Item as a child of a CheckboxGroup component.',
);
}
});

const childrenWithProps = Children.map(children, (child) => {
if (isValidElement(child)) {
return cloneElement(child, { ...styles.button });
}
return child;
});

return (
<tet.span data-testid="button-group" {...styles.container} {...rest}>
{childrenWithProps}
</tet.span>
);
};

ButtonGroup.Item = Button;
3 changes: 3 additions & 0 deletions src/components/ButtonGroup/index.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
export { ButtonGroup } from './ButtonGroup';
export type { ButtonGroupProps } from './ButtonGroup.props';
export { buttonGroupStyles } from './ButtonGroup.styles';
31 changes: 31 additions & 0 deletions src/components/ButtonGroup/stylesBuilder.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,31 @@
import type { ButtonGroupSize } from './ButtonGroup.props';
import { ButtonGroupConfig, defaultConfig } from './ButtonGroup.styles';

import { mergeConfigWithCustom } from '@/services';
import { BaseProps } from '@/types/BaseProps';

type ButtonGroupStyleBuilder = {
container: BaseProps;
button: BaseProps;
};

type ButtonGroupStyleBuilderInput = {
size: ButtonGroupSize;
custom?: ButtonGroupConfig;
};

export const stylesBuilder = ({
size,
custom,
}: ButtonGroupStyleBuilderInput): ButtonGroupStyleBuilder => {
const { button, ...container } = mergeConfigWithCustom({
defaultConfig,
custom,
});
const buttonStyles = { ...button, ...button.size[size] };

return {
container,
button: buttonStyles,
};
};
2 changes: 1 addition & 1 deletion src/components/Checkbox/Checkbox.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -3,10 +3,10 @@ import styled from '@xstyled/styled-components';
import { forwardRef, useCallback, useId, useMemo } from 'react';

import type { CheckboxProps } from './Checkbox.props';
import { useIndeterminate } from './hooks';
import { stylesBuilder } from './stylesBuilder';
import { HelperText } from '../HelperText';

import { useIndeterminate } from '@/hooks';
import { extractInputProps } from '@/services';
import { tet } from '@/tetrisly';
import { MarginProps } from '@/types/MarginProps';
Expand Down
1 change: 0 additions & 1 deletion src/components/Checkbox/hooks/index.ts

This file was deleted.

11 changes: 0 additions & 11 deletions src/components/Checkbox/hooks/useIconChecked.ts

This file was deleted.

20 changes: 20 additions & 0 deletions src/components/FileIcon/FileIcon.props.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,20 @@
import { BaseProps } from '@/types';

export type IconType =
| 'Sketch'
| 'Photoshop'
| 'Excel'
| 'Word'
| 'Pdf'
| 'Spreadsheet'
| 'Document'
| 'File'
| 'Archive'
| 'Figma';
export type Size = 'Large' | 'Medium';

export type FileIconProps = {
iconType: IconType;
size?: Size;
custom?: BaseProps;
};
31 changes: 31 additions & 0 deletions src/components/FileIcon/FileIcon.stories.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,31 @@
import { Meta, StoryObj } from '@storybook/react';

import { FileIcon } from './FileIcon';

import { FileIconDocs } from '@/docs-components/FileIconDocs';
import { TetDocs } from '@/docs-components/TetDocs';

const meta = {
title: 'File Icon',
component: FileIcon,
tags: ['autodocs'],
parameters: {
docs: {
page: () => (
<TetDocs docs="">
<FileIconDocs />
</TetDocs>
),
},
},
} satisfies Meta<typeof FileIcon>;

export default meta;
type Story = StoryObj<typeof meta>;

export const Default: Story = {
args: {
iconType: 'Archive',
size: 'Large',
},
};
9 changes: 9 additions & 0 deletions src/components/FileIcon/FileIcon.styles.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
import { BaseProps } from '@/types';

export type FileIconConfig = BaseProps;

export const defaultConfig = {} satisfies FileIconConfig;

export const fileIconStyles = {
defaultConfig,
};
24 changes: 24 additions & 0 deletions src/components/FileIcon/FileIcon.test.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
import { FileIcon } from './FileIcon';

import { customPropTester } from '@/tests/customPropTester';
import { render } from '@/tests/render';

const getFileIcon = (jsx: JSX.Element) => {
const { getByTestId } = render(jsx);

return getByTestId('file-icon');
};

describe('FileIcon', () => {
customPropTester(<FileIcon iconType="Photoshop" />, {
containerId: 'file-icon',
props: {
size: ['Large', 'Medium'],
},
});

it('should render the file icon', () => {
const fileIcon = getFileIcon(<FileIcon iconType="Photoshop" />);
expect(fileIcon).toBeInTheDocument();
});
});
23 changes: 23 additions & 0 deletions src/components/FileIcon/FileIcon.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,23 @@
import { useMemo, type FC } from 'react';

import { FileIconProps } from './FileIcon.props';
import { stylesBuilder } from './stylesBuilder';
import { renderProperIcon } from './utils';

import { tet } from '@/tetrisly';
import type { MarginProps } from '@/types';

export const FileIcon: FC<FileIconProps & MarginProps> = ({
iconType,
size = 'Large',
custom,
...restProps
}) => {
const styles = useMemo(() => stylesBuilder(custom), [custom]);

return (
<tet.div {...styles.container} data-testid="file-icon" {...restProps}>
{renderProperIcon(iconType, size)}
</tet.div>
);
};
2 changes: 2 additions & 0 deletions src/components/FileIcon/index.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
export { FileIcon } from './FileIcon';
export type { FileIconProps } from './FileIcon.props';
Loading

0 comments on commit 1fbe42e

Please sign in to comment.