Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/tujoworker/gatsby-plugin-wrap-pages

With this Gatsby Plugin you can wrap your pages declaratively
https://github.com/tujoworker/gatsby-plugin-wrap-pages

gatsby gatsby-plugin layout providers react

Last synced: 2 months ago
JSON representation

With this Gatsby Plugin you can wrap your pages declaratively

Awesome Lists containing this project

README

        

# Gatsby Plugin for wrapping pages

With this plugin you can add (nested) wrappers inside the pages directory. It will give you declaratively control of what pages you wrap with either HTML elements or React Providers.

```js
my-project/
├── src/
│ └── pages/
│ ├── wrap-pages.tsx // ← wrapPagesDeep()
│ ├── index.tsx
│ └── your-path/
│ ├── wrap-pages.tsx // ← wrapPagesDeep()
│ ├── index.tsx
│ ├── … // more pages
│ └── more-nested-paths/
│ ├── wrap-pages.js // ← wrapPages()
│ ├── index.js
│ └── … // more pages
├── package.json
└── gatsby-config.json
```

Why? Everything is possible with vanilla Gatsby – what this Plugin offers, is a different, yet declarative way of wrapping your pages.

It supports:

- SSR (SSG) and client-side rendering with the same wrapper
- Wrap your pages with HTML Elements, React Components or Providers
- Wrap pages in current directory
- Wrap pages deep (nested)
- Nested wrappers
- Gatsby Themes and Plugins
- Gatsby Plugins used as [Micro Frontends](#micro-frontends)
- Programmatically created pages
- Custom name of the wrapper file

Live [Demo](https://tujoworker.github.io/gatsby-plugin-wrap-pages/) 🚀 provided by the [example code](https://github.com/tujoworker/gatsby-plugin-wrap-pages/tree/main/example-basic).

## How to use

Create a file called `wrap-pages.js` (or `.tsx`) inside `/src/pages/...somewhere` with a named export function. That's it.

The parameters of that function are the same as Gatsby's API [`wrapPageElement`](https://www.gatsbyjs.com/docs/reference/config-files/gatsby-browser/#wrapPageElement), including the `element`, which we now can return – but modified.

### Wrap pages at the current scope

- Add a wrapper function named `wrapPages`:

```jsx
// Inside wrap-pages.js (or.tsx)
export function wrapPages({ element }) {
return

{element}

}
```

Pages in that specific directory will be wrapped with `

[current page]
`.

### Wrap nested pages

- Add a function named `wrapPagesDeep`:

```jsx
// Inside wrap-pages.js (or.tsx)
export function wrapPagesDeep({ element }) {
return {element}
}
```

All your pages, including the ones in nested directories, will be wrapped in ``.

**NB:** It will also be used for the current directory scope, if `wrapPages` is not given.

... and yes, you can have both:

```jsx
// wraps pages in the current scope
export function wrapPages({ element }) {
return {element}
}

// wraps all nested pages (deep)
export function wrapPagesDeep({ element }) {
return {element}
}
```

### Nested wrappers

You can have as many nested wrappers as you like:

```js
my-project/
├── src/
│ └── pages/
│ ├── wrap-pages.js // ← One
│ └── directory/
│ ├── wrap-pages.js // ← Two
│ ├── …
│ └── directory/
│ ├── wrap-pages.js // ← Three
│ └── …
```

## Install

```bash
npm install gatsby-plugin-wrap-pages
# or
yarn add gatsby-plugin-wrap-pages
```

… and add it to your `gatsby-config.js` file:

```diff
exports.plugins = [
+ 'gatsby-plugin-wrap-pages'
]
```

## Plugin Options

You want to call the `wrap-pages.js` (or `.tsx`) file something different?

```cjs
exports.plugins = [
{
resolve: 'gatsby-plugin-wrap-pages',
options: {
wrapperName: 'yourWrapperName.tsx', // string or array
},
},
]
```

## Micro Frontends

Gatsby can be used to build UX focused micro frontends, where everything is page based and optimized for a fantastic user and a11y experience – while still deliver micro frontend independence in terms of DX and dedicated developer teams.

But the only piece missing is to easily customize what layout and what data provider is used by every micro application.

Now, **gatsby-plugin-wrap-pages** can be included by every "mirco app" independently. They even can define what they want to call the wrapper files (`wrapperName`) by itself. Or if that matters, it can be used just by one micro app – even if the root application is not aware of this plugin.

There is an [example setup](https://github.com/tujoworker/gatsby-plugin-wrap-pages/tree/main/example-micro-frontends) in this repo.

Example structure:

```js
micro-app-a/
├── src/
│ └── pages/
│ └── micro-app-a/
│ ├── wrap-pages.tsx // ← will wrap everything inside this directory
│ ├── index.tsx
│ └── … // more pages and nested routes
micro-app-b/
├── src/
│ └── pages/
│ └── micro-app-b/
│ ├── wrap-pages.tsx // ← will wrap everything inside this directory
│ ├── index.tsx
│ └── … // more pages and nested routes
main-application/
├── src/
│ └── pages/
│ └── main-layout.tsx // ← wrapPagesDeep()
├── package.json
└── gatsby-config.json
```

## Programmatically created pages

If you create some pages with the Gatsby API `createPage` "programmatically", and you also want these pages to get wrapped, you can provide `wrapPageWith` in the page context. Simply use a relative path (with a dot) or an absolute path and point to the directories where these wrappers are located – typical inside `./src/pages/...`.

```diff
createPage({
path: '/page-path',
component: systemPath.resolve('./src/templates/your-page-component.js'),
context: {
+ wrapPageWith: './src/pages/where-my/wrapper-is/'
},
})
```

## FAQ

### About performance and size

_Does it negatively impact the performance of my production build?_

No. It should not have an impact negatively compared to other solutions done differently. The technique relies on function calls only.

_Does it increase the size of my production build?_

Not really. Every page gets assigned one or more hashes, but this should not really make your production build significantly larger.

_Does it negatively interfere with Webpack tree shaking?_

Not really. All the production code from the wrappers will be a part of the `app-[hash].js` bundle, because this code gets shared with several pages.

If you use a large amount of JavaScript on only one page, you should put this code inside the page it belongs to. Webpack will then create a dedicated bundle for that page only.

### Known issues in development mode

- When adding or deleting a `wrap-pages.js` file, you may see a "file did not exists" error masse for a second. It should normally resolve. Try a hard refresh – if not. In worst case, you may have to restart the development server. This also can happen when adding or deleting pages.
- On the initial first save to refresh, you may have to "save" twice to see force fast refresh to actually show the changes.

## How it works

This description is simplified and the order of all the steps is not 100% as described. Also, during development, we only touch actually effected files.

1. Check all available pages if there is a file called `wrap-pages` (_wrapperName_).

2. We don't want it to be a page, so we delete it from the pages map.

3. But we keep this "page" it in our local cache, together with the directory "scope" it is located in.

4. We also create a reference file inside the Gatsby cache folder: `.cache/wpe-scopes.js`. It includes the imports of our wrapper file along with a _hash_.

5. Now we compare the pages directory with a matching wrapper "scope". If there is a match, we put a _hash_ in the page context.

6. The plugin uses Gatsby's API `wrapPageElement` and checks if a matching _hash_ is in the current page context.

### WP_CACHE_PATH

During the Gatsby compilation, Webpack sets `WP_CACHE_PATH` with the Define Plugin.

While `wrap-pages.js` is using this to locate the currently used cache file (`.cache/wpe-scopes.js`) which includes all imports pointing to your wrappers.

When testing `wrap-pages.js` with Jest, we have to provide the `WP_CACHE_PATH` by setting this in the jest.config:

```js
// jest.config.js
...
globals: {
WP_CACHE_PATH: require.resolve('./__mocks__/.cache/wpe-scopes.js'),
}
```

### Why use onCreateDevServer?

Now, things would have been very much easier, if this plugin not would support development mode, with support for resolving file changes on the disk on the fly.

The Gatsby `onCreateDevServer` is the last one during warm up, so we add our file change listeners inside this API.

We listen for `CREATE_PAGE` to update everything related, both when a page gets created or a new wrapper file gets created.

We also listen for `unlink` so we can update related pages when a wrapper gets deleted.

### Development

To test the add and remove cycle, run:

1. $ `unlink example-micro-frontends/micro-app-home/src/pages/home-layout.tsx`

2. $ `git checkout -- example-micro-frontends/micro-app-home/src/pages/home-layout.tsx`

3. And repeat. Eventually refresh the browser.