2016-06-11 18:54:09 +02:00
|
|
|
import {
|
|
|
|
Editor,
|
|
|
|
Modifier,
|
|
|
|
ContentState,
|
|
|
|
convertFromHTML,
|
|
|
|
DefaultDraftBlockRenderMap,
|
|
|
|
DefaultDraftInlineStyle,
|
|
|
|
CompositeDecorator
|
|
|
|
} from 'draft-js';
|
2016-06-11 12:22:08 +02:00
|
|
|
import * as sdk from './index';
|
2016-05-27 06:45:55 +02:00
|
|
|
|
2016-05-28 08:28:22 +02:00
|
|
|
const BLOCK_RENDER_MAP = DefaultDraftBlockRenderMap.set('unstyled', {
|
2016-06-15 17:04:37 +02:00
|
|
|
element: 'span'
|
|
|
|
/*
|
|
|
|
draft uses <div> by default which we don't really like, so we're using <span>
|
|
|
|
this is probably not a good idea since <span> is not a block level element but
|
|
|
|
we're trying to fix things in contentStateToHTML below
|
|
|
|
*/
|
2016-05-28 08:28:22 +02:00
|
|
|
});
|
|
|
|
|
2016-06-11 18:54:09 +02:00
|
|
|
const STYLES = {
|
2016-05-27 06:45:55 +02:00
|
|
|
BOLD: 'strong',
|
|
|
|
CODE: 'code',
|
|
|
|
ITALIC: 'em',
|
|
|
|
STRIKETHROUGH: 's',
|
|
|
|
UNDERLINE: 'u'
|
|
|
|
};
|
|
|
|
|
2016-06-12 00:52:30 +02:00
|
|
|
const MARKDOWN_REGEX = {
|
|
|
|
LINK: /(?:\[([^\]]+)\]\(([^\)]+)\))|\<(\w+:\/\/[^\>]+)\>/g,
|
|
|
|
ITALIC: /([\*_])([\w\s]+?)\1/g,
|
|
|
|
BOLD: /([\*_])\1([\w\s]+?)\1\1/g
|
|
|
|
};
|
|
|
|
|
|
|
|
const USERNAME_REGEX = /@\S+:\S+/g;
|
|
|
|
const ROOM_REGEX = /#\S+:\S+/g;
|
|
|
|
|
2016-06-11 12:22:08 +02:00
|
|
|
export function contentStateToHTML(contentState: ContentState): string {
|
|
|
|
return contentState.getBlockMap().map((block) => {
|
|
|
|
let elem = BLOCK_RENDER_MAP.get(block.getType()).element;
|
|
|
|
let content = [];
|
2016-06-11 18:54:09 +02:00
|
|
|
block.findStyleRanges(
|
|
|
|
() => true, // always return true => don't filter any ranges out
|
|
|
|
(start, end) => {
|
|
|
|
// map style names to elements
|
2016-06-14 15:40:35 +02:00
|
|
|
let tags = block.getInlineStyleAt(start).map(style => STYLES[style]).filter(style => !!style);
|
2016-06-11 18:54:09 +02:00
|
|
|
// combine them to get well-nested HTML
|
|
|
|
let open = tags.map(tag => `<${tag}>`).join('');
|
|
|
|
let close = tags.map(tag => `</${tag}>`).reverse().join('');
|
|
|
|
// and get the HTML representation of this styled range (this .substring() should never fail)
|
2016-06-15 16:54:37 +02:00
|
|
|
let text = block.getText().substring(start, end);
|
|
|
|
// http://shebang.brandonmintern.com/foolproof-html-escaping-in-javascript/
|
|
|
|
let div = document.createElement('div');
|
|
|
|
div.appendChild(document.createTextNode(text));
|
|
|
|
let safeText = div.innerHTML;
|
|
|
|
content.push(`${open}${safeText}${close}`);
|
2016-06-11 18:54:09 +02:00
|
|
|
}
|
|
|
|
);
|
2016-05-27 06:45:55 +02:00
|
|
|
|
2016-06-15 16:54:37 +02:00
|
|
|
let result = `<${elem}>${content.join('')}</${elem}>`;
|
|
|
|
|
|
|
|
// dirty hack because we don't want block level tags by default, but breaks
|
|
|
|
if(elem === 'span')
|
|
|
|
result += '<br />';
|
|
|
|
return result;
|
2016-05-27 06:45:55 +02:00
|
|
|
}).join('');
|
|
|
|
}
|
|
|
|
|
2016-06-11 18:54:09 +02:00
|
|
|
export function HTMLtoContentState(html: string): ContentState {
|
2016-05-27 06:45:55 +02:00
|
|
|
return ContentState.createFromBlockArray(convertFromHTML(html));
|
|
|
|
}
|
2016-06-09 20:23:09 +02:00
|
|
|
|
2016-06-11 12:22:08 +02:00
|
|
|
/**
|
|
|
|
* Returns a composite decorator which has access to provided scope.
|
|
|
|
*/
|
2016-06-11 23:13:57 +02:00
|
|
|
export function getScopedRTDecorators(scope: any): CompositeDecorator {
|
2016-06-11 18:54:09 +02:00
|
|
|
let MemberAvatar = sdk.getComponent('avatars.MemberAvatar');
|
2016-06-09 20:23:09 +02:00
|
|
|
|
2016-06-11 18:54:09 +02:00
|
|
|
let usernameDecorator = {
|
2016-06-09 20:23:09 +02:00
|
|
|
strategy: (contentBlock, callback) => {
|
|
|
|
findWithRegex(USERNAME_REGEX, contentBlock, callback);
|
|
|
|
},
|
|
|
|
component: (props) => {
|
2016-06-11 12:22:08 +02:00
|
|
|
let member = scope.room.getMember(props.children[0].props.text);
|
2016-06-14 15:40:35 +02:00
|
|
|
// unused until we make these decorators immutable (autocomplete needed)
|
|
|
|
let name = member ? member.name : null;
|
2016-06-11 18:54:09 +02:00
|
|
|
let avatar = member ? <MemberAvatar member={member} width={16} height={16}/> : null;
|
2016-06-09 20:23:09 +02:00
|
|
|
return <span className="mx_UserPill">{avatar} {props.children}</span>;
|
|
|
|
}
|
|
|
|
};
|
2016-06-11 18:54:09 +02:00
|
|
|
let roomDecorator = {
|
2016-06-09 20:23:09 +02:00
|
|
|
strategy: (contentBlock, callback) => {
|
|
|
|
findWithRegex(ROOM_REGEX, contentBlock, callback);
|
|
|
|
},
|
|
|
|
component: (props) => {
|
|
|
|
return <span className="mx_RoomPill">{props.children}</span>;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2016-06-11 23:13:57 +02:00
|
|
|
return [usernameDecorator, roomDecorator];
|
2016-06-09 20:23:09 +02:00
|
|
|
}
|
|
|
|
|
2016-06-11 23:13:57 +02:00
|
|
|
export function getScopedMDDecorators(scope: any): CompositeDecorator {
|
|
|
|
let markdownDecorators = ['BOLD', 'ITALIC'].map(
|
|
|
|
(style) => ({
|
|
|
|
strategy: (contentBlock, callback) => {
|
|
|
|
return findWithRegex(MARKDOWN_REGEX[style], contentBlock, callback);
|
|
|
|
},
|
|
|
|
component: (props) => (
|
|
|
|
<span className={"mx_MarkdownElement mx_Markdown_" + style}>
|
|
|
|
{props.children}
|
|
|
|
</span>
|
|
|
|
)
|
|
|
|
}));
|
|
|
|
|
|
|
|
markdownDecorators.push({
|
|
|
|
strategy: (contentBlock, callback) => {
|
|
|
|
return findWithRegex(MARKDOWN_REGEX.LINK, contentBlock, callback);
|
|
|
|
},
|
|
|
|
component: (props) => (
|
|
|
|
<a href="#" className="mx_MarkdownElement mx_Markdown_LINK">
|
|
|
|
{props.children}
|
|
|
|
</a>
|
|
|
|
)
|
|
|
|
});
|
|
|
|
|
|
|
|
return markdownDecorators;
|
|
|
|
}
|
|
|
|
|
2016-06-11 18:54:09 +02:00
|
|
|
/**
|
|
|
|
* Utility function that looks for regex matches within a ContentBlock and invokes {callback} with (start, end)
|
|
|
|
* From https://facebook.github.io/draft-js/docs/advanced-topics-decorators.html
|
|
|
|
*/
|
|
|
|
function findWithRegex(regex, contentBlock: ContentBlock, callback: (start: number, end: number) => any) {
|
2016-06-09 20:23:09 +02:00
|
|
|
const text = contentBlock.getText();
|
|
|
|
let matchArr, start;
|
|
|
|
while ((matchArr = regex.exec(text)) !== null) {
|
|
|
|
start = matchArr.index;
|
|
|
|
callback(start, start + matchArr[0].length);
|
|
|
|
}
|
|
|
|
}
|
2016-06-11 18:54:09 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Passes rangeToReplace to modifyFn and replaces it in contentState with the result.
|
|
|
|
*/
|
2016-06-12 00:50:30 +02:00
|
|
|
export function modifyText(contentState: ContentState, rangeToReplace: SelectionState,
|
2016-06-14 15:40:35 +02:00
|
|
|
modifyFn: (text: string) => string, inlineStyle, entityKey): ContentState {
|
2016-06-12 00:50:30 +02:00
|
|
|
let getText = (key) => contentState.getBlockForKey(key).getText(),
|
|
|
|
startKey = rangeToReplace.getStartKey(),
|
|
|
|
startOffset = rangeToReplace.getStartOffset(),
|
|
|
|
endKey = rangeToReplace.getEndKey(),
|
|
|
|
endOffset = rangeToReplace.getEndOffset(),
|
2016-06-11 18:54:09 +02:00
|
|
|
text = "";
|
|
|
|
|
2016-06-12 00:50:30 +02:00
|
|
|
|
|
|
|
for(let currentKey = startKey;
|
|
|
|
currentKey && currentKey !== endKey;
|
|
|
|
currentKey = contentState.getKeyAfter(currentKey)) {
|
2016-06-14 15:58:51 +02:00
|
|
|
let blockText = getText(currentKey);
|
|
|
|
text += blockText.substring(startOffset, blockText.length);
|
2016-06-12 00:50:30 +02:00
|
|
|
|
|
|
|
// from now on, we'll take whole blocks
|
|
|
|
startOffset = 0;
|
2016-06-11 18:54:09 +02:00
|
|
|
}
|
|
|
|
|
2016-06-12 00:50:30 +02:00
|
|
|
// add remaining part of last block
|
|
|
|
text += getText(endKey).substring(startOffset, endOffset);
|
|
|
|
|
2016-06-14 15:40:35 +02:00
|
|
|
return Modifier.replaceText(contentState, rangeToReplace, modifyFn(text), inlineStyle, entityKey);
|
2016-06-11 18:54:09 +02:00
|
|
|
}
|