summaryrefslogtreecommitdiffstats
path: root/src/pages
Commit message (Expand)AuthorAgeFilesLines
...
* chore: create thematic page viewArmand Philippot2021-12-161-0/+77
* refactor: move SWR to blog pageArmand Philippot2021-12-161-5/+40
* chore: display year between blog postsArmand Philippot2021-12-161-1/+1
* chore: create contact pageArmand Philippot2021-12-161-0/+104
* chore: create legal notice viewArmand Philippot2021-12-151-0/+48
* chore: create CV viewArmand Philippot2021-12-151-0/+48
* chore: create single post viewArmand Philippot2021-12-151-0/+58
* fix: change branding title tag on homepageArmand Philippot2021-12-151-1/+1
* chore: add pagination (load more) to blog pageArmand Philippot2021-12-151-6/+9
* chore: retrieve posts list on blog pageArmand Philippot2021-12-151-0/+48
* chore: create homepage with graphql dataArmand Philippot2021-12-131-65/+15
* chore: load i18n translations using lingui providerArmand Philippot2021-12-132-1/+44
* chore: change app to use a layoutArmand Philippot2021-12-132-5/+12
* chore: add modern normalize and change global css to scss fileArmand Philippot2021-12-131-1/+1
* build: move global styles and pages inside src directoryArmand Philippot2021-12-123-0/+93
' href='#n178'>178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228
import { ExpandableWidget } from '@components/WidgetParts';
import { getIntlInstance } from '@utils/helpers/i18n';
import { useRouter } from 'next/router';
import { useEffect, useState } from 'react';
import { useIntl } from 'react-intl';
import styles from './Sharing.module.scss';

type Parameters = {
  content: string;
  image: string;
  title: string;
  url: string;
};

type Website = {
  id: string;
  name: string;
  parameters: Parameters;
  url: string;
};

const Sharing = ({ excerpt, title }: { excerpt: string; title: string }) => {
  const intl = useIntl();
  const [pageExcerpt, setPageExcerpt] = useState('');
  const [pageUrl, setPageUrl] = useState('');
  const [domainName, setDomainName] = useState('');
  const router = useRouter();

  useEffect(() => {
    const divEl = document.createElement('div');
    divEl.innerHTML = excerpt;
    const cleanExcerpt = divEl.textContent!;
    setPageExcerpt(cleanExcerpt);
  }, [excerpt]);

  useEffect(() => {
    const { protocol, hostname, port } = window.location;
    const currentPort = port ? `:${port}` : '';
    const fullUrl = `${protocol}//${hostname}${currentPort}${router.asPath}`;

    setDomainName(hostname);
    setPageUrl(fullUrl);
  }, [router.asPath]);

  const getSharingUrl = (website: Website): string => {
    const { id, parameters, url } = website;
    let sharingUrl = `${url}?`;
    let count = 0;

    for (const [key, value] of Object.entries(parameters)) {
      if (!value) continue;

      sharingUrl += count > 0 ? `&${value}=` : `${value}=`;

      switch (key) {
        case 'content':
          if (id === 'email') {
            const intro = intl.formatMessage({
              defaultMessage: 'Introduction:',
              description: 'Sharing: email content prefix',
            });
            const readMore = intl.formatMessage({
              defaultMessage: 'Read more here:',
              description: 'Sharing: content link prefix',
            });
            const body = `${intro}\n\n"${pageExcerpt}"\n\n${readMore} ${pageUrl}`;
            sharingUrl += encodeURI(body);
          } else {
            sharingUrl += encodeURI(pageExcerpt);
          }
          break;
        case 'title':
          const prefix =
            id === 'email'
              ? intl.formatMessage(
                  {
                    defaultMessage: 'Seen on {domainName}:',
                    description: 'Sharing: seen on text',
                  },
                  { domainName }
                )
              : '';
          sharingUrl += encodeURI(`${prefix} ${title}`);
          break;
        case 'url':
          sharingUrl += encodeURI(pageUrl);
          break;
        default:
          break;
      }

      count++;
    }

    return sharingUrl;
  };

  const websites = [
    {
      id: 'diaspora',
      name: intl.formatMessage({
        defaultMessage: 'Diaspora',
        description: 'Sharing: Diaspora',
      }),
      parameters: {
        content: '',
        image: '',
        title: 'title',
        url: 'url',
      },
      url: 'https://share.diasporafoundation.org/',
    },
    {
      id: 'facebook',
      name: intl.formatMessage({
        defaultMessage: 'Facebook',
        description: 'Sharing: Facebook',
      }),
      parameters: {
        content: '',
        image: '',
        title: '',
        url: 'u',
      },
      url: 'https://www.facebook.com/sharer/sharer.php',
    },
    {
      id: 'linkedin',
      name: intl.formatMessage({
        defaultMessage: 'LinkedIn',
        description: 'Sharing: LinkedIn',
      }),
      parameters: {
        content: '',
        image: '',
        title: '',
        url: 'url',
      },
      url: 'https://www.linkedin.com/sharing/share-offsite/',
    },
    {
      id: 'twitter',
      name: intl.formatMessage({
        defaultMessage: 'Twitter',
        description: 'Sharing: Twitter',
      }),
      parameters: {
        content: '',
        image: '',
        title: 'text',
        url: 'url',
      },
      url: 'https://twitter.com/intent/tweet',
    },
    {
      id: 'journal-du-hacker',
      name: intl.formatMessage({
        defaultMessage: 'Journal du hacker',
        description: 'Sharing: Journal du hacker',
      }),
      parameters: {
        content: '',
        image: '',
        title: 'title',
        url: 'url',
      },
      url: 'https://www.journalduhacker.net/stories/new',
    },
    {
      id: 'email',
      name: intl.formatMessage({
        defaultMessage: 'Email',
        description: 'Sharing: Email',
      }),
      parameters: {
        content: 'body',
        image: '',
        title: 'subject',
        url: '',
      },
      url: 'mailto:',
    },
  ];

  const getItems = () => {
    return websites.map((website) => {
      const { id, name } = website;
      const sharingUrl = getSharingUrl(website);
      const linkModifier = `link--${id}`;

      return (
        <li key={id}>
          <a
            href={sharingUrl}
            title={name}
            className={`${styles.link} ${styles[linkModifier]}`}
          >
            <span className="screen-reader-text">
              {intl.formatMessage(
                {
                  defaultMessage: 'Share on {name}',
                  description: 'Sharing: share on social network text',
                },
                { name }
              )}
            </span>
          </a>
        </li>
      );
    });
  };

  return (
    <ExpandableWidget
      title={intl.formatMessage({
        defaultMessage: 'Share',
        description: 'Sharing: widget title',
      })}
      expand={true}
    >
      <ul className={`${styles.list} ${styles['list--sharing']}`}>
        {getItems()}
      </ul>
    </ExpandableWidget>
  );
};

export default Sharing;