2016-06-16 19:20:36 -03:00
|
|
|
import React, { PropTypes } from 'react';
|
2016-08-01 16:41:55 -03:00
|
|
|
import { Editor } from 'slate';
|
|
|
|
import Markdown from 'slate-markdown-serializer';
|
|
|
|
const markdown = new Markdown();
|
2016-05-30 16:55:32 -07:00
|
|
|
|
2016-08-01 16:41:55 -03:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Slate Render Configuration
|
|
|
|
*/
|
|
|
|
|
|
|
|
// Define the default node type.
|
|
|
|
const DEFAULT_NODE = 'paragraph';
|
|
|
|
|
|
|
|
// Local node renderers.
|
|
|
|
const NODES = {
|
|
|
|
'block-quote': (props) => <blockquote {...props.attributes}>{props.children}</blockquote>,
|
|
|
|
'bulleted-list': props => <ul {...props.attributes}>{props.children}</ul>,
|
|
|
|
'heading1': props => <h1 {...props.attributes}>{props.children}</h1>,
|
|
|
|
'heading2': props => <h2 {...props.attributes}>{props.children}</h2>,
|
|
|
|
'list-item': props => <li {...props.attributes}>{props.children}</li>,
|
|
|
|
'paragraph': props => <p {...props.attributes}>{props.children}</p>,
|
|
|
|
'link': (props) => {
|
|
|
|
const { data } = props.node;
|
|
|
|
const href = data.get('href');
|
|
|
|
return <a {...props.attributes} href={href}>{props.children}</a>;
|
|
|
|
},
|
|
|
|
'image': (props) => {
|
|
|
|
const { node, state } = props;
|
|
|
|
const src = node.data.get('src');
|
|
|
|
return (
|
|
|
|
<img {...props.attributes} src={src} />
|
|
|
|
);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
// Local mark renderers.
|
|
|
|
const MARKS = {
|
|
|
|
bold: {
|
|
|
|
fontWeight: 'bold'
|
|
|
|
},
|
|
|
|
italic: {
|
|
|
|
fontStyle: 'italic'
|
|
|
|
},
|
|
|
|
code: {
|
|
|
|
fontFamily: 'monospace',
|
|
|
|
backgroundColor: '#eee',
|
|
|
|
padding: '3px',
|
|
|
|
borderRadius: '4px'
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
class MarkdownControl extends React.Component {
|
2016-05-30 16:55:32 -07:00
|
|
|
constructor(props) {
|
|
|
|
super(props);
|
|
|
|
this.state = {
|
2016-08-01 16:41:55 -03:00
|
|
|
state: markdown.deserialize(props.value || '')
|
2016-05-30 16:55:32 -07:00
|
|
|
};
|
2016-08-01 16:41:55 -03:00
|
|
|
|
|
|
|
this.hasMark = this.hasMark.bind(this);
|
|
|
|
this.hasBlock = this.hasBlock.bind(this);
|
2016-05-30 16:55:32 -07:00
|
|
|
this.handleChange = this.handleChange.bind(this);
|
2016-08-01 16:41:55 -03:00
|
|
|
this.handleDocumentChange = this.handleDocumentChange.bind(this);
|
|
|
|
this.onClickMark = this.onClickMark.bind(this);
|
|
|
|
this.onClickBlock = this.onClickBlock.bind(this);
|
|
|
|
this.renderToolbar = this.renderToolbar.bind(this);
|
|
|
|
this.renderMarkButton = this.renderMarkButton.bind(this);
|
|
|
|
this.renderBlockButton = this.renderBlockButton.bind(this);
|
|
|
|
this.renderNode = this.renderNode.bind(this);
|
|
|
|
this.renderMark = this.renderMark.bind(this);
|
2016-05-30 16:55:32 -07:00
|
|
|
}
|
|
|
|
|
2016-08-01 16:41:55 -03:00
|
|
|
/*
|
|
|
|
* Used to set toolbar buttons to active state
|
|
|
|
*/
|
|
|
|
hasMark(type) {
|
|
|
|
const { state } = this.state;
|
|
|
|
return state.marks.some(mark => mark.type == type);
|
2016-05-30 16:55:32 -07:00
|
|
|
}
|
2016-08-01 16:41:55 -03:00
|
|
|
hasBlock(type) {
|
|
|
|
const { state } = this.state;
|
|
|
|
return state.blocks.some(node => node.type == type);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Slate keeps track of selections, scroll position etc.
|
|
|
|
* So, onChange gets dispatched on every interaction (click, arrows, everything...)
|
|
|
|
* It also have an onDocumentChange, that get's dispached only when the actual
|
|
|
|
* content changes
|
|
|
|
*/
|
|
|
|
handleChange(state) {
|
|
|
|
this.setState({ state });
|
|
|
|
}
|
|
|
|
|
|
|
|
handleDocumentChange(document, state) {
|
|
|
|
this.props.onChange(markdown.serialize(state));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Toggle marks / blocks when button is clicked
|
|
|
|
*/
|
|
|
|
onClickMark(e, type) {
|
|
|
|
e.preventDefault();
|
|
|
|
let { state } = this.state;
|
|
|
|
|
|
|
|
state = state
|
|
|
|
.transform()
|
|
|
|
.toggleMark(type)
|
|
|
|
.apply();
|
|
|
|
|
|
|
|
this.setState({ state });
|
|
|
|
}
|
|
|
|
|
|
|
|
onClickBlock(e, type) {
|
|
|
|
e.preventDefault();
|
|
|
|
let { state } = this.state;
|
|
|
|
let transform = state.transform();
|
|
|
|
const { document } = state;
|
2016-05-30 16:55:32 -07:00
|
|
|
|
2016-08-01 16:41:55 -03:00
|
|
|
// Handle everything but list buttons.
|
|
|
|
if (type != 'bulleted-list' && type != 'numbered-list') {
|
|
|
|
const isActive = this.hasBlock(type);
|
|
|
|
const isList = this.hasBlock('list-item');
|
|
|
|
|
|
|
|
if (isList) {
|
|
|
|
transform = transform
|
|
|
|
.setBlock(isActive ? DEFAULT_NODE : type)
|
|
|
|
.unwrapBlock('bulleted-list')
|
|
|
|
.unwrapBlock('numbered-list');
|
|
|
|
}
|
|
|
|
|
|
|
|
else {
|
|
|
|
transform = transform
|
|
|
|
.setBlock(isActive ? DEFAULT_NODE : type);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Handle the extra wrapping required for list buttons.
|
|
|
|
else {
|
|
|
|
const isList = this.hasBlock('list-item');
|
|
|
|
const isType = state.blocks.some((block) => {
|
|
|
|
return !!document.getClosest(block, parent => parent.type == type);
|
|
|
|
});
|
|
|
|
|
|
|
|
if (isList && isType) {
|
|
|
|
transform = transform
|
|
|
|
.setBlock(DEFAULT_NODE)
|
|
|
|
.unwrapBlock('bulleted-list');
|
|
|
|
} else if (isList) {
|
|
|
|
transform = transform
|
|
|
|
.unwrapBlock(type == 'bulleted-list')
|
|
|
|
.wrapBlock(type);
|
|
|
|
} else {
|
|
|
|
transform = transform
|
|
|
|
.setBlock('list-item')
|
|
|
|
.wrapBlock(type);
|
|
|
|
}
|
2016-05-30 16:55:32 -07:00
|
|
|
}
|
2016-08-01 16:41:55 -03:00
|
|
|
|
|
|
|
state = transform.apply();
|
|
|
|
this.setState({ state });
|
|
|
|
}
|
|
|
|
|
|
|
|
renderToolbar() {
|
|
|
|
return (
|
|
|
|
<div className="menu toolbar-menu">
|
|
|
|
{this.renderMarkButton('bold', 'b')}
|
|
|
|
{this.renderMarkButton('italic', 'i')}
|
|
|
|
{this.renderMarkButton('code', 'code')}
|
|
|
|
{this.renderBlockButton('heading1', 'h1')}
|
|
|
|
{this.renderBlockButton('heading2', 'h2')}
|
|
|
|
{this.renderBlockButton('block-quote', 'blockquote')}
|
|
|
|
{this.renderBlockButton('bulleted-list', 'ul')}
|
|
|
|
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
renderMarkButton(type, icon) {
|
|
|
|
const isActive = this.hasMark(type);
|
|
|
|
const onMouseDown = e => this.onClickMark(e, type);
|
|
|
|
|
|
|
|
return (
|
|
|
|
<button onMouseDown={onMouseDown} data-active={isActive}>
|
|
|
|
{icon}
|
|
|
|
</button>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
renderBlockButton(type, icon) {
|
|
|
|
const isActive = this.hasBlock(type);
|
|
|
|
const onMouseDown = e => this.onClickBlock(e, type);
|
|
|
|
|
|
|
|
return (
|
|
|
|
<button onMouseDown={onMouseDown} data-active={isActive}>
|
|
|
|
{icon}
|
|
|
|
</button>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Return renderers for Slate
|
|
|
|
*/
|
|
|
|
renderNode(node) {
|
|
|
|
return NODES[node.type];
|
|
|
|
}
|
|
|
|
renderMark(mark) {
|
|
|
|
return MARKS[mark.type];
|
2016-05-30 16:55:32 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
render() {
|
|
|
|
return (
|
2016-08-01 16:41:55 -03:00
|
|
|
<div>
|
|
|
|
{this.renderToolbar()}
|
|
|
|
<div className="editor">
|
|
|
|
<Editor
|
|
|
|
placeholder={'Enter some rich text...'}
|
|
|
|
state={this.state.state}
|
|
|
|
renderNode={this.renderNode}
|
|
|
|
renderMark={this.renderMark}
|
|
|
|
onChange={this.handleChange}
|
|
|
|
onDocumentChange={this.handleDocumentChange}
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
);
|
2016-05-30 16:55:32 -07:00
|
|
|
}
|
|
|
|
}
|
2016-06-16 19:20:36 -03:00
|
|
|
|
2016-08-01 16:41:55 -03:00
|
|
|
export default MarkdownControl;
|
|
|
|
|
2016-06-16 19:20:36 -03:00
|
|
|
MarkdownControl.propTypes = {
|
|
|
|
onChange: PropTypes.func.isRequired,
|
|
|
|
value: PropTypes.node,
|
|
|
|
};
|