aboutsummaryrefslogtreecommitdiffstats
path: root/src/components/organisms/layout/summary.tsx
blob: 3624e5d1740c4df784b62c0a504a45793816681f (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
import ButtonLink from '@components/atoms/buttons/button-link';
import Heading, { type HeadingLevel } from '@components/atoms/headings/heading';
import Arrow from '@components/atoms/icons/arrow';
import Link from '@components/atoms/links/link';
import ResponsiveImage from '@components/molecules/images/responsive-image';
import Meta, { type MetaItem } from '@components/molecules/layout/meta';
import { VFC } from 'react';
import { useIntl } from 'react-intl';
import styles from './summary.module.scss';

export type Cover = {
  alt: string;
  height: number;
  url: string;
  width: number;
};

export type RequiredMetaKey = 'publication';

export type RequiredMeta = {
  [key in RequiredMetaKey]: MetaItem;
};

export type OptionalMetaKey =
  | 'author'
  | 'categories'
  | 'comments'
  | 'readingTime'
  | 'update';

export type OptionalMeta = {
  [key in OptionalMetaKey]?: MetaItem;
};

export type Meta = RequiredMeta & OptionalMeta;

export type SummaryProps = {
  cover?: Cover;
  excerpt: string;
  meta: Meta;
  title: string;
  titleLevel?: HeadingLevel;
  url: string;
};

/**
 * Summary component
 *
 * Render a page summary.
 */
const Summary: VFC<SummaryProps> = ({
  cover,
  excerpt,
  meta,
  title,
  titleLevel = 2,
  url,
}) => {
  const intl = useIntl();

  return (
    <article className={styles.wrapper}>
      {cover && (
        <ResponsiveImage
          alt={cover.alt}
          src={cover.url}
          width={cover.width}
          height={cover.height}
          className={styles.cover}
        />
      )}
      <header className={styles.header}>
        <Link href={url}>
          <Heading level={titleLevel} className={styles.title}>
            {title}
          </Heading>
        </Link>
      </header>
      <div className={styles.body}>
        {excerpt}
        <ButtonLink target={url} className={styles['read-more']}>
          {intl.formatMessage(
            {
              defaultMessage: 'Read more<a11y> about {title}</a11y>',
              description: 'Summary: read more link',
              id: 'Zpgv+f',
            },
            {
              title,
              a11y: (chunks: string) => (
                <span className="screen-reader-text">{chunks}</span>
              ),
            }
          )}
          <Arrow direction="right" />
        </ButtonLink>
      </div>
      <footer className={styles.footer}>
        <Meta data={meta} layout="column" className={styles.meta} />
      </footer>
    </article>
  );
};

export default Summary;