-
Notifications
You must be signed in to change notification settings - Fork 1.8k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat: Editor Core Packaging and Restructuring (#2358)
* initialized tiptap component with common tailwind config * added common tailwind config to web * abstracted upload and delete functions * removed tiptap pro extension * fixed types * removed old tailwind config and fixed plane package imports * exported tiptap editor with and without ref * updated package name to @plane/editor * finally fixed import errors * added turbo dependency for tiptap * reverted back types and fixed tailwind * migrated all components to use the common package * removed old tiptap dependency * improved dev experience to build the tiptap package before starting dev server * resolved lock life and missing deps * fixed dependency issue with react type resolution * chore: updated pulls build CI for using turbo builds * comment editor basic version added * new structure of editor components * refactored editor to not require workspace slug * added seperation of extensions and props * refactoring to LiteTextEditor and RichTextEditor * fixed global css issue with highlight js * refactoring tiptap to core/lite/rich text editor * read only editor support added * replaced all read-only instances * trimming html at start and end of content added * onSubmit on enterkey captured * removed absolute imports from editor/core package * removed absolute imports from lite-text-editor * removed absolute imports from rich-text-editor * fixed dependencies in editor package * fixed tailwind config for editor * Enter key behaviour added for Comments * fixed modal form issue * added comment editor with fixed menu * added support for range commands * modified turbo config for build pipeline of space and web projects * fixed shift enter behavior for lists * removed extra margin from access specifiers * removed tiptap instance from web * fixed bugs returning empty editor boxes * fixed toggle Underline behvaiour * updated bubble menu to use core package's utilities * added editor/core readme and fixed imports * fixed ts issues with link plugin * added usage of common dependance for slash commands * completed core package's documentation * fixed tsconfig by removing path aliases * Completed readme for rich-text-editor * Added rich text editor documentation * changed readme title of core package --------- Co-authored-by: Henit Chobisa <[email protected]>
- Loading branch information
1 parent
930a20e
commit 0a8b99a
Showing
131 changed files
with
3,753 additions
and
2,949 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -74,3 +74,7 @@ pnpm-lock.yaml | |
pnpm-workspace.yaml | ||
|
||
.npmrc | ||
|
||
|
||
## packages | ||
dist |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -6,7 +6,11 @@ | |
"workspaces": [ | ||
"web", | ||
"space", | ||
"packages/*" | ||
"packages/editor/*", | ||
"packages/eslint-config-custom", | ||
"packages/tailwind-config-custom", | ||
"packages/tsconfig", | ||
"packages/ui" | ||
], | ||
"scripts": { | ||
"build": "turbo run build", | ||
|
@@ -25,5 +29,8 @@ | |
"tailwindcss": "^3.3.3", | ||
"turbo": "latest" | ||
}, | ||
"resolutions": { | ||
"@types/react": "18.2.0" | ||
}, | ||
"packageManager": "[email protected]" | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,112 @@ | ||
# @plane/editor-core | ||
|
||
## Description | ||
|
||
The `@plane/editor-core` package serves as the foundation for our editor system. It provides the base functionality for our other editor packages, but it will not be used directly in any of the projects but only for extending other editors. | ||
|
||
## Utilities | ||
|
||
We provide a wide range of utilities for extending the core itself. | ||
|
||
1. Merging classes and custom styling | ||
2. Adding new extensions | ||
3. Adding custom props | ||
4. Base menu items, and their commands | ||
|
||
This allows for extensive customization and flexibility in the Editors created using our `editor-core` package. | ||
|
||
### Here's a detailed overview of what's exported | ||
|
||
1. useEditor - A hook that you can use to extend the Plane editor. | ||
|
||
| Prop | Type | Description | | ||
| --- | --- | --- | | ||
| `extensions` | `Extension[]` | An array of custom extensions you want to add into the editor to extend it's core features | | ||
| `editorProps` | `EditorProps` | Extend the editor props by passing in a custom props object | | ||
| `uploadFile` | `(file: File) => Promise<string>` | A function that handles file upload. It takes a file as input and handles the process of uploading that file. | | ||
| `deleteFile` | `(assetUrlWithWorkspaceId: string) => Promise<any>` | A function that handles deleting an image. It takes the asset url from your bucket and handles the process of deleting that image. | | ||
| `value` | `html string` | The initial content of the editor. | | ||
| `debouncedUpdatesEnabled` | `boolean` | If set to true, the `onChange` event handler is debounced, meaning it will only be invoked after the specified delay (default 1500ms) once the user has stopped typing. | | ||
| `onChange` | `(json: any, html: string) => void` | This function is invoked whenever the content of the editor changes. It is passed the new content in both JSON and HTML formats. | | ||
| `setIsSubmitting` | `(isSubmitting: "submitting" \| "submitted" \| "saved") => void` | This function is called to update the submission status. | | ||
| `setShouldShowAlert` | `(showAlert: boolean) => void` | This function is used to show or hide an alert in case of content not being "saved". | | ||
| `forwardedRef` | `any` | Pass this in whenever you want to control the editor's state from an external component | | ||
|
||
2. useReadOnlyEditor - A hook that can be used to extend a Read Only instance of the core editor. | ||
|
||
| Prop | Type | Description | | ||
| --- | --- | --- | | ||
| `value` | `string` | The initial content of the editor. | | ||
| `forwardedRef` | `any` | Pass this in whenever you want to control the editor's state from an external component | | ||
| `extensions` | `Extension[]` | An array of custom extensions you want to add into the editor to extend it's core features | | ||
| `editorProps` | `EditorProps` | Extend the editor props by passing in a custom props object | | ||
|
||
3. Items and Commands - H1, H2, H3, task list, quote, code block, etc's methods. | ||
|
||
4. UI Wrappers | ||
|
||
- `EditorContainer` - Wrap your Editor Container with this to apply base classes and styles. | ||
- `EditorContentWrapper` - Use this to get Editor's Content and base menus. | ||
|
||
5. Extending with Custom Styles | ||
|
||
```ts | ||
const customEditorClassNames = getEditorClassNames({ noBorder, borderOnFocus, customClassName }); | ||
``` | ||
|
||
## Core features | ||
|
||
- **Content Trimming**: The Editor’s content is now automatically trimmed of empty line breaks from the start and end before submitting it to the backend. This ensures cleaner, more consistent data. | ||
- **Value Cleaning**: The Editor’s value is cleaned at the editor core level, eliminating the need for additional validation before sending from our app. This results in cleaner code and less potential for errors. | ||
- **Turbo Pipeline**: Added a turbo pipeline for both dev and build tasks for projects depending on the editor package. | ||
|
||
```json | ||
"web#develop": { | ||
"cache": false, | ||
"persistent": true, | ||
"dependsOn": [ | ||
"@plane/lite-text-editor#build", | ||
"@plane/rich-text-editor#build" | ||
] | ||
}, | ||
"space#develop": { | ||
"cache": false, | ||
"persistent": true, | ||
"dependsOn": [ | ||
"@plane/lite-text-editor#build", | ||
"@plane/rich-text-editor#build" | ||
] | ||
}, | ||
"web#build": { | ||
"cache": true, | ||
"dependsOn": [ | ||
"@plane/lite-text-editor#build", | ||
"@plane/rich-text-editor#build" | ||
] | ||
}, | ||
"space#build": { | ||
"cache": true, | ||
"dependsOn": [ | ||
"@plane/lite-text-editor#build", | ||
"@plane/rich-text-editor#build" | ||
] | ||
}, | ||
|
||
``` | ||
|
||
## Base extensions included | ||
|
||
- BulletList | ||
- OrderedList | ||
- Blockquote | ||
- Code | ||
- Gapcursor | ||
- Link | ||
- Image | ||
- Basic Marks | ||
- Underline | ||
- TextStyle | ||
- Color | ||
- TaskList | ||
- Markdown | ||
- Table |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,76 @@ | ||
{ | ||
"name": "@plane/editor-core", | ||
"version": "0.0.1", | ||
"description": "Core Editor that powers Plane", | ||
"main": "./dist/index.mjs", | ||
"module": "./dist/index.mjs", | ||
"types": "./dist/index.d.mts", | ||
"files": [ | ||
"dist/**/*" | ||
], | ||
"exports": { | ||
".": { | ||
"types": "./dist/index.d.mts", | ||
"import": "./dist/index.mjs", | ||
"module": "./dist/index.mjs" | ||
} | ||
}, | ||
"scripts": { | ||
"build": "tsup", | ||
"dev": "tsup --watch", | ||
"check-types": "tsc --noEmit" | ||
}, | ||
"peerDependencies": { | ||
"react": "^18.2.0", | ||
"react-dom": "18.2.0", | ||
"next": "12.3.2", | ||
"next-themes": "^0.2.1" | ||
}, | ||
"dependencies": { | ||
"@blueprintjs/popover2": "^2.0.10", | ||
"@tiptap/core": "^2.1.7", | ||
"@tiptap/extension-color": "^2.1.11", | ||
"@tiptap/extension-image": "^2.1.7", | ||
"@tiptap/extension-link": "^2.1.7", | ||
"@tiptap/extension-table": "^2.1.6", | ||
"@tiptap/extension-table-cell": "^2.1.6", | ||
"@tiptap/extension-table-header": "^2.1.6", | ||
"@tiptap/extension-table-row": "^2.1.6", | ||
"@tiptap/extension-task-item": "^2.1.7", | ||
"@tiptap/extension-task-list": "^2.1.7", | ||
"@tiptap/extension-text-style": "^2.1.11", | ||
"@tiptap/extension-underline": "^2.1.7", | ||
"@tiptap/pm": "^2.1.7", | ||
"@tiptap/react": "^2.1.7", | ||
"@tiptap/starter-kit": "^2.1.10", | ||
"@types/react": "^18.2.5", | ||
"@types/react-dom": "18.0.11", | ||
"@types/node": "18.15.3", | ||
"class-variance-authority": "^0.7.0", | ||
"clsx": "^1.2.1", | ||
"eslint": "8.36.0", | ||
"eslint-config-next": "13.2.4", | ||
"eventsource-parser": "^0.1.0", | ||
"lucide-react": "^0.244.0", | ||
"react-markdown": "^8.0.7", | ||
"tailwind-merge": "^1.14.0", | ||
"tippy.js": "^6.3.7", | ||
"tiptap-markdown": "^0.8.2", | ||
"use-debounce": "^9.0.4" | ||
}, | ||
"devDependencies": { | ||
"eslint": "^7.32.0", | ||
"postcss": "^8.4.29", | ||
"tailwind-config-custom": "*", | ||
"tsconfig": "*", | ||
"tsup": "^7.2.0", | ||
"typescript": "4.9.5" | ||
}, | ||
"keywords": [ | ||
"editor", | ||
"rich-text", | ||
"markdown", | ||
"nextjs", | ||
"react" | ||
] | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,9 @@ | ||
// If you want to use other PostCSS plugins, see the following: | ||
// https://tailwindcss.com/docs/using-with-preprocessors | ||
|
||
module.exports = { | ||
plugins: { | ||
tailwindcss: {}, | ||
autoprefixer: {}, | ||
}, | ||
}; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,19 @@ | ||
// styles | ||
// import "./styles/tailwind.css"; | ||
// import "./styles/editor.css"; | ||
|
||
// utils | ||
export * from "./lib/utils"; | ||
export { startImageUpload } from "./ui/plugins/upload-image"; | ||
|
||
// components | ||
export { EditorContainer } from "./ui/components/editor-container"; | ||
export { EditorContentWrapper } from "./ui/components/editor-content"; | ||
|
||
// hooks | ||
export { useEditor } from "./ui/hooks/useEditor"; | ||
export { useReadOnlyEditor } from "./ui/hooks/useReadOnlyEditor"; | ||
|
||
// helper items | ||
export * from "./ui/menus/menu-items"; | ||
export * from "./lib/editor-commands"; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,91 @@ | ||
import { Editor, Range } from "@tiptap/core"; | ||
import { UploadImage } from "../types/upload-image"; | ||
import { startImageUpload } from "../ui/plugins/upload-image"; | ||
|
||
export const toggleHeadingOne = (editor: Editor, range?: Range) => { | ||
if (range) editor.chain().focus().deleteRange(range).setNode("heading", { level: 1 }).run(); | ||
else editor.chain().focus().toggleHeading({ level: 1 }).run() | ||
}; | ||
|
||
export const toggleHeadingTwo = (editor: Editor, range?: Range) => { | ||
if (range) editor.chain().focus().deleteRange(range).setNode("heading", { level: 2 }).run(); | ||
else editor.chain().focus().toggleHeading({ level: 2 }).run() | ||
}; | ||
|
||
export const toggleHeadingThree = (editor: Editor, range?: Range) => { | ||
if (range) editor.chain().focus().deleteRange(range).setNode("heading", { level: 3 }).run(); | ||
else editor.chain().focus().toggleHeading({ level: 3 }).run() | ||
}; | ||
|
||
export const toggleBold = (editor: Editor, range?: Range) => { | ||
if (range) editor.chain().focus().deleteRange(range).toggleBold().run(); | ||
else editor.chain().focus().toggleBold().run(); | ||
}; | ||
|
||
export const toggleItalic = (editor: Editor, range?: Range) => { | ||
if (range) editor.chain().focus().deleteRange(range).toggleItalic().run(); | ||
else editor.chain().focus().toggleItalic().run(); | ||
}; | ||
|
||
export const toggleUnderline = (editor: Editor, range?: Range) => { | ||
if (range) editor.chain().focus().deleteRange(range).toggleUnderline().run(); | ||
else editor.chain().focus().toggleUnderline().run(); | ||
}; | ||
|
||
export const toggleCode = (editor: Editor, range?: Range) => { | ||
if (range) editor.chain().focus().deleteRange(range).toggleCode().run(); | ||
else editor.chain().focus().toggleCode().run(); | ||
}; | ||
export const toggleOrderedList = (editor: Editor, range?: Range) => { | ||
if (range) editor.chain().focus().deleteRange(range).toggleOrderedList().run(); | ||
else editor.chain().focus().toggleOrderedList().run(); | ||
}; | ||
|
||
export const toggleBulletList = (editor: Editor, range?: Range) => { | ||
if (range) editor.chain().focus().deleteRange(range).toggleBulletList().run(); | ||
else editor.chain().focus().toggleBulletList().run(); | ||
}; | ||
|
||
export const toggleTaskList = (editor: Editor, range?: Range) => { | ||
if (range) editor.chain().focus().deleteRange(range).toggleTaskList().run(); | ||
else editor.chain().focus().toggleTaskList().run() | ||
}; | ||
|
||
export const toggleStrike = (editor: Editor, range?: Range) => { | ||
if (range) editor.chain().focus().deleteRange(range).toggleStrike().run(); | ||
else editor.chain().focus().toggleStrike().run(); | ||
}; | ||
|
||
export const toggleBlockquote = (editor: Editor, range?: Range) => { | ||
if (range) editor.chain().focus().deleteRange(range).toggleNode("paragraph", "paragraph").toggleBlockquote().run(); | ||
else editor.chain().focus().toggleNode("paragraph", "paragraph").toggleBlockquote().run(); | ||
}; | ||
|
||
export const insertTableCommand = (editor: Editor, range?: Range) => { | ||
if (range) editor.chain().focus().deleteRange(range).insertTable({ rows: 3, cols: 3, withHeaderRow: true }).run(); | ||
else editor.chain().focus().insertTable({ rows: 3, cols: 3, withHeaderRow: true }).run(); | ||
}; | ||
|
||
export const unsetLinkEditor = (editor: Editor) => { | ||
editor.chain().focus().unsetLink().run(); | ||
}; | ||
|
||
export const setLinkEditor = (editor: Editor, url: string) => { | ||
editor.chain().focus().setLink({ href: url }).run(); | ||
}; | ||
|
||
export const insertImageCommand = (editor: Editor, uploadFile: UploadImage, setIsSubmitting?: (isSubmitting: "submitting" | "submitted" | "saved") => void, range?: Range) => { | ||
if (range) editor.chain().focus().deleteRange(range).run(); | ||
const input = document.createElement("input"); | ||
input.type = "file"; | ||
input.accept = "image/*"; | ||
input.onchange = async () => { | ||
if (input.files?.length) { | ||
const file = input.files[0]; | ||
const pos = editor.view.state.selection.from; | ||
startImageUpload(file, editor.view, pos, uploadFile, setIsSubmitting); | ||
} | ||
}; | ||
input.click(); | ||
}; | ||
|
Oops, something went wrong.
0a8b99a
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Successfully deployed to the following URLs:
plane-sh-dev – ./space/
plane-space-dev.vercel.app
plane-sh-dev-git-develop-plane.vercel.app
plane-sh-dev-plane.vercel.app
0a8b99a
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Successfully deployed to the following URLs:
plane-dev – ./web/
plane-dev.vercel.app
plane-dev-plane.vercel.app
plane-dev-git-develop-plane.vercel.app
crew42.plane.so