Files
flipper/desktop/app/src/sandy-chrome/DesignComponentDemos.tsx
Michel Weststrate 84c6e05b8a Wire up tracking to Sandy Chrome
Summary: Wired up tracking to all chrome sections and some import UI elements

Reviewed By: jknoxville

Differential Revision: D25219089

fbshipit-source-id: c75bed91894609dafc5fcc6423a5228211fb92d8
2020-12-03 04:15:45 -08:00

431 lines
14 KiB
TypeScript

/**
* Copyright (c) Facebook, Inc. and its affiliates.
*
* This source code is licensed under the MIT license found in the
* LICENSE file in the root directory of this source tree.
*
* @format
*/
import React from 'react';
import {Typography, Card, Table, Collapse, Button, Tabs} from 'antd';
import {Layout, Link} from '../ui';
import {NUX, theme, Tracked, TrackingScope} from 'flipper-plugin';
import reactElementToJSXString from 'react-element-to-jsx-string';
import {CodeOutlined} from '@ant-design/icons';
const {Text} = Typography;
const demoStyle: Record<string, React.CSSProperties> = {
square: {
background: theme.successColor,
width: 50,
height: 50,
lineHeight: '50px',
textAlign: 'center',
},
border: {border: `1px dotted ${theme.primaryColor}`},
} as const;
type PreviewProps = {
title: string;
description?: React.ReactNode;
props: [string, React.ReactNode, React.ReactNode][];
demos: Record<string, React.ReactNode>;
};
const largeChild = (
<div style={{background: theme.warningColor}}>
<img src="https://fbflipper.com/img/mascot.png" height={500} />
</div>
);
const aButton = <Button>A button</Button>;
const aBox = <div style={{...demoStyle.square, width: 100}}>A fixed child</div>;
const aFixedWidthBox = (
<div style={{background: theme.primaryColor, width: 150, color: 'white'}}>
Fixed width box
</div>
);
const aFixedHeightBox = (
<div
style={{
background: theme.primaryColor,
height: 40,
lineHeight: '40px',
color: 'white',
}}>
Fixed height box
</div>
);
const aDynamicBox = (
<div style={{background: theme.warningColor, flex: 1}}>
A dynamic child (flex: 1)
</div>
);
const someText = <Text>Some text</Text>;
const demos: PreviewProps[] = [
{
title: 'Layout.Container',
description: `Layout.Container can be used to organize the UI in regions. It takes care of paddings and borders. Children will be arranged vertically. Use Layout.Horizontal instead for arranging children horizontally. If you need a margin on this component, try to wrap it in other Layout component instead.`,
props: [
['rounded', 'boolean (false)', 'Make the corners rounded'],
[
'padv / padh / pad',
Object.keys(theme.space).join(' | ') + ' | number | true',
'Short-hand to set the horizontal, vertical or both paddings. The keys correspond to the theme space settings. Using `true` picks the default horizontal / vertical padding for inline elements.',
],
[
'width / height',
'number',
'Set the width / height of this container in pixels. Use sparingly.',
],
[
'bordered',
'boolean (false)',
'This container will use a default border on all sides',
],
[
'borderTop / borderRight / borderBottom / borderLeft',
'boolean (false)',
'Use a standard padding on the top side',
],
[
'gap',
'true / number (0)',
'Set the spacing between children. If just set, theme.space.small will be used.',
],
[
'center',
'boolean (false)',
'If set, all children will use their own naturally width, and they will be centered horizontally in the Container. If not set, all children will be stretched to the width of the Container.',
],
],
demos: {
'Basic container with fixed dimensions': (
<Layout.Container style={demoStyle.square}></Layout.Container>
),
'Basic container with fixed height': (
<Layout.Container
style={{
height: 50,
background: theme.successColor,
}}></Layout.Container>
),
'bordered pad rounded': (
<Layout.Container
bordered
pad
rounded
style={{background: theme.backgroundDefault, width: 200}}>
<div style={demoStyle.square}>child</div>
</Layout.Container>
),
'Multiple children, gap={24}': (
<Layout.Container gap={24}>
{aButton}
{someText}
{aBox}
{aDynamicBox}
</Layout.Container>
),
'Multiple children icmw. pad center gap': (
<Layout.Container pad center gap>
{aButton}
{someText}
{aBox}
{aDynamicBox}
</Layout.Container>
),
},
},
{
title: 'Layout.Horizontal',
description:
'Use this component to arrange multiple items horizontally. All vanilla Container props can be used as well.',
props: [
[
'center',
'boolean (false)',
'If set, all children will use their own height, and they will be centered vertically in the layout. If not set, all children will be stretched to the height of the layout.',
],
],
demos: {
'Basic usage, gap="large"': (
<Layout.Horizontal gap="large">
{aButton}
{someText}
{aBox}
{aDynamicBox}
</Layout.Horizontal>
),
'Using flags: pad center gap={8} (great for toolbars and such)': (
<Layout.Horizontal pad center gap={8}>
{aButton}
{someText}
{aBox}
{aDynamicBox}
</Layout.Horizontal>
),
},
},
{
title: 'Layout.ScrollContainer',
description:
'Use this component to create an area that can be scrolled. The scrollable area will automatically consume all available space. ScrollContainer accepts all properties that Container accepts as well. Padding will be applied to the child rather than the parent.',
props: [
[
'horizontal / vertical',
'boolean',
'specifies in which directions the container should scroll. If none is specified the container will scroll in both directions',
],
[
'padv / padh / pad',
'see Container',
'Padding will be applied to the child',
],
],
demos: {
'Basic usage': (
<Layout.ScrollContainer style={{height: 100}}>
{largeChild}
</Layout.ScrollContainer>
),
'ScrollContainer + Vertical for vertical scroll only': (
<Layout.ScrollContainer
vertical
style={{
height: 100,
width: 100,
border: `2px solid ${theme.primaryColor}`,
}}>
<Layout.Container>
<Text ellipsis>
This text is truncated because it is too long and scroll is
vertical only...
</Text>
{largeChild}
</Layout.Container>
</Layout.ScrollContainer>
),
},
},
{
title: 'Layout.Top|Left|Right|Bottom',
description:
"Divides all available space over two children. The (top|left|right|bottom)-most first child will keep it's own dimensions, and positioned (top|left|right|bottom) of the other child. All remaining space will be assigned to the remaining child.",
props: [
[
'scrollable',
'boolean (false)',
'If set, the area of the second child will automatically be made scrollable.',
],
[
'center',
'boolean (false)',
'If set, all children will use their own height, and they will be centered vertically in the layout. If not set, all children will be stretched to the height of the layout.',
],
],
demos: {
'Layout.Top': (
<Layout.Top>
{aFixedHeightBox}
{aDynamicBox}
</Layout.Top>
),
'Layout.Left': (
<Layout.Left>
{aFixedWidthBox}
{aDynamicBox}
</Layout.Left>
),
'Layout.Right': (
<Layout.Right>
{aDynamicBox}
{aFixedWidthBox}
</Layout.Right>
),
'Layout.Bottom': (
<Layout.Bottom>
{aDynamicBox}
{aFixedHeightBox}
</Layout.Bottom>
),
'Layout.Top + Layout.ScrollContainer': (
<Layout.Container style={{height: 150}}>
<Layout.Top>
{aFixedHeightBox}
<Layout.ScrollContainer>{largeChild}</Layout.ScrollContainer>
</Layout.Top>
</Layout.Container>
),
'Layout.Left + Layout.ScrollContainer': (
<Layout.Container style={{height: 150}}>
<Layout.Left>
{aFixedWidthBox}
<Layout.ScrollContainer>{largeChild}</Layout.ScrollContainer>
</Layout.Left>
</Layout.Container>
),
'Layout.Right + Layout.ScrollContainer': (
<Layout.Container style={{height: 150}}>
<Layout.Right>
<Layout.ScrollContainer>{largeChild}</Layout.ScrollContainer>
{aFixedWidthBox}
</Layout.Right>
</Layout.Container>
),
'Layout.Bottom + Layout.ScrollContainer': (
<Layout.Container style={{height: 150}}>
<Layout.Bottom>
<Layout.ScrollContainer>{largeChild}</Layout.ScrollContainer>
{aFixedHeightBox}
</Layout.Bottom>
</Layout.Container>
),
},
},
{
title: 'NUX',
description:
'A component to provide a New-User-eXperience: Highlight new features to first time users. For tooltips that should stay available, use ToolTip from ANT design',
props: [
['title', 'string / React element', 'The tooltip contents'],
[
'placement',
<>
See{' '}
<Link href="https://ant.design/components/tooltip/#components-tooltip-demo-placement">
docs
</Link>
</>,
'(optional) on which side to place the tooltip',
],
],
demos: {
'NUX example': (
<NUX title="This button does something cool" placement="right">
<Button>Hello world</Button>
</NUX>
),
},
},
{
title: 'Tracked',
description:
'A component that tracks component interactions. For Facebook internal builds, global stats for these interactions will be tracked. Wrap this component around another element to track its events',
props: [
[
'events',
'string | string[] (default: "onClick")',
'The event(s) of the child component that should be tracked',
],
[
'action',
'string (optional)',
'Describes the element the user interacted with. Will by default be derived from the title, key or contents of the element',
],
],
demos: {
'Basic example': (
<Tracked>
<Button onClick={() => {}}>Test</Button>
</Tracked>
),
},
},
{
title: 'TrackingScope',
description:
'Describes more precisely the place in the UI for all underlying Tracked elements. Multiple Tracking scopes are automatically nested. Use the `withTrackingScope` HoC to automatically wrap a component definition in a tracking scope',
props: [
['scope', 'string', 'The name of the scope. For example "Login Dialog"'],
],
demos: {
'Basic example': (
<TrackingScope scope="tracking scope demo">
<Tracked>
<Button onClick={() => {}}>Test</Button>
</Tracked>
</TrackingScope>
),
},
},
];
function ComponentPreview({title, demos, description, props}: PreviewProps) {
return (
<Card title={title} size="small" type="inner">
<TrackingScope scope={title}>
<Layout.Container gap="small">
<Text type="secondary">{description}</Text>
<Collapse ghost>
<Collapse.Panel header="Examples" key="demos">
<Layout.Container gap="large">
{Object.entries(demos).map(([name, children]) => (
<div key={name}>
<Tabs type="line">
<Tabs.TabPane tab={name} key="1">
<div
style={{
background: theme.backgroundWash,
width: '100%',
}}>
{children}
</div>
</Tabs.TabPane>
<Tabs.TabPane tab={<CodeOutlined />} key="2">
<div
style={{
background: theme.backgroundWash,
width: '100%',
padding: theme.space.medium,
}}>
<pre>{reactElementToJSXString(children)}</pre>
</div>
</Tabs.TabPane>
</Tabs>
</div>
))}
</Layout.Container>
</Collapse.Panel>
<Collapse.Panel header="Props" key="props">
<Table
size="small"
pagination={false}
dataSource={props.map((prop) =>
Object.assign(prop, {key: prop[0]}),
)}
columns={[
{
title: 'Property',
dataIndex: 0,
width: 100,
},
{
title: 'Type and default',
dataIndex: 1,
width: 200,
},
{
title: 'Description',
dataIndex: 2,
},
]}
/>
</Collapse.Panel>
</Collapse>
</Layout.Container>
</TrackingScope>
</Card>
);
}
export const DesignComponentDemos = () => (
<Layout.Container gap={theme.space.large}>
{demos.map((demo) => (
<ComponentPreview key={demo.title} {...demo} />
))}
</Layout.Container>
);