/* eslint react/prop-types: 0, react/no-multi-comp: 0 */ import React from 'react'; import { List, Map } from 'immutable'; import MarkupIt from 'markup-it'; import markdownSyntax from 'markup-it/syntaxes/markdown'; import htmlSyntax from 'markup-it/syntaxes/html'; import reInline from 'markup-it/syntaxes/markdown/re/inline'; import { Icon } from '../UI'; /* * All Rich text widgets (Markdown, for example) should use Slate for text editing and * MarkupIt to convert between structured formats (Slate JSON, Markdown, HTML, etc.). * This module Processes and provides Slate nodes and MarkupIt syntaxes augmented with plugins */ let processedPlugins = List([]); const nodes = {}; let augmentedMarkdownSyntax = markdownSyntax; let augmentedHTMLSyntax = htmlSyntax; function processEditorPlugins(plugins) { // Since the plugin list is immutable, a simple comparisson is enough // to determine whether we need to process again. if (plugins === processedPlugins) return; plugins.forEach(plugin => { const basicRule = MarkupIt.Rule(plugin.id).regExp(plugin.pattern, (state, match) => ( { data: plugin.fromBlock(match) } )); const markdownRule = basicRule.toText((state, token) => ( plugin.toBlock(token.getData().toObject()) + '\n\n' )); const htmlRule = basicRule.toText((state, token) => ( plugin.toPreview(token.getData().toObject()) )); const nodeRenderer = (props) => { const { node, state } = props; const isFocused = state.selection.hasEdgeIn(node); const className = isFocused ? 'plugin active' : 'plugin'; return (
{ plugin.fields.map(field => `${field.label}: “${node.data.get(field.name)}”`) }
); }; augmentedMarkdownSyntax = augmentedMarkdownSyntax.addInlineRules(markdownRule); augmentedHTMLSyntax = augmentedHTMLSyntax.addInlineRules(htmlRule); nodes[plugin.id] = nodeRenderer; }); processedPlugins = plugins; } function processMediaProxyPlugins(getMedia) { const mediaProxyRule = MarkupIt.Rule('mediaproxy').regExp(reInline.link, (state, match) => { if (match[0].charAt(0) !== '!') { // Return if this is not an image return; } var imgData = Map({ alt: match[1], src: match[2], title: match[3] }).filter(Boolean); return { data: imgData }; }); const mediaProxyMarkdownRule = mediaProxyRule.toText((state, token) => { var data = token.getData(); var alt = data.get('alt', ''); var src = getMedia(data.get('src', '')); var title = data.get('title', ''); if (title) { return '![' + alt + '](' + src + ' "' + title + '")'; } else { return '![' + alt + '](' + src + ')'; } }); const mediaProxyHTMLRule = mediaProxyRule.toText((state, token) => { var data = token.getData(); var alt = data.get('alt', ''); var src = data.get('src', ''); return `${alt}`; }); nodes['mediaproxy'] = (props) => { /* eslint react/prop-types: 0 */ const { node, state } = props; const isFocused = state.selection.hasEdgeIn(node); const className = isFocused ? 'active' : null; const src = node.data.get('src'); return ( ); }; augmentedMarkdownSyntax = augmentedMarkdownSyntax.addInlineRules(mediaProxyMarkdownRule); augmentedHTMLSyntax = augmentedHTMLSyntax.addInlineRules(mediaProxyHTMLRule); } function getPlugins() { return processedPlugins.map(plugin => ( { id: plugin.id, icon: plugin.icon, fields: plugin.fields } )).toArray(); } function getNodes() { return nodes; } function getSyntaxes(getMedia) { if (getMedia) { processMediaProxyPlugins(getMedia); } return { markdown: augmentedMarkdownSyntax, html:augmentedHTMLSyntax }; } export { processEditorPlugins, getNodes, getSyntaxes, getPlugins };