These integration guides are not official documentation and the Strapi Support Team will not provide assistance with them.
Why Use ProseMirror?
ProseMirror is a modern, highly customizable rich text editor framework that transforms how developers build content management systems. Unlike traditional best WYSIWYG editors, ProseMirror represents documents as structured, editable JSON objects based on customizable schemas.
What sets ProseMirror apart:
- Modular Architecture: Load only the components you need, keeping your editor lightweight and performant.
- Extensible Plugin System: Add custom functionality like tables, math notation, or image handling with ease, boosting your productivity with Strapi plugins.
- Collaborative Editing: Build real-time collaboration through plugins like prosemirror-collab.
- Schema Validation: Define exact document structures to prevent invalid content, aiding in content modeling with Strapi.
- Performance: Handle complex documents without sacrificing speed.
The official ProseMirror documentation offers comprehensive resources and examples to get you started.
Why Use Strapi?
Strapi is the leading open-source headless CMS offering features, like customizable APIs, role-based permissions, multilingual support, etc. It simplifies content management and integrates effortlessly with modern frontend frameworks.
Explore the Strapi documentation for more details.
Strapi 5 Highlights
The out-of-the-box Strapi features allow you to get up and running in no time: 1. Single types: Create one-off pages that have a unique content structure. 2. Draft and Publish: Reduce the risk of publishing errors and streamline collaboration. 3. 100% TypeScript Support: Enjoy type safety & easy maintainability 4. Customizable API: With Strapi, you can just hop in your code editor and edit the code to fit your API to your needs. 5. Integrations: Strapi supports integrations with Cloudinary, SendGrid, Algolia, and others. 6. Editor interface: The editor allows you to pull in dynamic blocks of content. 7. Authentication: Secure and authorize access to your API with JWT or providers. 8. RBAC: Help maximize operational efficiency, reduce dev team support work, and safeguard against unauthorized access or configuration modifications. 9. i18n: Manage content in multiple languages. Easily query the different locales through the API. 10. Plugins: Customize and extend Strapi using plugins.
Learn more about Strapi 5 feature.
See Strapi in action with an interactive demo
Setup Strapi 5 Headless CMS
We are going to start by setting up our Strapi 5 project with the following command:
🖐️ Note: make sure that you have created a new directory for your project.
You can find the full documentation for Strapi 5 here.
Install Strapi
npx create-strapi-app@latest serverYou will be asked to choose if you would like to use Strapi Cloud we will choose to skip for now.
 Strapi   v5.6.0 🚀 Let's create your new project
 
We can't find any auth credentials in your Strapi config.
Create a free account on Strapi Cloud and benefit from:
- ✦ Blazing-fast ✦ deployment for your projects
- ✦ Exclusive ✦ access to resources to make your project successful
- An ✦ Awesome ✦ community and full enjoyment of Strapi's ecosystem
Start your 14-day free trial now!
? Please log in or sign up. 
  Login/Sign up 
❯ Skip After that, you will be asked how you would like to set up your project. We will choose the following options:
? Do you want to use the default database (sqlite) ? Yes
? Start with an example structure & data? Yes <-- make sure you say yes 
? Start with Typescript? Yes
? Install dependencies with npm? Yes
? Initialize a git repository? YesOnce everything is set up and all the dependencies are installed, you can start your Strapi server with the following command:
cd server
npm run developYou will be greeted with the Admin Create Account screen.
Go ahead and create your first Strapi user. All of this is local so you can use whatever you want.
Once you have created your user, you will be redirected to the Strapi Dashboard screen.
Publish Article Entries
Since we created our app with the example data, you should be able to navigate to your Article collection and see the data that was created for us.
Now, let's make sure that all of the data is published. If not, you can select all items via the checkbox and then click the Publish button.
Enable API Access
Once all your articles are published, we will expose our Strapi API for the Articles Collection. This can be done in Settings -> Users & Permissions plugin -> Roles -> Public -> Article.
You should have find and findOne selected.  If not, go ahead and select them.
Test API
Now, if we make a GET request to http://localhost:1337/api/articles, we should see the following data for our articles.
🖐️ Note: The article covers (images) are not returned. This is because the REST API by default does not populate any relations, media fields, components, or dynamic zones.. Learn more about REST API: Population & Field Selection.
So, let's get the article covers by using the populate=* parameter: http://localhost:1337/api/articles?populate=*
Getting Started with ProseMirror
Integrating ProseMirror with Strapi requires proper environment setup and a structured approach. This creates the foundation for a powerful rich-text editing experience in your application.
Installing ProseMirror and Adjusting Settings
Grab these core ProseMirror packages:
npm install prosemirror-state prosemirror-view prosemirror-modelEach package has a specific role:
- prosemirror-state: Manages editor state
- prosemirror-view: Handles rendering and user interactions
- prosemirror-model: Defines document schemas and transformations
Set up your environment variables in an .env file:
1STRAPI_HOST=localhost
2STRAPI_PORT=1337
3NODE_ENV=developmentWhen coding, you might wonder whether to use TypeScript vs JavaScript for Strapi. Both have their merits, so choose the one that best fits your project needs.
Code Implementation: Building Your ProseMirror Editor Within Strapi
With everything installed, let's build a basic ProseMirror editor integrated with Strapi:
1import { EditorState } from "prosemirror-state";
2import { EditorView } from "prosemirror-view";
3import { Schema } from "prosemirror-model";
4import { schema } from "prosemirror-schema-basic";
5
6// Set up editor state with basic schema
7const state = EditorState.create({ schema });
8
9// Set up editor view in the DOM
10const view = new EditorView(document.querySelector("#editor"), { state });Connecting ProseMirror to Your Strapi Backend
Integrate ProseMirror with Strapi by fetching and saving content, utilizing the evolution of APIs:
1// Fetch content from Strapi
2const fetchContent = async () => {
3  const response = await fetch("http://localhost:1337/api/blog-posts/1");
4  const data = await response.json();
5  
6  // Convert Strapi JSON to ProseMirror document
7  const content = JSON.parse(data.attributes.content);
8  const newState = EditorState.create({
9    schema,
10    doc: schema.nodeFromJSON(content)
11  });
12  
13  view.updateState(newState);
14};
15
16// Save content to Strapi
17const saveContent = async () => {
18  const content = JSON.stringify(view.state.doc.toJSON());
19  
20  await fetch("http://localhost:1337/api/blog-posts/1", {
21    method: "PUT",
22    headers: { 
23      "Content-Type": "application/json",
24      "Authorization": "Bearer YOUR_API_TOKEN" 
25    },
26    body: JSON.stringify({ 
27      data: { content } 
28    }),
29  });
30};Whether you're using REST and GraphQL with Strapi, you have flexibility in how you connect your frontend and backend.
This basic implementation lays the groundwork for integrating ProseMirror with Strapi. As you grow, you can add plugins, customize schemas, and build more sophisticated data mapping between ProseMirror documents and Strapi content types. Whether you're building a simple blog or a complex platform, combining ProseMirror with Strapi provides the flexibility needed for diverse applications, including those requiring a headless CMS for media.
Strapi Open Office Hours
If you have any questions about Strapi 5 or just would like to stop by and say hi, you can join us at Strapi's Discord Open Office Hours Monday through Friday at 12:30 pm - 1:30 pm CST: Strapi Discord Open Office Hours
For more details, visit the Strapi documentation and ProseMirror documentation.