Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

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.

Awesome Lists containing this project

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