1 + 1
[1] 2
This guide demonstrates an initial approach to displaying Quarto documents in a NextJS application using iframes. I am writing this post because I couldn’t find any solution to this issue on the interwebs.
While not the final solution, it provides a working foundation for integrating Quarto’s notebooks with NextJS’s modern web framework capabilities. The end goal is to move beyond iframes to a fully integrated solution where Quarto content is parsed into native React components, but this implementation serves as a practical starting point for those looking for a quick and dirty solution
app/
├── blog/
│ ├── [slug]/
│ │ ├── layout.tsx # Right sidebar with blog links
│ │ └── page.tsx # iframe container for Quarto content
│ ├── BlogSummary.tsx # Blog metadata and post info
│ └── page.tsx # Main blog listing page
└── public/
└── blogs/ # Quarto HTML files
Create your Quarto document with embedded resources enabled:
---
title: "Your Blog Title"
embed-resources: true
---
Run Quarto’s render command to generate a self-contained HTML file:
quarto render document.qmd
Place the generated HTML in your public folder:
mv document.html public/blogs/
export default function BlogLayout({
,
children: {
}: React.ReactNode
children
}) {return (
<div className="flex h-screen m-0 p-0">
<main className="flex-1 m-0 p-0">
{children}</main>
<aside className="w-48 bg-white shadow-lg h-screen overflow-y-auto border-l">
<div className="p-3">
<h2 className="text-lg font-semibold mb-3">All Posts</h2>
<nav>
.map((post) => (
{BlogPosts<a key={post.title} href={post.link}>
.title}
{post</a>
))}</nav>
</div>
</aside>
</div>
;
) }
interface PageProps {
: Promise<{
params: string;
slug>;
}
}
export default async function BlogPost({ params }: PageProps) {
const { slug } = await params;
return (
<div className="w-full h-screen">
<iframe
={`/blogs/${slug}.html`}
src="w-full h-full border-none"
className="Blog post content"
title={{
style: '100vh',
minHeight: '100%',
width: 0,
margin: 0,
padding: 'none'
border
}}/>
</div>
;
) }
Update BlogSummary.tsx with your post information:
interface BlogPost {
: string;
title: string;
description: string;
link: 'blog' | 'project';
type: string[];
tags: string;
featured: string;
image?: 'email' | 'github' | 'external';
linkType
}
export const BlogPosts: BlogPost[] = [
{: "Getting Quarto Started in NextJS",
title: "How to integrate Quarto with NextJS",
description: "/blog/quarto-nextjs",
link: "blog",
type: ["nextjs", "web dev", "quarto"],
tags: 'true',
featured: "/favicon.ico"
image
}; ]
When you click the Render button a document will be generated that includes both content and the output of embedded code. You can embed code like this:
1 + 1
[1] 2
You can add options to executable code like this
[1] 4
The echo: false
option disables the printing of code (only output is displayed).
library(ggplot2)
= seq(1:100)
x = seq(1:100)^2
y = data.frame(x,y)
df
ggplot(df,aes(x = x, y = y))+geom_point()
library(ggplot2)
library(plotly)
Attaching package: 'plotly'
The following object is masked from 'package:ggplot2':
last_plot
The following object is masked from 'package:stats':
filter
The following object is masked from 'package:graphics':
layout
= seq(1:100)
x = seq(1:100)^2
y = data.frame(x,y)
df
= ggplot(df,aes(x = x, y = y))+geom_point()
p
= ggplotly(p)
p
p
this is my \(s\) and this is my \(\phi\)