Skip to content

Commit f5540e4

Browse files
authoredSep 14, 2021
support ins/del in structured headers (#359)
1 parent 4786953 commit f5540e4

File tree

4 files changed

+85
-11
lines changed

4 files changed

+85
-11
lines changed
 

‎src/Clause.ts

+10-1
Original file line numberDiff line numberDiff line change
@@ -103,7 +103,16 @@ export default class Clause extends Builder {
103103
if (type === 'sdo' && (formattedHeader ?? header.innerHTML).includes('(')) {
104104
// SDOs are rendered without parameter lists in the header, for the moment
105105
const currentHeader = formattedHeader ?? header.innerHTML;
106-
header.innerHTML = currentHeader.substring(0, currentHeader.indexOf('(')).trim();
106+
header.innerHTML = (
107+
currentHeader.substring(0, currentHeader.indexOf('(')) +
108+
currentHeader.substring(currentHeader.lastIndexOf(')') + 1)
109+
).trim();
110+
if (
111+
header.children.length === 1 &&
112+
['INS', 'DEL', 'MARK'].includes(header.children[0].tagName)
113+
) {
114+
header.children[0].innerHTML = header.children[0].innerHTML.trim();
115+
}
107116
} else if (formattedHeader != null) {
108117
header.innerHTML = formattedHeader;
109118
}

‎src/header-parser.ts

+34-7
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,19 @@ export function parseStructuredHeaderH1(
77
): { name: string | null; formattedHeader: string | null; formattedParams: string | null } {
88
// parsing is intentionally permissive; the linter can do stricter checks
99
// TODO have the linter do checks
10+
11+
let wrapper = null;
1012
let headerText = header.innerHTML;
13+
let beforeContents = 0;
14+
const headerWrapperMatch = headerText.match(
15+
/^(?<beforeContents>\s*<(?<tag>ins|del|mark)>)(?<contents>.*)<\/\k<tag>>\s*$/is
16+
);
17+
if (headerWrapperMatch != null) {
18+
wrapper = headerWrapperMatch.groups!.tag;
19+
headerText = headerWrapperMatch.groups!.contents;
20+
beforeContents = headerWrapperMatch.groups!.beforeContents.length;
21+
}
22+
1123
const prefix = headerText.match(/^\s*(Static|Runtime) Semantics:\s*/);
1224
if (prefix != null) {
1325
headerText = headerText.substring(prefix[0].length);
@@ -28,7 +40,7 @@ export function parseStructuredHeaderH1(
2840
return { name: null, formattedHeader: null, formattedParams: null };
2941
}
3042

31-
type Param = { name: string; type: string | null };
43+
type Param = { name: string; type: string | null; wrapper: string | null };
3244
const name = parsed.groups!.name;
3345
let paramText = parsed.groups!.params ?? '';
3446
const params: Array<Param> = [];
@@ -42,13 +54,20 @@ export function parseStructuredHeaderH1(
4254
let offset = 0;
4355
for (const line of paramLines) {
4456
offset += line.length;
45-
const chunk = line.trim();
57+
let chunk = line.trim();
4658
if (chunk === '') {
4759
continue;
4860
}
61+
const wrapperMatch = chunk.match(/^<(ins|del|mark)>(.*)<\/\1>$/i);
62+
let paramWrapper = null;
63+
if (wrapperMatch != null) {
64+
paramWrapper = wrapperMatch[1];
65+
chunk = wrapperMatch[2];
66+
}
4967
++index;
5068
function getParameterOffset() {
5169
return (
70+
beforeContents +
5271
(prefix?.[0].length ?? 0) +
5372
parsed!.groups!.beforeParams.length +
5473
1 + // `beforeParams` does not include the leading `(`
@@ -96,10 +115,13 @@ export function parseStructuredHeaderH1(
96115
(optional ? optionalParams : params).push({
97116
name: paramName,
98117
type: paramType === 'unknown' ? null : paramType,
118+
wrapper: paramWrapper,
99119
});
100120
}
101121
const formattedPrefix = prefix == null ? '' : prefix[0].trim() + ' ';
102-
formattedHeader = `${formattedPrefix}${name} (${params.map(n => ' ' + n.name).join(',')}`;
122+
// prettier-ignore
123+
const printParam = (p: Param) => ` ${p.wrapper == null ? '' : `<${p.wrapper}>`}${p.name}${p.wrapper == null ? '' : `</${p.wrapper}>`}`;
124+
formattedHeader = `${formattedPrefix}${name} (${params.map(printParam).join(',')}`;
103125
if (optionalParams.length > 0) {
104126
formattedHeader +=
105127
optionalParams
@@ -134,17 +156,18 @@ export function parseStructuredHeaderH1(
134156
break;
135157
}
136158
paramText = text;
137-
(optional ? optionalParams : params).push({ name: match![1], type: null });
159+
(optional ? optionalParams : params).push({ name: match![1], type: null, wrapper: null });
138160
({ success, text } = eat(paramText, /^(\s*\])+|,/));
139161
if (success) {
140162
paramText = text;
141163
}
142164
}
143165
}
144166

145-
const printParam = (p: Param) => `${p.name}${p.type == null ? '' : ` (${p.type})`}`;
146-
const paramsWithTypes = params.map(printParam);
147-
const optionalParamsWithTypes = optionalParams.map(printParam);
167+
// prettier-ignore
168+
const printParamWithType = (p: Param) => `${p.wrapper == null ? '' : `<${p.wrapper}>`}${p.name}${p.type == null ? '' : ` (${p.type})`}${p.wrapper == null ? '' : `</${p.wrapper}>`}`;
169+
const paramsWithTypes = params.map(printParamWithType);
170+
const optionalParamsWithTypes = optionalParams.map(printParamWithType);
148171
let formattedParams = '';
149172
if (params.length === 0 && optionalParams.length === 0) {
150173
formattedParams = 'no arguments';
@@ -165,6 +188,10 @@ export function parseStructuredHeaderH1(
165188
}
166189
}
167190

191+
if (formattedHeader != null && wrapper != null) {
192+
formattedHeader = `<${wrapper}>${formattedHeader}</${wrapper}>`;
193+
}
194+
168195
return { name, formattedHeader, formattedParams };
169196
}
170197

‎test/baselines/generated-reference/structured-headers.html

+15-3
Original file line numberDiff line numberDiff line change
@@ -11,15 +11,27 @@ <h1><span class="secnum">1</span> ExampleAO ( param [ , param2 ] )</h1>
1111
<emu-alg><ol><li>Algorithm steps go here.</li></ol></emu-alg>
1212
</emu-clause>
1313

14+
<emu-clause id="sec-exampleao2" type="abstract operation" aoid="ExampleAO2">
15+
<h1><span class="secnum">2</span> <ins>ExampleAO2 ( param )</ins></h1>
16+
<p>The abstract operation ExampleAO2 takes argument param (an <emu-xref href="#integer"><a href="https://tc39.es/ecma262/#integer">integer</a></emu-xref>). It performs the following steps when called:</p>
17+
<emu-alg><ol><li>Algorithm steps go here.</li></ol></emu-alg>
18+
</emu-clause>
19+
20+
<emu-clause id="sec-exampleao3" type="abstract operation" aoid="ExampleAO3">
21+
<h1><span class="secnum">3</span> ExampleAO3 ( <del>param</del> )</h1>
22+
<p>The abstract operation ExampleAO3 takes argument <del>param (an <emu-xref href="#integer"><a href="https://tc39.es/ecma262/#integer">integer</a></emu-xref>)</del>. It performs the following steps when called:</p>
23+
<emu-alg><ol><li>Algorithm steps go here.</li></ol></emu-alg>
24+
</emu-clause>
25+
1426
<emu-clause id="sec-example-grammar">
15-
<h1><span class="secnum">2</span> This and That</h1>
27+
<h1><span class="secnum">4</span> This and That</h1>
1628
<emu-grammar type="definition"><emu-production name="PrimaryExpression" id="prod-PrimaryExpression">
1729
<emu-nt><a href="#prod-PrimaryExpression">PrimaryExpression</a></emu-nt> <emu-geq>:</emu-geq> <emu-rhs a="jo4mwtvh" id="prod-Joc9-5i9"><emu-t>this</emu-t></emu-rhs>
1830
<emu-rhs a="_ditw-4g" id="prod-QC7Z2SC4"><emu-t>that</emu-t></emu-rhs>
1931
</emu-production>
2032
</emu-grammar>
2133
<emu-clause id="sec-isthis" type="sdo" aoid="IsThis">
22-
<h1><span class="secnum">2.1</span> IsThis</h1>
34+
<h1><span class="secnum">4.1</span> IsThis</h1>
2335
<p>The <emu-xref href="#sec-algorithm-conventions-syntax-directed-operations"><a href="https://tc39.es/ecma262/#sec-algorithm-conventions-syntax-directed-operations">syntax-directed operation</a></emu-xref> IsThis takes no arguments. It is an example. It is defined piecewise over the following productions:</p>
2436
<emu-grammar><emu-production name="PrimaryExpression" collapsed="">
2537
<emu-nt><a href="#prod-PrimaryExpression">PrimaryExpression</a></emu-nt> <emu-geq>:</emu-geq> <emu-rhs a="jo4mwtvh" id="prod-oDKnYE2s"><emu-t>this</emu-t></emu-rhs>
@@ -33,7 +45,7 @@ <h1><span class="secnum">2.1</span> IsThis</h1>
3345
<emu-alg><ol><li>Return <emu-val>false</emu-val>.</li></ol></emu-alg>
3446
</emu-clause>
3547
<emu-clause id="sec-isthat" type="sdo" aoid="IsThat">
36-
<h1><span class="secnum">2.2</span> IsThat</h1>
48+
<h1><span class="secnum">4.2</span> IsThat</h1>
3749
<p>The <emu-xref href="#sec-algorithm-conventions-syntax-directed-operations"><a href="https://tc39.es/ecma262/#sec-algorithm-conventions-syntax-directed-operations">syntax-directed operation</a></emu-xref> IsThat takes argument <var>ignored</var> (an example). It is defined piecewise over the following productions:</p>
3850
<emu-grammar><emu-production name="PrimaryExpression" collapsed="">
3951
<emu-nt><a href="#prod-PrimaryExpression">PrimaryExpression</a></emu-nt> <emu-geq>:</emu-geq> <emu-rhs a="jo4mwtvh" id="prod-QDgm4qgx"><emu-t>this</emu-t></emu-rhs>

‎test/baselines/sources/structured-headers.html

+26
Original file line numberDiff line numberDiff line change
@@ -23,6 +23,32 @@ <h1>
2323
</emu-alg>
2424
</emu-clause>
2525

26+
<emu-clause id="sec-exampleao2" type="abstract operation">
27+
<h1>
28+
<ins>ExampleAO2 (
29+
param : an integer,
30+
)</ins>
31+
</h1>
32+
<dl class='header'>
33+
</dl>
34+
<emu-alg>
35+
1. Algorithm steps go here.
36+
</emu-alg>
37+
</emu-clause>
38+
39+
<emu-clause id="sec-exampleao3" type="abstract operation">
40+
<h1>
41+
ExampleAO3 (
42+
<del>param : an integer,</del>
43+
)
44+
</h1>
45+
<dl class='header'>
46+
</dl>
47+
<emu-alg>
48+
1. Algorithm steps go here.
49+
</emu-alg>
50+
</emu-clause>
51+
2652
<emu-clause id="sec-example-grammar">
2753
<h1>This and That</h1>
2854
<emu-grammar type="definition">

0 commit comments

Comments
 (0)
Please sign in to comment.