nuxt-strapi-blocks-renderer
Version:
Renderer for the strapi CMS blocks text content element.
437 lines (324 loc) • 11.2 kB
Markdown
<p align="center">
<img width="320" src="https://raw.githubusercontent.com/freb97/nuxt-strapi-blocks-renderer/refs/heads/main/.github/public/nuxt-strapi-blocks-renderer.png">
</p>
# Nuxt Strapi Blocks Renderer
[![Github Actions][github-actions-src]][github-actions-href]
[![NPM version][npm-version-src]][npm-version-href]
[![NPM downloads][npm-downloads-src]][npm-downloads-href]
[![NPM last update][npm-last-update-src]][npm-last-update-href]
[![License][license-src]][license-href]
Fully customizable module for Nuxt 3 & 4 to render the "Blocks" rich text editor element from Strapi CMS.
The implementation is based on Strapi's [Blocks React Renderer](https://github.com/strapi/blocks-react-renderer/).
- ✨ [Release notes](/CHANGELOG.md)
- 🏀 [Online stackblitz playground](https://stackblitz.com/github/freb97/nuxt-strapi-blocks-renderer?file=playground%2Fcustom%2Fpages%2Findex.vue)
## Installation
To install the Nuxt Strapi Blocks Renderer module, run the following command:
```bash
npx nuxt@latest module add nuxt-strapi-blocks-renderer
```
<details>
<summary>Or install it manually</summary>
1. Install the module:
```bash
npm install nuxt-strapi-blocks-renderer
```
2. Add the module to `nuxt.config.{ts|js}`:
```typescript
export default defineNuxtConfig({
modules: [ 'nuxt-strapi-blocks-renderer' ]
})
```
</details>
## Usage
To render text, use the `StrapiBlocksText` component:
```vue
<StrapiBlocksText :nodes="blockNodes" />
```
In this example, the `blockNodes` are taken from the JSON response which Strapi provides when using the Blocks rich
text editor element:
```vue
<script setup lang="ts">
import type { BlockNode } from '#strapi-blocks-renderer/types';
import type { Restaurant } from '~/types';
const route = useRoute();
const { findOne } = useStrapi();
// Fetch restaurants data from Strapi
const response = await findOne<Restaurant>('restaurants', route.params.id);
// Obtain blocks text nodes from description field
const blockNodes: BlockNode[] = response.data.attributes.description;
</script>
<template>
<!-- Render blocks text -->
<StrapiBlocksText :nodes="blockNodes" />
</template>
```
To use the `useStrapi` composable, install the [Strapi Nuxt module](https://strapi.nuxtjs.org/).
### Advanced Usage
In situations where your project requires specific styling or behavior for certain HTML tags such as `<a>`, `<p>`,
and others, you can override the default rendering components used by the Nuxt Strapi Blocks Renderer.
This flexibility allows you to tailor the rendering to align with your project's unique design and functional needs.
#### Global Component Registration
First, ensure that your components are globally registered in your Nuxt app.
This step is crucial for your custom components to be recognized and used by the renderer.
In your Nuxt configuration (`nuxt.config.{js|ts}`), add:
```ts
export default defineNuxtConfig({
components: {
dirs: [
{
path: '~/components/blocks',
pathPrefix: false,
global: true,
},
// also include ~/components to ensure other local components are resolved properly
'~/components'
],
},
})
```
> [!IMPORTANT]
> It's important to include `~/components` in the dirs array.
> Omitting it may cause locally scoped components (outside of `~/components/blocks`) not to resolve correctly.
#### Customizing the Paragraph Tag
To customize the rendering of the paragraph (`<p>`) tag, you need to create a corresponding Vue component.
The name of the component follows a predefined pattern: `'StrapiBlocksText' + [NodeName] + 'Node.vue'`.
To override the default paragraph tag, we create a file called `StrapiBlocksTextParagraphNode.vue`.
```html
<!-- components/blocks/StrapiBlocksTextParagraphNode.vue -->
<template>
<p class="my-custom-class-for-p">
<slot />
</p>
</template>
```
This component assigns a custom class `my-custom-class-for-p` to the paragraph tag, which can be styled as needed.
The prefix for the custom components can be adjusted in your `nuxt.config.{js|ts}`:
```ts
export default defineNuxtConfig({
strapiBlocksRenderer: {
prefix: 'MyCustomPrefix',
blocksPrefix: 'MyCustomBlocksPrefix',
},
})
```
With this configuration, the `StrapiBlocksText` component becomes `MyCustomPrefixStrapiBlocksText` and the custom
paragraph node component would be named `MyCustomBlocksPrefixParagraphNode`.
#### Other Custom Tags
You can apply similar customizations to all other HTML tags used by the renderer:
<details open>
<summary>Headings</summary>
Custom heading tags (`<h1>`, `<h2>`, `<h3>`, etc.):
```html
<!-- components/blocks/StrapiBlocksTextHeading1Node.vue -->
<template>
<h1 class="my-custom-class-for-h1">
<slot />
</h1>
</template>
<!-- components/blocks/StrapiBlocksTextHeading2Node.vue -->
<template>
<h2 class="my-custom-class-for-h2">
<slot />
</h2>
</template>
```
This pattern also extends to the `h3`, `h4`, `h5` and `h6` tags.
</details>
<details>
<summary>Lists</summary>
Custom list tags (`<ol>`, `<ul>` and `<li>`):
```html
<!-- components/blocks/StrapiBlocksTextOrderedListNode.vue -->
<template>
<ol class="my-custom-class-for-ol">
<slot />
</ol>
</template>
<!-- components/blocks/StrapiBlocksTextUnorderedListNode.vue -->
<template>
<ul class="my-custom-class-for-ul">
<slot />
</ul>
</template>
<!-- components/blocks/StrapiBlocksTextListItemInlineNode.vue -->
<template>
<li class="my-custom-class-for-li">
<slot />
</li>
</template>
```
</details>
<details>
<summary>Blockquotes and Codes</summary>
Custom blockquote and code tags (`<blockquote>`, `<pre>`):
```html
<!-- components/blocks/StrapiBlocksTextQuoteNode.vue -->
<template>
<blockquote class="my-custom-class-for-blockquote">
<slot />
</blockquote>
</template>
<!-- components/blocks/StrapiBlocksTextCodeNode.vue -->
<script setup lang="ts">
const props = defineProps<{
language?: string;
}>();
</script>
<template>
<pre class="my-custom-class-for-pre"><slot /></pre>
</template>
```
</details>
<details>
<summary>Inline text nodes</summary>
Custom inline text nodes (`<strong>`, `<em>`, `<u>`, `<del>`, `<code>`):
```html
<!-- components/blocks/StrapiBlocksTextBoldInlineNode.vue -->
<template>
<strong class="my-custom-class-for-strong">
<slot />
</strong>
</template>
<!-- components/blocks/StrapiBlocksTextItalicInlineNode -->
<template>
<em class="my-custom-class-for-em">
<slot />
</em>
</template>
<!-- components/blocks/StrapiBlocksTextUnderlineInlineNode -->
<template>
<u class="my-custom-class-for-u">
<slot />
</u>
</template>
<!-- components/blocks/StrapiBlocksTextStrikethroughInlineNode -->
<template>
<del class="my-custom-class-for-del">
<slot />
</del>
</template>
<!-- components/blocks/StrapiBlocksTextCodeInlineNode.vue -->
<template>
<code class="my-custom-class-for-code">
<slot />
</code>
</template>
```
</details>
<details>
<summary>Links</summary>
Custom link tag (`<a>`):
```html
<!-- components/blocks/StrapiBlocksTextLinkInlineNode.vue -->
<script setup lang="ts">
const props = defineProps<{
url: string;
}>();
</script>
<template>
<a :href="props.url" class="my-custom-class-for-a">
<slot />
</a>
</template>
```
When rendering a link tag, the url gets passed as the `url` component property.
</details>
<details>
<summary>Images</summary>
Custom image tag (`<img>`):
```html
<!-- components/blocks/StrapiBlocksTextImageNode.vue -->
<script setup lang="ts">
const props = defineProps<{
image: any;
}>();
</script>
<template>
<img
class="my-custom-class-for-img"
:src="props.image.url"
:alt="props.image.alternativeText"
:width="props.image.width"
:height="props.image.height"
>
</template>
```
When rendering an image tag, the image object gets passed as the `image` component property.
You can also use different image components here, i.e. `NuxtImg` or others.
</details>
## Development
### Dependencies
To install the dependencies, run the `install` command:
```bash
npm install
```
The project requires Node.js and NPM to run.
You can either install these manually on your system or if you have the nix package manager installed, use the
provided nix-shell with the following command:
```bash
nix-shell
```
This will automatically install the needed software and start up a shell.
### Type stubs
To generate the type stubs for the nuxt module, run the `dev:prepare` command:
```bash
npm run dev:prepare
```
### Development server
To start the development server with the provided text components, run the `dev` command:
```bash
npm run dev
```
This will boot up the playground with the default text components.
To start the development server using custom text components, overriding the provided components,
use the `dev:custom` command:
```bash
npm run dev:custom
```
### Quality
#### Linter
To run ESLint, use the following command:
```bash
npm run lint:es
```
#### Type checks
To run the TypeScript type checks, use the following command:
```bash
npm run lint:types
```
#### Unit Tests
To run the Vitest unit tests, run the following command:
```bash
npm run test
````
### Build
To build the module, first install all dependencies and [generate the type stubs](#type-stubs).
Then run the build script:
```bash
npm run build
```
The module files will be output to the `dist` folder.
### Release
To release a new version of the strapi blocks renderer nuxt module, take the following steps:
1. Increment version number in the `package.json` file
2. Add changelog entry for the new version number
3. Run linters and unit tests
4. Build the nuxt module
```bash
npm run build
```
5. Log in to NPM using your access token
6. Run the `release` command
```bash
npm run release
```
[github-actions-src]: https://github.com/freb97/nuxt-strapi-blocks-renderer/actions/workflows/ci.yml/badge.svg
[github-actions-href]: https://github.com/freb97/nuxt-strapi-blocks-renderer/actions
[npm-version-src]: https://img.shields.io/npm/v/nuxt-strapi-blocks-renderer/latest.svg?style=flat&colorA=18181B&colorB=31C553
[npm-version-href]: https://npmjs.com/package/nuxt-strapi-blocks-renderer
[npm-downloads-src]: https://img.shields.io/npm/dm/nuxt-strapi-blocks-renderer.svg?style=flat&colorA=18181B&colorB=31C553
[npm-downloads-href]: https://npmjs.com/package/nuxt-strapi-blocks-renderer
[npm-last-update-src]: https://img.shields.io/npm/last-update/nuxt-strapi-blocks-renderer.svg?style=flat&colorA=18181B&colorB=31C553
[npm-last-update-href]: https://npmjs.com/package/nuxt-strapi-blocks-renderer
[license-src]: https://img.shields.io/github/license/freb97/nuxt-strapi-blocks-renderer.svg?style=flat&colorA=18181B&colorB=31C553
[license-href]: https://github.com/freb97/nuxt-strapi-blocks-renderer/blob/main/LICENSE