Β
Β
Β
Β
live demo: https://jukebox.candycode.com
Explore or clone this repository to see how all the pieces of our Next.js Conf talk fit together in a fun Next.js jukebox application powered by Tailwind CSS and Framer Motion.
Then continue reading below to learn how to apply these techniques to your own web projects. It assumes you're using the Next.js app router with Tailwind CSS, but can be adapted to work in other contexts as well. See the official Next.js docs for more details.
Self-hosting Google Fonts on your own domain can be done by importing the font from the next/font/google
bundle.
// ~/app/layout.js
import { Roboto } from 'next/font/google';
import cx from 'classnames';
export default function App({ children }) {
return (
<html lang="en">
<body className={cx(roboto.variable, 'any other classes')}>{children}</body>
</html>
);
}
const roboto = Roboto({
weight: ['400', '700', '900'],
style: ['normal', 'italic'],
subsets: ['latin'],
display: 'swap',
variable: '--font-roboto',
});
// tailwind.config.js
module.exports = {
theme: {
extend: {
fontFamily: {
roboto: 'var(--font-roboto), serif',
},
},
},
};
// component.js
export const Component = () => {
return <div className="font-roboto font-bold">This is in Roboto bold</div>;
};
If you want to supply your own local font files, the syntax for next/font/local
is a bit different.
// ~/app/layout.js
import localFont from 'next/font/local';
import cx from 'classnames';
export default function App({ children }) {
return (
<html lang="en">
<body className={cx(foundersGrotesk.variable, 'any other classes')}>{children}</body>
</html>
);
}
const foundersGrotesk = localFont({
src: [
{
path: '../fonts/founders-grotesk-regular.woff2',
weight: '400',
style: 'normal',
display: 'swap',
},
// add each weight and style combination
],
variable: '--founders-grotesk',
});
// tailwind.config.js
module.exports = {
theme: {
extend: {
fontFamily: {
'founders-grotesk': 'var(--font-founders-grotesk), serif',
},
},
},
};
// component.js
export const Component = () => {
return <div className="font-founders-grotesk">This is in Founders Grotesk</div>;
};
If you want to ensure that your type is only ever rendered in the custom web font even for those on slower connections, you can change the font display property to block rendering the type until the web font is loaded.
- display: 'swap',
+ display: 'block',
The required props are determined based on whether you're loading a static file from the public
directory (below), loading from a headless CMS, or importing an image file directly.
// component.js
import Image from 'next/image';
export const Component = () => {
return (
<>
<Image src="/jukebox.png" height={640} width={480} alt="A jukebox in a dive bar" />
</>
);
};
For images that appear above the fold of a web page, you'll want to to opt out of lazy loading to ensure the image loads eagerly as fast as possible.
- <Image />
+ <Image priority />
Instead of leaving an empty space before images load, add a solid color or low resolution preview as a placeholder. The blur data hash may be supplied automatically from your CMS, but could also be generated manually using third-party tools.
- <Image />
+ <Image placeholder="blur" blurDataURL="..." />
// ~/app/section/[slug]/page.js
export default function SectionPage({ params }) {
return (
<>
<div>content for {params.slug}</div>
</>
);
}
// ~/app/@modal/(.)section/[slug]/page.js
import { Modal } from '~/components/modal';
export default function SectionModal({ params }) {
return (
<Modal>
<div>content for {params.slug}</div>
</Modal>
);
}
// ~/app/@modal/default.js
export default function Default() {
return null;
}
// ~/app/layout.js
export default function App(props) {
return (
<html lang="en">
<body>
{props.children}
{props.modal}
</body>
</html>
);
}
If an element is shared between two pages, you can animate a transition using Framer Motion.
// ~/app/page.js
import Link from 'next/link';
import { motion } from 'framer-motion';
export default function HomePage() {
const slug = 'yourPageSlug';
return (
<>
<h1>Home page</h1>
<section>
<motion.h3 layout layoutId={`${slug}-headline`}>
{slug}
</motion.h3>
<Link href={`/section/${slug}`}>Visit {slug}</Link>
</section>
</>
);
}
// ~/app/section/[slug]/page.js
import { motion } from 'framer-motion';
export default function SectionPage({ params }) {
return (
<div>
<motion.h1 layout layoutId={`${params.slug}-headline`}>
{params.slug}
</motion.h1>
</div>
);
}
// ~/app/@modal/(.)section/[slug]/page.js
import { motion } from 'framer-motion';
import { Modal } from '~/components/modal';
export default function SectionModal({ params }) {
return (
<Modal>
<div>
<motion.h2 layout layoutId={`${params.slug}-headline`}>
{params.slug}
</motion.h2>
</div>
</Modal>
);
}
// ~/app/section/[slug]/opengraph-image.js
import { ImageResponse } from 'next/og';
export default async function OpenGraphImage({ params }) {
const foundersGroteskRegular = fetch(
new URL('../../../fonts/founders-grotesk-regular.otf', import.meta.url),
).then((res) => res.arrayBuffer());
return new ImageResponse(
(
<div
style={{
display: 'flex',
justifyContent: 'center',
alignItems: 'center',
width: '100%',
height: '100%',
background: 'black',
fontSize: 128,
color: 'white',
}}
>
{params.slug}
</div>
),
{
...size,
fonts: [
{
name: 'Founders Grotesk',
data: await foundersGroteskRegular,
weight: 400,
style: 'normal',
},
],
},
);
}
export const size = {
width: 1200,
height: 630,
};
export const contentType = 'image/png';
export const alt = 'OpenGraph image alternate text';
export const runtime = 'edge';