import React, { PropTypes } from 'react'; import { Editor } from 'slate'; import Markdown from 'slate-markdown-serializer'; const markdown = new Markdown(); /* * Slate Render Configuration */ // Define the default node type. const DEFAULT_NODE = 'paragraph'; // Local node renderers. const NODES = { 'block-quote': (props) =>
{props.children}, 'bulleted-list': props =>
{props.children}
, 'link': (props) => { const { data } = props.node; const href = data.get('href'); return {props.children}; }, 'image': (props) => { const { node, state } = props; const src = node.data.get('src'); return ( ); } }; // 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 { constructor(props) { super(props); this.state = { state: markdown.deserialize(props.value || '') }; this.hasMark = this.hasMark.bind(this); this.hasBlock = this.hasBlock.bind(this); this.handleChange = this.handleChange.bind(this); 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); } /* * Used to set toolbar buttons to active state */ hasMark(type) { const { state } = this.state; return state.marks.some(mark => mark.type == type); } 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; // 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); } } state = transform.apply(); this.setState({ state }); } renderToolbar() { return (