mirror of https://github.com/lukechilds/docs.git
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
225 lines
6.2 KiB
225 lines
6.2 KiB
import React from 'react';
|
|
import { Box, Flex, BoxProps, color, Grid, space } from '@blockstack/ui';
|
|
import { border, onlyText, transition } from '@common/utils';
|
|
import { useTouchable } from '@common/hooks/use-touchable';
|
|
import { Caption, Text } from '@components/typography';
|
|
import Link from 'next/link';
|
|
import routes from '@common/routes';
|
|
import { Img } from '@components/mdx/image';
|
|
import { css } from '@styled-system/css';
|
|
import { getCapsizeStyles, getHeadingStyles } from '@components/mdx/typography';
|
|
const Image = ({
|
|
src,
|
|
isHovered,
|
|
size,
|
|
...rest
|
|
}: BoxProps & { src?: string; isHovered?: boolean }) => (
|
|
<Box
|
|
flexShrink={0}
|
|
style={{
|
|
willChange: 'transform',
|
|
}}
|
|
width="100%"
|
|
size={size}
|
|
{...rest}
|
|
>
|
|
<Img
|
|
flexShrink={0}
|
|
borderRadius="12px"
|
|
src={src}
|
|
width="100%"
|
|
minWidth={size}
|
|
size={size}
|
|
mx="0 !important"
|
|
my="0 !important"
|
|
/>
|
|
</Box>
|
|
);
|
|
|
|
const Title = ({ children, ...props }: BoxProps) => (
|
|
<Text
|
|
css={css({
|
|
...getHeadingStyles('h3'),
|
|
})}
|
|
{...props}
|
|
>
|
|
{children}
|
|
</Text>
|
|
);
|
|
|
|
const Description = ({ children, ...props }) => (
|
|
<Text
|
|
{...props}
|
|
css={css({
|
|
...getCapsizeStyles(16, 26),
|
|
mt: space('base-tight'),
|
|
color: color('text-body'),
|
|
})}
|
|
>
|
|
{children}
|
|
</Text>
|
|
);
|
|
|
|
const FloatingLink = ({ href, ...props }: any) => (
|
|
<Link href={href} passHref>
|
|
<Box as="a" position="absolute" size="100%" left={0} top={0} />
|
|
</Link>
|
|
);
|
|
const InlineCard = ({ page }) => {
|
|
const { hover, active, bind } = useTouchable({
|
|
behavior: 'link',
|
|
});
|
|
return (
|
|
<Flex
|
|
border={border()}
|
|
flexDirection={['column', 'row', 'row', 'row']}
|
|
p={space('base-loose')}
|
|
borderRadius="12px"
|
|
align="center"
|
|
transition={transition()}
|
|
boxShadow={hover ? 'mid' : 'none'}
|
|
position="relative"
|
|
{...bind}
|
|
>
|
|
<Box flexShrink={0} size="64px" overflow="hidden" borderRadius={'12px'}>
|
|
<Image
|
|
transition={transition('0.45s')}
|
|
transform={(hover || active) && 'scale(1.12)'}
|
|
style={{ willChange: 'transform' }}
|
|
size="64px"
|
|
src={page.images.sm}
|
|
/>
|
|
</Box>
|
|
<Flex
|
|
flexDirection="column"
|
|
ml={space(['none', 'base', 'base', 'base'])}
|
|
mt={space(['base', 'none', 'none', 'none'])}
|
|
textAlign={['center', 'left', 'left', 'left']}
|
|
>
|
|
<Flex align="baseline">
|
|
<Title
|
|
width={['100%', 'unset', 'unset', 'unset']}
|
|
color={hover ? color('accent') : color('text-title')}
|
|
mb={space('extra-tight')}
|
|
>
|
|
{page.title || page.headings[0]}
|
|
</Title>
|
|
{page.tags?.length ? (
|
|
<Flex
|
|
position={['absolute', 'static', 'static', 'static']}
|
|
top={space('base-loose')}
|
|
right={space('base-loose')}
|
|
>
|
|
{page.tags.map((tag, key) => (
|
|
<Flex
|
|
ml={space('tight')}
|
|
borderRadius="18px"
|
|
px={space('base-tight')}
|
|
height="20px"
|
|
align="center"
|
|
justify="center"
|
|
fontSize="12px"
|
|
bg={color('border')}
|
|
textTransform="capitalize"
|
|
color={color('invert')}
|
|
transition={transition()}
|
|
key={key}
|
|
>
|
|
{tag}
|
|
</Flex>
|
|
))}
|
|
</Flex>
|
|
) : null}
|
|
</Flex>
|
|
<Description>{page.description}</Description>
|
|
</Flex>
|
|
<FloatingLink href={`${page.path}`} />
|
|
</Flex>
|
|
);
|
|
};
|
|
|
|
const GridCardImage: React.FC<BoxProps & { isHovered?: boolean; page: any }> = React.memo(
|
|
({ isHovered, page, ...props }) => (
|
|
<Box
|
|
bg="#9985FF"
|
|
position="relative"
|
|
borderRadius="12px"
|
|
mb={space('loose')}
|
|
overflow="hidden"
|
|
{...props}
|
|
>
|
|
<Grid style={{ placeItems: 'center' }} height="0px" paddingTop="56.25%">
|
|
<Image
|
|
width="102%"
|
|
size="102%"
|
|
transition={transition('0.45s')}
|
|
transform={isHovered && 'scale(1.08)'}
|
|
style={{ willChange: 'transform' }}
|
|
src={page?.images?.large}
|
|
position="absolute"
|
|
left={'-2%'}
|
|
top={'-2%'}
|
|
/>
|
|
</Grid>
|
|
</Box>
|
|
)
|
|
);
|
|
|
|
const GridCardDetails: React.FC<BoxProps & { isHovered?: boolean; page: any }> = React.memo(
|
|
({ isHovered, page, ...props }) => (
|
|
<>
|
|
<Flex alignItems="flex-start" justifyContent="flex-start" flexDirection="column">
|
|
<Title color="currentColor" mb={space('tight')}>
|
|
{page.title || page.headings[0]}
|
|
</Title>
|
|
<Description>{page.description}</Description>
|
|
</Flex>
|
|
<FloatingLink href={`${page.path}`} />
|
|
</>
|
|
)
|
|
);
|
|
|
|
const GridCard: React.FC<BoxProps & { page?: any }> = React.memo(({ page, ...rest }) => {
|
|
const { hover, active, bind } = useTouchable({
|
|
behavior: 'link',
|
|
});
|
|
return (
|
|
<Box
|
|
position="relative"
|
|
color={color('text-title')}
|
|
_hover={{ color: color('accent') }}
|
|
{...rest}
|
|
{...bind}
|
|
>
|
|
<GridCardImage page={page} isHovered={hover || active} />
|
|
<GridCardDetails page={page} />
|
|
</Box>
|
|
);
|
|
});
|
|
|
|
export const PageReference: React.FC<BoxProps> = React.memo(({ children }) => {
|
|
const content = onlyText(children).trim();
|
|
const [variant, _paths] = content.includes('\n') ? content.split('\n') : ['default', content];
|
|
const paths = _paths.includes(', ') ? _paths.split(', ') : [_paths];
|
|
if (!routes) return null;
|
|
|
|
const pages = paths.map(path => routes?.find(route => route.path === path)).filter(page => page);
|
|
return (
|
|
<Grid
|
|
width="100%"
|
|
mt={space('extra-loose')}
|
|
gridColumnGap={space('extra-loose')}
|
|
gridRowGap={space('extra-loose')}
|
|
gridTemplateColumns={[
|
|
`repeat(1, 1fr)`,
|
|
`repeat(${pages.length === 1 ? 1 : 2}, 1fr)`,
|
|
`repeat(${pages.length === 1 ? 1 : 2}, 1fr)`,
|
|
`repeat(${pages.length === 1 ? 1 : 3}, 1fr)`,
|
|
]}
|
|
>
|
|
{pages.map(page =>
|
|
variant === 'inline' ? <InlineCard page={page} /> : <GridCard page={page} />
|
|
)}
|
|
</Grid>
|
|
);
|
|
});
|
|
|