Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/brijeshb42/medium-draft
📝 A medium like Rich Text Editor built on draft-js with a focus on keyboard shortcuts.
https://github.com/brijeshb42/medium-draft
draft-js editor medium-editor rich-text-editor
Last synced: about 15 hours ago
JSON representation
📝 A medium like Rich Text Editor built on draft-js with a focus on keyboard shortcuts.
- Host: GitHub
- URL: https://github.com/brijeshb42/medium-draft
- Owner: brijeshb42
- License: mit
- Archived: true
- Created: 2016-04-17T07:53:56.000Z (almost 9 years ago)
- Default Branch: master
- Last Pushed: 2021-05-08T22:25:43.000Z (over 3 years ago)
- Last Synced: 2024-10-01T09:01:40.515Z (4 months ago)
- Topics: draft-js, editor, medium-editor, rich-text-editor
- Language: JavaScript
- Homepage: https://bitwiser.in/medium-draft/
- Size: 1.66 MB
- Stars: 1,726
- Watchers: 37
- Forks: 245
- Open Issues: 51
-
Metadata Files:
- Readme: README.md
- Changelog: CHANGELOG.md
- License: LICENSE
Awesome Lists containing this project
- awesome-draft-js - Medium Draft - Medium-like rich text editor with a focus on keyboard shortcuts. (Standalone Editors Built on Draft.js)
README
Latest development is going on in this [branch](https://github.com/brijeshb42/medium-draft/tree/djs-pl-ed-ts).
# medium-draft - [demo](http://bitwiser.in/medium-draft/)
[![npm version](https://badge.fury.io/js/medium-draft.svg)](https://www.npmjs.com/package/medium-draft) [![Build Status](https://travis-ci.org/brijeshb42/medium-draft.svg?branch=master)](https://travis-ci.org/brijeshb42/medium-draft)
A medium like rich text editor built upon [draft-js](https://facebook.github.io/draft-js/) with an emphasis on eliminating mouse usage by adding relevant keyboard shortcuts.
[Documentation](https://github.com/brijeshb42/medium-draft/wiki) in progress.
Install the beta version using
`npm install medium-draft@beta`
### Features
- Focus on keyboard shortcuts and auto transform of text blocks.
- Image addition with support for rich text captioning.
- Minimize mouse usage.
- Autolists.
- Proper handling of `RETURN` presses.
- It also has implementations of some custom blocks like:
- `caption` - Can be used as a caption for media blocks like image or video instead of nested `draft-js` instances for simplicity.
- `block-quote-caption` - Caption for `blockquote`s.
- `todo` - Todo text with a checkbox.
- Easily customizable toolbar via `toolbarConfig` for the following block and inline styles. Defaults to all. Case sensitive.
- `block: ['ordered-list-item', 'unordered-list-item', 'blockquote', 'header-three', 'todo']`
- `inline: ['BOLD', 'ITALIC', 'UNDERLINE', 'hyperlink', 'HIGHLIGHT']`##### Following are the keyboard shortcuts to toggle block types (Alt and CTRL for Windows/Linux and Option and Command for OSX)
* Alt/Option +* 1 - Toggle Ordered list item
* * - Toggle Unordered list item
* # - Toggle Header-three.
* < - Toggle Caption block.
* > - Toggle unstyled or paragraph block.
* H - Highlight selection.##### Other Shortcuts
* CMD/CTRL + K -> Add Link
* CMD/CTRL + SHIFT + K -> Remove link if cursor is inside a word with link.##### Editor level commands
These commands are not a part of the core editor but have been implemented in the example code that uses the `medium-draft` editor.
* Command/CTRL + S - Save current data to `localstorage`.
* Alt + Shift + L - Load previously saved data from `localstorage`.##### Special characters while typing: While typing in an empty block, if the content matches one of the following, that particular block's type and look will be changed to the corresponding block specified below
* `--` - If current block is `blockquote`, it will be changed to `block-quote-caption`, else `caption`.
* `*.` `(An asterisk and a period)` - `unordered-list-item`.
* `*` `(An asterisk and a space)` - `unordered-list-item`.
* `-` `(A hyphen and a space)` - `unordered-list-item`.
* `1.` `(The number 1 and a period)` - `unordered-list-item`.
* `##` - `header-two`.
* `[]` - `todo`.
* `==` - `unstyled`.### Installation
- **npm**.
- `npm install medium-draft`.
- `import Editor from 'medium-draft'`
- **Browser**
- Include `` in ``
- Include ``. **medium-draft** is available in the global object as `MediumDraft`.### Usage
`medium-draft` sits on top of `draft-js` with some built in functionalities and blocks. Its API is almost the same as that of `draft-js`. You can take a look at [the demo editor's code](https://github.com/brijeshb42/medium-draft/tree/master/src/example.js) to see the implementation.
#### CSS
Include the css that comes with the library in your HTML -
```html```
If you are using `webpack` for bundling, you can import the CSS like this in your JS code
```javascript
import 'medium-draft/lib/index.css';
```If you are using `sideButtons`, you will also need to include the css for `font-awesome` -
```html
```
or something equivalent.
#### JS (ES6)
At the minimum, you need to provide `editorState` and `onChange` props, the same as `draft-js`.
```javascript
import React from 'react';
import ReactDOM from 'react-dom';// if using webpack
// import 'medium-draft/lib/index.css';import {
Editor,
createEditorState,
} from 'medium-draft';class App extends React.Component {
constructor(props) {
super(props);this.state = {
editorState: createEditorState(), // for empty content
};/*
this.state = {
editorState: createEditorState(data), // with content
};
*/this.onChange = (editorState) => {
this.setState({ editorState });
};this.refsEditor = React.createRef();
}
componentDidMount() {
this.refsEditor.current.focus();
}render() {
const { editorState } = this.state;
return (
);
}
};ReactDOM.render(
,
document.getElementById('app')
);
```### Customizing side buttons
`medium-draft`'s `Editor` accepts a prop called `sideButtons`. By default, there is only one (image) button, but you can add more. The `sideButtons` prop must be an array of objects with each object having the following signature:
```js
{
"title": "unique-button-name",
"component": ButtonComponent
}
```For ex:
```js
{
"title": "Image",
"component": ImageSideButton
}
```Example code:
Right now, the image button simply adds an image inside the editor using `URL.createObjectURL`. But if you would like to first upload the image to your server and then add that image to the editor, you can follow one of the 2 methods:
1. Either extend the default `ImageSideButton` component that comes with `medium-draft`.
2. Or create your own component with the complete functionality yourself.
For simplicity, we will follow the first method. If you study the [implementation](src/components/sides/image.js) of `ImageSideButton`, you will see an `onChange` method that receives the file chooser event where the seleced files are available as `event.target.files`. We will simply override this method as we don't want to customize anything else. Also note that each side button component receives `getEditorState` function (returns the draft `editorState`), `setEditorState(newEditorState)` function (sets the new editorState) and `close` function which you need to call manually to close the side buttons list:
```javascript
import React from 'react';
import {
ImageSideButton,
Block,
addNewBlock,
createEditorState,
Editor,
} from 'medium-draft';
import 'isomorphic-fetch';class CustomImageSideButton extends ImageSideButton {
/*
We will only check for first file and also whether
it is an image or not.
*/
onChange(e) {
const file = e.target.files[0];
if (file.type.indexOf('image/') === 0) {
// This is a post request to server endpoint with image as `image`
const formData = new FormData();
formData.append('image', file);
fetch('/your-server-endpoint', {
method: 'POST',
body: formData,
}).then((response) => {
if (response.status === 200) {
// Assuming server responds with
// `{ "url": "http://example-cdn.com/image.jpg"}`
return response.json().then(data => {
if (data.url) {
this.props.setEditorState(addNewBlock(
this.props.getEditorState(),
Block.IMAGE, {
src: data.url,
}
));
}
});
}
});
}
this.props.close();
}}
// Now pass this component instead of default prop to Editor example above.
class App extends React.Component {
constructor(props) {
super(props);this.sideButtons = [{
title: 'Image',
component: CustomImageSideButton,
}];this.state = {
editorState: createEditorState(), // for empty content
};/*
this.state = {
editorState: createEditorState(data), // with content
};
*/this.onChange = (editorState) => {
this.setState({ editorState });
};this.refsEditor = React.createRef()
}
componentDidMount() {
this.refsEditor.current.focus();
}render() {
const { editorState } = this.state;
return (
);
}
};
```### Removing side buttons
To remove the side buttons entirely, so that the circular add button never appears, just pass an empty array:
`sideButtons={[]}`
### Customizing toolbar
There are three props you can use to customize the buttons in the toolbar that appears whenever you select text within the editor:
- `blockButtons`
- `inlineButtons`
- `toolbarConfig`The default block-level editor buttons are `['header-three', 'unordered-list-item', 'ordered-list-item', 'blockquote', 'todo']`, and the default inline editor buttons `['BOLD', 'ITALIC', 'UNDERLINE', 'HIGHLIGHT', 'hyperlink']`.
For example, if you want to keep the default block buttons and add a few more, you can do something like the following:
```js
import { BLOCK_BUTTONS } from 'medium-draft';const blockButtons = [{
label: 'H1',
style: 'header-one',
icon: 'header',
description: 'Heading 1',
},
{
label: 'H2',
style: 'header-two',
icon: 'header',
description: 'Heading 2',
}].concat(BLOCK_BUTTONS);// in your component
```
If you want to __remove__ some buttons or __reorder__ them, you could use functions like `array.slice` on the default `BLOCK_BUTTONS` and `INLINE_BUTTONS`, but this is probably more trouble than it's worth.
For this purpose it's better to use the `toolbarConfig` prop:
```js
// custom ordering for block and inline buttons, and removes some buttons
const toolbarConfig = {
block: ['unordered-list-item', 'header-one', 'header-three'],
inline: ['BOLD', 'UNDERLINE', 'hyperlink'],
}```
The strings inside the `block` and `inline` arrays must match the `style` attribute inside `blockButtons` and `inlineButtons` arrays.
To summarize: if you need add, remove, and reorder buttons, it's probably easiest to use `blockButtons`, `inlineButtons`, and `toolbarConfig` together.
#### Supply Your Own Toolbar
If the toolbar customization props aren't sufficient to get the behavior you want, you can inject your own toolbar with the `ToolbarComponent` prop.
This pattern is called [component injection](https://reactpatterns.github.io/Component-injection/). Your `ToolbarComponent` receives the same props as the default toolbar.
If you want to write your own toolbar component, a good place to start is [with the default component](https://github.com/brijeshb42/medium-draft/blob/master/src/components/toolbar.js).
### Render data to HTML
The feature to export HTML is available from version `0.4.1` onwards.
`medium-draft` uses [draft-convert](https://github.com/hubspot/draft-convert) (which in turn uses react-dom-server) to render `draft-js`'s `editorState` to HTML.
The exporter is not a part of the core library. If you want to use `medium-draft-exporter`, follow these steps -
#### Browserify/webpack
- `npm install draft-convert`.
`draft-convert` is part of `peerDependencies` of `medium-draft`.
##### Code
```js
import mediumDraftExporter from 'medium-draft/lib/exporter';
const editorState = /* your draft editorState */;
const renderedHTML = mediumDraftExporter(editorState.getCurrentContent());
/* Use renderedHTML */
```#### Browser
- Add the following scripts before your js code.
```html
```
The exporter is available as `MediumDraftExporter` global;
- JS
```js
var mediumDraftExporter = MediumDraftExporter.default;
const editorState = /* your draft editorState */;
const renderedHTML = mediumDraftExporter(editorState.getCurrentContent());
/* Use renderedHTML */
```The `medium-draft-exporter` also comes with a preset CSS if you want to apply some basic styles to the rendered HTML.
- In webpack, as part of your rendered HTML's page, use this-
```js
import 'medium-draft/lib/basic.css'
```- In browser, in your rendered html's page, you can include this stylesheet link
```html
```### Load HTML exported using `medium-draft-exporter` to `editorState`
The feature to export HTML is available from version `0.5.3` onwards.
`medium-draft` uses [draft-convert](https://github.com/hubspot/draft-convert) (which in turn uses react-dom-server) to render `draft-js`'s `editorState` to HTML.
The importer is not a part of the core library. If you want to use `medium-draft-importer`, follow these steps -
#### Browserify/webpack
- `npm install draft-convert`.
`draft-convert` is part of `peerDependencies` of `medium-draft`.
##### Code
```js
import { convertToRaw } from 'draft-js';
import { createEditorState } from 'medium-draft';
import mediumDraftImporter from 'medium-draft/lib/importer';const html = /* your previously exported html */;
const editorState = createEditorState(convertToRaw(mediumDraftImporter(html)));
// Use this editorState
```#### Browser
- Add the following scripts before your js code.
```html
```
The importer is available as `MediumDraftImporter` global;
- JS
```js
const { convertToRaw } = Draft;
const { createEditorState } = MediumDraft;
const mediumDraftImporter = MediumDraftImporter.default;
const html = /* your previously exported html */;
const editorState = createEditorState(convertToRaw(mediumDraftImporter(html)));
// Use this editorState
```### Issues
- [x] Write an exporter to export draft data to HTML specifically for `medium-draft`.
- [ ] Figure out a way to show placeholder text for empty image captions.
- [x] Currently, the toolbar that appears when text is selected needs to be fixed regarding its position in the viewport.### Developer
- Clone this repo `git clone https://github.com/brijeshb42/medium-draft.git`.
- Install node packages `npm install react react-dom draft-convert && npm install`.
- Start local demo `npm run dev`. This will start a local server on port `8080`.
- Build using `npm run build`.#### LICENSE
MIT