diff --git a/examples/cms-agilitycms/.env.local.example b/examples/cms-agilitycms/.env.local.example new file mode 100644 index 0000000000000..ccb9dd33a8fbe --- /dev/null +++ b/examples/cms-agilitycms/.env.local.example @@ -0,0 +1,4 @@ +NEXT_EXAMPLE_CMS_AGILITY_GUID= +NEXT_EXAMPLE_CMS_AGILITY_API_FETCH_KEY= +NEXT_EXAMPLE_CMS_AGILITY_API_PREVIEW_KEY= +NEXT_EXAMPLE_CMS_AGILITY_SECURITY_KEY= \ No newline at end of file diff --git a/examples/cms-agilitycms/.gitignore b/examples/cms-agilitycms/.gitignore new file mode 100644 index 0000000000000..1f6cbab32ca91 --- /dev/null +++ b/examples/cms-agilitycms/.gitignore @@ -0,0 +1,2 @@ +.env*.local +.vercel \ No newline at end of file diff --git a/examples/cms-agilitycms/README.md b/examples/cms-agilitycms/README.md new file mode 100644 index 0000000000000..350ab5e780641 --- /dev/null +++ b/examples/cms-agilitycms/README.md @@ -0,0 +1,325 @@ +# A statically generated blog example using Next.js and Agility CMS + +This example showcases Next.js's [Static Generation](https://nextjs.org/docs/basic-features/pages) feature using [Agility CMS](https://www.agilitycms.com) as the data source. + +> `IMPORTANT` - This example uses Agility CMS's [**Page Management**](https://agilitycms.com/resources/posts/page-management-in-agility-cms-vs-other-headless-cmss) features. This means that the CMS ultimately drives what pages are available and what content is on each page. This enables **Editors** to focus on managing their pages, while allowing you, (the **Developer**) to focus on building UI components for the editors to compose their pages. + +## Demo + +- **Live**: [https://next-blog-agilitycms.now.sh/](https://next-blog-agilitycms.now.sh/) +- **Preview Mode**: [https://next-blog-agilitycms.now.sh/?agilitypreviewkey=...](https://next-blog-agilitycms.now.sh/?agilitypreviewkey=GzL%2fio1pLkfKc9BR1%2fC1cDQeKjL0AkwrTAJ22q3UEjAcOhyrqZejDkDv4kMlBKqrEuQxsuRyiP%2bUaykDYlJ%2fJg%3d%3d) + +### Related examples + +- [Agility CMS Sample Starter](https://github.com/agility/agilitycms-next-starter-ssg) +- [Blog Starter](/examples/blog-starter) +- [Sanity](/examples/cms-sanity) +- [TakeShape](/examples/cms-takeshape) +- [Prismic](/examples/cms-prismic) + +## How to use + +### Using `create-next-app` + +Execute [`create-next-app`](https://github.com/zeit/next.js/tree/canary/packages/create-next-app) with [npm](https://docs.npmjs.com/cli/init) or [Yarn](https://yarnpkg.com/lang/en/docs/cli/create/) to bootstrap the example: + +```bash +npm init next-app --example cms-agilitycms cms-agilitycms-app +# or +yarn create next-app --example cms-agilitycms cms-agilitycms-app +``` + +### Download manually + +Download the example: + +```bash +curl https://codeload.github.com/zeit/next.js/tar.gz/canary | tar -xz --strip=2 next.js-canary/examples/cms-agilitycms +cd cms-agilitycms +``` + +## Configuration + +### How is this Different from Other CMS Examples? + +The key principle behind Agility CMS is that **Editors** should have full control of their pages and what content is on each page without getting into code. + +This means you'll not only be definining **Content** for your `Posts` and `Authors`, but you'll also be defining UI Components to compose your pages. This site will consist of a single **Page Template** and a collection of **Modules** that represent the UI components you see on the page. + +> **NOTE** - `Modules` and `Page Templates` in Agility CMS simply correspond to `React Components` in your website. + +Once you've gone through the steps below, you'll be able to dynamically manage pages (and what is on them) directly through the CMS without requiring a developer. + +### Step 1. Create an account and a project on `Agility CMS` + +First, [create an account on Agility CMS](https://agilitycms.com). + +After creating an account you'll be asked to create a new project. Use any name of your liking as the **Project Name** and select the **Blank (advanced users)** template to create a blank Agility CMS instance. + +### Step 2. Create an `Author` Content Definition + +From within the Agility CMS Content Manager, navigate to **Settings** > **Content Definitions** and click **New** to create a new **Content Definition**. + +- The **Title** should be `Author`. This will also pre-populate **Reference Name** for you. + +Next, add these fields via the **Form Builder** tab (you don't have to modify any other settings): + +- `Name` - Set **Field Label** to `Name` and **Field Type** to `Text` +- `Picture` - Set **Field Label** to `Picture` and **Field Type** to `Image` + +When you are done, click **Save & Close** to save your `Author` content definition. + +### Step 3. Create a `List` based on your `Author` Content Definition + +From within the Agility CMS Content Manager, navigate to **Shared Content** and click the **+ (New)** button, then fill the form like so: + +- **Type** should be `Content List` +- **Content Definition** should be **Author** +- **Display Name** should be set to **Authors**. This will also pre-populate **Reference Name** for you. + +### Step 4. Create a `Post` Content Definiton + +From within the Agility CMS Content Manager, navigate to **Settings** > **Content Definitions** and click **New** to create a new **Content Definition**. + +- The **Title** should be `Post`. + +Next, add these fields via the **Form Builder** tab (you don't have to modify any other settings): + +- `Title` - Set **Field Type** to `Text` +- `Slug` - Set **Field Type** to `Text` +- `Date` - Set **Field Type** to `Date/Time` +- `AuthorID` - Set **Field Type** to `Number` and enable **Hide field from input form** +- `Author` - Do the following: + - **Field Type** - `Linked Content` + - **Content Definition** - `Author` + - **Content View** - `Shared Content` + - **Shared Content** - `Authors` + - **Render As** - `Dropdown List` + - **Save Value To Field** - `AuthorID` +- `Excerpt` - Set **Field Type** to `Text` +- `Content` - Set **Field Type** to `HTML` +- `Cover Image` - Set **Field Type** to `Image` + +When you are done, click **Save & Close** to save your `Post` content definition. + +### Step 5. Create a `Dynamic Page List` based on your `Posts` Content Definition + +From within the Agility CMS Content Manager, navigate to **Shared Content** and click the **+ (New)** button, then fill the form like so: + +- **Type** should be `Dynamic Page List` +- **Content Definition** should be `Post` +- **Display Name** should be `Posts`. This will also pre-populate **Reference Name** for you + +### Step 6. Populate Content + +Go to **Shared Content**, select the **Authors** list and click the **+ New** button to create a new content item: + +- You just need **1 Author content item**. +- Use dummy data for the text. +- For the image, you can download one from [Unsplash](https://unsplash.com/). + +Click on **Save** and **Publish** once you're done. + +Next, select the **Posts** list and click the **+ New** button to create a new content item: + +- We recommend creating at least **2 Post content items**. +- Use dummy data for the text. +- You can write markdown for the **Content** field. +- For the images, you can download ones from [Unsplash](https://unsplash.com/). +- Pick the **Author** you created earlier. + +For each post content item, you need to click `Publish` after saving. If not, the post will be in the `Staging` state. + +### Step 7. Define your `Intro` Module + +Navigate to **Settings** > **Module Definitions** and click **New** to create a new **Module Definition**. + +- Set **Title** to `Intro` +- Set **Description** to `Displays an intro message.` + +In this case, we are not adding any fields to control the output or behaviour, since the content is actually hard-coded in the template. + +Click **Save & Close** to save the definition. + +### Step 8. Define your `Hero Post` Module + +Navigate to **Settings** > **Module Definitions** and click **New** to create a new **Module Definition**. + +- Set **Title** to `Hero Post` +- Set **Description** to `Displays the latest Post.` + +In this case, we are not adding any fields to control the output or behaviour, since the latest post will be used by default and all of the data is associated to the post itself. + +Click **Save & Close** to save the definition. + +### Step 9. Define your `More Stories` Module + +Navigate to **Settings** > **Module Definitions** and click **New** to create a new **Module Definition**. + +- Set **Title** to `More Stories` +- Set **Description** to `Displays a listing of Posts.` + +Next, add the following field: + +- `Title` - Set **Field Type** to `Text` + +Click **Save & Close** to save the definition. + +### Step 10. Define your `Post Details` Module + +Navigate to **Settings** > **Module Definitions** and click **New** to create a new **Module Definition**. + +- Set **Title** to `Post Details` +- Set **Description** to `Displays the details of a Post.` + +In this case, we are not adding any fields to control the output or behaviour, since the data is associated to the post itself. + +Click **Save & Close** to save the definition. + +### Step 11. Define a `One Column` Page Template + +Navigate to **Settings** > **Page Templates** and click **New** to create a new **Page Template**. + +- **Name** should be `One Column Template` +- **Digital Channel Type** should be `Website` +- Under **Module Zones** click `+ (New)` + - Set **Display Name** to `Main Content Zone`, it will populate **Reference Name** for you + - Click `Save` to apply the `Main Content Zone` + +Click **Save & Close** to save the page template. + +### Step 12. Add a new Page called `home` + +Navigate to **Pages** and click the **+ (New)** button in the page tree to create a new **Page**. + +- Set **Type** to `Page` +- Set **Page Template** to `One Column Template` +- Set **Menu Text** to `Home` - **Page Title** and **Page Name** fields will be auto-populated. + +Click **Save** to create the `/home` page. + +Next, let's add the `Intro`, `Hero Post` and `More Stories` modules to the `Main Content Zone` of the `home` page: + +- Click the **+ (New)** button on `Main Content Zone` and select `Intro` to add the module to the page +- Click **Save & Close** on the module to return back to the page + +- Click the **+ (New)** button on `Main Content Zone` and select `Hero Post` to add the module to the page +- Click **Save & Close** on the module to return back to the page + +- Click the **+ (New)** button on `Main Content Zone` and select `More Stories` to add the module to the page + - Set **Title** to `More Stories` +- Click **Save & Close** on the module to return back to the page + +Then click **Publish** on the page in order to publish the page and all of its modules. + +### Step 13. Add a new Folder called `posts` + +Navigate to **Pages** and click the `Website` channel, then click the **+ (New)** button in the page tree to create a new **Folder** in the root of the site: + +- Set **Type** to `Folder` +- Set **Menu Text** to `Posts`, **Folder Name** will be auto-populated to `posts` + +Click **Save** to create the `/posts` folder. + +**Important:** Click **Publish** on the folder. + +### Step 14. Add a new Dynamic Page called `posts-dynamic` + +Navigate to **Pages** and select the existing `/posts` folder. Click the **+ (New)** button in the page tree to create a new **Dynamic Page** underneath the `posts` page. + +- Set **Type** to `Dynamic Page` +- Set **Page Template** to `One Column Template` +- Set **Build Pages From** to `Posts` +- Set **Sitemap Label** to `posts-dynamic` +- Set **Page Path Formula** to `##Slug##` +- Set **Page Title Formula** and **Menu Text Formula** to `##Title##` + +Click **Save** to create the `/posts/posts-dynamic` dynamic page. + +Next, let's add the `Post Details` and `More Stories` modules to the `Main Content Zone` of the `posts-dynamic` page: + +- Click the **+ (New)** button on `Main Content Zone` and select `Post Details` to add the module to the page +- Click the **+ (New)** button on `Main Content Zone` and select `More Stories` to add the module to the page + - Set **Title** to `More Stories` +- Click **Save & Close** on the module to return back to the `posts-dynamic` page + +Then click **Publish** on the page in order to publish the page and all of its modules. + +### Step 15. Set up environment variables + +Copy the `.env.local.example` file in this directory to `.env.local` (which will be ignored by Git): + +```bash +cp .env.local.example .env.local +``` + +Go to the **Getting Started** section from the menu and click on **API Keys**. You should see a new modal called `Content API Details`, then click in the **Show API Key(s)** button within it. + +Then set each variable on `.env.local`: + +- `NEXT_EXAMPLE_CMS_AGILITY_GUID` should be the **Instance GUID** field +- `NEXT_EXAMPLE_CMS_AGILITY_API_FETCH_KEY` should be the **Live API Key** field +- `NEXT_EXAMPLE_CMS_AGILITY_API_PREVIEW_KEY` should be the **Preview API Key** field - this is used when the site is in [Preview Mode](https://nextjs.org/docs/advanced-features/preview-mode) and allows your site to pull the latest content, regardless of whether it is published or not. +- `NEXT_EXAMPLE_CMS_AGILITY_SECURITY_KEY` should be the **Security Key** field that can be found in **Settings** > **Global Security** - this is used to communicate between the CMS and your site to validate [Preview Mode](https://nextjs.org/docs/advanced-features/preview-mode) + +Your `.env.local` file should look like this: + +```bash +NEXT_EXAMPLE_CMS_AGILITY_GUID=... +NEXT_EXAMPLE_CMS_AGILITY_API_FETCH_KEY=... +NEXT_EXAMPLE_CMS_AGILITY_API_PREVIEW_KEY=... +NEXT_EXAMPLE_CMS_AGILITY_SECURITY_KEY=... +``` + +### Step 16. Run Next.js in development mode + +```bash +npm install +npm run dev + +# or + +yarn install +yarn dev +``` + +Your blog should be up and running on [http://localhost:3000](http://localhost:3000)! If it doesn't work, post on [GitHub discussions](https://github.com/zeit/next.js/discussions). + +### Step 17. Deploy on Vercel + +You can deploy this app to the cloud with [Vercel](https://vercel.com/import?filter=next.js&utm_source=github&utm_medium=readme&utm_campaign=next-example) ([Documentation](https://nextjs.org/docs/deployment)). + +To deploy on Vercel, you need to set the environment variables using the [Vercel CLI](https://vercel.com/download) ([Documentation](https://vercel.com/docs/now-cli#commands/secrets)). + +Install [Vercel CLI](https://vercel.com/download), log in to your account from the CLI, and run the following commands to add the environment variables. Replace each variable with the corresponding strings in `.env.local`: + +``` +vercel secrets add next_example_cms_agility_guid +vercel secrets add next_example_cms_agility_api_fetch_key +vercel secrets add next_example_cms_agility_api_preview_key +vercel secrets add next_example_cms_agility_security_key +``` + +Then push the project to GitHub/GitLab/Bitbucket and [import to Vercel](https://vercel.com/import?filter=next.js&utm_source=github&utm_medium=readme&utm_campaign=next-example) to deploy. + +### Step 18. Try preview mode + +Now that you've deployed your app to Vercel, take note of the URL of your deployed site. This will be registered in Agility CMS so that when editors click the `Preview` button within Agility CMS, your app is loaded in **Preview Mode**. Learn more about [NextJS Preview Mode](https://nextjs.org/docs/advanced-features/preview-mode)). + +To enable the Preview Mode, you'll need to add your site to **Domain Configuration** in Agility CMS: + +- Go to **Settings** > **Domain Configuration** +- Click on the existing channel in the list called `Website` +- Click on the **+ (New)** button to add a new domain: + - Set **Name** to `Production` + - Set **Domain URL** to the URL of your production deployment, it should look like `https://.now.sh` + - Enable **Preview Domain** + - Click **Save** to save your settings + +Go to one of your `Posts` and update the title. For example, you can add `[Staging]` in front of the title. Click **Save**, but **DO NOT** click **Publish**. By doing this, the post will be in the staging state. + +To enter **Preview Mode**, click the `Preview` button on the details of your `Post`. This redirects you to the `/` page, however you will now be in **Preview Mode** so you can navigate to your `Post` you want to view on the website. + +You should now be able to see the updated title. To exit the preview mode, you can click **Click here to exit preview mode** at the top. + +> NOTE - To set up preview on a specific `Post` (as opposed to the `/` page), click on the **Settings** tab of the `Post` list in **Shared Content**. For **Item Preview Page** set it to `~/posts/posts-dynamic` and for **Item Preview Query String Parameter** set it to `contentid`. diff --git a/examples/cms-agilitycms/components/alert.js b/examples/cms-agilitycms/components/alert.js new file mode 100644 index 0000000000000..3530e66e59c8d --- /dev/null +++ b/examples/cms-agilitycms/components/alert.js @@ -0,0 +1,42 @@ +import Container from './container' +import cn from 'classnames' +import { EXAMPLE_PATH } from '../lib/constants' + +export default function Alert({ preview }) { + return ( +
+ +
+ {preview ? ( + <> + This is page is a preview.{' '} + + Click here + {' '} + to exit preview mode. + + ) : ( + <> + The source code for this blog is{' '} + + available on GitHub + + . + + )} +
+
+
+ ) +} diff --git a/examples/cms-agilitycms/components/avatar.js b/examples/cms-agilitycms/components/avatar.js new file mode 100644 index 0000000000000..b0f2588908306 --- /dev/null +++ b/examples/cms-agilitycms/components/avatar.js @@ -0,0 +1,12 @@ +export default function Avatar({ name, picture }) { + return ( +
+ {name} +
{name}
+
+ ) +} diff --git a/examples/cms-agilitycms/components/container.js b/examples/cms-agilitycms/components/container.js new file mode 100644 index 0000000000000..fc1c29dfb0747 --- /dev/null +++ b/examples/cms-agilitycms/components/container.js @@ -0,0 +1,3 @@ +export default function Container({ children }) { + return
{children}
+} diff --git a/examples/cms-agilitycms/components/cover-image.js b/examples/cms-agilitycms/components/cover-image.js new file mode 100644 index 0000000000000..9bb01798efb4d --- /dev/null +++ b/examples/cms-agilitycms/components/cover-image.js @@ -0,0 +1,28 @@ +//import { Image } from 'react-datocms' +import Image from '../lib/components/image' +import cn from 'classnames' +import Link from 'next/link' + +export default function CoverImage({ title, responsiveImage, slug }) { + const image = ( + + ) + return ( +
+ {slug ? ( + + {image} + + ) : ( + image + )} +
+ ) +} diff --git a/examples/cms-agilitycms/components/date.js b/examples/cms-agilitycms/components/date.js new file mode 100644 index 0000000000000..eac5681378bfd --- /dev/null +++ b/examples/cms-agilitycms/components/date.js @@ -0,0 +1,6 @@ +import { parseISO, format } from 'date-fns' + +export default function Date({ dateString }) { + const date = parseISO(dateString) + return +} diff --git a/examples/cms-agilitycms/components/footer.js b/examples/cms-agilitycms/components/footer.js new file mode 100644 index 0000000000000..dbde8ff306efd --- /dev/null +++ b/examples/cms-agilitycms/components/footer.js @@ -0,0 +1,30 @@ +import Container from './container' +import { EXAMPLE_PATH } from '../lib/constants' + +export default function Footer() { + return ( + + ) +} diff --git a/examples/cms-agilitycms/components/header.js b/examples/cms-agilitycms/components/header.js new file mode 100644 index 0000000000000..562e7e3eebb6a --- /dev/null +++ b/examples/cms-agilitycms/components/header.js @@ -0,0 +1,12 @@ +import Link from 'next/link' + +export default function Header() { + return ( +

+ + Blog + + . +

+ ) +} diff --git a/examples/cms-agilitycms/components/hero-post.js b/examples/cms-agilitycms/components/hero-post.js new file mode 100644 index 0000000000000..2b742838cd9cb --- /dev/null +++ b/examples/cms-agilitycms/components/hero-post.js @@ -0,0 +1,49 @@ +import Link from 'next/link' +import Avatar from '../components/avatar' +import Date from '../components/date' +import CoverImage from '../components/cover-image' + +export default function HeroPost({ + title, + coverImage, + date, + excerpt, + author, + slug, +}) { + return ( +
+
+ +
+
+
+

+ + {title} + +

+
+ +
+
+ {author && ( +
+

{excerpt}

+ +
+ )} +
+
+ ) +} + +// The data returned here will be send as `props` to the component +HeroPost.getCustomInitialProps = async function ({ client }) { + const post = await client.getLatestPost() + return post +} diff --git a/examples/cms-agilitycms/components/intro.js b/examples/cms-agilitycms/components/intro.js new file mode 100644 index 0000000000000..5931b3c5961bd --- /dev/null +++ b/examples/cms-agilitycms/components/intro.js @@ -0,0 +1,28 @@ +import { CMS_NAME, CMS_URL } from '../lib/constants' + +export default function Intro() { + return ( +
+

+ Blog. +

+

+ A statically generated blog example using{' '} + + Next.js + {' '} + and{' '} + + {CMS_NAME} + + . +

+
+ ) +} diff --git a/examples/cms-agilitycms/components/layout.js b/examples/cms-agilitycms/components/layout.js new file mode 100644 index 0000000000000..99d95353131e0 --- /dev/null +++ b/examples/cms-agilitycms/components/layout.js @@ -0,0 +1,16 @@ +import Alert from '../components/alert' +import Footer from '../components/footer' +import Meta from '../components/meta' + +export default function Layout({ preview, children }) { + return ( + <> + +
+ +
{children}
+
+