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

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>
);
});