securemark 0.294.7 → 0.294.9
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +8 -0
- package/dist/index.js +185 -205
- package/markdown.d.ts +13 -36
- package/package.json +1 -1
- package/src/combinator/control/constraint/block.ts +2 -2
- package/src/combinator/control/constraint/line.ts +7 -5
- package/src/combinator/control/manipulation/convert.ts +2 -1
- package/src/combinator/control/manipulation/fence.ts +4 -4
- package/src/combinator/control/manipulation/surround.ts +9 -9
- package/src/combinator/data/parser/some.ts +1 -1
- package/src/combinator/data/parser/union.ts +6 -2
- package/src/parser/api/bind.test.ts +0 -1
- package/src/parser/api/normalize.test.ts +5 -8
- package/src/parser/api/normalize.ts +4 -2
- package/src/parser/autolink.ts +1 -2
- package/src/parser/block/blockquote.ts +3 -3
- package/src/parser/block/extension/fig.ts +4 -1
- package/src/parser/block/heading.ts +12 -2
- package/src/parser/block/reply/quote.ts +1 -2
- package/src/parser/block/ulist.ts +1 -1
- package/src/parser/block.ts +0 -4
- package/src/parser/header.ts +28 -40
- package/src/parser/inline/annotation.ts +2 -3
- package/src/parser/inline/autolink/account.ts +49 -17
- package/src/parser/inline/autolink/anchor.test.ts +0 -1
- package/src/parser/inline/autolink/anchor.ts +16 -15
- package/src/parser/inline/autolink/email.test.ts +1 -1
- package/src/parser/inline/autolink/email.ts +10 -11
- package/src/parser/inline/autolink/hashnum.ts +17 -13
- package/src/parser/inline/autolink/hashtag.ts +19 -15
- package/src/parser/inline/autolink/url.ts +24 -19
- package/src/parser/inline/autolink.ts +1 -2
- package/src/parser/inline/bracket.ts +14 -14
- package/src/parser/inline/deletion.ts +2 -1
- package/src/parser/inline/emphasis.ts +2 -1
- package/src/parser/inline/emstrong.ts +2 -1
- package/src/parser/inline/extension/index.ts +4 -4
- package/src/parser/inline/extension/indexer.ts +1 -1
- package/src/parser/inline/extension/label.ts +1 -1
- package/src/parser/inline/extension/placeholder.ts +4 -3
- package/src/parser/inline/html.ts +4 -4
- package/src/parser/inline/htmlentity.ts +2 -2
- package/src/parser/inline/insertion.ts +2 -1
- package/src/parser/inline/italic.ts +2 -1
- package/src/parser/inline/link.ts +12 -24
- package/src/parser/inline/mark.ts +2 -1
- package/src/parser/inline/math.ts +4 -2
- package/src/parser/inline/media.ts +28 -33
- package/src/parser/inline/reference.ts +4 -4
- package/src/parser/inline/remark.ts +2 -1
- package/src/parser/inline/ruby.ts +3 -4
- package/src/parser/inline/strong.ts +2 -1
- package/src/parser/inline/template.ts +10 -10
- package/src/parser/segment.ts +2 -2
- package/src/parser/source/escapable.ts +3 -4
- package/src/parser/source/line.ts +3 -1
- package/src/parser/source/text.ts +5 -10
- package/src/parser/source/unescapable.ts +2 -4
- package/src/parser/source.ts +1 -2
- package/src/parser/inline/autolink/channel.ts +0 -44
|
@@ -1,25 +1,57 @@
|
|
|
1
1
|
import { AutolinkParser } from '../../inline';
|
|
2
2
|
import { State, Backtrack } from '../../context';
|
|
3
3
|
import { List, Data } from '../../../combinator/data/parser';
|
|
4
|
-
import {
|
|
5
|
-
import {
|
|
4
|
+
import { some, state, constraint, verify, surround, lazy } from '../../../combinator';
|
|
5
|
+
import { parse } from '../link';
|
|
6
|
+
import { emoji } from './hashtag';
|
|
7
|
+
import { str } from '../../source';
|
|
6
8
|
import { define } from 'typed-dom/dom';
|
|
7
9
|
|
|
8
10
|
// https://example/@user must be a user page or a redirect page going there.
|
|
11
|
+
// https://example/@user?ch=a+b must be a user channel page or a redirect page going there.
|
|
9
12
|
|
|
10
|
-
export const account: AutolinkParser.AccountParser = lazy(() =>
|
|
13
|
+
export const account: AutolinkParser.AccountParser = lazy(() => constraint(State.autolink, state(State.autolink,
|
|
11
14
|
surround(
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
15
|
+
surround(
|
|
16
|
+
/(?<![0-9a-z])@/yi,
|
|
17
|
+
str(/[0-9a-z](?:[.-](?=[0-9a-z])|[0-9a-z]){0,254}\/|/yi),
|
|
18
|
+
str(/[a-z][0-9a-z]*(?:[-.][0-9a-z]+)*(?![-.]?[0-9a-z@]|>>|:\S)/yi),
|
|
19
|
+
false,
|
|
20
|
+
[3 | Backtrack.autolink]),
|
|
21
|
+
some(surround(
|
|
22
|
+
'#',
|
|
23
|
+
verify(
|
|
24
|
+
str(new RegExp([
|
|
25
|
+
/(?!['_])(?:[^\p{C}\p{S}\p{P}\s]|emoji|'(?=[0-9A-Za-z])|_(?=[^\p{C}\p{S}\p{P}\s]|emoji))+/yu.source,
|
|
26
|
+
].join('|').replace(/emoji/g, emoji.source), 'yu')),
|
|
27
|
+
([{ value }]) => /^[0-9]{0,4}[^0-9]/.test(value)),
|
|
28
|
+
new RegExp([
|
|
29
|
+
/(?![0-9a-z@]|>>|:\S|[^\p{C}\p{S}\p{P}\s]|emoji)/yu.source,
|
|
30
|
+
].join('|').replace(/emoji/g, emoji.source), 'yu'),
|
|
31
|
+
false,
|
|
32
|
+
[3 | Backtrack.autolink])),
|
|
33
|
+
'',
|
|
34
|
+
false, [],
|
|
35
|
+
([[{ value: host }, { value: account }], nodes], context) => {
|
|
36
|
+
const hashes = nodes.foldl((acc, { value }) => acc + '#' + value, '');
|
|
37
|
+
const param = nodes.foldl((acc, { value }) => acc ? acc + '+' + value : value, '');
|
|
38
|
+
return new List([
|
|
39
|
+
new Data(define(
|
|
40
|
+
parse(
|
|
41
|
+
new List([new Data(`@${host}${account}${hashes}`)]),
|
|
42
|
+
new List([new Data(host ? `https://${host}@${account}?ch=${param}` : `/@${account}?ch=${param}`)]),
|
|
43
|
+
context),
|
|
44
|
+
{ class: 'channel' }))
|
|
45
|
+
]);
|
|
46
|
+
},
|
|
47
|
+
([[{ value: host }, { value: account }]], context) => {
|
|
48
|
+
if (context.source[context.position] === '#') return;
|
|
49
|
+
return new List([
|
|
50
|
+
new Data(define(
|
|
51
|
+
parse(
|
|
52
|
+
new List([new Data(`@${host}${account}`)]),
|
|
53
|
+
new List([new Data(host ? `https://${host}@${account}` : `/@${account}`)]),
|
|
54
|
+
context),
|
|
55
|
+
{ class: 'account' }))
|
|
56
|
+
]);
|
|
57
|
+
}))));
|
|
@@ -32,7 +32,6 @@ describe('Unit: parser/inline/autolink/anchor', () => {
|
|
|
32
32
|
assert.deepStrictEqual(inspect(parser('>>0-A'), ctx), [['<a class="anchor" href="?at=0-A">>>0-A</a>'], '']);
|
|
33
33
|
assert.deepStrictEqual(inspect(parser('>>0--a'), ctx), [['<a class="anchor" href="?at=0">>>0</a>'], '--a']);
|
|
34
34
|
assert.deepStrictEqual(inspect(parser('>>2000-0131-2359-59999'), ctx), [['<a class="anchor" href="?at=2000-0131-2359-59999">>>2000-0131-2359-59999</a>'], '']);
|
|
35
|
-
assert.deepStrictEqual(inspect(parser('>>A/2000-0131-2359-59'), ctx), [['<a class="anchor" href="/@A/timeline?at=2000-0131-2359-59">>>A/2000-0131-2359-59</a>'], '']);
|
|
36
35
|
});
|
|
37
36
|
|
|
38
37
|
});
|
|
@@ -1,8 +1,9 @@
|
|
|
1
1
|
import { AutolinkParser } from '../../inline';
|
|
2
2
|
import { State, Backtrack } from '../../context';
|
|
3
3
|
import { List, Data } from '../../../combinator/data/parser';
|
|
4
|
-
import {
|
|
5
|
-
import {
|
|
4
|
+
import { state, constraint, surround, lazy } from '../../../combinator';
|
|
5
|
+
import { parse } from '../link';
|
|
6
|
+
import { str } from '../../source';
|
|
6
7
|
import { define } from 'typed-dom/dom';
|
|
7
8
|
|
|
8
9
|
// Timeline(pseudonym): user/tid
|
|
@@ -15,18 +16,18 @@ import { define } from 'typed-dom/dom';
|
|
|
15
16
|
// 内部表現はUnixTimeに統一する(時系列順)
|
|
16
17
|
// 外部表現は投稿ごとに投稿者の投稿時のタイムゾーンに統一する(非時系列順)
|
|
17
18
|
|
|
18
|
-
export const anchor: AutolinkParser.AnchorParser = lazy(() =>
|
|
19
|
-
|
|
19
|
+
export const anchor: AutolinkParser.AnchorParser = lazy(() => constraint(State.autolink, state(State.autolink,
|
|
20
|
+
surround(
|
|
20
21
|
/(?<![0-9a-z])>>/yi,
|
|
21
|
-
/
|
|
22
|
+
str(/[0-9a-z]+(?:-[0-9a-z]+)*(?!-?[0-9a-z@#]|>>|:\S)/yi),
|
|
23
|
+
'',
|
|
22
24
|
false,
|
|
23
|
-
[3 | Backtrack.autolink]
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
([{ value }]) => new List([new Data(define(value, { class: 'anchor' }))]))))));
|
|
25
|
+
[3 | Backtrack.autolink],
|
|
26
|
+
([, [{ value }]], context) =>
|
|
27
|
+
new List([
|
|
28
|
+
new Data(define(parse(
|
|
29
|
+
new List([new Data(`>>${value}`)]),
|
|
30
|
+
new List([new Data(`?at=${value}`)]),
|
|
31
|
+
context),
|
|
32
|
+
{ class: 'anchor' }))
|
|
33
|
+
])))));
|
|
@@ -27,8 +27,8 @@ describe('Unit: parser/inline/autolink/email', () => {
|
|
|
27
27
|
assert.deepStrictEqual(inspect(parser('a..b@c'), ctx), undefined);
|
|
28
28
|
assert.deepStrictEqual(inspect(parser('a++b@c'), ctx), undefined);
|
|
29
29
|
assert.deepStrictEqual(inspect(parser('a@b.c:d'), ctx), undefined);
|
|
30
|
+
assert.deepStrictEqual(inspect(parser('a@b.domain.com:c'), ctx), undefined);
|
|
30
31
|
assert.deepStrictEqual(inspect(parser('a@http://host'), ctx), undefined);
|
|
31
|
-
assert.deepStrictEqual(inspect(parser(`a@${'b'.repeat(64)}`), ctx), undefined);
|
|
32
32
|
assert.deepStrictEqual(inspect(parser(' a@b'), ctx), undefined);
|
|
33
33
|
});
|
|
34
34
|
|
|
@@ -1,21 +1,20 @@
|
|
|
1
1
|
import { AutolinkParser } from '../../inline';
|
|
2
2
|
import { State, Backtrack } from '../../context';
|
|
3
3
|
import { List, Data } from '../../../combinator/data/parser';
|
|
4
|
-
import {
|
|
4
|
+
import { state, constraint, verify, surround } from '../../../combinator';
|
|
5
5
|
import { str } from '../../source';
|
|
6
6
|
import { html } from 'typed-dom/dom';
|
|
7
7
|
|
|
8
8
|
// https://html.spec.whatwg.org/multipage/input.html
|
|
9
9
|
|
|
10
|
-
export const email: AutolinkParser.EmailParser =
|
|
11
|
-
|
|
10
|
+
export const email: AutolinkParser.EmailParser = constraint(State.autolink, state(State.autolink,
|
|
11
|
+
surround(
|
|
12
|
+
/(?<![0-9a-z][_.+-]?|[@#])(?=[0-9a-z])/yi,
|
|
12
13
|
verify(
|
|
13
|
-
str(/[0-9a-z](?:[_.+-](?=[0-9a-z])|[0-9a-z]){0,
|
|
14
|
-
([{ value }]) => value.length <=
|
|
14
|
+
str(/[0-9a-z](?:[_.+-](?=[0-9a-z])|[0-9a-z]){0,63}@[0-9a-z](?:[.-](?=[0-9a-z])|[0-9a-z]){0,254}(?![.-]?[0-9a-z@#]|>>|:\S)/yi),
|
|
15
|
+
([{ value }]) => value.length <= 254),
|
|
16
|
+
'',
|
|
15
17
|
false,
|
|
16
|
-
[3 | Backtrack.autolink]
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
({ context: { source } }) =>
|
|
20
|
-
new List([new Data(html('a', { class: 'email', href: `mailto:${source}` }, source))])
|
|
21
|
-
]))));
|
|
18
|
+
[3 | Backtrack.autolink],
|
|
19
|
+
([, [{ value }]]) =>
|
|
20
|
+
new List([new Data(html('a', { class: 'email', href: `mailto:${value}` }, value))]))));
|
|
@@ -1,24 +1,28 @@
|
|
|
1
1
|
import { AutolinkParser } from '../../inline';
|
|
2
2
|
import { State, Backtrack } from '../../context';
|
|
3
3
|
import { List, Data } from '../../../combinator/data/parser';
|
|
4
|
-
import {
|
|
5
|
-
import {
|
|
4
|
+
import { state, constraint, surround, lazy } from '../../../combinator';
|
|
5
|
+
import { parse } from '../link';
|
|
6
6
|
import { emoji } from './hashtag';
|
|
7
|
+
import { str } from '../../source';
|
|
7
8
|
import { define } from 'typed-dom/dom';
|
|
8
9
|
|
|
9
|
-
export const hashnum: AutolinkParser.HashnumParser = lazy(() =>
|
|
10
|
-
|
|
10
|
+
export const hashnum: AutolinkParser.HashnumParser = lazy(() => constraint(State.autolink, state(State.autolink,
|
|
11
|
+
surround(
|
|
11
12
|
new RegExp([
|
|
12
13
|
/(?<![^\p{C}\p{S}\p{P}\s]|emoji)#/yu.source,
|
|
13
14
|
].join('|').replace(/emoji/g, emoji.source), 'yu'),
|
|
14
|
-
new RegExp([
|
|
15
|
+
str(new RegExp([
|
|
15
16
|
/[0-9]{1,9}(?![0-9a-z@#]|>>|:\S|[^\p{C}\p{S}\p{P}\s]|emoji)/yu.source,
|
|
16
|
-
].join('|').replace(/emoji/g, emoji.source), 'yu'),
|
|
17
|
+
].join('|').replace(/emoji/g, emoji.source), 'yu')),
|
|
18
|
+
'',
|
|
17
19
|
false,
|
|
18
|
-
[1 | Backtrack.autolink]
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
20
|
+
[1 | Backtrack.autolink],
|
|
21
|
+
([, [{ value }]], context) =>
|
|
22
|
+
new List([
|
|
23
|
+
new Data(define(parse(
|
|
24
|
+
new List([new Data(`#${value}`)]),
|
|
25
|
+
new List([new Data(value)]),
|
|
26
|
+
context),
|
|
27
|
+
{ class: 'hashnum', href: null }))
|
|
28
|
+
])))));
|
|
@@ -1,8 +1,8 @@
|
|
|
1
1
|
import { AutolinkParser } from '../../inline';
|
|
2
2
|
import { State, Backtrack } from '../../context';
|
|
3
3
|
import { List, Data } from '../../../combinator/data/parser';
|
|
4
|
-
import {
|
|
5
|
-
import {
|
|
4
|
+
import { state, constraint, verify, surround, lazy } from '../../../combinator';
|
|
5
|
+
import { parse } from '../link';
|
|
6
6
|
import { str } from '../../source';
|
|
7
7
|
import { define } from 'typed-dom/dom';
|
|
8
8
|
|
|
@@ -11,22 +11,26 @@ import { define } from 'typed-dom/dom';
|
|
|
11
11
|
// https://github.com/tc39/proposal-regexp-unicode-property-escapes#matching-emoji
|
|
12
12
|
export const emoji = /\p{Emoji_Modifier_Base}\p{Emoji_Modifier}?|\p{Emoji_Presentation}|\p{Emoji}\uFE0F|\u200D/u;
|
|
13
13
|
|
|
14
|
-
export const hashtag: AutolinkParser.HashtagParser = lazy(() =>
|
|
15
|
-
|
|
14
|
+
export const hashtag: AutolinkParser.HashtagParser = lazy(() => constraint(State.autolink, state(State.autolink,
|
|
15
|
+
surround(
|
|
16
16
|
new RegExp([
|
|
17
17
|
/(?<![^\p{C}\p{S}\p{P}\s]|emoji)#/yu.source,
|
|
18
18
|
].join('|').replace(/emoji/g, emoji.source), 'yu'),
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
19
|
+
verify(
|
|
20
|
+
str(new RegExp([
|
|
21
|
+
/(?!['_])(?:[^\p{C}\p{S}\p{P}\s]|emoji|'(?=[0-9A-Za-z])|_(?=[^\p{C}\p{S}\p{P}\s]|emoji))+/yu.source,
|
|
22
|
+
].join('|').replace(/emoji/g, emoji.source), 'yu')),
|
|
23
|
+
([{ value }]) => /^[0-9]{0,4}[^0-9]/.test(value)),
|
|
22
24
|
new RegExp([
|
|
23
25
|
/(?![0-9a-z@#]|>>|:\S|[^\p{C}\p{S}\p{P}\s]|emoji)/yu.source,
|
|
24
26
|
].join('|').replace(/emoji/g, emoji.source), 'yu'),
|
|
25
|
-
false,
|
|
26
|
-
[3 | Backtrack.autolink]
|
|
27
|
-
([{ value }]) =>
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
27
|
+
false,
|
|
28
|
+
[3 | Backtrack.autolink],
|
|
29
|
+
([, [{ value }]], context) =>
|
|
30
|
+
new List([
|
|
31
|
+
new Data(define(parse(
|
|
32
|
+
new List([new Data(`#${value}`)]),
|
|
33
|
+
new List([new Data(`/hashtags/${value}`)]),
|
|
34
|
+
context),
|
|
35
|
+
{ class: 'hashtag' }))
|
|
36
|
+
])))));
|
|
@@ -1,9 +1,9 @@
|
|
|
1
1
|
import { AutolinkParser } from '../../inline';
|
|
2
2
|
import { State, Recursion, Backtrack } from '../../context';
|
|
3
|
-
import { List } from '../../../combinator/data/parser';
|
|
4
|
-
import { union, tails, some, recursion, precedence, state, constraint, verify, focus, rewrite,
|
|
3
|
+
import { List, Data } from '../../../combinator/data/parser';
|
|
4
|
+
import { union, tails, some, recursion, precedence, state, constraint, verify, focus, rewrite, surround, open, lazy } from '../../../combinator';
|
|
5
5
|
import { inline } from '../../inline';
|
|
6
|
-
import {
|
|
6
|
+
import { parse } from '../link';
|
|
7
7
|
import { unescsource, str } from '../../source';
|
|
8
8
|
|
|
9
9
|
export const url: AutolinkParser.UrlParser = lazy(() => rewrite(
|
|
@@ -16,10 +16,8 @@ export const url: AutolinkParser.UrlParser = lazy(() => rewrite(
|
|
|
16
16
|
false,
|
|
17
17
|
[3 | Backtrack.autolink]),
|
|
18
18
|
union([
|
|
19
|
-
constraint(State.autolink, state(State.autolink,
|
|
20
|
-
|
|
21
|
-
unsafelink,
|
|
22
|
-
false))),
|
|
19
|
+
constraint(State.autolink, state(State.autolink, ({ context }) =>
|
|
20
|
+
new List([new Data(parse(new List(), new List([new Data(context.source)]), context))]))),
|
|
23
21
|
open(str(/[^:]+/y), some(inline)),
|
|
24
22
|
])));
|
|
25
23
|
|
|
@@ -28,21 +26,28 @@ export const lineurl: AutolinkParser.UrlParser.LineUrlParser = lazy(() => focus(
|
|
|
28
26
|
tails([
|
|
29
27
|
str('!'),
|
|
30
28
|
union([
|
|
31
|
-
constraint(State.autolink, state(State.autolink,
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
29
|
+
constraint(State.autolink, state(State.autolink, ({ context }) => {
|
|
30
|
+
const { source, position } = context;
|
|
31
|
+
context.position -= source[0] === '!' ? 1 : 0;
|
|
32
|
+
context.position += source.length;
|
|
33
|
+
return new List([
|
|
34
|
+
new Data(parse(
|
|
35
|
+
new List(),
|
|
36
|
+
new List([new Data(source.slice(position))]),
|
|
37
|
+
context))
|
|
38
|
+
]);
|
|
39
|
+
})),
|
|
35
40
|
open(str(/[^:]+/y), some(inline)),
|
|
36
41
|
]),
|
|
37
42
|
])));
|
|
38
43
|
|
|
39
44
|
const bracket: AutolinkParser.UrlParser.BracketParser = lazy(() => union([
|
|
40
|
-
surround(str('('), recursion(Recursion.terminal, some(union([bracket, unescsource]), ')')), str(')'),
|
|
41
|
-
undefined, () => new List()
|
|
42
|
-
surround(str('['), recursion(Recursion.terminal, some(union([bracket, unescsource]), ']')), str(']'),
|
|
43
|
-
undefined, () => new List()
|
|
44
|
-
surround(str('{'), recursion(Recursion.terminal, some(union([bracket, unescsource]), '}')), str('}'),
|
|
45
|
-
undefined, () => new List()
|
|
46
|
-
surround(str('"'), precedence(2, recursion(Recursion.terminal, some(unescsource, '"'))), str('"'),
|
|
47
|
-
undefined, () => new List()
|
|
45
|
+
surround(str('('), recursion(Recursion.terminal, some(union([bracket, unescsource]), ')')), str(')'),
|
|
46
|
+
true, [3 | Backtrack.autolink], undefined, () => new List()),
|
|
47
|
+
surround(str('['), recursion(Recursion.terminal, some(union([bracket, unescsource]), ']')), str(']'),
|
|
48
|
+
true, [3 | Backtrack.autolink], undefined, () => new List()),
|
|
49
|
+
surround(str('{'), recursion(Recursion.terminal, some(union([bracket, unescsource]), '}')), str('}'),
|
|
50
|
+
true, [3 | Backtrack.autolink], undefined, () => new List()),
|
|
51
|
+
surround(str('"'), precedence(2, recursion(Recursion.terminal, some(unescsource, '"'))), str('"'),
|
|
52
|
+
true, [3 | Backtrack.autolink], undefined, () => new List()),
|
|
48
53
|
]));
|
|
@@ -3,7 +3,6 @@ import { State } from '../context';
|
|
|
3
3
|
import { state, lazy } from '../../combinator';
|
|
4
4
|
import { url, lineurl } from './autolink/url';
|
|
5
5
|
import { email } from './autolink/email';
|
|
6
|
-
import { channel } from './autolink/channel';
|
|
7
6
|
import { account } from './autolink/account';
|
|
8
7
|
import { hashtag } from './autolink/hashtag';
|
|
9
8
|
import { hashnum } from './autolink/hashnum';
|
|
@@ -18,7 +17,7 @@ export const autolink: AutolinkParser = lazy(() =>
|
|
|
18
17
|
const fst = source[position];
|
|
19
18
|
switch (fst) {
|
|
20
19
|
case '@':
|
|
21
|
-
return
|
|
20
|
+
return account(input);
|
|
22
21
|
case '#':
|
|
23
22
|
return hashtag(input) || hashnum(input);
|
|
24
23
|
case '>':
|
|
@@ -39,34 +39,35 @@ const p1 = lazy(() => surround(
|
|
|
39
39
|
precedence(1, recursion(Recursion.bracket, some(inline, ')', [[')', 1]]))),
|
|
40
40
|
str(')'),
|
|
41
41
|
true,
|
|
42
|
+
[2 | Backtrack.bracket],
|
|
42
43
|
([as, bs = new List(), cs], { source, position, range = 0 }) => {
|
|
43
44
|
const str = source.slice(position - range + 1, position - 1);
|
|
44
45
|
return indexA.test(str)
|
|
45
46
|
? new List([new Data(as.head!.value), new Data(str), new Data(cs.head!.value)])
|
|
46
47
|
: new List([new Data(html('span', { class: 'paren' }, defrag(unwrap(as.import(bs as List<Data<string>>).import(cs)))))]);
|
|
47
48
|
},
|
|
48
|
-
([as, bs = new List()]) => as.import(bs as List<Data<string>>)
|
|
49
|
-
[2 | Backtrack.bracket]));
|
|
49
|
+
([as, bs = new List()]) => as.import(bs as List<Data<string>>)));
|
|
50
50
|
|
|
51
51
|
const p2 = lazy(() => surround(
|
|
52
52
|
str('('),
|
|
53
53
|
precedence(1, recursion(Recursion.bracket, some(inline, ')', [[')', 1]]))),
|
|
54
54
|
str(')'),
|
|
55
55
|
true,
|
|
56
|
+
[2 | Backtrack.bracket],
|
|
56
57
|
([as, bs = [], cs], { source, position, range = 0 }) => {
|
|
57
58
|
const str = source.slice(position - range + 1, position - 1);
|
|
58
59
|
return indexF.test(str)
|
|
59
60
|
? new List([new Data(as.head!.value), new Data(str), new Data(cs.head!.value)])
|
|
60
61
|
: new List([new Data(html('span', { class: 'paren' }, defrag(unwrap(as.import(bs as List<Data<string>>).import(cs)))))]);
|
|
61
62
|
},
|
|
62
|
-
([as, bs = new List()]) => as.import(bs as List<Data<string>>)
|
|
63
|
-
[2 | Backtrack.bracket]));
|
|
63
|
+
([as, bs = new List()]) => as.import(bs as List<Data<string>>)));
|
|
64
64
|
|
|
65
65
|
const s1 = lazy(() => surround(
|
|
66
66
|
str('['),
|
|
67
67
|
precedence(1, recursion(Recursion.bracket, some(inline, ']', [[']', 1]]))),
|
|
68
68
|
str(']'),
|
|
69
69
|
true,
|
|
70
|
+
[2 | Backtrack.bracket],
|
|
70
71
|
([as, bs = new List(), cs], context) => {
|
|
71
72
|
if (context.state! & State.link) {
|
|
72
73
|
const { source, position, range = 0 } = context;
|
|
@@ -89,35 +90,34 @@ const s1 = lazy(() => surround(
|
|
|
89
90
|
}
|
|
90
91
|
return as.import(bs as List<Data<string>>).import(cs);
|
|
91
92
|
},
|
|
92
|
-
([as, bs = new List()]) => as.import(bs as List<Data<string>>)
|
|
93
|
-
[2 | Backtrack.bracket]));
|
|
93
|
+
([as, bs = new List()]) => as.import(bs as List<Data<string>>)));
|
|
94
94
|
|
|
95
95
|
const s2 = lazy(() => surround(
|
|
96
96
|
str('['),
|
|
97
97
|
precedence(1, recursion(Recursion.bracket, some(inline, ']', [[']', 1]]))),
|
|
98
98
|
str(']'),
|
|
99
99
|
true,
|
|
100
|
+
[2 | Backtrack.bracket],
|
|
100
101
|
undefined,
|
|
101
|
-
([as, bs = new List()]) => as.import(bs as List<Data<string>>)
|
|
102
|
-
[2 | Backtrack.bracket]));
|
|
102
|
+
([as, bs = new List()]) => as.import(bs as List<Data<string>>)));
|
|
103
103
|
|
|
104
104
|
const c1 = lazy(() => surround(
|
|
105
105
|
str('{'),
|
|
106
106
|
precedence(1, recursion(Recursion.bracket, some(inline, '}', [['}', 1]]))),
|
|
107
107
|
str('}'),
|
|
108
108
|
true,
|
|
109
|
+
[2 | Backtrack.bracket],
|
|
109
110
|
undefined,
|
|
110
|
-
([as, bs = new List()]) => as.import(bs as List<Data<string>>)
|
|
111
|
-
[2 | Backtrack.bracket]));
|
|
111
|
+
([as, bs = new List()]) => as.import(bs as List<Data<string>>)));
|
|
112
112
|
|
|
113
113
|
const c2 = lazy(() => surround(
|
|
114
114
|
str('{'),
|
|
115
115
|
precedence(1, recursion(Recursion.bracket, some(inline, '}', [['}', 1]]))),
|
|
116
116
|
str('}'),
|
|
117
117
|
true,
|
|
118
|
+
[2 | Backtrack.bracket],
|
|
118
119
|
undefined,
|
|
119
|
-
([as, bs = new List()]) => as.import(bs as List<Data<string>>)
|
|
120
|
-
[2 | Backtrack.bracket]));
|
|
120
|
+
([as, bs = new List()]) => as.import(bs as List<Data<string>>)));
|
|
121
121
|
|
|
122
122
|
const d1 = lazy(() => surround(
|
|
123
123
|
str('"'),
|
|
@@ -125,6 +125,6 @@ const d1 = lazy(() => surround(
|
|
|
125
125
|
precedence(2, recursion(Recursion.bracket, some(inline, /["\n]/y, [['"', 2], ['\n', 3]]))),
|
|
126
126
|
str('"'),
|
|
127
127
|
true,
|
|
128
|
+
[2 | Backtrack.bracket],
|
|
128
129
|
undefined,
|
|
129
|
-
([as, bs = new List()]) => as.import(bs as List<Data<string>>)
|
|
130
|
-
[2 | Backtrack.bracket]));
|
|
130
|
+
([as, bs = new List()]) => as.import(bs as List<Data<string>>)));
|
|
@@ -15,7 +15,8 @@ export const deletion: DeletionParser = lazy(() =>
|
|
|
15
15
|
some(inline, blankWith('\n', '~~')),
|
|
16
16
|
open('\n', some(inline, '~'), true),
|
|
17
17
|
]))),
|
|
18
|
-
'~~',
|
|
18
|
+
'~~',
|
|
19
|
+
false, [],
|
|
19
20
|
([, bs], { buffer }) => buffer!.import(bs),
|
|
20
21
|
([, bs], { buffer }) => bs && buffer!.import(bs).push(new Data(Command.Cancel)) && buffer!),
|
|
21
22
|
nodes => new List([new Data(html('del', defrag(unwrap(nodes))))]))));
|
|
@@ -17,6 +17,7 @@ export const emphasis: EmphasisParser = lazy(() => surround(
|
|
|
17
17
|
some(inline, blankWith('*')),
|
|
18
18
|
open(some(inline, '*'), inline),
|
|
19
19
|
]))))),
|
|
20
|
-
str('*'),
|
|
20
|
+
str('*'),
|
|
21
|
+
false, [],
|
|
21
22
|
([, bs]) => new List([new Data(html('em', defrag(unwrap(bs))))]),
|
|
22
23
|
([as, bs]) => bs && as.import(bs as List<Data<string>>)));
|
|
@@ -32,7 +32,8 @@ export const emstrong: EmStrongParser = lazy(() =>
|
|
|
32
32
|
some(inline, blankWith('*')),
|
|
33
33
|
open(some(inline, '*'), inline),
|
|
34
34
|
])))),
|
|
35
|
-
str(/\*{1,3}/y),
|
|
35
|
+
str(/\*{1,3}/y),
|
|
36
|
+
false, [],
|
|
36
37
|
([, bs, cs], context): Result<Node<EmStrongParser>, Context<EmStrongParser>> => {
|
|
37
38
|
assert(cs.length === 1);
|
|
38
39
|
const { buffer = new List() } = context;
|
|
@@ -22,12 +22,12 @@ export const index: IndexParser = lazy(() => constraint(State.index, fmap(indexe
|
|
|
22
22
|
]), ']', [[']', 1]])))),
|
|
23
23
|
str(']'),
|
|
24
24
|
false,
|
|
25
|
+
[3 | Backtrack.bracket],
|
|
25
26
|
([, bs], context) =>
|
|
26
27
|
context.linebreak === 0 && trimBlankNodeEnd(bs).length > 0
|
|
27
28
|
? new List([new Data(html('a', { 'data-index': dataindex(bs) }, defrag(unwrap(bs))))])
|
|
28
29
|
: undefined,
|
|
29
|
-
undefined,
|
|
30
|
-
[3 | Backtrack.bracket])),
|
|
30
|
+
undefined)),
|
|
31
31
|
ns => {
|
|
32
32
|
if (ns.length === 1) {
|
|
33
33
|
const el = ns.head!.value as HTMLElement;
|
|
@@ -54,14 +54,14 @@ export const signature: IndexParser.SignatureParser = lazy(() => validate('|', s
|
|
|
54
54
|
]), ']'),
|
|
55
55
|
/(?=])/y,
|
|
56
56
|
false,
|
|
57
|
+
[3 | Backtrack.bracket],
|
|
57
58
|
([, ns], context) => {
|
|
58
59
|
const index = identity('index', undefined, ns.foldl((acc, { value }) => acc + value, ''))?.slice(7);
|
|
59
60
|
return index && context.linebreak === 0
|
|
60
61
|
? new List([new Data(html('span', { class: 'indexer', 'data-index': index }))])
|
|
61
62
|
: undefined;
|
|
62
63
|
},
|
|
63
|
-
([as, bs]) => bs && as.import(bs)
|
|
64
|
-
[3 | Backtrack.bracket])));
|
|
64
|
+
([as, bs]) => bs && as.import(bs))));
|
|
65
65
|
|
|
66
66
|
export function dataindex(nodes: List<Data<string | HTMLElement>>): string | undefined {
|
|
67
67
|
let node = nodes.last;
|
|
@@ -14,7 +14,7 @@ export const segment: ExtensionParser.LabelParser.SegmentParser = clear(union([
|
|
|
14
14
|
|
|
15
15
|
export const label: ExtensionParser.LabelParser = constraint(State.label, fmap(
|
|
16
16
|
union([
|
|
17
|
-
surround('[', body, ']', false,
|
|
17
|
+
surround('[', body, ']', false, [1 | Backtrack.bracket, 1]),
|
|
18
18
|
body,
|
|
19
19
|
]),
|
|
20
20
|
([{ value }]) => new List([
|
|
@@ -17,7 +17,9 @@ export const placeholder: ExtensionParser.PlaceholderParser = lazy(() => surroun
|
|
|
17
17
|
str(/\[[:^|]/y),
|
|
18
18
|
precedence(1, recursion(Recursion.inline,
|
|
19
19
|
tightStart(some(union([inline]), ']', [[']', 1]])))),
|
|
20
|
-
str(']'),
|
|
20
|
+
str(']'),
|
|
21
|
+
false,
|
|
22
|
+
[3 | Backtrack.bracket],
|
|
21
23
|
(_, context) => new List([
|
|
22
24
|
new Data(html('span',
|
|
23
25
|
{
|
|
@@ -26,5 +28,4 @@ export const placeholder: ExtensionParser.PlaceholderParser = lazy(() => surroun
|
|
|
26
28
|
},
|
|
27
29
|
context.source.slice(context.position - context.range!, context.position)))
|
|
28
30
|
]),
|
|
29
|
-
([as, bs]) => bs && as.import(bs as List<Data<string>>)
|
|
30
|
-
[3 | Backtrack.bracket]));
|
|
31
|
+
([as, bs]) => bs && as.import(bs as List<Data<string>>)));
|
|
@@ -26,7 +26,7 @@ export const html: HTMLParser = lazy(() => validate(/<[a-z]+(?=[ >])/yi,
|
|
|
26
26
|
str(/<(?:area|base|br|col|embed|hr|img|input|link|meta|source|track|wbr)(?=[ >])/y),
|
|
27
27
|
some(union([attribute])),
|
|
28
28
|
open(str(/ ?/y), str('>'), true),
|
|
29
|
-
true,
|
|
29
|
+
true, [],
|
|
30
30
|
([as, bs = new List(), cs], context) =>
|
|
31
31
|
new List([new Data(elem(as.head!.value.slice(1), false, [...unwrap(as.import(bs).import(cs))], new List(), new List(), context))]),
|
|
32
32
|
([as, bs = new List()], context) =>
|
|
@@ -38,7 +38,7 @@ export const html: HTMLParser = lazy(() => validate(/<[a-z]+(?=[ >])/yi,
|
|
|
38
38
|
surround<HTMLParser.TagParser, string>(
|
|
39
39
|
surround(
|
|
40
40
|
str(`<${tag}`), some(attribute), open(str(/ ?/y), str('>'), true),
|
|
41
|
-
true,
|
|
41
|
+
true, [],
|
|
42
42
|
([as, bs = new List(), cs]) => as.import(bs).import(cs),
|
|
43
43
|
([as, bs = new List()]) => as.import(bs)),
|
|
44
44
|
// 不可視のHTML構造が可視構造を変化させるべきでない。
|
|
@@ -51,7 +51,7 @@ export const html: HTMLParser = lazy(() => validate(/<[a-z]+(?=[ >])/yi,
|
|
|
51
51
|
open('\n', some(inline, `</${tag}>`), true),
|
|
52
52
|
])))),
|
|
53
53
|
str(`</${tag}>`),
|
|
54
|
-
true,
|
|
54
|
+
true, [],
|
|
55
55
|
([as, bs = new List(), cs], context) =>
|
|
56
56
|
new List([new Data(elem(tag, true, [...unwrap(as)], bs, cs, context))]),
|
|
57
57
|
([as, bs = new List()], context) =>
|
|
@@ -63,7 +63,7 @@ export const html: HTMLParser = lazy(() => validate(/<[a-z]+(?=[ >])/yi,
|
|
|
63
63
|
str(/<[a-z]+(?=[ >])/yi),
|
|
64
64
|
some(union([attribute])),
|
|
65
65
|
open(str(/ ?/y), str('>'), true),
|
|
66
|
-
true,
|
|
66
|
+
true, [],
|
|
67
67
|
([as, bs = new List(), cs], context) =>
|
|
68
68
|
new List([new Data(elem(as.head!.value.slice(1), false, [...unwrap(as.import(bs).import(cs))], new List(), new List(), context))]),
|
|
69
69
|
([as, bs = new List()], context) =>
|
|
@@ -9,11 +9,11 @@ import { html } from 'typed-dom/dom';
|
|
|
9
9
|
export const unsafehtmlentity: UnsafeHTMLEntityParser = surround(
|
|
10
10
|
str('&'), str(/[0-9A-Za-z]+/y), str(';'),
|
|
11
11
|
false,
|
|
12
|
+
[3 | Backtrack.bracket],
|
|
12
13
|
([as, bs, cs]) =>
|
|
13
14
|
new List([new Data(parser(as.head!.value + bs.head!.value + cs.head!.value))]),
|
|
14
15
|
([as, bs]) =>
|
|
15
|
-
new List([new Data(as.head!.value + (bs?.head?.value ?? ''))])
|
|
16
|
-
[3 | Backtrack.bracket]);
|
|
16
|
+
new List([new Data(as.head!.value + (bs?.head?.value ?? ''))]));
|
|
17
17
|
|
|
18
18
|
export const htmlentity: HTMLEntityParser = fmap(
|
|
19
19
|
union([unsafehtmlentity]),
|
|
@@ -15,7 +15,8 @@ export const insertion: InsertionParser = lazy(() =>
|
|
|
15
15
|
some(inline, blankWith('\n', '++')),
|
|
16
16
|
open('\n', some(inline, '+'), true),
|
|
17
17
|
]))),
|
|
18
|
-
'++',
|
|
18
|
+
'++',
|
|
19
|
+
false, [],
|
|
19
20
|
([, bs], { buffer }) => buffer!.import(bs),
|
|
20
21
|
([, bs], { buffer }) => bs && buffer!.import(bs).push(new Data(Command.Cancel)) && buffer!),
|
|
21
22
|
nodes => new List([new Data(html('ins', defrag(unwrap(nodes))))]))));
|
|
@@ -18,7 +18,8 @@ export const italic: ItalicParser = lazy(() =>
|
|
|
18
18
|
some(inline, blankWith('///')),
|
|
19
19
|
open(some(inline, '/'), inline),
|
|
20
20
|
])))),
|
|
21
|
-
'///',
|
|
21
|
+
'///',
|
|
22
|
+
false, [],
|
|
22
23
|
([, bs], { buffer }) => buffer!.import(bs),
|
|
23
24
|
([, bs], { buffer }) => bs && buffer!.import(bs).push(new Data(Command.Cancel)) && buffer!),
|
|
24
25
|
nodes => new List([new Data(html('i', defrag(unwrap(nodes))))]))));
|