Skip to content

Commit

Permalink
chore: add example for next 13 with SSR
Browse files Browse the repository at this point in the history
  • Loading branch information
chrishelgert committed Sep 5, 2023
1 parent ac9a8bc commit e9eb949
Show file tree
Hide file tree
Showing 24 changed files with 4,028 additions and 5 deletions.
8 changes: 8 additions & 0 deletions examples/next-13-app-router-ssr/.env.local.example
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
# This is the Space ID from your Contentful space.
CONTENTFUL_SPACE_ID=
# This is the Content Delivery API - access token, which is used for fetching published data from your Contentful space.
CONTENTFUL_ACCESS_TOKEN=
# This is the Content Preview API - access token, which is used for fetching draft data from your Contentful space.
CONTENTFUL_PREVIEW_ACCESS_TOKEN=
# This can be any value you want. It must be URL friendly as it will be send as a query parameter to enable draft mode.
CONTENTFUL_PREVIEW_SECRET=
3 changes: 3 additions & 0 deletions examples/next-13-app-router-ssr/.eslintrc.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
{
"extends": "next/core-web-vitals"
}
40 changes: 40 additions & 0 deletions examples/next-13-app-router-ssr/.gitignore
Original file line number Diff line number Diff line change
@@ -0,0 +1,40 @@
# See https://help.github.com/articles/ignoring-files/ for more about ignoring files.

# dependencies
/node_modules
/.pnp
.pnp.js

# testing
/coverage

# next.js
/.next/
/out/

# production
/build
/lib/live-preview.mjs
/lib/live-preview.umd.js

# misc
.DS_Store
*.pem

# debug
npm-debug.log*
yarn-debug.log*
yarn-error.log*

# local env files
.env*.local

# vercel
.vercel

# typescript
*.tsbuildinfo
next-env.d.ts

.next
.env
42 changes: 42 additions & 0 deletions examples/next-13-app-router-ssr/README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,42 @@
# Next.js GraphQL Contentful live preview SDK example

This is an example project that demonstrates how to use the `@contentful/live-preview` SDK with a Next.js application that uses the GraphQL API. The live preview SDK provides live updates functionality for content changes and the inspector mode for your Contentful space.

## 1. Installation

Install the dependencies:

```bash
npm install
```

## 2. Environment variables

To run this project, you will need to add the following environment variables to your `.env.local` file:

- `CONTENTFUL_SPACE_ID`: This is the Space ID from your Contentful space.
- `CONTENTFUL_ACCESS_TOKEN`: This is the Content Delivery API - access token, which is used for fetching **published** data from your Contentful space.
- `CONTENTFUL_PREVIEW_ACCESS_TOKEN`: This is the Content Preview API - access token, which is used for fetching **draft** data from your Contentful space.
- `CONTENTFUL_PREVIEW_SECRET`: This can be any value you want. It must be URL friendly as it will be send as a query parameter to enable draft mode.

## 3. Setting up the content model

You will need to set up a content model within your Contentful space. For this project, we need a `Post` content type with the following fields:

- `slug`
- `title`
- `description`

Once you've set up the `Post` content model, you can populate it with some example entries.

## 4. Setting up Content preview URL

In order to enable the live preview feature in your local development environment, you need to set up the Content preview URL in your Contentful space.

`http://localhost:3000/api/draft?secret=<CONTENTFUL_PREVIEW_SECRET>&slug={entry.fields.slug}`

Replace `<CONTENTFUL_PREVIEW_SECRET>` with its respective value in `.env.local`.

## 5. Running the project locally

To run the project locally, you can use the `npm run dev` command. You can now use the live preview feature.
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
import { draftMode } from 'next/headers';

export async function GET(request: Request) {
draftMode().disable();
return new Response('Draft mode is disabled');
}
45 changes: 45 additions & 0 deletions examples/next-13-app-router-ssr/app/api/draft/route.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,45 @@
// route handler with secret and slug
import { getPreviewPostBySlug } from '@/lib/api-graphql';
import { cookies, draftMode } from 'next/headers';
import { redirect } from 'next/navigation';

export async function GET(request: Request) {
// Parse query string parameters
const { searchParams } = new URL(request.url);
const secret = searchParams.get('secret');
const slug = searchParams.get('slug');

// This secret should only be known to this route handler and the CMS
if (secret !== process.env.CONTENTFUL_PREVIEW_SECRET || !slug) {
return new Response('Invalid token', { status: 401 });
}

// Fetch the headless CMS to check if the provided `slug` exists
// getPostBySlug would implement the required fetching logic to the headless CMS
const post = await getPreviewPostBySlug(slug);

// If the slug doesn't exist prevent draft mode from being enabled
if (!post) {
return new Response('Invalid slug', { status: 401 });
}

// Enable Draft Mode by setting the cookie
draftMode().enable();

// Override cookie header for draft mode for usage in live-preview
// https://github.com/vercel/next.js/issues/49927
const cookieStore = cookies();
const cookie = cookieStore.get('__prerender_bypass')!;
cookies().set({
name: '__prerender_bypass',
value: cookie?.value,
httpOnly: true,
path: '/',
secure: true,
sameSite: 'none',
});

// Redirect to the path from the fetched post
// We don't redirect to searchParams.slug as that might lead to open redirect vulnerabilities
redirect(`/posts/${post.slug}`);
}
25 changes: 25 additions & 0 deletions examples/next-13-app-router-ssr/app/components/post-layout.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,25 @@
'use client';

import {
useContentfulInspectorMode,
useContentfulLiveUpdates,
} from '@contentful/live-preview/react';
import ErrorPage from 'next/error';
import { Post } from '../../lib/api-graphql';

export default function PostLayout({ post }: { post: Post | undefined }) {
const updatedPost = useContentfulLiveUpdates(post);

const inspectorProps = useContentfulInspectorMode({ entryId: post?.sys.id });

if (!updatedPost) {
return <ErrorPage statusCode={404} />;
}

return (
<>
<h1 {...inspectorProps({ fieldId: 'title' })}>{updatedPost?.title || ''}</h1>
<p {...inspectorProps({ fieldId: 'description' })}>{updatedPost?.description || ''}</p>
</>
);
}
Binary file added examples/next-13-app-router-ssr/app/favicon.ico
Binary file not shown.
26 changes: 26 additions & 0 deletions examples/next-13-app-router-ssr/app/layout.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,26 @@
import type { Metadata } from 'next';
import { Inter } from 'next/font/google';
import { draftMode } from 'next/headers';
import Script from 'next/script';
import './globals.css';
import { Providers } from './providers';

const inter = Inter({ subsets: ['latin'] });

export const metadata: Metadata = {
title: 'Create Next App',
description: 'Generated by create next app',
};

export default function RootLayout({ children }: { children: React.ReactNode }) {
const { isEnabled } = draftMode();

return (
<html lang="en">
<body className={inter.className}>
<Providers isEnabled={isEnabled}>{children}</Providers>
</body>
{isEnabled && <Script src="./live-preview.mjs" />}
</html>
);
}
32 changes: 32 additions & 0 deletions examples/next-13-app-router-ssr/app/page.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,32 @@
import { getAllPostsForHome } from '@/lib/api-graphql';
import { Metadata } from 'next';
import { draftMode } from 'next/headers';
import Link from 'next/link';

export const metadata: Metadata = {
title: 'Contentful live preview example with Next.js and GraphQL',
};

export default async function Home() {
const { isEnabled } = draftMode();

const posts = await getAllPostsForHome(isEnabled);

if (!posts || posts.length === 0) {
return (
<main>
<p>No posts found.</p>
</main>
);
}

return (
<>
{posts.map((post) => (
<Link href={`/posts/${post.slug}`} key={post.sys.id}>
{post.title}
</Link>
))}
</>
);
}
20 changes: 20 additions & 0 deletions examples/next-13-app-router-ssr/app/posts/[slug]/page.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,20 @@
import { draftMode } from 'next/headers';
import { getAllPostsWithSlug, getPost } from '../../../lib/api-graphql';

import PostLayout from '../../components/post-layout';

export default async function Post({ params }: { params: { slug: string } }) {
const { isEnabled } = draftMode();

const { post } = await getPost(params.slug, isEnabled);

return <PostLayout post={post} />;
}

export async function generateStaticParams() {
const posts = await getAllPostsWithSlug();

return posts?.map((post) => ({
slug: post.slug,
}));
}
20 changes: 20 additions & 0 deletions examples/next-13-app-router-ssr/app/providers.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,20 @@
import { ContentfulLivePreviewProvider } from '@contentful/live-preview/react';
import React from 'react';

export function Providers({
children,
isEnabled,
}: {
children: React.ReactNode;
isEnabled: boolean;
}) {
return (
<ContentfulLivePreviewProvider
locale="en-US"
enableInspectorMode={isEnabled}
enableLiveUpdates={false}
>
{children}
</ContentfulLivePreviewProvider>
);
}
Loading

0 comments on commit e9eb949

Please sign in to comment.