Movatterモバイル変換


[0]ホーム

URL:


Skip to content

Navigation Menu

Sign in
Appearance settings

Search code, repositories, users, issues, pull requests...

Provide feedback

We read every piece of feedback, and take your input very seriously.

Saved searches

Use saved searches to filter your results more quickly

Sign up
Appearance settings

A plugin for Strapi Headless CMS that provides navigation / menu builder feature with their possibility to control the audience and different output structure renderers like (flat, tree and RFR - ready for handling by Redux First Router)

License

NotificationsYou must be signed in to change notification settings

VirtusLab-Open-Source/strapi-plugin-navigation

Repository files navigation

Logo - Strapi Navigation plugin

Strapi - Navigation plugin

Create consumable navigation with a simple and straightforward visual builder

GitHub package.json versionMonthly download on NPMCircleCIcodecov.io

UI preview

Strapi Navigation Plugin provides a website navigation / menu builder feature forStrapi Headless CMS admin panel. Navigation has the possibility to control the audience and can be consumed by the website with different output structure renderers:

  • Flat
  • Tree (nested)
  • RFR (ready for handling by Redux First Router)

Table of Contents

  1. 💎 Versions
  2. ✨ Features
  3. ⏳ Installation
  4. 🖐 Requirements
  5. 🔧 Basic Configuration
  6. 🔧 GraphQL Configuration
  7. 👤 RBAC
  8. 🔐 Authorization strategy
  9. 🕸️ Public API specification
  10. 🔌 Extensions
  11. 🌿 Model lifecycle hooks
  12. 🧹 REST Cache
  13. 🧩 Examples
  14. 💬 FAQ
  15. 🤝 Contributing
  16. 👨‍💻 Community support

💎 Versions

✨ Features

  • Navigation Public API: Simple and ready for use API endpoint for consuming the navigation structure you've created
  • Visual builder: Elegant and easy to use visual builder
  • Any Content Type relation: Navigation can by linked to any of your Content Types by default. Simply, you're controlling it and also limiting available content types by configuration props
  • Different types of navigation items: Create navigation with items linked to internal types, to external links or wrapper elements to keep structure clean
  • Multiple navigations: Create as many Navigation containers as you want, setup them and use in the consumer application
  • Light / Dark mode compatible: By design we're supporting Strapi ☀️ Light / 🌙 Dark modes
  • Webhooks integration: Changes to navigation will trigger 'entry.update' or 'entry.create' webhook events.
  • Customizable: Possibility to customize the options like: available Content Types, Maximum level for "attach to menu", Additional fields (audience)

⏳ Installation

Via Strapi Marketplace

As a ✅verified plugin by Strapi team we're available on theStrapi Marketplace as well asIn-App Marketplace where you can follow the installation instructions.

Strapi In-App Marketplace

Via command line

It's recommended to useyarn to install this plugin within your Strapi project.You can install yarn with these docs.

yarn add strapi-plugin-navigation@latest

After successful installation you've to re-build your Strapi instance. To archive that simply use:

yarn buildyarn develop

TheUI Navigation plugin should appear in thePlugins section of Strapi sidebar after you run app again.

You can manage your multiple navigation containers by going to theNavigation manage view by clicking "Manage" button.

Navigation Manager View

As a next step you must configure your the plugin by the way you want to. SeeConfiguration section.

All done. Enjoy 🎉

🖐 Requirements

Complete installation requirements are exact same as for Strapi itself and can be found in the documentation underInstallation Requirements.

Supported Strapi versions:

  • Strapi v5.10.3 (recently tested)
  • Strapi v5.x

This plugin is designed forStrapi v5 and is not working with v4.x. To get version forStrapi v4 install versionv4.x.

We recommend always using the latest version of Strapi to start your new projects.

🔧 Configuration

To start your journey withNavigation plugin you must first setup it using the dedicated Settings page or for any version, put your configuration inconfig/plugins.{js|ts}. Anyway we're recommending the click-through option where your configuration is going to be properly validated.

Settings page

On the dedicated page, you will be able to set up all crucial properties which drive the plugin and customize each individual collection for whichNavigation plugin should be enabled.

Plugin configuration

NoteThe default configuration for your plugin is fetched fromconfig/plugins.js or, if the file is not there, directly from the plugin itself. If you would like to customize the default state to which you might revert, please follow the next section.

File

Config for this plugin is stored as a part of theconfig/plugins.{js|ts} orconfig/<env>/plugins.{js|ts} file. You can use the following snippet to make sure that the config structure is correct. If you've got already configurations for other plugins stores by this way, you can use thenavigation along with them.

module.exports=({ env})=>({// ...navigation:{enabled:true,config:{additionalFields:['audience',{name:'my_custom_field',type:'boolean',label:'My custom field'}],contentTypes:['api::page.page'],contentTypesNameFields:{'api::page.page':['title']},pathDefaultFields:{'api::page.page':['slug']},allowedLevels:2,gql:{...},}}});

Properties

  • additionalFields - Additional fields for navigation items. More here
  • allowedLevels - Maximum level for which you're able to mark item as "Menu attached"
  • contentTypes - UIDs of related content types
  • contentTypesNameFields - Definition of content type title fields like'api::<collection name>.<content type name>': ['field_name_1', 'field_name_2'], if not set titles are pulled from fields like['title', 'subject', 'name'].TIP - Proper content type uid you can find in the URL of Content Manager where you're managing relevant entities like:admin/content-manager/collectionType/< THE UID HERE >?page=1&pageSize=10&sort=Title:ASC&plugins[i18n][locale]=en
  • pathDefaultFields - The attribute to copy the default path from per content type. Syntax:'api::<collection name>.<content type name>': ['url_slug', 'path']
  • gql - If you're using GraphQL that's the right place to put all necessary settings. More here
  • cascadeMenuAttached - If you don't want "Menu attached" to cascade on child items set this valueDisabled.

Properties

Additional Fields

It is advised to configure additional fields through the plugin's Settings Page. There you can find the table of custom fields and toggle input for the audience field. When enabled, the audience field can be customized through the content manager. Custom fields can be added, edited, toggled, and removed with the use of the table provided on the Settings Page. When removing custom fields be advised that their values in navigation items will be lost. Disabling the custom fields will not affect the data and can be done with no consequence of loosing information.

Creating configuration for additional fields with theconfig.(js|ts) file should be done with caution. Config object contains theadditionalFields property of typeArray<CustomField | 'audience'>, where CustomField is of type

typeCustomField=|{type:'string'|'boolean'|'media';name:string;label:string;description?:string;placeholder?:string;required?:boolean;enabled?:boolean;}|{type:'select';name:string;label:string;description?:string;placeholder?:string;multi:boolean;options:string[];required?:boolean;enabled?:boolean;};

When creating custom fields be advised that thename property has to be unique. When editing a custom field it is advised not to edit itsname andtype properties. After config has been restored the custom fields that are not present inconfig.js file will be deleted and their values in navigation items will be lost.

🔧 GQL Configuration

Using navigation with GraphQL requires both plugins to be installed and working. You can find installation guide for GraphQL pluginhere. To properly configure GQL to work with navigation you should providegql prop. This should contain union types that will be used to define GQL response format for your data while fetching:

Important!If you're usingconfig/plugins.js to configure your plugins , please putnavigation property beforegraphql. Otherwise types are not going to be properly added to GraphQL Schema. That's because of dynamic types which base on plugin configuration which are added onbootstrap stage, notregister. This is not valid if you're usinggraphql plugin without any custom configuration, so most of cases in real.

master:Intitems: [NavigationItem]related:NavigationRelated

This prop should look as follows:

gql:{navigationItemRelated:['<your GQL related content types>'],},

for example:

gql:{navigationItemRelated:['Page','UploadFile'],},

wherePage andUploadFile are your type names for theContent Types you're referring by navigation items relations.

👤 RBAC

Plugin provides granular permissions based onStrapi RBAC functionality within the editorial interface &Admin API. Those settings are editable via theSetings ->Administration Panel ->Roles.

For any role different thanSuper Admin, to access theNavigation panel you must set following permissions:

Mandatory permissions

  • Plugins ->Navigation ->Read - gives you the access toNavigation Panel

Other permissions

  • Plugins ->Navigation ->Update - with this permission user is able to change Navigation structure
  • Plugins ->Navigation ->Settings - special permission for users that should be able to change plugin settings

🔐 Authorization strategy

Is applied forPublic API both for REST and GraphQL. You can manage is by two different ways. Those settings are editable via theSetings ->Users & Permissions Plugin ->Roles.

User based

  • Public - as per description it's default role for any not authenticated user. By enablingPublic API of the plugin here you're making itfully public, withoutany permissions check.
  • Authenticated - as per description this is default role for Strapi Users. If you enablePublic API here, for any call made you must use the User authentication token asBearer <token>.

Token based

  • Full Access - gives full access to every Strapi Content API including our plugin endpoints as well.
  • Custom - granural access management to every Strapi Content API endpoints as well as pluginPublic API -(recomended approach)

Note: Token usage &amp Read-Only tokensIf you're aiming to use token based approach, for every call you must provide proper token in headers asBearer <token>.

Important: As the Read-Only tokens are dedicated to support justfind andfindAll endpoints from Strapi Content API, they are not covering access to pluginPublic APIrender andrenderChild endpoints. We recommend to use theCustom token type for fully granural and secured approach instead ofFull Access ones.

Reference:Strapi - API Tokens

Base Navigation Item model

Flat

{"id":1,"documentId":"njx99iv4p4txuqp307ye8625","title":"News","type":"INTERNAL","path":"news","externalPath":null,"uiRouterKey":"News","menuAttached":false,"parent":8,// Parent Navigation Item 'id', null in case of root level"master":1,// Navigation 'id'"createdAt":"2020-09-29T13:29:19.086Z","updatedAt":"2020-09-29T13:29:19.128Z","related": {/*<Content Type model >*/  },"audience": []}

Tree

{"title":"News","menuAttached":true,"path":"/news","type":"INTERNAL","uiRouterKey":"news","slug":"benefits","external":false,"related": {// <Content Type model >  },"items": [    {"title":"External url","menuAttached":true,"path":"http://example.com","type":"EXTERNAL","uiRouterKey":"generic","external":true    }//  < Tree Navigation Item models >  ]}

RFR

{"id":"News","title":"News","related": {"contentType":"page","collectionName":"pages","id":1  },"path":"/news","slug":"news","parent":null,// Parent Navigation Item 'id', null in case of root level"menuAttached":true}

🕸️ Public API specification

Plugin supports bothREST API andGraphQL API exposed by Strapi.

Query Params

  • navigationIdOrSlug - ID or slug for which your navigation structure is generated like for REST API:

    https://localhost:1337/api/navigation/render/njx99iv4p4txuqp307ye8625 >https://localhost:1337/api/navigation/render/main-menu

  • type - Enum value representing structure type of returned navigation:

    https://localhost:1337/api/navigation/render/njx99iv4p4txuqp307ye8625?type=FLAT

  • menu (menuOnly for GQL) - Boolean value for querying only navigation items that are attached to menu should be rendered eg.

    https://localhost:1337/api/navigation/render/njx99iv4p4txuqp307ye8625?menu=true

  • path - String value for querying navigation items by its path:

    https://localhost:1337/api/navigation/render/njx99iv4p4txuqp307ye8625?path=/home/about-us

REST API

GET <host>/api/navigation/?locale=<locale>&orderBy=<orderBy>&orderDirection=<orderDirection>

NOTE: All params are optional

Example URL:https://localhost:1337/api/navigation?locale=en

Example response body

[  {"id":383,"documentId":"njx99iv4p4txuqp307ye8625","name":"Floor","slug":"floor-pl","visible":true,"createdAt":"2023-09-29T12:45:54.399Z","updatedAt":"2023-09-29T13:44:08.702Z","locale":"pl"  },  {"id":384,"documentId":"njx99iv4p4txuqp307ye8625","name":"Floor","slug":"floor-fr","visible":true,"createdAt":"2023-09-29T12:45:54.399Z","updatedAt":"2023-09-29T13:44:08.725Z","locale":"fr"  },  {"id":382,"documentId":"njx99iv4p4txuqp307ye8625","name":"Floor","slug":"floor","visible":true,"createdAt":"2023-09-29T12:45:54.173Z","updatedAt":"2023-09-29T13:44:08.747Z","locale":"en"  },  {"id":374,"documentId":"njx99iv4p4txuqp307ye8625","name":"Main navigation","slug":"main-navigation-pl","visible":true,"createdAt":"2023-09-29T12:22:30.373Z","updatedAt":"2023-09-29T13:44:08.631Z","locale":"pl"  },  {"id":375,"documentId":"njx99iv4p4txuqp307ye8625","name":"Main navigation","slug":"main-navigation-fr","visible":true,"createdAt":"2023-09-29T12:22:30.373Z","updatedAt":"2023-09-29T13:44:08.658Z","locale":"fr"  },  {"id":373,"documentId":"njx99iv4p4txuqp307ye8625","name":"Main navigation","slug":"main-navigation","visible":true,"createdAt":"2023-09-29T12:22:30.356Z","updatedAt":"2023-09-29T13:44:08.680Z","locale":"en"  }]

GET <host>/api/navigation/render/<navigationIdOrSlug>?type=<type>

Return a rendered navigation structure depends on passed type (TREE,RFR or nothing to render asFLAT).

Example URL:https://localhost:1337/api/navigation/render/njx99iv4p4txuqp307ye8625

Example response body

[  {"id":1,"documentId":"njx99iv4p4txuqp307ye8625","title":"News","type":"INTERNAL","path":"news","externalPath":null,"uiRouterKey":"News","menuAttached":false,"parent":null,"master":1,"created_at":"2020-09-29T13:29:19.086Z","updated_at":"2020-09-29T13:29:19.128Z","related": {"__contentType":"Page","id":1,"documentId":"njx99iv4p4txuqp307ye8625","title":"News"// ...    }  }// ...]

Example URL:https://localhost:1337/api/navigation/render/njx99iv4p4txuqp307ye8625?type=TREE

Example response body

[  {"title":"News","menuAttached":true,"path":"/news","type":"INTERNAL","uiRouterKey":"news","slug":"benefits","external":false,"related": {"__contentType":"Page","id":1,"title":"News"// ...    },"items": [      {"title":"External url","menuAttached":true,"path":"http://example.com","type":"EXTERNAL","uiRouterKey":"generic","external":true      }// ...    ]  }// ...]

Example URL:https://localhost:1337/api/navigation/render/njx99iv4p4txuqp307ye8625?type=RFR

Example response body

{"pages": {"News": {"id":"News","title":"News","related": {"contentType":"page","collectionName":"pages","id":1      },"path":"/news","slug":"news","parent":null,"menuAttached":true    },"Community": {"id":"Community","title":"Community","related": {"contentType":"page","collectionName":"pages","id":2      },"path":"/community","slug":"community","parent":null,"menuAttached":true    },"Highlights": {"id":"Highlights","title":"Highlights","related": {"contentType":"page","collectionName":"pages","id":3      },"path":"/community/highlights","slug":"community-highlights","parent":"Community","menuAttached":false    }// ...  },"nav": {"root": [      {"label":"News","type":"internal","page":"News"      },      {"label":"Community","type":"internal","page":"Community"      },      {"label":"External url","type":"external","url":"http://example.com"      }// ...    ],"Community": [      {"label":"Highlights","type":"internal","page":"Highlights"      }// ...    ]// ...  }}

GraphQL API

Same asREST API returns a rendered navigation structure depends on passed type (TREE,RFR or nothing to render asFLAT).

Example request

query {renderNavigation(navigationIdOrSlug:"main-navigation",type: TREE,menuOnly:false) {idtitlepathrelated {id__typename...onPage {Title      }...onWithFlowType {Name      }    }items {idtitlepathrelated {id__typename...onPage {Title        }...onWithFlowType {Name        }      }    }  }}

Example response

{"data": {"renderNavigation": [      {"id":8,"title":"Test page","path":"/test-path","related": {"id":3,"__typename":"WithFlowType","Name":"Test"        },"items": [          {"id":11,"title":"Nested","path":"/test-path/nested-one","related": {"id":1,"__typename":"Page","Title":"Eg. Page title"            }          }        ]      },      {"id":10,"title":"Another page","path":"/another","related": {"__typename":"Page","Title":"Eg. Page title"        },"items": []      }    ]  }}

🔌 Extensions

Server-side schema

On bootstrap you can customise Zod validators. Not all schemas are available at the moment, consultCommonService to see all available updaters.

Example:

constnavigationCommonService=strapi.plugin('navigation').service('common');navigationCommonService.updateUpdateNavigationSchema((schema:ZodObject)=>{returnschema.refine((data)=>{if(!data.visible){returnfalse;}returntrue;},{message:"Hidden navigation updated."});});

Slug generation

Slug generation is available as a controller and service. If you have custom requirements outside of what this plugin provides you can add your own logic withplugins extensions.

For example:

// path: /admin/src/index.jsmodule.exports={// ...bootstrap({ strapi}){constnavigationCommonService=strapi.plugin('navigation').service('common');constoriginalGetSlug=navigationCommonService.getSlug;constpreprocess=(q)=>{returnq+'suffix';};navigationCommonService.getSlug=(query)=>{returnoriginalGetSlug(preprocess(query));};},};

Model lifecycle hooks

Navigation plugin allows to register lifecycle hooks forNavigation andNavigationItem content types.

You can read more about lifecycle hookshere. (You can set a listener for all of the hooks).

Lifecycle hooks can be register either inregister() orbootstrap() methods of your server. You can register more than one listener for a specified lifecycle hook. For example: you want to do three things on navigation item creation and do not want to handle all of these actions in one big function. You can split logic in as many listeners as you want.

Listeners can by sync andasync.

Be aware that lifecycle hooks registered inregister() may be fired by plugin's bootstrapping. If you want listen to events triggered after server's startup usebootstrap().

Example:

constnavigationCommonService=strapi.plugin('navigation').service('common');navigationCommonService.registerLifecycleHook({callback:async({ action, result})=>{constsaveResult=awaitlogIntoSystem(action,result);console.log(saveResult);},contentTypeName:'navigation',hookName:'afterCreate',});navigationCommonService.registerLifecycleHook({callback:async({ action, result})=>{constsaveResult=awaitlogIntoSystem(action,result);console.log(saveResult);},contentTypeName:'navigation-item',hookName:'afterCreate',});

🧹 REST Cache

Note: Yet using the4.x compatible version of the plugin. Integration migration expected once the maintenance team release their5.x compatible version.

If your strapi server usesREST Cache plugin this plugin can take integrate with it. All you need to do is to enable it in configuration of Navigation plugin. After integration is enabled all client calls will be wrapped with caching middleware.

In admin panel new controls will be available. Cache clearing is done manually or after cache will timeout(rest-cache plugin's settings are used).

Navigation edit screen will have"Clear cache" button.

Navigation management modal items will also have icon button for clearing the cache.

🧩 Examples

Live example of plugin usage can be found in theVirtusLab Strapi Examples repository.

💬 FAQ

GraphQL tricks

Q: I would like to use GraphQL schemas but I'm not gettingrenderNavigation query or even proper types as Navigation, NavigationItem etc. What should I do?

A: There is a one trick you might try. Strapi by default is ordering plugins by the way which takesstrapi-plugin-graphql to initialize earlier than other plugins so types might not be injected. If you don't have it yet, please createconfig/plugins.{js|ts} file and put there following lines (putgraphql at the end):

module.exports={navigation:{enabled:true},graphql:{enabled:true},};

If you already got it, make sure thatnavigation plugin is inserted beforegraphql. That should do the job.

🤝 Contributing to the plugin

How to start?

Feel free to fork and make a Pull Request to this plugin project. All the input is warmly welcome!

  1. Clone repository

    git clone git@github.com:VirtusLab-Open-Source/strapi-plugin-navigation.git
  2. Runinstall &watch:link command

    // Install all dependenciesyarninstall// Watch for file changes using `plugin-sdk` and follow the instructions provided by this official Strapi developer toolyarn watch:link
  3. Within the Strapi project, modifyconfig/plugins.{js|ts} forimgix

//...'navigation':{enabled:true,//...}//...
  1. Run your Strapi instance

:octocat: Core team

@CodeVoyager@cyp3rius

🌟 Strapi Community

@jorrit

👨‍💻 Community support

For general help using Strapi, please refer tothe official Strapi documentation. For additional help, you can use one of these channels to ask a question:

  • Discord We're present on official Strapi Discord workspace. Find us by[VirtusLab] prefix and DM.
  • Slack - VirtusLab Open Source We're present on a public channel #strapi-molecules
  • GitHub (Bug reports, Contributions, Questions and Discussions)
  • E-mail - we will respond back as soon as possible

📝 License

MIT License Copyright (c)VirtusLab Sp. z o.o. &Strapi Solutions.

About

A plugin for Strapi Headless CMS that provides navigation / menu builder feature with their possibility to control the audience and different output structure renderers like (flat, tree and RFR - ready for handling by Redux First Router)

Topics

Resources

License

Stars

Watchers

Forks


[8]ページ先頭

©2009-2025 Movatter.jp