misskey/src/mfm/parse/index.ts

105 lines
2.6 KiB
TypeScript
Raw Normal View History

2016-12-29 07:49:51 +09:00
/**
* Misskey Text Analyzer
*/
2018-06-18 14:28:43 +09:00
import { TextElementBold } from './elements/bold';
2018-08-03 23:27:37 +09:00
import { TextElementBig } from './elements/big';
2018-06-18 14:28:43 +09:00
import { TextElementCode } from './elements/code';
import { TextElementEmoji } from './elements/emoji';
import { TextElementHashtag } from './elements/hashtag';
import { TextElementInlineCode } from './elements/inline-code';
import { TextElementLink } from './elements/link';
import { TextElementMention } from './elements/mention';
import { TextElementQuote } from './elements/quote';
import { TextElementSearch } from './elements/search';
import { TextElementTitle } from './elements/title';
import { TextElementUrl } from './elements/url';
2018-08-05 12:33:51 +09:00
import { TextElementMotion } from './elements/motion';
2018-06-17 19:55:39 +09:00
2016-12-29 07:49:51 +09:00
const elements = [
2018-08-03 23:27:37 +09:00
require('./elements/big'),
2016-12-29 07:49:51 +09:00
require('./elements/bold'),
2018-04-19 15:05:39 +09:00
require('./elements/title'),
2016-12-29 07:49:51 +09:00
require('./elements/url'),
2017-03-18 01:16:32 +09:00
require('./elements/link'),
2016-12-29 07:49:51 +09:00
require('./elements/mention'),
2017-02-09 01:07:06 +09:00
require('./elements/hashtag'),
2017-02-11 23:41:57 +09:00
require('./elements/code'),
require('./elements/inline-code'),
2018-02-27 12:32:01 +09:00
require('./elements/quote'),
2018-04-22 07:21:51 +09:00
require('./elements/emoji'),
2018-08-05 12:33:51 +09:00
require('./elements/search'),
require('./elements/motion')
2018-06-17 19:55:39 +09:00
].map(element => element.default as TextElementProcessor);
2018-06-18 14:28:43 +09:00
export type TextElement = { type: 'text', content: string }
2018-06-17 19:55:39 +09:00
| TextElementBold
2018-08-03 23:27:37 +09:00
| TextElementBig
2018-06-17 19:55:39 +09:00
| TextElementCode
| TextElementEmoji
| TextElementHashtag
| TextElementInlineCode
| TextElementLink
| TextElementMention
| TextElementQuote
| TextElementSearch
| TextElementTitle
2018-08-05 12:33:51 +09:00
| TextElementUrl
| TextElementMotion;
2018-11-03 22:40:12 +09:00
export type TextElementProcessor = (text: string, isBegin: boolean) => TextElement | TextElement[];
2016-12-29 07:49:51 +09:00
2018-06-17 19:55:39 +09:00
export default (source: string): TextElement[] => {
2018-07-01 13:46:34 +09:00
if (source == null || source == '') {
2016-12-29 07:49:51 +09:00
return null;
}
2018-06-17 19:55:39 +09:00
const tokens: TextElement[] = [];
2016-12-29 07:49:51 +09:00
2018-06-17 19:55:39 +09:00
function push(token: TextElement) {
2016-12-29 07:49:51 +09:00
if (token != null) {
tokens.push(token);
source = source.substr(token.content.length);
}
}
let i = 0;
// パース
while (source != '') {
const parsed = elements.some(el => {
2018-11-03 22:40:12 +09:00
let _tokens = el(source, i == 0);
2018-02-27 12:32:01 +09:00
if (_tokens) {
if (!Array.isArray(_tokens)) {
_tokens = [_tokens];
2016-12-29 07:49:51 +09:00
}
2018-02-27 12:32:01 +09:00
_tokens.forEach(push);
2016-12-29 07:49:51 +09:00
return true;
2017-03-18 20:05:11 +09:00
} else {
return false;
2016-12-29 07:49:51 +09:00
}
});
if (!parsed) {
push({
type: 'text',
content: source[0]
});
}
i++;
}
// テキストを纏める
return tokens.reduce((a, b) => {
2018-06-17 19:55:39 +09:00
if (a.length && a[a.length - 1].type == 'text' && b.type == 'text') {
2016-12-29 07:49:51 +09:00
const tail = a.pop();
return a.concat({
type: 'text',
content: tail.content + b.content
});
} else {
return a.concat(b);
}
2018-06-17 19:55:39 +09:00
}, [] as TextElement[]);
2017-03-18 20:05:11 +09:00
};