Skip to content

Commit

Permalink
[card] Revert change to footer button alignment, add new footerAction…
Browse files Browse the repository at this point in the history
…Alignment prop (left | right), default to right for backwards compat
  • Loading branch information
qq99 committed Nov 6, 2019
1 parent dcdce79 commit 8322d6c
Show file tree
Hide file tree
Showing 7 changed files with 74 additions and 6 deletions.
2 changes: 2 additions & 0 deletions UNRELEASED.md
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,8 @@
### Enhancements

- Updated the type of the `title` prop in `ChoiceList` from `string` to `ReactNode` ([#2355](https://github.com/Shopify/polaris-react/pull/2355))
- Updated `Card` footer actions to be right aligned by default again. Added `footerActionAlignment` prop to control the footer action alignemnt, but defaults to `'right'` ([#2407](https://github.com/Shopify/polaris-react/pull/2407))
- Added `reverse` prop to `<ButtonGroup>` to control visual render order ([#2407](https://github.com/Shopify/polaris-react/pull/2407))

### Bug fixes

Expand Down
4 changes: 4 additions & 0 deletions src/components/ButtonGroup/ButtonGroup.scss
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,10 @@ $item-spacing: spacing(tight);
margin-left: (-1 * $item-spacing);
}

.Reverse {
flex-direction: row-reverse;
}

.Item {
margin-top: $item-spacing;
margin-left: $item-spacing;
Expand Down
4 changes: 4 additions & 0 deletions src/components/ButtonGroup/ButtonGroup.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -13,19 +13,23 @@ export interface ButtonGroupProps {
connectedTop?: boolean;
/** Button components */
children?: React.ReactNode;
//* Reverse the visual order of the buttons while maintaining DOM order */
reverse?: boolean;
}

export function ButtonGroup({
children,
segmented,
fullWidth,
connectedTop,
reverse = false,
}: ButtonGroupProps) {
const className = classNames(
styles.ButtonGroup,
segmented && styles.segmented,
fullWidth && styles.fullWidth,
connectedTop && styles.connectedTop,
reverse && styles.Reverse,
);

const contents = elementChildren(children).map((child, index) => (
Expand Down
6 changes: 5 additions & 1 deletion src/components/Card/Card.scss
Original file line number Diff line number Diff line change
Expand Up @@ -93,13 +93,17 @@

.Footer {
display: flex;
justify-content: flex-start;
justify-content: flex-end;
padding: 0 spacing() spacing();

@include page-content-when-not-fully-condensed {
padding: 0 spacing(loose) spacing(loose);
}

&.LeftJustified {
justify-content: flex-start;
}

.Section-subdued + & {
border-top: border();
padding: spacing(loose);
Expand Down
14 changes: 11 additions & 3 deletions src/components/Card/Card.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -29,6 +29,8 @@ export interface CardProps {
secondaryFooterActions?: ComplexAction[];
/** The content of the disclosure button rendered when there is more than one secondary footer action */
secondaryFooterActionsDisclosureText?: string;
/** Alignment of the footer actions on the card, defaults to right */
footerActionAlignment?: 'right' | 'left';
}

// TypeScript can't generate types that correctly infer the typing of
Expand All @@ -49,6 +51,7 @@ export const Card: React.FunctionComponent<CardProps> & {
primaryFooterAction,
secondaryFooterActions,
secondaryFooterActionsDisclosureText,
footerActionAlignment = 'right',
}: CardProps) {
const i18n = useI18n();

Expand Down Expand Up @@ -94,10 +97,15 @@ export const Card: React.FunctionComponent<CardProps> & {

const footerMarkup =
primaryFooterActionMarkup || secondaryFooterActionsMarkup ? (
<div className={styles.Footer}>
<ButtonGroup>
{secondaryFooterActionsMarkup}
<div
className={classNames(
styles.Footer,
footerActionAlignment === 'left' && styles.LeftJustified,
)}
>
<ButtonGroup reverse={footerActionAlignment === 'right'}>
{primaryFooterActionMarkup}
{secondaryFooterActionsMarkup}
</ButtonGroup>
</div>
) : null;
Expand Down
2 changes: 1 addition & 1 deletion src/components/Card/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -237,7 +237,7 @@ Use for less important card actions, or actions merchants may do before reviewin

<!-- content-for: web -->

Use footer actions for a card’s most important actions, or actions merchants should do after reviewing the contents of the card. For example, merchants should review the contents of a shipment before an important action like adding tracking information.
Use footer actions for a card’s most important actions, or actions merchants should do after reviewing the contents of the card. For example, merchants should review the contents of a shipment before an important action like adding tracking information. Footer actions can be left or right aligned with the `footerActionAlignment` prop.

<!-- /content-for -->

Expand Down
48 changes: 47 additions & 1 deletion src/components/Card/tests/Card.test.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,15 @@ import {
trigger,
findByTestID,
} from 'test-utilities/legacy';
import {Card, Badge, Button, Popover, ActionList} from 'components';
import {mountWithApp} from 'test-utilities';
import {
Card,
Badge,
Button,
ButtonGroup,
Popover,
ActionList,
} from 'components';
import {WithinContentContext} from '../../../utilities/within-content-context';
import {Section} from '../components';

Expand Down Expand Up @@ -74,6 +82,44 @@ describe('<Card />', () => {
expect(card.find(Card.Header)).toHaveLength(1);
});

describe('footerActionAlignment prop', () => {
it('renders right-aligned if not supplied', () => {
const card = mountWithApp(
<Card primaryFooterAction={{content: 'test action'}}>
<p>Some card content.</p>
</Card>,
);

expect(card).toContainReactComponent(ButtonGroup, {reverse: true});
});

it('renders right-aligned if set to "right"', () => {
const card = mountWithApp(
<Card
primaryFooterAction={{content: 'test action'}}
footerActionAlignment="right"
>
<p>Some card content.</p>
</Card>,
);

expect(card).toContainReactComponent(ButtonGroup, {reverse: true});
});

it('renders left-aligned if set to "left"', () => {
const card = mountWithApp(
<Card
primaryFooterAction={{content: 'test action'}}
footerActionAlignment="left"
>
<p>Some card content.</p>
</Card>,
);

expect(card).toContainReactComponent(ButtonGroup, {reverse: false});
});
});

it('renders a primary footer action', () => {
const card = mountWithAppProvider(
<Card primaryFooterAction={{content: 'test action'}}>
Expand Down

0 comments on commit 8322d6c

Please sign in to comment.