forked from Ivasoft/mattermost-mobile
* Convert value to string
* Add default value prop
(cherry picked from commit ed4b100962)
Co-authored-by: Miguel Alatzar <migbot@users.noreply.github.com>
498 lines
15 KiB
JavaScript
498 lines
15 KiB
JavaScript
// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
|
|
// See LICENSE.txt for license information.
|
|
|
|
import {Parser, Node} from 'commonmark';
|
|
import Renderer from 'commonmark-react-renderer';
|
|
import React, {PureComponent} from 'react';
|
|
import PropTypes from 'prop-types';
|
|
import {
|
|
Platform,
|
|
Text,
|
|
View,
|
|
} from 'react-native';
|
|
|
|
import AtMention from 'app/components/at_mention';
|
|
import ChannelLink from 'app/components/channel_link';
|
|
import Emoji from 'app/components/emoji';
|
|
import FormattedText from 'app/components/formatted_text';
|
|
import Hashtag from 'app/components/markdown/hashtag';
|
|
import CustomPropTypes from 'app/constants/custom_prop_types';
|
|
import {blendColors, concatStyles, makeStyleSheetFromTheme} from 'app/utils/theme';
|
|
import {getScheme} from 'app/utils/url';
|
|
|
|
import MarkdownBlockQuote from './markdown_block_quote';
|
|
import MarkdownCodeBlock from './markdown_code_block';
|
|
import MarkdownImage from './markdown_image';
|
|
import MarkdownLink from './markdown_link';
|
|
import MarkdownList from './markdown_list';
|
|
import MarkdownListItem from './markdown_list_item';
|
|
import MarkdownTable from './markdown_table';
|
|
import MarkdownTableImage from './markdown_table_image';
|
|
import MarkdownTableRow from './markdown_table_row';
|
|
import MarkdownTableCell from './markdown_table_cell';
|
|
import {
|
|
addListItemIndices,
|
|
combineTextNodes,
|
|
highlightMentions,
|
|
pullOutImages,
|
|
} from './transform';
|
|
|
|
export default class Markdown extends PureComponent {
|
|
static propTypes = {
|
|
autolinkedUrlSchemes: PropTypes.array.isRequired,
|
|
baseTextStyle: CustomPropTypes.Style,
|
|
blockStyles: PropTypes.object,
|
|
channelMentions: PropTypes.object,
|
|
imagesMetadata: PropTypes.object,
|
|
isEdited: PropTypes.bool,
|
|
isReplyPost: PropTypes.bool,
|
|
isSearchResult: PropTypes.bool,
|
|
mentionKeys: PropTypes.array.isRequired,
|
|
minimumHashtagLength: PropTypes.number.isRequired,
|
|
onChannelLinkPress: PropTypes.func,
|
|
onHashtagPress: PropTypes.func,
|
|
onPermalinkPress: PropTypes.func,
|
|
onPostPress: PropTypes.func,
|
|
postId: PropTypes.string,
|
|
textStyles: PropTypes.object,
|
|
theme: PropTypes.object.isRequired,
|
|
value: PropTypes.string.isRequired,
|
|
disableHashtags: PropTypes.bool,
|
|
disableAtMentions: PropTypes.bool,
|
|
disableChannelLink: PropTypes.bool,
|
|
disableAtChannelMentionHighlight: PropTypes.bool,
|
|
disableGallery: PropTypes.bool,
|
|
};
|
|
|
|
static defaultProps = {
|
|
textStyles: {},
|
|
blockStyles: {},
|
|
onLongPress: () => true,
|
|
disableHashtags: false,
|
|
disableAtMentions: false,
|
|
disableChannelLink: false,
|
|
disableAtChannelMentionHighlight: false,
|
|
disableGallery: false,
|
|
value: '',
|
|
};
|
|
|
|
constructor(props) {
|
|
super(props);
|
|
|
|
this.parser = this.createParser();
|
|
this.renderer = this.createRenderer();
|
|
}
|
|
|
|
createParser = () => {
|
|
return new Parser({
|
|
urlFilter: this.urlFilter,
|
|
minimumHashtagLength: this.props.minimumHashtagLength,
|
|
});
|
|
};
|
|
|
|
urlFilter = (url) => {
|
|
const scheme = getScheme(url);
|
|
|
|
return !scheme || this.props.autolinkedUrlSchemes.indexOf(scheme) !== -1;
|
|
};
|
|
|
|
createRenderer = () => {
|
|
return new Renderer({
|
|
renderers: {
|
|
text: this.renderText,
|
|
|
|
emph: Renderer.forwardChildren,
|
|
strong: Renderer.forwardChildren,
|
|
del: Renderer.forwardChildren,
|
|
code: this.renderCodeSpan,
|
|
link: this.renderLink,
|
|
image: this.renderImage,
|
|
atMention: this.renderAtMention,
|
|
channelLink: this.renderChannelLink,
|
|
emoji: this.renderEmoji,
|
|
hashtag: this.renderHashtag,
|
|
|
|
paragraph: this.renderParagraph,
|
|
heading: this.renderHeading,
|
|
codeBlock: this.renderCodeBlock,
|
|
blockQuote: this.renderBlockQuote,
|
|
|
|
list: this.renderList,
|
|
item: this.renderListItem,
|
|
|
|
hardBreak: this.renderHardBreak,
|
|
thematicBreak: this.renderThematicBreak,
|
|
softBreak: this.renderSoftBreak,
|
|
|
|
htmlBlock: this.renderHtml,
|
|
htmlInline: this.renderHtml,
|
|
|
|
table: this.renderTable,
|
|
table_row: this.renderTableRow,
|
|
table_cell: this.renderTableCell,
|
|
|
|
mention_highlight: Renderer.forwardChildren,
|
|
|
|
editedIndicator: this.renderEditedIndicator,
|
|
},
|
|
renderParagraphsInLists: true,
|
|
getExtraPropsForNode: this.getExtraPropsForNode,
|
|
});
|
|
};
|
|
|
|
getExtraPropsForNode = (node) => {
|
|
const extraProps = {
|
|
continue: node.continue,
|
|
index: node.index,
|
|
};
|
|
|
|
if (node.type === 'image') {
|
|
extraProps.reactChildren = node.react.children;
|
|
extraProps.linkDestination = node.linkDestination;
|
|
}
|
|
|
|
return extraProps;
|
|
};
|
|
|
|
computeTextStyle = (baseStyle, context) => {
|
|
return concatStyles(baseStyle, context.map((type) => this.props.textStyles[type]));
|
|
};
|
|
|
|
renderText = ({context, literal}) => {
|
|
if (context.indexOf('image') !== -1) {
|
|
// If this text is displayed, it will be styled by the image component
|
|
return (
|
|
<Text testID='markdown_text'>
|
|
{literal}
|
|
</Text>
|
|
);
|
|
}
|
|
|
|
// Construct the text style based off of the parents of this node since RN's inheritance is limited
|
|
const style = this.computeTextStyle(this.props.baseTextStyle, context);
|
|
|
|
return (
|
|
<Text
|
|
testID='markdown_text'
|
|
style={style}
|
|
>
|
|
{literal}
|
|
</Text>
|
|
);
|
|
};
|
|
|
|
renderCodeSpan = ({context, literal}) => {
|
|
return <Text style={this.computeTextStyle([this.props.baseTextStyle, this.props.textStyles.code], context)}>{literal}</Text>;
|
|
};
|
|
|
|
renderImage = ({linkDestination, reactChildren, context, src}) => {
|
|
if (!this.props.imagesMetadata) {
|
|
return null;
|
|
}
|
|
|
|
if (context.indexOf('table') !== -1) {
|
|
// We have enough problems rendering images as is, so just render a link inside of a table
|
|
return (
|
|
<MarkdownTableImage
|
|
disable={this.props.disableGallery}
|
|
imagesMetadata={this.props.imagesMetadata}
|
|
postId={this.props.postId}
|
|
source={src}
|
|
>
|
|
{reactChildren}
|
|
</MarkdownTableImage>
|
|
);
|
|
}
|
|
|
|
return (
|
|
<MarkdownImage
|
|
disable={this.props.disableGallery}
|
|
errorTextStyle={[this.computeTextStyle(this.props.baseTextStyle, context), this.props.textStyles.error]}
|
|
linkDestination={linkDestination}
|
|
imagesMetadata={this.props.imagesMetadata}
|
|
isReplyPost={this.props.isReplyPost}
|
|
postId={this.props.postId}
|
|
source={src}
|
|
>
|
|
{reactChildren}
|
|
</MarkdownImage>
|
|
);
|
|
};
|
|
|
|
renderAtMention = ({context, mentionName}) => {
|
|
if (this.props.disableAtMentions) {
|
|
return this.renderText({context, literal: `@${mentionName}`});
|
|
}
|
|
|
|
const style = getStyleSheet(this.props.theme);
|
|
|
|
return (
|
|
<AtMention
|
|
mentionStyle={this.props.textStyles.mention}
|
|
textStyle={[this.computeTextStyle(this.props.baseTextStyle, context), style.atMentionOpacity]}
|
|
isSearchResult={this.props.isSearchResult}
|
|
mentionName={mentionName}
|
|
onPostPress={this.props.onPostPress}
|
|
mentionKeys={this.props.mentionKeys}
|
|
/>
|
|
);
|
|
};
|
|
|
|
renderChannelLink = ({context, channelName}) => {
|
|
if (this.props.disableChannelLink) {
|
|
return this.renderText({context, literal: `~${channelName}`});
|
|
}
|
|
|
|
return (
|
|
<ChannelLink
|
|
linkStyle={this.props.textStyles.link}
|
|
textStyle={this.computeTextStyle(this.props.baseTextStyle, context)}
|
|
onChannelLinkPress={this.props.onChannelLinkPress}
|
|
channelName={channelName}
|
|
channelMentions={this.props.channelMentions}
|
|
/>
|
|
);
|
|
};
|
|
|
|
renderEmoji = ({context, emojiName, literal}) => {
|
|
return (
|
|
<Emoji
|
|
emojiName={emojiName}
|
|
literal={literal}
|
|
textStyle={this.computeTextStyle(this.props.baseTextStyle, context)}
|
|
/>
|
|
);
|
|
};
|
|
|
|
renderHashtag = ({context, hashtag}) => {
|
|
if (this.props.disableHashtags) {
|
|
return this.renderText({context, literal: `#${hashtag}`});
|
|
}
|
|
|
|
return (
|
|
<Hashtag
|
|
hashtag={hashtag}
|
|
linkStyle={this.props.textStyles.link}
|
|
onHashtagPress={this.props.onHashtagPress}
|
|
/>
|
|
);
|
|
};
|
|
|
|
renderParagraph = ({children, first}) => {
|
|
if (!children || children.length === 0) {
|
|
return null;
|
|
}
|
|
|
|
const style = getStyleSheet(this.props.theme);
|
|
const blockStyle = [style.block];
|
|
if (!first) {
|
|
blockStyle.push(this.props.blockStyles.adjacentParagraph);
|
|
}
|
|
return (
|
|
<View style={blockStyle}>
|
|
<Text>
|
|
{children}
|
|
</Text>
|
|
</View>
|
|
);
|
|
};
|
|
|
|
renderHeading = ({children, level}) => {
|
|
const containerStyle = [
|
|
getStyleSheet(this.props.theme).block,
|
|
this.props.blockStyles[`heading${level}`],
|
|
];
|
|
const textStyle = this.props.blockStyles[`heading${level}Text`];
|
|
return (
|
|
<View style={containerStyle}>
|
|
<Text style={textStyle}>
|
|
{children}
|
|
</Text>
|
|
</View>
|
|
);
|
|
};
|
|
|
|
renderCodeBlock = (props) => {
|
|
// These sometimes include a trailing newline
|
|
const content = props.literal.replace(/\n$/, '');
|
|
|
|
return (
|
|
<MarkdownCodeBlock
|
|
content={content}
|
|
language={props.language}
|
|
textStyle={this.props.textStyles.codeBlock}
|
|
/>
|
|
);
|
|
};
|
|
|
|
renderBlockQuote = ({children, ...otherProps}) => {
|
|
return (
|
|
<MarkdownBlockQuote
|
|
iconStyle={this.props.blockStyles.quoteBlockIcon}
|
|
{...otherProps}
|
|
>
|
|
{children}
|
|
</MarkdownBlockQuote>
|
|
);
|
|
};
|
|
|
|
renderList = ({children, start, tight, type}) => {
|
|
return (
|
|
<MarkdownList
|
|
ordered={type !== 'bullet'}
|
|
start={start}
|
|
tight={tight}
|
|
>
|
|
{children}
|
|
</MarkdownList>
|
|
);
|
|
};
|
|
|
|
renderListItem = ({children, context, ...otherProps}) => {
|
|
const level = context.filter((type) => type === 'list').length;
|
|
|
|
return (
|
|
<MarkdownListItem
|
|
bulletStyle={this.props.baseTextStyle}
|
|
level={level}
|
|
{...otherProps}
|
|
>
|
|
{children}
|
|
</MarkdownListItem>
|
|
);
|
|
};
|
|
|
|
renderHardBreak = () => {
|
|
return <Text>{'\n'}</Text>;
|
|
};
|
|
|
|
renderThematicBreak = () => {
|
|
return <View style={this.props.blockStyles.horizontalRule}/>;
|
|
};
|
|
|
|
renderSoftBreak = () => {
|
|
return <Text>{'\n'}</Text>;
|
|
};
|
|
|
|
renderHtml = (props) => {
|
|
let rendered = this.renderText(props);
|
|
|
|
if (props.isBlock) {
|
|
const style = getStyleSheet(this.props.theme);
|
|
|
|
rendered = (
|
|
<View style={style.block}>
|
|
{rendered}
|
|
</View>
|
|
);
|
|
}
|
|
|
|
return rendered;
|
|
};
|
|
|
|
renderTable = ({children, numColumns}) => {
|
|
return (
|
|
<MarkdownTable
|
|
numColumns={numColumns}
|
|
>
|
|
{children}
|
|
</MarkdownTable>
|
|
);
|
|
};
|
|
|
|
renderTableRow = (args) => {
|
|
return <MarkdownTableRow {...args}/>;
|
|
};
|
|
|
|
renderTableCell = (args) => {
|
|
return <MarkdownTableCell {...args}/>;
|
|
};
|
|
|
|
renderLink = ({children, href}) => {
|
|
return (
|
|
<MarkdownLink
|
|
href={href}
|
|
onPermalinkPress={this.props.onPermalinkPress}
|
|
>
|
|
{children}
|
|
</MarkdownLink>
|
|
);
|
|
};
|
|
|
|
renderEditedIndicator = ({context}) => {
|
|
let spacer = '';
|
|
if (context[0] === 'paragraph') {
|
|
spacer = ' ';
|
|
}
|
|
|
|
const style = getStyleSheet(this.props.theme);
|
|
const styles = [
|
|
this.props.baseTextStyle,
|
|
style.editedIndicatorText,
|
|
];
|
|
|
|
return (
|
|
<Text
|
|
style={styles}
|
|
>
|
|
{spacer}
|
|
<FormattedText
|
|
id='post_message_view.edited'
|
|
defaultMessage='(edited)'
|
|
/>
|
|
</Text>
|
|
);
|
|
};
|
|
|
|
render() {
|
|
let ast = this.parser.parse(this.props.value.toString());
|
|
|
|
ast = combineTextNodes(ast);
|
|
ast = addListItemIndices(ast);
|
|
ast = pullOutImages(ast);
|
|
ast = highlightMentions(ast, this.props.mentionKeys);
|
|
|
|
if (this.props.isEdited) {
|
|
const editIndicatorNode = new Node('edited_indicator');
|
|
if (ast.lastChild && ['heading', 'paragraph'].includes(ast.lastChild.type)) {
|
|
ast.lastChild.appendChild(editIndicatorNode);
|
|
} else {
|
|
const node = new Node('paragraph');
|
|
node.appendChild(editIndicatorNode);
|
|
|
|
ast.appendChild(node);
|
|
}
|
|
}
|
|
|
|
return this.renderer.render(ast);
|
|
}
|
|
}
|
|
|
|
const getStyleSheet = makeStyleSheetFromTheme((theme) => {
|
|
// Android has trouble giving text transparency depending on how it's nested,
|
|
// so we calculate the resulting colour manually
|
|
const editedOpacity = Platform.select({
|
|
ios: 0.3,
|
|
android: 1.0,
|
|
});
|
|
const editedColor = Platform.select({
|
|
ios: theme.centerChannelColor,
|
|
android: blendColors(theme.centerChannelBg, theme.centerChannelColor, 0.3),
|
|
});
|
|
|
|
return {
|
|
block: {
|
|
alignItems: 'flex-start',
|
|
flexDirection: 'row',
|
|
flexWrap: 'wrap',
|
|
},
|
|
editedIndicatorText: {
|
|
color: editedColor,
|
|
opacity: editedOpacity,
|
|
},
|
|
atMentionOpacity: {
|
|
opacity: 1,
|
|
},
|
|
};
|
|
});
|