UNPKG

nuxt-strapi-blocks-renderer

Version:

Renderer for the strapi CMS blocks text content element.

437 lines (324 loc) 11.2 kB
<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