mirror of
https://github.com/facebook/docusaurus.git
synced 2025-06-04 11:52:39 +02:00
268 lines
7.9 KiB
TypeScript
268 lines
7.9 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.
|
|
*/
|
|
|
|
import React from 'react';
|
|
import clsx from 'clsx';
|
|
import LiteYouTubeEmbed from 'react-lite-youtube-embed';
|
|
import 'react-lite-youtube-embed/dist/LiteYouTubeEmbed.css';
|
|
import Link from '@docusaurus/Link';
|
|
import Translate, {translate} from '@docusaurus/Translate';
|
|
import useDocusaurusContext from '@docusaurus/useDocusaurusContext';
|
|
import useBaseUrl, {useBaseUrlUtils} from '@docusaurus/useBaseUrl';
|
|
|
|
import Image from '@theme/IdealImage';
|
|
import Layout from '@theme/Layout';
|
|
|
|
import Tweet from '@site/src/components/Tweet';
|
|
import Tweets, {type TweetItem} from '@site/src/data/tweets';
|
|
import Quotes from '@site/src/data/quotes';
|
|
import Features, {type FeatureItem} from '@site/src/data/features';
|
|
|
|
import styles from './styles.module.css';
|
|
|
|
function HeroBanner() {
|
|
return (
|
|
<div className={styles.hero} data-theme="dark">
|
|
<div className={styles.heroInner}>
|
|
<h1 className={styles.heroProjectTagline}>
|
|
<img
|
|
alt={translate({message: 'Docusaurus with Keytar'})}
|
|
className={styles.heroLogo}
|
|
src={useBaseUrl('/img/docusaurus_keytar.svg')}
|
|
width="200"
|
|
height="200"
|
|
/>
|
|
<span
|
|
className={styles.heroTitleTextHtml}
|
|
// eslint-disable-next-line react/no-danger
|
|
dangerouslySetInnerHTML={{
|
|
__html: translate({
|
|
id: 'homepage.hero.title',
|
|
message:
|
|
'Build <b>optimized</b> websites <b>quickly</b>, focus on your <b>content</b>',
|
|
description:
|
|
'Home page hero title, can contain simple html tags',
|
|
}),
|
|
}}
|
|
/>
|
|
</h1>
|
|
<div className={styles.indexCtas}>
|
|
<Link className="button button--primary" to="/docs">
|
|
<Translate>Get Started</Translate>
|
|
</Link>
|
|
<Link className="button button--info" to="https://docusaurus.new">
|
|
<Translate>Try a Demo</Translate>
|
|
</Link>
|
|
<span className={styles.indexCtasGitHubButtonWrapper}>
|
|
<iframe
|
|
className={styles.indexCtasGitHubButton}
|
|
src="https://ghbtns.com/github-btn.html?user=facebook&repo=docusaurus&type=star&count=true&size=large"
|
|
width={160}
|
|
height={30}
|
|
title="GitHub Stars"
|
|
/>
|
|
</span>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
function MigrationAnnouncement() {
|
|
return (
|
|
<div
|
|
className={clsx(styles.announcement, styles.announcementDark)}
|
|
data-theme="dark">
|
|
<div className={styles.announcementInner}>
|
|
<Translate
|
|
values={{
|
|
docusaurusV1Link: (
|
|
<Link to="https://v1.docusaurus.io/">
|
|
<Translate>Docusaurus v1</Translate>
|
|
</Link>
|
|
),
|
|
migrationGuideLink: (
|
|
<Link to="/docs/migration">
|
|
<Translate>v1 to v2 migration guide</Translate>
|
|
</Link>
|
|
),
|
|
}}>
|
|
{`Coming from {docusaurusV1Link}? Check out our {migrationGuideLink}.`}
|
|
</Translate>
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
function TweetsSection() {
|
|
const tweetColumns: Array<Array<TweetItem>> = [[], [], []];
|
|
Tweets.filter((tweet) => tweet.showOnHomepage).forEach((tweet, i) =>
|
|
tweetColumns[i % 3]!.push(tweet),
|
|
);
|
|
|
|
return (
|
|
<div className={clsx(styles.section, styles.sectionAlt)}>
|
|
<div className="container">
|
|
<h2 className={clsx('margin-bottom--lg', 'text--center')}>
|
|
<Translate>Loved by many engineers</Translate>
|
|
</h2>
|
|
<div className={clsx('row', styles.tweetsSection)}>
|
|
{tweetColumns.map((tweetItems, i) => (
|
|
<div className="col col--4" key={i}>
|
|
{tweetItems.map((tweet) => (
|
|
<Tweet {...tweet} key={tweet.url} />
|
|
))}
|
|
</div>
|
|
))}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
function QuotesSection() {
|
|
return (
|
|
<div className={clsx(styles.section)}>
|
|
<div className="container">
|
|
<div className="row">
|
|
{Quotes.map((quote) => (
|
|
<div className="col" key={quote.name}>
|
|
<div className="avatar avatar--vertical margin-bottom--sm">
|
|
<Image
|
|
alt={quote.name}
|
|
className="avatar__photo avatar__photo--xl"
|
|
img={quote.thumbnail}
|
|
style={{overflow: 'hidden'}}
|
|
/>
|
|
<div className="avatar__intro padding-top--sm">
|
|
<div className="avatar__name">{quote.name}</div>
|
|
<small className="avatar__subtitle">{quote.title}</small>
|
|
</div>
|
|
</div>
|
|
<p className="text--center text--italic padding-horiz--md">
|
|
{quote.text}
|
|
</p>
|
|
</div>
|
|
))}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
function VideoContainer() {
|
|
return (
|
|
<div className="container text--center margin-bottom--xl">
|
|
<div className="row">
|
|
<div className="col">
|
|
<h2>
|
|
<Translate>Check it out in the intro video</Translate>
|
|
</h2>
|
|
<div className="video-container">
|
|
<LiteYouTubeEmbed
|
|
id="_An9EsKPhp0"
|
|
params="autoplay=1&autohide=1&showinfo=0&rel=0"
|
|
title="Explain Like I'm 5: Docusaurus"
|
|
poster="maxresdefault"
|
|
webp
|
|
/>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
function Feature({
|
|
feature,
|
|
className,
|
|
}: {
|
|
feature: FeatureItem;
|
|
className?: string;
|
|
}) {
|
|
const {withBaseUrl} = useBaseUrlUtils();
|
|
|
|
return (
|
|
<div className={clsx('col', className)}>
|
|
<img
|
|
className={styles.featureImage}
|
|
alt={feature.title}
|
|
width={Math.floor(feature.image.width)}
|
|
height={Math.floor(feature.image.height)}
|
|
src={withBaseUrl(feature.image.src)}
|
|
loading="lazy"
|
|
/>
|
|
<h3 className={clsx(styles.featureHeading)}>{feature.title}</h3>
|
|
<p className="padding-horiz--md">{feature.text}</p>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
function FeaturesContainer() {
|
|
const firstRow = Features.slice(0, 3);
|
|
const secondRow = Features.slice(3);
|
|
|
|
return (
|
|
<div className="container text--center">
|
|
<h2>
|
|
<Translate>Main features</Translate>
|
|
</h2>
|
|
<div className="row margin-bottom--lg">
|
|
{firstRow.map((feature, idx) => (
|
|
<Feature feature={feature} key={idx} />
|
|
))}
|
|
</div>
|
|
<div className="row">
|
|
{secondRow.map((feature, idx) => (
|
|
<Feature
|
|
feature={feature}
|
|
key={idx}
|
|
className={clsx('col--4', idx === 0 && 'col--offset-2')}
|
|
/>
|
|
))}
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
export default function Home(): JSX.Element {
|
|
const {
|
|
siteConfig: {customFields, tagline},
|
|
} = useDocusaurusContext();
|
|
const {description} = customFields as {description: string};
|
|
return (
|
|
<Layout title={tagline} description={description}>
|
|
<main>
|
|
<div>
|
|
<div className={styles.banner}>
|
|
<Translate
|
|
id="homepage.banner"
|
|
values={{
|
|
link: (
|
|
<Link to="https://opensource.facebook.com/support-ukraine">
|
|
<Translate id="homepage.banner.link">
|
|
Help Provide Humanitarian Aid to Ukraine
|
|
</Translate>
|
|
</Link>
|
|
),
|
|
}}>
|
|
{'Support Ukraine 🇺🇦 {link}.'}
|
|
</Translate>
|
|
</div>
|
|
</div>
|
|
<HeroBanner />
|
|
<MigrationAnnouncement />
|
|
<div className={styles.section}>
|
|
<VideoContainer />
|
|
<FeaturesContainer />
|
|
</div>
|
|
<TweetsSection />
|
|
<QuotesSection />
|
|
</main>
|
|
</Layout>
|
|
);
|
|
}
|