fix: markdown

This commit is contained in:
Liu Jiangyi
2020-06-22 12:01:08 +08:00
parent b8b6dcf68d
commit b37dd82498
5 changed files with 393 additions and 19 deletions

View File

@@ -2,7 +2,14 @@ import React, {Component, PureComponent} from 'react';
import {format_time,Time,TitleLine} from './infrastructure/widgets';
import {THUHOLE_API_ROOT} from './flows_api';
import HtmlToReact from 'html-to-react'
import './Common.css';
import { URL_PID_RE, URL_RE, split_text } from './text_splitter';
import ProcessNodeDefinitions from 'html-to-react/lib/process-node-definitions';
import ReactMarkdown from 'react-markdown'
import htmlParser from 'react-markdown/plugins/html-parser'
export {format_time,Time,TitleLine};
@@ -26,11 +33,13 @@ export function ColoredSpan(props) {
)
}
function normalize_url(url) {
return /^https?:\/\//.test(url) ? url : 'http://'+url;
}
export class HighlightedText extends PureComponent {
render() {
function normalize_url(url) {
return /^https?:\/\//.test(url) ? url : 'http://'+url;
}
return (
<pre>
{this.props.parts.map((part,idx)=>{
@@ -51,6 +60,75 @@ export class HighlightedText extends PureComponent {
}
}
// props: text, show_pid, color_picker
export class HighlightedMarkdown extends PureComponent {
render() {
const processDefs = new HtmlToReact.ProcessNodeDefinitions(React)
const astInst = [
{
shouldProcessNode (node) {
return node.name === 'link' // urls
},
processNode (node) {
const originalHref = node.attribs.href
if (URL_PID_RE.test(originalHref)) { // url_pid
return (
<span className="url-pid-link" title={originalHref}>/##</span>
)
} else if (URL_RE.test(originalHref)) { // url
return (
<a href={normalize_url(originalHref)} target="_blank" rel="noopener">
{originalHref}
</a>
)
} else {// other cases are invalid, e.g. javascript:alert, etc
return (<a></a>)
}
}
},
{
shouldProcessNode (node) {
return node.name === 'text' // pid, nickname, search
},
processNode (node) {
const originalText = node.data
const splitted = split_text(originalText, [
['pid',PID_RE],
['nickname',NICKNAME_RE],
])
return (
<>
{splitted.map(([rule, p], idx) => {
<span key={p}>
{rule==='pid' ? <a href={'#'+p} onClick={(e)=>{e.preventDefault(); this.props.show_pid(p.substring(1));}}>{p}</a> :
rule==='nickname' ? <ColoredSpan colors={this.props.color_picker.get(p)}>{p}</ColoredSpan> :
rule==='search' ? <span className="search-query-highlight">{p}</span> :
p}
</span>
})}
</>
)
}
},
{
shouldProcessNode: () => true,
processNode: processDefs.processDefaultNode
}
]
return (
<ReactMarkdown source={this.props.text} astPlugins={
[
htmlParser({
isValidNode(node) { return node.type !== 'script' },
processingInstructions: astInst
})
]
}/>
)
}
}
window.TEXTAREA_BACKUP={};
export class SafeTextarea extends Component {

View File

@@ -2,7 +2,7 @@ import React, {Component, PureComponent} from 'react';
import copy from 'copy-to-clipboard';
import {ColorPicker} from './color_picker';
import {split_text, NICKNAME_RE, PID_RE, URL_RE, URL_PID_RE} from './text_splitter';
import {format_time, build_highlight_re, Time, TitleLine, HighlightedText, ClickHandler, ColoredSpan} from './Common';
import {format_time, build_highlight_re, Time, TitleLine, HighlightedText, ClickHandler, ColoredSpan, HighlightedMarkdown} from './Common';
import './Flows.css';
import LazyLoad from './react-lazyload/src';
import {AudioWidget} from './AudioWidget';
@@ -130,12 +130,6 @@ class FlowItem extends PureComponent {
render() {
let props=this.props;
let parts=props.parts||split_text(renderMd(props.info.text),[
['url_pid',URL_PID_RE],
['url',URL_RE],
['pid',PID_RE],
['nickname',NICKNAME_RE],
]);
return (
<div className={'flow-item'+(props.is_quote ? ' flow-item-quote' : '')}>
{!!props.is_quote &&
@@ -176,7 +170,7 @@ class FlowItem extends PureComponent {
<Time stamp={props.info.timestamp} />
</div>
<div className="box-content">
<HighlightedText parts={parts} color_picker={props.color_picker} show_pid={props.show_pid} />
<HighlightedMarkdown text={props.info.text} color_picker={props.color_picker} show_pid={props.show_pid} />
{props.info.type==='image' &&
<p className="img">
{props.img_clickable ?

View File

@@ -1,8 +0,0 @@
import MarkdownIt from 'markdown-it'
let md = new MarkdownIt({
html: false,
linkify: false // avoid collision with text_splitter.js
})
export default (text) => md.render(text)