Introduction

A ready to use text editor build for svelte with tiptap and shadcn ui. Creating a tiptap editor from scratch is a pain. This package provides a ready-to-use editor with all the features you need. You can install the package in your project but it will not give you flexibility. The recommended way is to use cli which installs all the dependencies for you and pastes the shad-editor component in your project.


Installation

The main focus of this project is to provide ready to use component in your current project. You need to install the shadcn ui and it’s dependencies.

Pre-Requisites - ⚠️ Important

Since, this project uses shadcn ui, you need to install shadcn in your svelte project. The editor uses tailwind typography plugin which is not included in shadcn. So, you need to install it manually.

1. Adding shadcn and it’s components

For installation of shadcn, please follow thier official installation guide.

The editor usages tooltip, dropdown menu, button, separator, input, etc. Add them in your svelte project.

# using npm
npx shadcn-svelte@next add dropdown-menu button tooltip input popover separator
# using pnpm
pnpm dlx shadcn-svelte@next add dropdown-menu button tooltip input popover separator

2. Adding Tailwind Typography

For installation of tailwind typography, please follow thier official installation guide.

3. Adding Lucide Icons (Optional, CLI will do it for you)

Add lucide icons in your svelte project.

# using npm
npm install lucide-svelte
# using pnpm
pnpm install lucide-svelte

4. Adding Mode Watcher (Optional, CLI will do it for you)

Mode watcher is used to detect the mode of the editor. Add it in your svelte project.

# using npm
npm install mode-watcher
# using pnpm
pnpm install mode-watcher

5. Installation

Recommended way is to use cli which installs all the dependencies for you and pastes the shad-editor component in your project. This method gives you full independence and flexibility. You can further customize the editor as per your need.

# using npm
npx shadeditor init
# using pnpm
pnpm dlx shadeditor init

Usage

You can import the shad-editor component in any of the file. It should work just like any other svelte component.

<script lang="ts">
	import { browser } from '$app/environment';
	import ShadEditor from '$lib/components/shad-editor/shad-editor.svelte';
	import { writable } from 'svelte/store';

	let localStorageContent = '';

	if (browser) {
		localStorageContent = localStorage.getItem('content') || '';
	}

	const content = writable(localStorageContent);

	content.subscribe((value) => {
		console.log('Content Changed');
		if (!browser) return;
		localStorage.setItem('content', value);
	});
</script>

<main class="my-10 flex h-full w-full flex-col items-center justify-center">
	<ShadEditor class="h-[40rem]" content={$content} />
</main>

Placeholder

The editor has built in placeholder which let you show a text on new line. This is an official tiptap extension.

You can find default implementation in shad-editor.svelte

Placeholder.configure({
	emptyEditorClass: 'is-empty',
	placeholder: 'Write something …'
})

You can also show node specific placeholder e.g. for headings.

Placeholder.configure({
	emptyEditorClass: 'is-empty',
	placeholder: ({ node }) => {
		if (node.type.name === 'heading') {
			return 'What’s the title?';
		}
		return 'Write something or use toolbar to insert something ...';
	}
})

Markdown Support

The editor has built in support for markdown. You can use **Bold**, _italic_, ~~strikethrough~~, >Quotes, # Heading 1, ## Heading 2, ### Heading 3, - Unordered List, 1. Ordered List, [] Task List and more.

try it out in the editor.


Code Block Extended

ShadEditor utilizes lowlight for syntax highlighting. For the code syntax highlighting, one-light and one-dark themes are added with respect to the theme of website. It usage tailwind css light and dark mode.

Info: By Default, the tiptap code block extension does not support code copy and language selection.

If you see the code, there has been added an extended code block component in shad-editor/custom/code-extended.svelte.

In shad-editor.svelte, the code block extension is added with lowlight option in extensions array.

CodeBlockLowlight.configure({
	lowlight
})
.extend({
	addNodeView() {
		return SvelteNodeViewRenderer(CodeExtended);
	}
}),

This extension provides a dropdown menu to select the language of the code block with a copy button.

try it out in the editor.


Image Extended

The image extension is extended with a resize, align, caption, Duplicate and Delete options. The code file for this extension can be found in shad-editor/custom/Extentions/ImageExtention.ts and component for this extension can be found in shad-editor/custom/image-extended-component.svelte.

You can find the implementation in shad-editor.svelte file in extensions array.

ImageExtension

try it out in the editor.

Credit to Shadcn Editor


Table

Table extension is default tiptap extension. You can edit, add, delete, merge and split rows, cells and columns.

try it out in the editor.


Search And Replace

Search and Replace provides an advanced search and replace functionality. This is a custom extension which is not a part of tiptap. You can find and replace the text in the editor. You can utilize find next, find previous, replace and replace all functionality. One can find the extenstion file in shad-editor/custom/Extentions/SearchAndReplace.ts.

You can find the implementation in shad-editor.svelte file in extensions array.

SearchAndReplace

try it out in the editor.

Credit to sereneinserenade


Links

Links extension is default tiptap extension. You can add and remove links with ease. By Default, the links do not open on click. You can make them open on click by adding openOnClick option.

You can edit the implementation in shad-editor.svelte file in extensions array.

Link.configure({
	openOnClick: true,
	openOnClick: false,
	autolink: true,
	defaultProtocol: 'https',
	HTMLAttributes: {
		target: '_blank',
		rel: 'noopener noreferrer'
	}
})

try it out in the editor.


Drag Handle

The editor has built in support for drag handle. You can drag the block to move it up and down. The nodes can be auto joined on drag. You can find the looking handle on the left side of each block. You can grab it and move the block to any position. The document will auto scroll on drag.

try it out in the editor.


HTML Content

The editor output can be stored as HTML. You can bind the content props of the editor component with a string variable. The content will be updated on each transaction. A simple example is shown below.

<script lang="ts">
	import ShadEditor from '$lib/components/shad-editor/shad-editor.svelte';
	import { writable } from 'svelte/store';

	// Initialize the content with empty string
	const content = writable("");

	content.subscribe(async (value) => {
		console.log('Content Changed');
		// Save the content to your storage
		await saveData(value);
	});

	omMount(async() => {
		let rawData: string = await loadData();
		content.set(rawData);
	});

</script>

<main class="my-10 flex h-full w-full flex-col items-center justify-center">
	<ShadEditor class="h-[40rem]" content={$content} />
</main>

Info: Under the hood, the content is being set to HTML in shad-editor.svelte file. You can find this on onTransaction function.

onTransaction: (transaction) => {
	editor = transaction.editor;
	content = editor.getHTML();
}

JSON Content

The editor output can be stored as JSON. You can bind the content props of the editor component with a string variable. The content will be updated on each transaction. A simple example is shown below.

<script lang="ts">
	import ShadEditor from '$lib/components/shad-editor/shad-editor.svelte';
	import { writable } from 'svelte/store';
	import { type JSONContent } from '@tiptap/core';

	// Initialize the content with empty string
	const content = writable<JSONContent[]>();

	content.subscribe(async (value) => {
		console.log('Content Changed');
		// Save the content to your storage
		await saveData(value);
	});

	omMount(async() => {
		let rawData: JSONContent[] = await loadData();
		content.set(rawData);
	});

</script>

<main class="my-10 flex h-full w-full flex-col items-center justify-center">
	<ShadEditor class="h-[40rem]" content={$content} />
</main>

Info: Under the hood, the content can be set to JSON in shad-editor.svelte file. You can edit this on onTransaction function.

onTransaction: (transaction) => {
	editor = transaction.editor;
	content = editor.getHTML();
	content = editor.getJSON();
}