lbry-desktop/ui/component/common/markdown-preview.jsx

176 lines
4.5 KiB
React
Raw Normal View History

// @flow
import * as React from 'react';
2020-09-17 11:32:31 -04:00
import classnames from 'classnames';
import remark from 'remark';
import remarkAttr from 'remark-attr';
import remarkStrip from 'strip-markdown';
import remarkEmoji from 'remark-emoji';
import remarkBreaks from 'remark-breaks';
import reactRenderer from 'remark-react';
import ExternalLink from 'component/externalLink';
import defaultSchema from 'hast-util-sanitize/lib/github.json';
import { formatedLinks, inlineLinks } from 'util/remark-lbry';
import { Link } from 'react-router-dom';
import { formatLbryUrlForWeb } from 'util/url';
import EmbedPlayButton from 'component/embedPlayButton';
2019-11-07 14:39:22 -05:00
type SimpleTextProps = {
children?: React.Node,
};
type SimpleLinkProps = {
href?: string,
title?: string,
children?: React.Node,
noDataStore?: boolean,
};
type MarkdownProps = {
strip?: boolean,
content: ?string,
promptLinks?: boolean,
noDataStore?: boolean,
2020-09-17 11:32:31 -04:00
className?: string,
};
const SimpleText = (props: SimpleTextProps) => {
return <span>{props.children}</span>;
};
const SimpleLink = (props: SimpleLinkProps) => {
const { title, children } = props;
const { href, noDataStore } = props;
if (!href) {
return children || null;
}
if (!href.startsWith('lbry://')) {
return (
<a href={href} title={title} target={'_blank'} rel={'noreferrer noopener'}>
{children}
</a>
);
}
const [uri, search] = href.split('?');
const urlParams = new URLSearchParams(search);
const embed = urlParams.get('embed');
if (embed) {
2020-07-30 17:20:18 -04:00
// Decode this since users might just copy it from the url bar
const decodedUri = decodeURI(uri);
return noDataStore ? (
<div className="embed__inline-button-preview">
<pre>{decodedUri}</pre>
</div>
) : (
<EmbedPlayButton uri={decodedUri} />
);
}
const webLink = formatLbryUrlForWeb(uri);
// using Link after formatLbryUrl to handle "/" vs "#/"
// for web and desktop scenarios respectively
return noDataStore ? (
// Dummy link (no 'href')
<a title={title}>{children}</a>
) : (
<Link
title={title}
to={webLink}
onClick={e => {
e.stopPropagation();
}}
>
{children}
</Link>
);
};
// Use github sanitation schema
const schema = { ...defaultSchema };
// Extend sanitation schema to support lbry protocol
schema.protocols.href.push('lbry');
schema.attributes.a.push('embed');
2020-04-29 17:37:14 -04:00
const REPLACE_REGEX = /(<iframe\s+src=["'])(.*?(?=))(["']\s*><\/iframe>)/g;
const MarkdownPreview = (props: MarkdownProps) => {
2020-09-17 11:32:31 -04:00
const { content, strip, promptLinks, noDataStore, className } = props;
const strippedContent = content
2020-04-29 17:37:14 -04:00
? content.replace(REPLACE_REGEX, (iframeHtml, y, iframeSrc) => {
// Let the browser try to create an iframe to see if the markup is valid
const outer = document.createElement('div');
outer.innerHTML = iframeHtml;
const iframe = ((outer.querySelector('iframe'): any): ?HTMLIFrameElement);
if (iframe) {
const src = iframe.src;
if (src && src.startsWith('lbry://')) {
return `${src}?embed=true`;
}
}
return iframeHtml;
})
: '';
const remarkOptions: Object = {
sanitize: schema,
fragment: React.Fragment,
remarkReactComponents: {
a: promptLinks ? ExternalLink : linkProps => <SimpleLink {...linkProps} noDataStore={noDataStore} />,
// Workaraund of remarkOptions.Fragment
div: React.Fragment,
},
};
const remarkAttrOpts = {
scope: 'extended',
elements: ['link'],
extend: { link: ['embed'] },
defaultValue: true,
};
// Strip all content and just render text
if (strip) {
// Remove new lines and extra space
remarkOptions.remarkReactComponents.p = SimpleText;
return (
<span dir="auto" className="markdown-preview">
{
remark()
.use(remarkStrip)
.use(reactRenderer, remarkOptions)
.processSync(content).contents
}
</span>
);
}
return (
2020-09-17 11:32:31 -04:00
<div dir="auto" className={classnames('markdown-preview', className)}>
{
remark()
.use(remarkAttr, remarkAttrOpts)
// Remark plugins for lbry urls
// Note: The order is important
.use(formatedLinks)
.use(inlineLinks)
// Emojis
.use(remarkEmoji)
// Render new lines without needing spaces.
.use(remarkBreaks)
.use(reactRenderer, remarkOptions)
.processSync(strippedContent).contents
}
</div>
);
2019-11-07 14:39:22 -05:00
};
export default MarkdownPreview;