Create presentation decks using MDX, React, and Next.js.
- π½ React-based Slideshow
- βοΈ Write using Markdown, React components, even HTML!
- π¨ Themeable with CSS
- π Swipe to change slides
- β»οΈ Sync slides between browser tabs
- π¨βπ» Presentation Mode
- π Speaker Notes
- βοΈ MDX mode for use within a website (Airview)
- Clone the project:
git clone https://github.com/AirWalk-Digital/mdx-deck-etherpad
- Install dependencies:
yarn
- Run the dev server:
npm run dev
oryarn dev
- Edit the first slide in
/markdwon/test.mdx
and save to see changes!
or
- Browse to Etherpad here
- Create a new pad
- [browse pads](http://localhost:3000/
or
use the docker image with Etherpad:
source .env* && docker run -d --env ETHERPAD_API_KEY=$ETHERPAD_API_KEY --env ETHERPAD_BASE_URL=$ETHERPAD_BASE_URL -p 8080:3000 --name mdx-deck ghcr.io/airwalk-digital/mdx-deck-etherpad:local
[browse pads](http://localhost:8080/
When you're done, run npm run build && npm run export
or yarn build && yarn export
will create a static app you can deploy anywhere (or use locally). See below for more details.
This project is easy to build locally or using a host with build services (like Netlify or Now).
- βοΈ Run the build process:
yarn export
- π Upload the static contents of
out
folder to host/CDN (or run theout/index.html
locally)
The default theme includes a title, date, author (and link to the author's website) in of the <Header>
component. You can edit this data inside the site.config.js
file.
You can use JSX in a few ways in your MDX files:
- You can use the syntax with HTML (
<button style={{ color: "red" }}>
) - You can import React component from other files (
import Button from "../components/Button"
). Then you can use that component anywhere in that MDX file. The path to the component is relative to the MDX file. - You can use any React component imported into the
<MDXProvider>
(inside/components/MDXProvider.js
). This allows you to use the component without importing it inside each MDX file. - You can define React components inside MDX files and then use them. MDX supports the use of JS inside files, like exporting variables, or in this case β defining new functions.
const Button = () => <button style={{ color: "red" }}>
Check out the MDX docs for more information on the syntax.
You can create new slide pages by making new .mdx
files inside the /markdown/
directory. Each slide is separated by a markdown divider (---
).
You have to wrap the contents of each
.mdx
file with a<SlidePage>
component. This is what breaks up each slide page into individual slides using the markdown dividers. [Note: this isn't required now]
[Note: this isn't implemented yet] Slide pages represent real pages - so page-2.mdx
== your-site.com/page-2
. By using the next
prop on the <SlidePage>
component you can control the next page you navigate to (after you go through all the slides).
// Navigates to your-site.com/your-custom-page after all slides complete
<SlidePage next="your-custom-page">
Theming is accomplished with CSS custom properties and/or Styled Components.
Design tokens are stored as CSS custom properties inside the SlidePage layout (/layouts/SlidePage.jsx
), which are injected into the app using Styled Component's global styling utility. There you can change the color of text, background colors, fonts, etc.
The actual CSS styles of the Slideshow are also stored in the SlidePage layout. There you can change the padding of slides, alignment of quotes, etc.
When the Markdown is parsed into HTML, you can replace HTML with React components. These "swaps" are handled by the <MDXProvider>
component. You can import custom components and swap elements (like a <button>
with <Button>
), or import components to use inside MDX (great for creating things like 2-col layouts with component). There you can change the syntax highlighting or find any custom MDX components. It's recommended to use Styled Components there to create custom components.
[Note: this isn't implemented yet]
Presentation mode allows you to view the current slide alongside any "speaker notes" included inside the slide.
- Press
ALT/OPT + P
to toggle presentation mode on and off. - You can also add the
mode
query parameter to the URL (e.g.http://localhost:3000/slides/1?mode=presentation
).
The most common way to use presentation mode:
- Open two browser tabs with any slide page.
- Activate presentation mode in one tab.
- Navigate through slides - they'll be synced in both tabs!
This way you can view your presentation on one monitor, while displaying the slides on another screen. Or if you're livestreaming, you can capture the slideshow window in software like OBS, and view the presentation window privately.
Note that syncing between tabs is achieved using
localStorage
. If you want to use an "Incognito" browser, make sure both tabs are "incognito" or they won't sync.
Speaker notes are only displayed during presentation mode. This allows you to write private notes to yourself that you can see in "presentation" mode, while the audience only sees the other slide content in "slideshow" mode.
Speaker notes can contain Markdown, MDX/JSX, and even HTML (as JSX). The notes are displayed in a scrollable window to the side of slide content during "presenation" mode.
To create speaker notes, you use the <SpeakerNotes>
component inside of your MDX files. No need to import it, it's automatically imported into any MDX slide page. You can also use it multiple times within the same slide, all the notes (per slide) will be combined.
Here's an example:
Slide content would go here.
<SpeakerNotes>
Private notes here.
# Even Markdown!
</SpeakerNotes>
MDX allows you to use JSX inline or import components, but if you want to use a React component across all slides without importing it, you can use the <MDXProvider>
component. This component wraps the app in a "context" that provides MDX with components to pass into the parser.
This also lets you replace Markdown parsed HTML elements with React components, like replacing ## Headings
with <Heading as="h2">
instead of the default <h2>
. This comes in handy if you have a React component library and you want to use it's primitives like <Text>
for paragraphs.
You can pass new components, or swap HTML elements inside the mdComponents
object in the /components/MDXProvider.jsx
file:
const mdComponents = {
h1: (props) => <h1 {...props} />,
CustomButton,
};
You can use any Framer Motion component:
<motion.div
animate={{scale: 2}}
transition={{ duration: 0.5 }}
>
# Your animated header
</motion.div>
Displays content centered and larger, commonly for the first slide in a deck.
<Cover>
# NextJS MDX Deck
Create presentations using Next & React & MDX.
</Cover>
Reveals content step by step as the user does next/prev slide button.
<Steps>
<Step>Something</Step>
<Step>happens</Step>
<Step>one</Step>
<Step>by</Step>
<Step>one</Step>
<Step>easy to add stuff</Step>
</Steps>
Custom order
You can also define the order explicitly:
<Steps>
<Step order={3}>Third</Step>
<Step order={2}>Second</Step>
<Step order={1}>First</Step>
</Steps>
Duration
You can control the animation duration using the prop and providing an integer (representing seconds):
<Steps>
{/** 3 seconds **/}
<Step duration={3}>Something</Step>
<Step>happens</Step>
</Steps>
To learn more about MDX, take a look at the following resources:
To learn more about Next.js, take a look at the following resources:
- Next.js Documentation - learn about Next.js features and API.
- Learn Next.js - an interactive Next.js tutorial.