diff --git a/IBMPlexMono-Bold-SlashedZero.woff2 b/IBMPlexMono-Bold-SlashedZero.woff2 new file mode 100644 index 00000000..de464a69 Binary files /dev/null and b/IBMPlexMono-Bold-SlashedZero.woff2 differ diff --git a/IBMPlexMono-BoldItalic-SlashedZero.woff2 b/IBMPlexMono-BoldItalic-SlashedZero.woff2 new file mode 100644 index 00000000..1697592f Binary files /dev/null and b/IBMPlexMono-BoldItalic-SlashedZero.woff2 differ diff --git a/IBMPlexMono-Italic-SlashedZero.woff2 b/IBMPlexMono-Italic-SlashedZero.woff2 new file mode 100644 index 00000000..48be3cb8 Binary files /dev/null and b/IBMPlexMono-Italic-SlashedZero.woff2 differ diff --git a/IBMPlexMono-Regular-SlashedZero.woff2 b/IBMPlexMono-Regular-SlashedZero.woff2 new file mode 100644 index 00000000..7307bc2c Binary files /dev/null and b/IBMPlexMono-Regular-SlashedZero.woff2 differ diff --git a/IBMPlexSans-Bold-SlashedZero.woff2 b/IBMPlexSans-Bold-SlashedZero.woff2 new file mode 100644 index 00000000..5fd0afd2 Binary files /dev/null and b/IBMPlexSans-Bold-SlashedZero.woff2 differ diff --git a/IBMPlexSans-BoldItalic-SlashedZero.woff2 b/IBMPlexSans-BoldItalic-SlashedZero.woff2 new file mode 100644 index 00000000..1b6ab9c1 Binary files /dev/null and b/IBMPlexSans-BoldItalic-SlashedZero.woff2 differ diff --git a/IBMPlexSans-Italic-SlashedZero.woff2 b/IBMPlexSans-Italic-SlashedZero.woff2 new file mode 100644 index 00000000..acd30ca7 Binary files /dev/null and b/IBMPlexSans-Italic-SlashedZero.woff2 differ diff --git a/IBMPlexSans-Regular-SlashedZero.woff2 b/IBMPlexSans-Regular-SlashedZero.woff2 new file mode 100644 index 00000000..c853ac4c Binary files /dev/null and b/IBMPlexSans-Regular-SlashedZero.woff2 differ diff --git a/IBMPlexSerif-Bold-SlashedZero.woff2 b/IBMPlexSerif-Bold-SlashedZero.woff2 new file mode 100644 index 00000000..103597e0 Binary files /dev/null and b/IBMPlexSerif-Bold-SlashedZero.woff2 differ diff --git a/IBMPlexSerif-BoldItalic-SlashedZero.woff2 b/IBMPlexSerif-BoldItalic-SlashedZero.woff2 new file mode 100644 index 00000000..2ae74334 Binary files /dev/null and b/IBMPlexSerif-BoldItalic-SlashedZero.woff2 differ diff --git a/IBMPlexSerif-Italic-SlashedZero.woff2 b/IBMPlexSerif-Italic-SlashedZero.woff2 new file mode 100644 index 00000000..37eedb96 Binary files /dev/null and b/IBMPlexSerif-Italic-SlashedZero.woff2 differ diff --git a/IBMPlexSerif-Regular-SlashedZero.woff2 b/IBMPlexSerif-Regular-SlashedZero.woff2 new file mode 100644 index 00000000..65592bce Binary files /dev/null and b/IBMPlexSerif-Regular-SlashedZero.woff2 differ diff --git a/ecma-header.png b/ecma-header.png new file mode 100644 index 00000000..a61c2b59 Binary files /dev/null and b/ecma-header.png differ diff --git a/ecma-logo.png b/ecma-logo.png new file mode 100644 index 00000000..b65fb873 Binary files /dev/null and b/ecma-logo.png differ diff --git a/ecmarkup.css b/ecmarkup.css new file mode 100644 index 00000000..4523c149 --- /dev/null +++ b/ecmarkup.css @@ -0,0 +1,1432 @@ +/* IBM Plex fonts (Latin subset) have been downloaded from Google Fonts and modified to add support back in for the `zero` substitution (slashed zeroes) */ + +/* https://fonts.googleapis.com/css2?family=IBM%20Plex%20Serif:ital,wght@0,400;0,700;1,400;1,700&display=swap */ +@font-face { + font-family: 'IBM Plex Serif'; + font-style: normal; + font-weight: 400; + font-display: swap; + src: local(IBM Plex Serif Regular), local(IBMPlexSerif-Regular), url(./IBMPlexSerif-Regular-SlashedZero.woff2) format('woff2'); +} +@font-face { + font-family: 'IBM Plex Serif'; + font-style: normal; + font-weight: 700; + font-display: swap; + src: local(IBM Plex Serif Bold), local(IBMPlexSerif-Bold), url(./IBMPlexSerif-Bold-SlashedZero.woff2) format('woff2'); +} +@font-face { + font-family: 'IBM Plex Serif'; + font-style: italic; + font-weight: 400; + font-display: swap; + src: local(IBM Plex Serif Italic), local(IBMPlexSerif-Italic), url(./IBMPlexSerif-Italic-SlashedZero.woff2) format('woff2'); +} +@font-face { + font-family: 'IBM Plex Serif'; + font-style: italic; + font-weight: 700; + font-display: swap; + src: local(IBM Plex Serif Bold Italic), local(IBMPlexSerif-BoldItalic), url(./IBMPlexSerif-BoldItalic-SlashedZero.woff2) format('woff2'); +} + +/* https://fonts.googleapis.com/css2?family=IBM%20Plex%20Sans:ital,wght@0,400;0,700;1,400;1,700&display=swap */ +@font-face { + font-family: 'IBM Plex Sans'; + font-style: normal; + font-weight: 400; + font-display: swap; + src: local(IBM Plex Sans Regular), local(IBMPlexSans-Regular), url(./IBMPlexSans-Regular-SlashedZero.woff2) format('woff2'); +} +@font-face { + font-family: 'IBM Plex Sans'; + font-style: normal; + font-weight: 700; + font-display: swap; + src: local(IBM Plex Sans Bold), local(IBMPlexSans-Bold), url(./IBMPlexSans-Bold-SlashedZero.woff2) format('woff2'); +} +@font-face { + font-family: 'IBM Plex Sans'; + font-style: italic; + font-weight: 400; + font-display: swap; + src: local(IBM Plex Sans Italic), local(IBMPlexSans-Italic), url(./IBMPlexSans-Italic-SlashedZero.woff2) format('woff2'); +} +@font-face { + font-family: 'IBM Plex Sans'; + font-style: italic; + font-weight: 700; + font-display: swap; + src: local(IBM Plex Sans Bold Italic), local(IBMPlexSans-BoldItalic), url(./IBMPlexSans-BoldItalic-SlashedZero.woff2) format('woff2'); +} + +/* https://fonts.googleapis.com/css2?family=IBM%20Plex%20Mono:ital,wght@0,400;0,700;1,400;1,700&display=swap */ +@font-face { + font-family: 'IBM Plex Mono'; + font-style: normal; + font-weight: 400; + font-display: swap; + src: local(IBM Plex Mono Regular), local(IBMPlexMono-Regular), url(./IBMPlexMono-Regular-SlashedZero.woff2) format('woff2'); +} +@font-face { + font-family: 'IBM Plex Mono'; + font-style: normal; + font-weight: 700; + font-display: swap; + src: local(IBM Plex Mono Bold), local(IBMPlexMono-Bold), url(./IBMPlexMono-Bold-SlashedZero.woff2) format('woff2'); +} +@font-face { + font-family: 'IBM Plex Mono'; + font-style: italic; + font-weight: 400; + font-display: swap; + src: local(IBM Plex Mono Italic), local(IBMPlexMono-Italic), url(./IBMPlexMono-Italic-SlashedZero.woff2) format('woff2'); +} +@font-face { + font-family: 'IBM Plex Mono'; + font-style: italic; + font-weight: 700; + font-display: swap; + src: local(IBM Plex Mono Bold Italic), local(IBMPlexMono-BoldItalic), url(./IBMPlexMono-BoldItalic-SlashedZero.woff2) format('woff2'); +} + +html { + background-color: #fff; +} + +body { + display: flex; + word-wrap: break-word; + font-size: 18px; + line-height: 1.5; + font-family: + IBM Plex Serif, + serif; + font-variant-numeric: slashed-zero; + padding: 0; + margin: 0; + color: #111; +} + +#spec-container { + padding: 0 20px; + max-width: 80rem; + margin: 0 auto; + flex-grow: 1; + flex-basis: 66%; + box-sizing: border-box; + overflow: hidden; + padding-bottom: 1em; +} + +body.oldtoc { + margin: 0 auto; +} + +span[aria-hidden='true'] { + font-size: 0; + white-space: pre; +} + +a { + text-decoration: none; + color: #206ca7; +} + +a:visited { + color: #206ca7; +} + +a:hover { + text-decoration: underline; + color: #239dee; +} + +a.e-user-code, +span.e-user-code { + white-space: nowrap; +} + +a.e-user-code::before, +span.e-user-code::before { + display: none; + color: brown; + background-color: white; + border: 2pt solid brown; + padding: 0.3ex; + margin: 0 0.25em 0 0; + line-height: normal; + vertical-align: middle; + text-transform: uppercase; + font-family: + IBM Plex Sans, + sans-serif; + font-weight: 900; + font-size: x-small; +} + +a.e-user-code:hover::before, +span.e-user-code:hover::before { + background-color: brown; + color: white; +} + +html.show-ao-annotations a.e-user-code::before, +html.show-ao-annotations span.e-user-code::before { + display: inline-block; +} + +a.e-user-code::before, +span.e-user-code::before { + content: 'UC'; +} + +code { + font-weight: bold; + font-family: + Comic Code, + IBM Plex Mono, + monospace; + white-space: pre; +} + +pre code { + font-weight: inherit; +} + +pre code.hljs { + background-color: #fff; + margin: 0; + padding: 0; +} + +ol.toc { + list-style: none; + padding-left: 0; +} + +ol.toc ol.toc { + padding-left: 2ex; + list-style: none; +} + +var { + border-radius: 5px; + color: #218379; + transition: all 0.25s ease-out; + cursor: pointer; + padding: 0 4px; + margin: 0 -4px; + mix-blend-mode: multiply; +} +var.field { + font: inherit; + color: inherit; +} +/* suppress line break opportunities between `.` and `[[FieldName]]` */ +var.field::before { + content: '\2060' +} + +var.referenced { + color: inherit; +} + +var.referenced0 { + background-color: #ffff6c; +} +var.referenced1 { + background-color: #ffa4a8; +} +var.referenced2 { + background-color: #96e885; +} +var.referenced3 { + background-color: #3eeed2; +} +var.referenced4 { + background-color: #eacfb6; +} +var.referenced5 { + background-color: #82ddff; +} +var.referenced6 { + background-color: #ffbcf2; +} + +emu-const { + font-family: + IBM Plex Sans, + sans-serif; + font-variant: small-caps; + text-transform: uppercase; +} + +emu-val { + font-weight: bold; +} + +/* depth 1 */ +emu-alg ol, +/* depth 4 */ +emu-alg ol ol ol ol, +emu-alg ol.nested-thrice, +emu-alg ol.nested-twice ol, +emu-alg ol.nested-once ol ol { + list-style-type: decimal; +} + +/* depth 2 */ +emu-alg ol ol, +emu-alg ol.nested-once, +/* depth 5 */ +emu-alg ol ol ol ol ol, +emu-alg ol.nested-four-times, +emu-alg ol.nested-thrice ol, +emu-alg ol.nested-twice ol ol, +emu-alg ol.nested-once ol ol ol { + list-style-type: lower-alpha; +} + +/* depth 3 */ +emu-alg ol ol ol, +emu-alg ol.nested-twice, +emu-alg ol.nested-once ol, +/* depth 6 */ +emu-alg ol ol ol ol ol ol, +emu-alg ol.nested-lots, +emu-alg ol.nested-four-times ol, +emu-alg ol.nested-thrice ol ol, +emu-alg ol.nested-twice ol ol ol, +emu-alg ol.nested-once ol ol ol ol, +/* depth 7+ */ +emu-alg ol.nested-lots ol { + list-style-type: lower-roman; +} + +emu-eqn { + display: block; + margin-left: 4em; +} + +emu-eqn.inline { + display: inline; + margin: 0; +} + +emu-eqn div:first-child { + margin-left: -2em; +} + +emu-note { + margin: 1em 0; + display: flex; + gap: 1em; + flex-direction: row; + color: inherit; + border-left: 5px solid #52e052; + background: #e9fbe9; + padding: 10px 10px 10px 0; + overflow-x: auto; +} + +emu-note > span.note { + white-space: nowrap; + flex-grow: 0; + flex-shrink: 1; + text-transform: uppercase; + padding-left: 5px; +} + +emu-note[type='editor'] { + border-left-color: #faa; +} + +emu-note > div.note-contents { + flex-grow: 1; + flex-shrink: 1; + overflow: auto; +} + +emu-note > div.note-contents > p:first-of-type { + margin-top: 0; +} + +emu-note > div.note-contents > p:last-of-type { + margin-bottom: 0; +} + +emu-table:not(.code) td code { + white-space: normal; +} + +emu-figure { + display: block; +} + +emu-example { + display: block; + margin: 1em 3em; +} + +emu-example figure figcaption { + margin-top: 0.5em; + text-align: left; +} + +emu-figure figure, +emu-example figure, +emu-table figure { + display: flex; + flex-direction: column; + align-items: center; +} + +emu-production { + display: block; +} + +emu-grammar[type='example'] emu-production, +emu-grammar[type='definition'] emu-production { + margin-top: 1em; + margin-bottom: 1em; + margin-left: 5ex; +} + +emu-grammar.inline, +emu-production.inline, +emu-grammar.inline emu-production emu-rhs, +emu-production.inline emu-rhs, +emu-grammar[collapsed] emu-production emu-rhs { + display: inline; + padding-left: 1ex; + margin-left: 0; +} + +emu-production[collapsed] emu-rhs { + display: inline; + padding-left: 0.5ex; + margin-left: 0; +} + +emu-grammar[collapsed] emu-production, +emu-production[collapsed] { + margin: 0; +} + +emu-constraints { + font-size: 0.75em; + margin-right: 0.5ex; +} + +emu-gann { + margin-right: 0.5ex; +} + +emu-gann emu-t:last-child, +emu-gann emu-gprose:last-child, +emu-gann emu-nt:last-child { + margin-right: 0; +} + +emu-geq { + margin-left: 0.5ex; + font-weight: bold; +} + +emu-oneof { + font-weight: bold; + margin-left: 0.5ex; +} + +emu-nt { + display: inline-block; + font-style: italic; + white-space: nowrap; + text-indent: 0; +} + +emu-nt a, +emu-nt a:visited { + color: #333; +} + +emu-rhs emu-nt { + margin-right: 0.5ex; +} + +emu-t { + display: inline-block; + font-family: + IBM Plex Mono, + monospace; + font-weight: bold; + white-space: nowrap; + text-indent: 0; +} + +emu-production emu-t { + margin-right: 0.5ex; +} + +emu-rhs { + display: block; + padding-left: 75px; + text-indent: -25px; +} + +emu-production:not([collapsed]) emu-rhs { + border: 0.2ex dashed transparent; +} + +emu-production:not([collapsed]) emu-rhs:hover { + border-color: #888; + background-color: #f0f0f0; +} + +emu-mods { + font-size: 0.85em; + vertical-align: sub; + font-style: normal; + font-weight: normal; +} + +emu-params, +emu-opt { + margin-right: 1ex; + font-family: + IBM Plex Mono, + monospace; +} + +emu-params, +emu-constraints { + color: #2aa198; +} + +emu-opt { + color: #b58900; +} + +emu-gprose { + font-size: 0.9em; + font-family: + IBM Plex Sans, + sans-serif; +} + +emu-production emu-gprose { + margin-right: 1ex; +} + +h1.shortname { + color: #f60; + font-size: 1.5em; + margin: 0; +} + +h1.version { + color: #f60; + font-size: 1.5em; +} + +h1.title { + color: #f60; +} + +h1, +h2, +h3, +h4, +h5, +h6 { + position: relative; +} + +h1 .secnum { + text-decoration: none; + margin-right: 5px; +} + +h1 span.title { + order: 2; +} + +h1 { + font-size: 2.67em; + margin-bottom: 0; + line-height: 1em; +} +h2 { + font-size: 2em; +} +h3 { + font-size: 1.56em; +} +h4 { + font-size: 1.25em; +} +h5 { + font-size: 1.11em; +} +h6 { + font-size: 1em; +} + +pre code.hljs { + background: transparent; +} + +emu-clause[id], +emu-annex[id], +emu-intro[id] { + scroll-margin-top: 2ex; +} + +emu-intro h1, +emu-clause h1, +emu-annex h1 { + font-size: 2em; +} +emu-intro h2, +emu-clause h2, +emu-annex h2 { + font-size: 1.56em; +} +emu-intro h3, +emu-clause h3, +emu-annex h3 { + font-size: 1.25em; +} +emu-intro h4, +emu-clause h4, +emu-annex h4 { + font-size: 1.11em; +} +emu-intro h5, +emu-clause h5, +emu-annex h5 { + font-size: 1em; +} +emu-intro h6, +emu-clause h6, +emu-annex h6 { + font-size: 0.9em; +} +emu-intro emu-intro h1, +emu-clause emu-clause h1, +emu-annex emu-annex h1 { + font-size: 1.56em; +} +emu-intro emu-intro h2, +emu-clause emu-clause h2, +emu-annex emu-annex h2 { + font-size: 1.25em; +} +emu-intro emu-intro h3, +emu-clause emu-clause h3, +emu-annex emu-annex h3 { + font-size: 1.11em; +} +emu-intro emu-intro h4, +emu-clause emu-clause h4, +emu-annex emu-annex h4 { + font-size: 1em; +} +emu-intro emu-intro h5, +emu-clause emu-clause h5, +emu-annex emu-annex h5 { + font-size: 0.9em; +} +emu-intro emu-intro emu-intro h1, +emu-clause emu-clause emu-clause h1, +emu-annex emu-annex emu-annex h1 { + font-size: 1.25em; +} +emu-intro emu-intro emu-intro h2, +emu-clause emu-clause emu-clause h2, +emu-annex emu-annex emu-annex h2 { + font-size: 1.11em; +} +emu-intro emu-intro emu-intro h3, +emu-clause emu-clause emu-clause h3, +emu-annex emu-annex emu-annex h3 { + font-size: 1em; +} +emu-intro emu-intro emu-intro h4, +emu-clause emu-clause emu-clause h4, +emu-annex emu-annex emu-annex h4 { + font-size: 0.9em; +} +emu-intro emu-intro emu-intro emu-intro h1, +emu-clause emu-clause emu-clause emu-clause h1, +emu-annex emu-annex emu-annex emu-annex h1 { + font-size: 1.11em; +} +emu-intro emu-intro emu-intro emu-intro h2, +emu-clause emu-clause emu-clause emu-clause h2, +emu-annex emu-annex emu-annex emu-annex h2 { + font-size: 1em; +} +emu-intro emu-intro emu-intro emu-intro h3, +emu-clause emu-clause emu-clause emu-clause h3, +emu-annex emu-annex emu-annex emu-annex h3 { + font-size: 0.9em; +} +emu-intro emu-intro emu-intro emu-intro emu-intro h1, +emu-clause emu-clause emu-clause emu-clause emu-clause h1, +emu-annex emu-annex emu-annex emu-annex emu-annex h1 { + font-size: 1em; +} +emu-intro emu-intro emu-intro emu-intro emu-intro h2, +emu-clause emu-clause emu-clause emu-clause emu-clause h2, +emu-annex emu-annex emu-annex emu-annex emu-annex h2 { + font-size: 0.9em; +} +emu-intro emu-intro emu-intro emu-intro emu-intro emu-intro h1, +emu-clause emu-clause emu-clause emu-clause emu-clause emu-clause h1, +emu-annex emu-annex emu-annex emu-annex emu-annex emu-annex h1 { + font-size: 0.9em; +} + +emu-clause, +emu-intro, +emu-annex { + display: block; +} + +/* these values are twice the font-size for the

titles for clauses */ +emu-intro, +emu-clause, +emu-annex { + margin-top: 4em; +} +emu-intro emu-intro, +emu-clause emu-clause, +emu-annex emu-annex { + margin-top: 3.12em; +} +emu-intro emu-intro emu-intro, +emu-clause emu-clause emu-clause, +emu-annex emu-annex emu-annex { + margin-top: 2.5em; +} +emu-intro emu-intro emu-intro emu-intro, +emu-clause emu-clause emu-clause emu-clause, +emu-annex emu-annex emu-annex emu-annex { + margin-top: 2.22em; +} +emu-intro emu-intro emu-intro emu-intro emu-intro, +emu-clause emu-clause emu-clause emu-clause emu-clause, +emu-annex emu-annex emu-annex emu-annex emu-annex { + margin-top: 2em; +} +emu-intro emu-intro emu-intro emu-intro emu-intro emu-intro, +emu-clause emu-clause emu-clause emu-clause emu-clause emu-clause, +emu-annex emu-annex emu-annex emu-annex emu-annex emu-annex { + margin-top: 1.8em; +} + +#spec-container > emu-intro:first-of-type, +#spec-container > emu-clause:first-of-type, +#spec-container > emu-annex:first-of-type { + margin-top: 0; +} + +/* Figures and tables */ +figure { + display: block; + overflow-x: auto; + margin: 1.5em 0; +} +figure object { + display: block; + margin: 0 auto; +} +figure table.real-table { + margin: 0 auto; +} +figure figcaption { + display: block; + color: #555555; + font-weight: bold; + text-align: center; +} + +emu-table table { + margin: 0 auto; +} + +emu-table table, +table.real-table { + border-collapse: collapse; +} + +emu-table td, +emu-table th, +table.real-table td, +table.real-table th { + border: 1px solid black; + padding: 0.4em; + vertical-align: baseline; +} +emu-table th, +emu-table thead td, +table.real-table th { + background-color: #eeeeee; +} + +emu-table td { + background: #fff; +} + +/* Note: the left content edges of table.lightweight-table >tbody >tr >td + and div.display line up. */ +table.lightweight-table { + border-collapse: collapse; + margin: 0 0 0 1.5em; +} +table.lightweight-table td, +table.lightweight-table th { + border: none; + padding: 0 0.5em; + vertical-align: baseline; +} + +/* for non-clause-like link targets, apply a fading highlight + and a persistent focus-associated highlight */ +@keyframes highlight-target-bg { + 0% { + background-color: rgba(249, 241, 172, 1); + } + 100% { + background-color: rgba(249, 241, 172, 0); + } +} +#spec-container :target:not(emu-annex, emu-clause, emu-intro, emu-note, body) { + animation: highlight-target-bg 2.5s ease-out; +} +#spec-container :target:focus-within:not(:has(:not(a))) { + animation: none; + background-color: rgba(249, 241, 172, 1); +} + +/* diff styles */ +ins { + background-color: #e0f8e0; + text-decoration: none; + border-bottom: 1px solid #396; +} + +del { + background-color: #fee; +} + +ins.block, +del.block, +emu-production > ins, +emu-production > del, +emu-grammar > ins, +emu-grammar > del { + display: block; +} +emu-rhs > ins, +emu-rhs > del { + display: inline; +} + +tr.ins > td > ins { + border-bottom: none; +} + +tr.ins > td { + background-color: #e0f8e0; +} + +tr.del > td { + background-color: #fee; +} + +/* Menu Styles */ +#menu-toggle { + font-size: 2em; + + position: fixed; + top: 0; + left: 0; + width: 1.5em; + height: 1.5em; + z-index: 3; + visibility: hidden; + color: #1567a2; + background-color: #fff; + + line-height: 1.5em; + text-align: center; + -webkit-touch-callout: none; + -webkit-user-select: none; + -khtml-user-select: none; + -moz-user-select: none; + -ms-user-select: none; + user-select: none; + + cursor: pointer; +} + +#menu { + display: flex; + flex-direction: column; + width: 33%; + height: 100vh; + max-width: 500px; + box-sizing: border-box; + background-color: #ddd; + overflow: hidden; + transition: opacity 0.1s linear; + padding: 0 5px; + position: fixed; + left: 0; + top: 0; + border-right: 2px solid #bbb; + + z-index: 2; +} + +.menu-spacer { + flex-basis: 33%; + max-width: 500px; + flex-grow: 0; + flex-shrink: 0; +} + +#menu a { + color: #1567a2; +} + +#menu.active { + display: flex; + opacity: 1; + z-index: 2; +} + +#menu-pins { + flex-grow: 1; + display: none; +} + +#menu-pins.active { + display: block; +} + +#menu-pins .unpin-all { + border: none; + background: #ccc; + border-radius: 4px; + height: 18px; + font-size: 12px; + margin: 0 5px 0 10px; + font-family: IBM Plex Sans; +} +#menu-pins .unpin-all:hover { + background: #ddd; +} + +#menu-pins-list { + margin: 0; + padding: 0; + counter-reset: pins-counter; +} + +#menu-pins-list > li { + display: flex; + align-items: stretch; + gap: 4px; + margin: 3px 1px; + border-radius: 4px; +} + +#menu-pins-list > li > a { + flex-grow: 1; + align-self: center; +} + +#menu-pins-list > li:before, +#menu-pins-list > li > .unpin { + flex-shrink: 0; + flex-grow: 0; + text-align: center; + padding: 1px 3px; + border-radius: 4px; + background: none; + border: none; +} +#menu-pins-list > li:before, +#menu-pins-list > li > .unpin:hover { + background: #ccc; +} + +#menu-pins-list > li > .unpin, +#menu-pins .unpin-all { + cursor: pointer; +} +#menu-pins-list > li > .unpin:hover, +#menu-pins .unpin-all:hover { + color: #bb1212; +} + +#menu-pins-list > li:before { + content: counter(pins-counter); + counter-increment: pins-counter; + font-size: 16px; +} + +#menu-toc > ol { + padding: 0; + flex-grow: 1; +} + +#menu-toc > ol li { + padding: 0; +} + +#menu-toc > ol, +#menu-toc > ol ol { + list-style-type: none; + margin: 0; + padding: 0; +} + +#menu-toc > ol ol { + padding-left: 0.75em; +} + +#menu-toc li { + text-overflow: ellipsis; + overflow: hidden; + white-space: nowrap; +} + +#menu-toc .item-toggle { + display: inline-block; + transform: rotate(0deg); + transition: transform 0.1s ease; + background: url(); + background-repeat: no-repeat; + background-position: center; + background-size: auto 50%; + text-align: center; + width: 1em; + + color: transparent; + + -webkit-touch-callout: none; + -webkit-user-select: none; + -khtml-user-select: none; + -moz-user-select: none; + -ms-user-select: none; + user-select: none; + + cursor: pointer; +} + +#menu-toc .item-toggle-none { + display: inline-block; + width: 1em; +} + +#menu-toc li.active > .item-toggle, +#menu-toc li.revealed > .item-toggle { + transform: rotate(90deg); +} + +#menu-toc li > ol { + display: none; +} + +#menu-toc li.active > ol { + display: block; +} + +#menu-toc li.revealed > a { + background-color: #ccc; + font-weight: bold; +} + +#menu-toc li.revealed-leaf > a { + color: #206ca7; +} + +#menu-toc li.revealed > ol { + display: block; +} + +#menu-toc li > a { + padding: 2px 5px; +} + +#menu > * { + margin-bottom: 5px; +} + +.menu-pane-header { + padding: 2px 8px; + background-color: #bbb; + font-weight: bold; + letter-spacing: 1px; + flex-grow: 0; + flex-shrink: 0; + font-size: 80%; + user-select: none; +} + +.menu-pane-header emu-opt, +.menu-pane-header emu-t, +.menu-pane-header emu-nt { + margin-right: 0px; + display: inline; + color: inherit; +} + +.menu-pane-header emu-rhs { + display: inline; + padding-left: 0px; + text-indent: 0px; +} + +.menu-pane-header emu-geq { + margin-left: 0px; +} + +a.menu-pane-header-production { + color: inherit; +} + +.menu-pane-header-production { + text-transform: none; + letter-spacing: 1.5px; + padding-left: 0.5em; +} + +#menu-toc { + display: flex; + flex-direction: column; + width: 100%; + overflow: hidden; + flex-grow: 1; +} + +#menu-toc ol.toc { + overflow-x: hidden; + overflow-y: auto; +} + +#menu-search { + position: relative; + flex-grow: 0; + flex-shrink: 0; + width: 100%; + + display: flex; + flex-direction: column; + + max-height: 300px; +} + +#menu-trace-list { + display: none; +} + +#menu-search-box { + box-sizing: border-box; + display: block; + width: 100%; + margin: 5px 0 0 0; + font-size: 1em; + padding: 2px; + background-color: #bbb; + border: 1px solid #999; +} + +#menu-search-results { + overflow-x: hidden; + overflow-y: auto; +} + +li.menu-search-result-clause:before { + content: 'clause'; + width: 40px; + display: inline-block; + text-align: right; + padding-right: 1ex; + color: #666; + font-size: 75%; +} +li.menu-search-result-op:before { + content: 'op'; + width: 40px; + display: inline-block; + text-align: right; + padding-right: 1ex; + color: #666; + font-size: 75%; +} + +li.menu-search-result-prod:before { + content: 'prod'; + width: 40px; + display: inline-block; + text-align: right; + padding-right: 1ex; + color: #666; + font-size: 75%; +} + +li.menu-search-result-term:before { + content: 'term'; + width: 40px; + display: inline-block; + text-align: right; + padding-right: 1ex; + color: #666; + font-size: 75%; +} + +#menu-search-results ul { + padding: 0 5px; + margin: 0; +} + +#menu-search-results li { + white-space: nowrap; + text-overflow: ellipsis; +} + +#menu-trace-list { + counter-reset: item; + margin: 0 0 0 20px; + padding: 0; +} +#menu-trace-list li { + display: block; + white-space: nowrap; +} + +#menu-trace-list li .secnum:after { + content: ' '; +} +#menu-trace-list li:before { + content: counter(item) ' '; + background-color: #222; + counter-increment: item; + color: #999; + width: 20px; + height: 20px; + line-height: 20px; + display: inline-block; + text-align: center; + margin: 2px 4px 2px 0; +} + +@media (max-width: 1000px) { + body { + margin: 0; + display: block; + } + + #menu { + display: none; + padding-top: 3em; + width: 450px; + } + + #menu.active { + position: fixed; + height: 100%; + left: 0; + top: 0; + right: 300px; + } + + #menu-toggle { + visibility: visible; + } + + #spec-container { + padding: 0 5px; + } + + #references-pane-spacer { + display: none; + } +} + +@media only screen and (max-width: 800px) { + #menu { + width: 100%; + } + + h1 .secnum:empty { + margin: 0; + padding: 0; + } +} + +/* Toolbox */ +.toolbox-container { + position: absolute; + display: none; + padding-bottom: 7px; +} + +.toolbox-container.active { + display: inline-block; +} + +.toolbox { + position: relative; + background: #ddd; + border: 1px solid #aaa; + color: #eee; + padding: 5px 7px; + border-radius: 3px; +} + +.toolbox a { + text-decoration: none; + padding: 0 3px; + white-space: nowrap; +} + +.toolbox a:hover { + text-decoration: underline; +} + +.toolbox:after, +.toolbox:before { + top: 100%; + left: 15px; + border: solid transparent; + content: ' '; + height: 0; + width: 0; + position: absolute; + pointer-events: none; +} + +.toolbox:after { + border-color: rgba(0, 0, 0, 0); + border-top-color: #ddd; + border-width: 10px; + margin-left: -10px; +} +.toolbox:before { + border-color: rgba(204, 204, 204, 0); + border-top-color: #aaa; + border-width: 12px; + margin-left: -12px; +} + +#references-pane-container { + position: fixed; + bottom: 0; + left: 0; + right: 0; + display: none; + background-color: #ddd; + z-index: 1; +} + +#references-pane-table-container { + overflow-x: hidden; + overflow-y: auto; + min-height: 35px; + max-height: 85vh; +} + +#references-pane { + flex-grow: 1; + overflow: hidden; + display: flex; + flex-direction: column; +} + +#references-pane > .menu-pane-header { + cursor: row-resize; +} + +#references-pane-container.active { + display: flex; +} + +#references-pane-close:after { + content: '\2716'; + float: right; + cursor: pointer; +} + +#references-pane table tbody { + vertical-align: baseline; +} + +#references-pane table tr td:first-child { + text-align: right; + padding-right: 5px; +} + +[normative-optional], +[deprecated], +[legacy] { + border-left: 5px solid #ff6600; + padding: 0.5em; + background: #ffeedd; +} + +.attributes-tag { + text-transform: uppercase; + color: #884400; +} + +.attributes-tag a { + color: #884400; +} + +/* Shortcuts help dialog */ + +#shortcuts-help { + position: fixed; + left: 5%; + margin: 0 auto; + right: 5%; + z-index: 10; + top: 10%; + top: calc(5vw + 5vh); + padding: 30px 90px; + max-width: 500px; + outline: solid 10000px rgba(255, 255, 255, 0.6); + border-radius: 5px; + border-width: 1px 1px 0 1px; + background-color: #ddd; + display: none; +} + +#shortcuts-help.active { + display: block; +} + +#shortcuts-help ul { + padding: 0; +} + +#shortcuts-help li { + display: flex; + justify-content: space-between; +} + +#shortcuts-help code { + padding: 3px 10px; + border-radius: 3px; + border-width: 1px 1px 0 1px; + border-color: #bbb; + background-color: #eee; + box-shadow: inset 0 -1px 0 #ccc; +} diff --git a/ecmarkup.js b/ecmarkup.js new file mode 100644 index 00000000..c12d5dd5 --- /dev/null +++ b/ecmarkup.js @@ -0,0 +1,1609 @@ +'use strict'; +let sdoBox = { + init() { + this.$alternativeId = null; + this.$outer = document.createElement('div'); + this.$outer.classList.add('toolbox-container'); + this.$container = document.createElement('div'); + this.$container.classList.add('toolbox'); + this.$displayLink = document.createElement('a'); + this.$displayLink.setAttribute('href', '#'); + this.$displayLink.textContent = 'Syntax-Directed Operations'; + this.$displayLink.addEventListener('click', e => { + e.preventDefault(); + e.stopPropagation(); + referencePane.showSDOs(sdoMap[this.$alternativeId] || {}, this.$alternativeId); + }); + this.$container.appendChild(this.$displayLink); + this.$outer.appendChild(this.$container); + document.body.appendChild(this.$outer); + }, + + activate(el) { + clearTimeout(this.deactiveTimeout); + Toolbox.deactivate(); + this.$alternativeId = el.id; + let numSdos = Object.keys(sdoMap[this.$alternativeId] || {}).length; + this.$displayLink.textContent = 'Syntax-Directed Operations (' + numSdos + ')'; + this.$outer.classList.add('active'); + let top = el.offsetTop - this.$outer.offsetHeight; + let left = el.offsetLeft + 50 - 10; // 50px = padding-left(=75px) + text-indent(=-25px) + this.$outer.setAttribute('style', 'left: ' + left + 'px; top: ' + top + 'px'); + if (top < document.body.scrollTop) { + this.$container.scrollIntoView(); + } + }, + + deactivate() { + clearTimeout(this.deactiveTimeout); + this.$outer.classList.remove('active'); + }, +}; + +document.addEventListener('DOMContentLoaded', () => { + if (typeof sdoMap == 'undefined') { + console.error('could not find sdo map'); + return; + } + sdoBox.init(); + + let insideTooltip = false; + sdoBox.$outer.addEventListener('pointerenter', () => { + insideTooltip = true; + }); + sdoBox.$outer.addEventListener('pointerleave', () => { + insideTooltip = false; + sdoBox.deactivate(); + }); + + sdoBox.deactiveTimeout = null; + [].forEach.call(document.querySelectorAll('emu-grammar[type=definition] emu-rhs'), node => { + node.addEventListener('pointerenter', function () { + sdoBox.activate(this); + }); + + node.addEventListener('pointerleave', () => { + sdoBox.deactiveTimeout = setTimeout(() => { + if (!insideTooltip) { + sdoBox.deactivate(); + } + }, 500); + }); + }); + + document.addEventListener( + 'keydown', + debounce(e => { + if (e.code === 'Escape') { + sdoBox.deactivate(); + } + }), + ); +}); + +'use strict'; +function Search(menu) { + this.menu = menu; + this.$search = document.getElementById('menu-search'); + this.$searchBox = document.getElementById('menu-search-box'); + this.$searchResults = document.getElementById('menu-search-results'); + + this.loadBiblio(); + + document.addEventListener('keydown', this.documentKeydown.bind(this)); + + this.$searchBox.addEventListener( + 'keydown', + debounce(this.searchBoxKeydown.bind(this), { stopPropagation: true }), + ); + this.$searchBox.addEventListener( + 'keyup', + debounce(this.searchBoxKeyup.bind(this), { stopPropagation: true }), + ); + + // Perform an initial search if the box is not empty. + if (this.$searchBox.value) { + this.search(this.$searchBox.value); + } +} + +Search.prototype.loadBiblio = function () { + if (typeof biblio === 'undefined') { + console.error('could not find biblio'); + this.biblio = { refToClause: {}, entries: [] }; + } else { + this.biblio = biblio; + this.biblio.clauses = this.biblio.entries.filter(e => e.type === 'clause'); + this.biblio.byId = this.biblio.entries.reduce((map, entry) => { + map[entry.id] = entry; + return map; + }, {}); + let refParentClause = Object.create(null); + this.biblio.refParentClause = refParentClause; + let refsByClause = this.biblio.refsByClause; + Object.keys(refsByClause).forEach(clause => { + refsByClause[clause].forEach(ref => { + refParentClause[ref] = clause; + }); + }); + } +}; + +Search.prototype.documentKeydown = function (e) { + if (e.key === '/') { + e.preventDefault(); + e.stopPropagation(); + this.triggerSearch(); + } +}; + +Search.prototype.searchBoxKeydown = function (e) { + e.stopPropagation(); + e.preventDefault(); + if (e.keyCode === 191 && e.target.value.length === 0) { + e.preventDefault(); + } else if (e.keyCode === 13) { + e.preventDefault(); + this.selectResult(); + } +}; + +Search.prototype.searchBoxKeyup = function (e) { + if (e.keyCode === 13 || e.keyCode === 9) { + return; + } + + this.search(e.target.value); +}; + +Search.prototype.triggerSearch = function () { + if (this.menu.isVisible()) { + this._closeAfterSearch = false; + } else { + this._closeAfterSearch = true; + this.menu.show(); + } + + this.$searchBox.focus(); + this.$searchBox.select(); +}; +// bit 12 - Set if the result starts with searchString +// bits 8-11: 8 - number of chunks multiplied by 2 if cases match, otherwise 1. +// bits 1-7: 127 - length of the entry +// General scheme: prefer case sensitive matches with fewer chunks, and otherwise +// prefer shorter matches. +function relevance(result) { + let relevance = 0; + + relevance = Math.max(0, 8 - result.match.chunks) << 7; + + if (result.match.caseMatch) { + relevance *= 2; + } + + if (result.match.prefix) { + relevance += 2048; + } + + relevance += Math.max(0, 255 - result.key.length); + + return relevance; +} + +Search.prototype.search = function (searchString) { + if (searchString === '') { + this.displayResults([]); + this.hideSearch(); + return; + } else { + this.showSearch(); + } + + if (searchString.length === 1) { + this.displayResults([]); + return; + } + + let results; + + if (/^[\d.]*$/.test(searchString)) { + results = this.biblio.clauses + .filter(clause => clause.number.substring(0, searchString.length) === searchString) + .map(clause => ({ key: getKey(clause), entry: clause })); + } else { + results = []; + + for (let i = 0; i < this.biblio.entries.length; i++) { + let entry = this.biblio.entries[i]; + let key = getKey(entry); + if (!key) { + // biblio entries without a key aren't searchable + continue; + } + + let match = fuzzysearch(searchString, key); + if (match) { + results.push({ key, entry, match }); + } + } + + results.forEach(result => { + result.relevance = relevance(result, searchString); + }); + + results = results.sort((a, b) => b.relevance - a.relevance); + } + + if (results.length > 50) { + results = results.slice(0, 50); + } + + this.displayResults(results); +}; +Search.prototype.hideSearch = function () { + this.$search.classList.remove('active'); +}; + +Search.prototype.showSearch = function () { + this.$search.classList.add('active'); +}; + +Search.prototype.selectResult = function () { + let $first = this.$searchResults.querySelector('li:first-child a'); + + if ($first) { + document.location = $first.getAttribute('href'); + } + + this.$searchBox.value = ''; + this.$searchBox.blur(); + this.displayResults([]); + this.hideSearch(); + + if (this._closeAfterSearch) { + this.menu.hide(); + } +}; + +Search.prototype.displayResults = function (results) { + if (results.length > 0) { + this.$searchResults.classList.remove('no-results'); + + let html = ''; + + this.$searchResults.innerHTML = html; + } else { + this.$searchResults.innerHTML = ''; + this.$searchResults.classList.add('no-results'); + } +}; + +function getKey(item) { + if (item.key) { + return item.key; + } + switch (item.type) { + case 'clause': + return item.title || item.titleHTML; + case 'production': + return item.name; + case 'op': + return item.aoid; + case 'term': + return item.term; + case 'table': + case 'figure': + case 'example': + case 'note': + return item.caption; + case 'step': + return item.id; + default: + throw new Error("Can't get key for " + item.type); + } +} + +function Menu() { + this.$toggle = document.getElementById('menu-toggle'); + this.$menu = document.getElementById('menu'); + this.$toc = document.querySelector('menu-toc > ol'); + this.$pins = document.querySelector('#menu-pins'); + this.$pinList = document.getElementById('menu-pins-list'); + this.$toc = document.querySelector('#menu-toc > ol'); + this.$specContainer = document.getElementById('spec-container'); + this.search = new Search(this); + + this._pinnedIds = {}; + this.loadPinEntries(); + + // unpin all button + document + .querySelector('#menu-pins .unpin-all') + .addEventListener('click', this.unpinAll.bind(this)); + + // individual unpinning buttons + this.$pinList.addEventListener('click', this.pinListClick.bind(this)); + + // toggle menu + this.$toggle.addEventListener('click', this.toggle.bind(this)); + + // keydown events for pinned clauses + document.addEventListener('keydown', this.documentKeydown.bind(this)); + + // toc expansion + let tocItems = this.$menu.querySelectorAll('#menu-toc li'); + for (let i = 0; i < tocItems.length; i++) { + let $item = tocItems[i]; + $item.addEventListener('click', event => { + $item.classList.toggle('active'); + event.stopPropagation(); + }); + } + + // close toc on toc item selection + let tocLinks = this.$menu.querySelectorAll('#menu-toc li > a'); + for (let i = 0; i < tocLinks.length; i++) { + let $link = tocLinks[i]; + $link.addEventListener('click', event => { + this.toggle(); + event.stopPropagation(); + }); + } + + // update active clause on scroll + window.addEventListener('scroll', debounce(this.updateActiveClause.bind(this))); + this.updateActiveClause(); + + // prevent menu scrolling from scrolling the body + this.$toc.addEventListener('wheel', e => { + let target = e.currentTarget; + let offTop = e.deltaY < 0 && target.scrollTop === 0; + if (offTop) { + e.preventDefault(); + } + let offBottom = e.deltaY > 0 && target.offsetHeight + target.scrollTop >= target.scrollHeight; + + if (offBottom) { + e.preventDefault(); + } + }); +} + +Menu.prototype.documentKeydown = function (e) { + e.stopPropagation(); + if (e.keyCode === 80) { + this.togglePinEntry(); + } else if (e.keyCode >= 48 && e.keyCode < 58) { + this.selectPin((e.keyCode - 9) % 10); + } +}; + +Menu.prototype.updateActiveClause = function () { + this.setActiveClause(findActiveClause(this.$specContainer)); +}; + +Menu.prototype.setActiveClause = function (clause) { + this.$activeClause = clause; + this.revealInToc(this.$activeClause); +}; + +Menu.prototype.revealInToc = function (path) { + let current = this.$toc.querySelectorAll('li.revealed'); + for (let i = 0; i < current.length; i++) { + current[i].classList.remove('revealed'); + current[i].classList.remove('revealed-leaf'); + } + + current = this.$toc; + let index = 0; + outer: while (index < path.length) { + let children = current.children; + for (let i = 0; i < children.length; i++) { + if ('#' + path[index].id === children[i].children[1].hash) { + children[i].classList.add('revealed'); + if (index === path.length - 1) { + children[i].classList.add('revealed-leaf'); + let rect = children[i].getBoundingClientRect(); + // this.$toc.getBoundingClientRect().top; + let tocRect = this.$toc.getBoundingClientRect(); + if (rect.top + 10 > tocRect.bottom) { + this.$toc.scrollTop = + this.$toc.scrollTop + (rect.top - tocRect.bottom) + (rect.bottom - rect.top); + } else if (rect.top < tocRect.top) { + this.$toc.scrollTop = this.$toc.scrollTop - (tocRect.top - rect.top); + } + } + current = children[i].querySelector('ol'); + index++; + continue outer; + } + } + console.log('could not find location in table of contents', path); + break; + } +}; + +function findActiveClause(root, path) { + path = path || []; + + let visibleClauses = getVisibleClauses(root, path); + let midpoint = Math.floor(window.innerHeight / 2); + + for (let [$clause, path] of visibleClauses) { + let { top: clauseTop, bottom: clauseBottom } = $clause.getBoundingClientRect(); + let isFullyVisibleAboveTheFold = + clauseTop > 0 && clauseTop < midpoint && clauseBottom < window.innerHeight; + if (isFullyVisibleAboveTheFold) { + return path; + } + } + + visibleClauses.sort(([, pathA], [, pathB]) => pathB.length - pathA.length); + for (let [$clause, path] of visibleClauses) { + let { top: clauseTop, bottom: clauseBottom } = $clause.getBoundingClientRect(); + let $header = $clause.querySelector('h1'); + let clauseStyles = getComputedStyle($clause); + let marginTop = Math.max( + 0, + parseInt(clauseStyles['margin-top']), + parseInt(getComputedStyle($header)['margin-top']), + ); + let marginBottom = Math.max(0, parseInt(clauseStyles['margin-bottom'])); + let crossesMidpoint = + clauseTop - marginTop <= midpoint && clauseBottom + marginBottom >= midpoint; + if (crossesMidpoint) { + return path; + } + } + + return path; +} + +function getVisibleClauses(root, path) { + let childClauses = getChildClauses(root); + path = path || []; + + let result = []; + + let seenVisibleClause = false; + for (let $clause of childClauses) { + let { top: clauseTop, bottom: clauseBottom } = $clause.getBoundingClientRect(); + let isPartiallyVisible = + (clauseTop > 0 && clauseTop < window.innerHeight) || + (clauseBottom > 0 && clauseBottom < window.innerHeight) || + (clauseTop < 0 && clauseBottom > window.innerHeight); + + if (isPartiallyVisible) { + seenVisibleClause = true; + let innerPath = path.concat($clause); + result.push([$clause, innerPath]); + result.push(...getVisibleClauses($clause, innerPath)); + } else if (seenVisibleClause) { + break; + } + } + + return result; +} + +function* getChildClauses(root) { + for (let el of root.children) { + switch (el.nodeName) { + // descend into + case 'EMU-IMPORT': + yield* getChildClauses(el); + break; + + // accept , , and + case 'EMU-CLAUSE': + case 'EMU-INTRO': + case 'EMU-ANNEX': + yield el; + } + } +} + +Menu.prototype.toggle = function () { + this.$menu.classList.toggle('active'); +}; + +Menu.prototype.show = function () { + this.$menu.classList.add('active'); +}; + +Menu.prototype.hide = function () { + this.$menu.classList.remove('active'); +}; + +Menu.prototype.isVisible = function () { + return this.$menu.classList.contains('active'); +}; + +Menu.prototype.showPins = function () { + this.$pins.classList.add('active'); +}; + +Menu.prototype.hidePins = function () { + this.$pins.classList.remove('active'); +}; + +Menu.prototype.addPinEntry = function (id) { + let entry = this.search.biblio.byId[id]; + if (!entry) { + // id was deleted after pin (or something) so remove it + delete this._pinnedIds[id]; + this.persistPinEntries(); + return; + } + + let text; + if (entry.type === 'clause') { + let prefix; + if (entry.number) { + prefix = entry.number + ' '; + } else { + prefix = ''; + } + text = `${prefix}${entry.titleHTML}`; + } else { + text = getKey(entry); + } + + let link = `${text}`; + this.$pinList.innerHTML += `
  • ${link}
  • `; + + if (Object.keys(this._pinnedIds).length === 0) { + this.showPins(); + } + this._pinnedIds[id] = true; + this.persistPinEntries(); +}; + +Menu.prototype.removePinEntry = function (id) { + let item = this.$pinList.querySelector(`li[data-section-id="${id}"]`); + this.$pinList.removeChild(item); + delete this._pinnedIds[id]; + if (Object.keys(this._pinnedIds).length === 0) { + this.hidePins(); + } + + this.persistPinEntries(); +}; + +Menu.prototype.unpinAll = function () { + for (let id of Object.keys(this._pinnedIds)) { + this.removePinEntry(id); + } +}; + +Menu.prototype.pinListClick = function (event) { + if (event?.target?.classList.contains('unpin')) { + let id = event.target.parentNode.dataset.sectionId; + if (id) { + this.removePinEntry(id); + } + } +}; + +Menu.prototype.persistPinEntries = function () { + try { + if (!window.localStorage) return; + } catch (e) { + return; + } + + localStorage.pinEntries = JSON.stringify(Object.keys(this._pinnedIds)); +}; + +Menu.prototype.loadPinEntries = function () { + try { + if (!window.localStorage) return; + } catch (e) { + return; + } + + let pinsString = window.localStorage.pinEntries; + if (!pinsString) return; + let pins = JSON.parse(pinsString); + for (let i = 0; i < pins.length; i++) { + this.addPinEntry(pins[i]); + } +}; + +Menu.prototype.togglePinEntry = function (id) { + if (!id) { + id = this.$activeClause[this.$activeClause.length - 1].id; + } + + if (this._pinnedIds[id]) { + this.removePinEntry(id); + } else { + this.addPinEntry(id); + } +}; + +Menu.prototype.selectPin = function (num) { + if (num >= this.$pinList.children.length) return; + document.location = this.$pinList.children[num].children[0].href; +}; + +let menu; + +document.addEventListener('DOMContentLoaded', init); + +function debounce(fn, opts) { + opts = opts || {}; + let timeout; + return function (e) { + if (opts.stopPropagation) { + e.stopPropagation(); + } + let args = arguments; + if (timeout) { + clearTimeout(timeout); + } + timeout = setTimeout(() => { + timeout = null; + fn.apply(this, args); + }, 150); + }; +} + +let CLAUSE_NODES = ['EMU-CLAUSE', 'EMU-INTRO', 'EMU-ANNEX']; +function findContainer($elem) { + let parentClause = $elem.parentNode; + while (parentClause && CLAUSE_NODES.indexOf(parentClause.nodeName) === -1) { + parentClause = parentClause.parentNode; + } + return parentClause; +} + +function findLocalReferences(parentClause, name) { + let vars = parentClause.querySelectorAll('var'); + let references = []; + + for (let i = 0; i < vars.length; i++) { + let $var = vars[i]; + + if ($var.innerHTML === name) { + references.push($var); + } + } + + return references; +} + +let REFERENCED_CLASSES = Array.from({ length: 7 }, (x, i) => `referenced${i}`); +function chooseHighlightIndex(parentClause) { + let counts = REFERENCED_CLASSES.map($class => parentClause.getElementsByClassName($class).length); + // Find the earliest index with the lowest count. + let minCount = Infinity; + let index = null; + for (let i = 0; i < counts.length; i++) { + if (counts[i] < minCount) { + minCount = counts[i]; + index = i; + } + } + return index; +} + +function toggleFindLocalReferences($elem) { + let parentClause = findContainer($elem); + let references = findLocalReferences(parentClause, $elem.innerHTML); + if ($elem.classList.contains('referenced')) { + references.forEach($reference => { + $reference.classList.remove('referenced', ...REFERENCED_CLASSES); + }); + } else { + let index = chooseHighlightIndex(parentClause); + references.forEach($reference => { + $reference.classList.add('referenced', `referenced${index}`); + }); + } +} + +function installFindLocalReferences() { + document.addEventListener('click', e => { + if (e.target.nodeName === 'VAR') { + toggleFindLocalReferences(e.target); + } + }); +} + +document.addEventListener('DOMContentLoaded', installFindLocalReferences); + +// The following license applies to the fuzzysearch function +// The MIT License (MIT) +// Copyright © 2015 Nicolas Bevacqua +// Copyright © 2016 Brian Terlson +// Permission is hereby granted, free of charge, to any person obtaining a copy of +// this software and associated documentation files (the "Software"), to deal in +// the Software without restriction, including without limitation the rights to +// use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of +// the Software, and to permit persons to whom the Software is furnished to do so, +// subject to the following conditions: + +// The above copyright notice and this permission notice shall be included in all +// copies or substantial portions of the Software. + +// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS +// FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR +// COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER +// IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN +// CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. +function fuzzysearch(searchString, haystack, caseInsensitive) { + let tlen = haystack.length; + let qlen = searchString.length; + let chunks = 1; + let finding = false; + + if (qlen > tlen) { + return false; + } + + if (qlen === tlen) { + if (searchString === haystack) { + return { caseMatch: true, chunks: 1, prefix: true }; + } else if (searchString.toLowerCase() === haystack.toLowerCase()) { + return { caseMatch: false, chunks: 1, prefix: true }; + } else { + return false; + } + } + + let j = 0; + outer: for (let i = 0; i < qlen; i++) { + let nch = searchString[i]; + while (j < tlen) { + let targetChar = haystack[j++]; + if (targetChar === nch) { + finding = true; + continue outer; + } + if (finding) { + chunks++; + finding = false; + } + } + + if (caseInsensitive) { + return false; + } + + return fuzzysearch(searchString.toLowerCase(), haystack.toLowerCase(), true); + } + + return { caseMatch: !caseInsensitive, chunks, prefix: j <= qlen }; +} + +let referencePane = { + init() { + this.$container = document.createElement('div'); + this.$container.setAttribute('id', 'references-pane-container'); + + let $spacer = document.createElement('div'); + $spacer.setAttribute('id', 'references-pane-spacer'); + $spacer.classList.add('menu-spacer'); + + this.$pane = document.createElement('div'); + this.$pane.setAttribute('id', 'references-pane'); + + this.$container.appendChild($spacer); + this.$container.appendChild(this.$pane); + + this.$header = document.createElement('div'); + this.$header.classList.add('menu-pane-header'); + this.$headerText = document.createElement('span'); + this.$header.appendChild(this.$headerText); + this.$headerRefId = document.createElement('a'); + this.$header.appendChild(this.$headerRefId); + this.$header.addEventListener('pointerdown', e => { + this.dragStart(e); + }); + + this.$closeButton = document.createElement('span'); + this.$closeButton.setAttribute('id', 'references-pane-close'); + this.$closeButton.addEventListener('click', () => { + this.deactivate(); + }); + this.$header.appendChild(this.$closeButton); + + this.$pane.appendChild(this.$header); + this.$tableContainer = document.createElement('div'); + this.$tableContainer.setAttribute('id', 'references-pane-table-container'); + + this.$table = document.createElement('table'); + this.$table.setAttribute('id', 'references-pane-table'); + + this.$tableBody = this.$table.createTBody(); + + this.$tableContainer.appendChild(this.$table); + this.$pane.appendChild(this.$tableContainer); + + if (menu != null) { + menu.$specContainer.appendChild(this.$container); + } + }, + + activate() { + this.$container.classList.add('active'); + }, + + deactivate() { + this.$container.classList.remove('active'); + this.state = null; + }, + + showReferencesFor(entry) { + this.activate(); + this.state = { type: 'ref', id: entry.id }; + this.$headerText.textContent = 'References to '; + let newBody = document.createElement('tbody'); + let previousId; + let previousCell; + let dupCount = 0; + this.$headerRefId.innerHTML = getKey(entry); + this.$headerRefId.setAttribute('href', makeLinkToId(entry.id)); + this.$headerRefId.style.display = 'inline'; + (entry.referencingIds || []) + .map(id => { + let cid = menu.search.biblio.refParentClause[id]; + let clause = menu.search.biblio.byId[cid]; + if (clause == null) { + throw new Error('could not find clause for id ' + cid); + } + return { id, clause }; + }) + .sort((a, b) => sortByClauseNumber(a.clause, b.clause)) + .forEach(record => { + if (previousId === record.clause.id) { + previousCell.innerHTML += ` (${dupCount + 2})`; + dupCount++; + } else { + let row = newBody.insertRow(); + let cell = row.insertCell(); + cell.innerHTML = record.clause.number; + cell = row.insertCell(); + cell.innerHTML = `${record.clause.titleHTML}`; + previousCell = cell; + previousId = record.clause.id; + dupCount = 0; + } + }, this); + this.$table.removeChild(this.$tableBody); + this.$tableBody = newBody; + this.$table.appendChild(this.$tableBody); + this.autoSize(); + }, + + showSDOs(sdos, alternativeId) { + let rhs = document.getElementById(alternativeId); + let parentName = rhs.parentNode.getAttribute('name'); + let colons = rhs.parentNode.querySelector('emu-geq'); + rhs = rhs.cloneNode(true); + rhs.querySelectorAll('emu-params,emu-constraints').forEach(e => { + e.remove(); + }); + rhs.querySelectorAll('[id]').forEach(e => { + e.removeAttribute('id'); + }); + rhs.querySelectorAll('a').forEach(e => { + e.parentNode.replaceChild(document.createTextNode(e.textContent), e); + }); + + this.$headerText.innerHTML = `Syntax-Directed Operations for
    ${parentName} ${colons.outerHTML} `; + this.$headerText.querySelector('a').append(rhs); + this.showSDOsBody(sdos, alternativeId); + }, + + showSDOsBody(sdos, alternativeId) { + this.activate(); + this.state = { type: 'sdo', id: alternativeId, html: this.$headerText.innerHTML }; + this.$headerRefId.style.display = 'none'; + let newBody = document.createElement('tbody'); + Object.keys(sdos).forEach(sdoName => { + let pair = sdos[sdoName]; + let clause = pair.clause; + let ids = pair.ids; + let first = ids[0]; + let row = newBody.insertRow(); + let cell = row.insertCell(); + cell.innerHTML = clause; + cell = row.insertCell(); + let html = '' + sdoName + ''; + for (let i = 1; i < ids.length; ++i) { + html += ' (' + (i + 1) + ')'; + } + cell.innerHTML = html; + }); + this.$table.removeChild(this.$tableBody); + this.$tableBody = newBody; + this.$table.appendChild(this.$tableBody); + this.autoSize(); + }, + + autoSize() { + this.$tableContainer.style.height = + Math.min(250, this.$table.getBoundingClientRect().height) + 'px'; + }, + + dragStart(pointerDownEvent) { + let startingMousePos = pointerDownEvent.clientY; + let startingHeight = this.$tableContainer.getBoundingClientRect().height; + let moveListener = pointerMoveEvent => { + if (pointerMoveEvent.buttons === 0) { + removeListeners(); + return; + } + let desiredHeight = startingHeight - (pointerMoveEvent.clientY - startingMousePos); + this.$tableContainer.style.height = Math.max(0, desiredHeight) + 'px'; + }; + let listenerOptions = { capture: true, passive: true }; + let removeListeners = () => { + document.removeEventListener('pointermove', moveListener, listenerOptions); + this.$header.removeEventListener('pointerup', removeListeners, listenerOptions); + this.$header.removeEventListener('pointercancel', removeListeners, listenerOptions); + }; + document.addEventListener('pointermove', moveListener, listenerOptions); + this.$header.addEventListener('pointerup', removeListeners, listenerOptions); + this.$header.addEventListener('pointercancel', removeListeners, listenerOptions); + }, +}; + +let Toolbox = { + init() { + this.$outer = document.createElement('div'); + this.$outer.classList.add('toolbox-container'); + this.$container = document.createElement('div'); + this.$container.classList.add('toolbox'); + this.$outer.appendChild(this.$container); + this.$permalink = document.createElement('a'); + this.$permalink.textContent = 'Permalink'; + this.$pinLink = document.createElement('a'); + this.$pinLink.textContent = 'Pin'; + this.$pinLink.setAttribute('href', '#'); + this.$pinLink.addEventListener('click', e => { + e.preventDefault(); + e.stopPropagation(); + menu.togglePinEntry(this.entry.id); + this.$pinLink.textContent = menu._pinnedIds[this.entry.id] ? 'Unpin' : 'Pin'; + }); + + this.$refsLink = document.createElement('a'); + this.$refsLink.setAttribute('href', '#'); + this.$refsLink.addEventListener('click', e => { + e.preventDefault(); + e.stopPropagation(); + referencePane.showReferencesFor(this.entry); + }); + this.$container.appendChild(this.$permalink); + this.$container.appendChild(document.createTextNode(' ')); + this.$container.appendChild(this.$pinLink); + this.$container.appendChild(document.createTextNode(' ')); + this.$container.appendChild(this.$refsLink); + document.body.appendChild(this.$outer); + }, + + activate(el, entry, target) { + if (el === this._activeEl) return; + sdoBox.deactivate(); + this.active = true; + this.entry = entry; + this.$pinLink.textContent = menu._pinnedIds[entry.id] ? 'Unpin' : 'Pin'; + this.$outer.classList.add('active'); + this.top = el.offsetTop - this.$outer.offsetHeight; + this.left = el.offsetLeft - 10; + this.$outer.setAttribute('style', 'left: ' + this.left + 'px; top: ' + this.top + 'px'); + this.updatePermalink(); + this.updateReferences(); + this._activeEl = el; + if (this.top < document.body.scrollTop && el === target) { + // don't scroll unless it's a small thing (< 200px) + this.$outer.scrollIntoView(); + } + }, + + updatePermalink() { + this.$permalink.setAttribute('href', makeLinkToId(this.entry.id)); + }, + + updateReferences() { + this.$refsLink.textContent = `References (${(this.entry.referencingIds || []).length})`; + }, + + activateIfMouseOver(e) { + let ref = this.findReferenceUnder(e.target); + if (ref && (!this.active || e.pageY > this._activeEl.offsetTop)) { + let entry = menu.search.biblio.byId[ref.id]; + this.activate(ref.element, entry, e.target); + } else if ( + this.active && + (e.pageY < this.top || e.pageY > this._activeEl.offsetTop + this._activeEl.offsetHeight) + ) { + this.deactivate(); + } + }, + + findReferenceUnder(el) { + while (el) { + let parent = el.parentNode; + if (el.nodeName === 'EMU-RHS' || el.nodeName === 'EMU-PRODUCTION') { + return null; + } + if ( + el.nodeName === 'H1' && + parent.nodeName.match(/EMU-CLAUSE|EMU-ANNEX|EMU-INTRO/) && + parent.id + ) { + return { element: el, id: parent.id }; + } else if (el.nodeName === 'EMU-NT') { + if ( + parent.nodeName === 'EMU-PRODUCTION' && + parent.id && + parent.id[0] !== '_' && + parent.firstElementChild === el + ) { + // return the LHS non-terminal element + return { element: el, id: parent.id }; + } + return null; + } else if ( + el.nodeName.match(/EMU-(?!CLAUSE|XREF|ANNEX|INTRO)|DFN/) && + el.id && + el.id[0] !== '_' + ) { + if ( + el.nodeName === 'EMU-FIGURE' || + el.nodeName === 'EMU-TABLE' || + el.nodeName === 'EMU-EXAMPLE' + ) { + // return the figcaption element + return { element: el.children[0].children[0], id: el.id }; + } else { + return { element: el, id: el.id }; + } + } + el = parent; + } + }, + + deactivate() { + this.$outer.classList.remove('active'); + this._activeEl = null; + this.active = false; + }, +}; + +function sortByClauseNumber(clause1, clause2) { + let c1c = clause1.number.split('.'); + let c2c = clause2.number.split('.'); + + for (let i = 0; i < c1c.length; i++) { + if (i >= c2c.length) { + return 1; + } + + let c1 = c1c[i]; + let c2 = c2c[i]; + let c1cn = Number(c1); + let c2cn = Number(c2); + + if (Number.isNaN(c1cn) && Number.isNaN(c2cn)) { + if (c1 > c2) { + return 1; + } else if (c1 < c2) { + return -1; + } + } else if (!Number.isNaN(c1cn) && Number.isNaN(c2cn)) { + return -1; + } else if (Number.isNaN(c1cn) && !Number.isNaN(c2cn)) { + return 1; + } else if (c1cn > c2cn) { + return 1; + } else if (c1cn < c2cn) { + return -1; + } + } + + if (c1c.length === c2c.length) { + return 0; + } + return -1; +} + +function makeLinkToId(id) { + let hash = '#' + id; + if (typeof idToSection === 'undefined' || !idToSection[id]) { + return hash; + } + let targetSec = idToSection[id]; + return (targetSec === 'index' ? './' : targetSec + '.html') + hash; +} + +function doShortcut(e) { + if (!(e.target instanceof HTMLElement)) { + return; + } + let target = e.target; + let name = target.nodeName.toLowerCase(); + if (name === 'textarea' || name === 'input' || name === 'select' || target.isContentEditable) { + return; + } + if (e.altKey || e.ctrlKey || e.metaKey) { + return; + } + if (e.key === 'm' && usesMultipage) { + let pathParts = location.pathname.split('/'); + let hash = location.hash; + if (pathParts[pathParts.length - 2] === 'multipage') { + if (hash === '') { + let sectionName = pathParts[pathParts.length - 1]; + if (sectionName.endsWith('.html')) { + sectionName = sectionName.slice(0, -5); + } + if (idToSection['sec-' + sectionName] !== undefined) { + hash = '#sec-' + sectionName; + } + } + location = pathParts.slice(0, -2).join('/') + '/' + hash; + } else { + location = 'multipage/' + hash; + } + } else if (e.key === 'u') { + document.documentElement.classList.toggle('show-ao-annotations'); + } else if (e.key === '?') { + document.getElementById('shortcuts-help').classList.toggle('active'); + } +} + +function init() { + if (document.getElementById('menu') == null) { + return; + } + menu = new Menu(); + let $container = document.getElementById('spec-container'); + $container.addEventListener( + 'mouseover', + debounce(e => { + Toolbox.activateIfMouseOver(e); + }), + ); + document.addEventListener( + 'keydown', + debounce(e => { + if (e.code === 'Escape') { + if (Toolbox.active) { + Toolbox.deactivate(); + } + document.getElementById('shortcuts-help').classList.remove('active'); + } + }), + ); +} + +document.addEventListener('keypress', doShortcut); + +document.addEventListener('DOMContentLoaded', () => { + Toolbox.init(); + referencePane.init(); +}); + +// preserve state during navigation + +function getTocPath(li) { + let path = []; + let pointer = li; + while (true) { + let parent = pointer.parentElement; + if (parent == null) { + return null; + } + let index = [].indexOf.call(parent.children, pointer); + if (index == -1) { + return null; + } + path.unshift(index); + pointer = parent.parentElement; + if (pointer == null) { + return null; + } + if (pointer.id === 'menu-toc') { + break; + } + if (pointer.tagName !== 'LI') { + return null; + } + } + return path; +} + +function activateTocPath(path) { + try { + let pointer = document.getElementById('menu-toc'); + for (let index of path) { + pointer = pointer.querySelector('ol').children[index]; + } + pointer.classList.add('active'); + } catch (e) { + // pass + } +} + +function getActiveTocPaths() { + return [...menu.$menu.querySelectorAll('.active')].map(getTocPath).filter(p => p != null); +} + +function initTOCExpansion(visibleItemLimit) { + // Initialize to a reasonable amount of TOC expansion: + // * Expand any full-breadth nesting level up to visibleItemLimit. + // * Expand any *single-item* level while under visibleItemLimit (even if that pushes over it). + + // Limit to initialization by bailing out if any parent item is already expanded. + const tocItems = Array.from(document.querySelectorAll('#menu-toc li')); + if (tocItems.some(li => li.classList.contains('active') && li.querySelector('li'))) { + return; + } + + const selfAndSiblings = maybe => Array.from(maybe?.parentNode.children ?? []); + let currentLevelItems = selfAndSiblings(tocItems[0]); + let availableCount = visibleItemLimit - currentLevelItems.length; + while (availableCount > 0 && currentLevelItems.length) { + const nextLevelItems = currentLevelItems.flatMap(li => selfAndSiblings(li.querySelector('li'))); + availableCount -= nextLevelItems.length; + if (availableCount > 0 || currentLevelItems.length === 1) { + // Expand parent items of the next level down (i.e., current-level items with children). + for (const ol of new Set(nextLevelItems.map(li => li.parentNode))) { + ol.closest('li').classList.add('active'); + } + } + currentLevelItems = nextLevelItems; + } +} + +function initState() { + if (typeof menu === 'undefined' || window.navigating) { + return; + } + const storage = typeof sessionStorage !== 'undefined' ? sessionStorage : Object.create(null); + if (storage.referencePaneState != null) { + let state = JSON.parse(storage.referencePaneState); + if (state != null) { + if (state.type === 'ref') { + let entry = menu.search.biblio.byId[state.id]; + if (entry != null) { + referencePane.showReferencesFor(entry); + } + } else if (state.type === 'sdo') { + let sdos = sdoMap[state.id]; + if (sdos != null) { + referencePane.$headerText.innerHTML = state.html; + referencePane.showSDOsBody(sdos, state.id); + } + } + delete storage.referencePaneState; + } + } + + if (storage.activeTocPaths != null) { + document.querySelectorAll('#menu-toc li.active').forEach(li => li.classList.remove('active')); + let active = JSON.parse(storage.activeTocPaths); + active.forEach(activateTocPath); + delete storage.activeTocPaths; + } else { + initTOCExpansion(20); + } + + if (storage.searchValue != null) { + let value = JSON.parse(storage.searchValue); + menu.search.$searchBox.value = value; + menu.search.search(value); + delete storage.searchValue; + } + + if (storage.tocScroll != null) { + let tocScroll = JSON.parse(storage.tocScroll); + menu.$toc.scrollTop = tocScroll; + delete storage.tocScroll; + } +} + +document.addEventListener('DOMContentLoaded', initState); + +window.addEventListener('pageshow', initState); + +window.addEventListener('beforeunload', () => { + if (!window.sessionStorage || typeof menu === 'undefined') { + return; + } + sessionStorage.referencePaneState = JSON.stringify(referencePane.state || null); + sessionStorage.activeTocPaths = JSON.stringify(getActiveTocPaths()); + sessionStorage.searchValue = JSON.stringify(menu.search.$searchBox.value); + sessionStorage.tocScroll = JSON.stringify(menu.$toc.scrollTop); +}); + +'use strict'; + +// Manually prefix algorithm step list items with hidden counter representations +// corresponding with their markers so they get selected and copied with content. +// We read list-style-type to avoid divergence with the style sheet, but +// for efficiency assume that all lists at the same nesting depth use the same +// style (except for those associated with replacement steps). +// We also precompute some initial items for each supported style type. +// https://w3c.github.io/csswg-drafts/css-counter-styles/ + +const lowerLetters = Array.from({ length: 26 }, (_, i) => + String.fromCharCode('a'.charCodeAt(0) + i), +); +// Implement the lower-alpha 'alphabetic' algorithm, +// adjusting for indexing from 0 rather than 1. +// https://w3c.github.io/csswg-drafts/css-counter-styles/#simple-alphabetic +// https://w3c.github.io/csswg-drafts/css-counter-styles/#alphabetic-system +const lowerAlphaTextForIndex = i => { + let S = ''; + for (const N = lowerLetters.length; i >= 0; i--) { + S = lowerLetters[i % N] + S; + i = Math.floor(i / N); + } + return S; +}; + +const weightedLowerRomanSymbols = Object.entries({ + m: 1000, + cm: 900, + d: 500, + cd: 400, + c: 100, + xc: 90, + l: 50, + xl: 40, + x: 10, + ix: 9, + v: 5, + iv: 4, + i: 1, +}); +// Implement the lower-roman 'additive' algorithm, +// adjusting for indexing from 0 rather than 1. +// https://w3c.github.io/csswg-drafts/css-counter-styles/#simple-numeric +// https://w3c.github.io/csswg-drafts/css-counter-styles/#additive-system +const lowerRomanTextForIndex = i => { + let value = i + 1; + let S = ''; + for (const [symbol, weight] of weightedLowerRomanSymbols) { + if (!value) break; + if (weight > value) continue; + const reps = Math.floor(value / weight); + S += symbol.repeat(reps); + value -= weight * reps; + } + return S; +}; + +// Memoize pure index-to-text functions with an exposed cache for fast retrieval. +const makeCounter = (pureGetTextForIndex, precomputeCount = 30) => { + const cache = Array.from({ length: precomputeCount }, (_, i) => pureGetTextForIndex(i)); + const getTextForIndex = i => { + if (i >= cache.length) cache[i] = pureGetTextForIndex(i); + return cache[i]; + }; + return { getTextForIndex, cache }; +}; + +const counterByStyle = { + __proto__: null, + decimal: makeCounter(i => String(i + 1)), + 'lower-alpha': makeCounter(lowerAlphaTextForIndex), + 'upper-alpha': makeCounter(i => lowerAlphaTextForIndex(i).toUpperCase()), + 'lower-roman': makeCounter(lowerRomanTextForIndex), + 'upper-roman': makeCounter(i => lowerRomanTextForIndex(i).toUpperCase()), +}; +const fallbackCounter = makeCounter(() => '?'); +const counterByDepth = []; + +function addStepNumberText( + ol, + depth = 0, + indent = '', + special = [...ol.classList].some(c => c.startsWith('nested-')), +) { + let counter = !special && counterByDepth[depth]; + if (!counter) { + const counterStyle = getComputedStyle(ol)['list-style-type']; + counter = counterByStyle[counterStyle]; + if (!counter) { + console.warn('unsupported list-style-type', { + ol, + counterStyle, + id: ol.closest('[id]')?.getAttribute('id'), + }); + counterByStyle[counterStyle] = fallbackCounter; + counter = fallbackCounter; + } + if (!special) { + counterByDepth[depth] = counter; + } + } + const { cache, getTextForIndex } = counter; + let i = (Number(ol.getAttribute('start')) || 1) - 1; + for (const li of ol.children) { + const marker = document.createElement('span'); + const markerText = i < cache.length ? cache[i] : getTextForIndex(i); + const extraIndent = ' '.repeat(markerText.length + 2); + marker.textContent = `${indent}${markerText}. `; + marker.setAttribute('aria-hidden', 'true'); + marker.setAttribute('class', 'list-marker'); + const attributesContainer = li.querySelector('.attributes-tag'); + if (attributesContainer == null) { + li.prepend(marker); + } else { + attributesContainer.insertAdjacentElement('afterend', marker); + } + for (const sublist of li.querySelectorAll(':scope > ol')) { + addStepNumberText(sublist, depth + 1, indent + extraIndent, special); + } + i++; + } +} + +document.addEventListener('DOMContentLoaded', () => { + document.querySelectorAll('emu-alg > ol').forEach(ol => { + addStepNumberText(ol); + }); +}); + +// Omit indendation when copying a single algorithm step. +document.addEventListener('copy', evt => { + // Construct a DOM from the selection. + const doc = document.implementation.createHTMLDocument(''); + const domRoot = doc.createElement('div'); + const html = evt.clipboardData.getData('text/html'); + if (html) { + domRoot.innerHTML = html; + } else { + const selection = getSelection(); + const singleRange = selection?.rangeCount === 1 && selection.getRangeAt(0); + const container = singleRange?.commonAncestorContainer; + if (!container?.querySelector?.('.list-marker')) { + return; + } + domRoot.append(singleRange.cloneContents()); + } + + // Preserve the indentation if there is no hidden list marker, or if selection + // of more than one step is indicated by either multiple such markers or by + // visible text before the first one. + const listMarkers = domRoot.querySelectorAll('.list-marker'); + if (listMarkers.length !== 1) { + return; + } + const treeWalker = document.createTreeWalker(domRoot, undefined, { + acceptNode(node) { + return node.nodeType === Node.TEXT_NODE || node === listMarkers[0] + ? NodeFilter.FILTER_ACCEPT + : NodeFilter.FILTER_SKIP; + }, + }); + while (treeWalker.nextNode()) { + const node = treeWalker.currentNode; + if (node.nodeType === Node.ELEMENT_NODE) break; + if (/\S/u.test(node.data)) return; + } + + // Strip leading indentation from the plain text representation. + evt.clipboardData.setData('text/plain', domRoot.textContent.trimStart()); + if (!html) { + evt.clipboardData.setData('text/html', domRoot.innerHTML); + } + evt.preventDefault(); +}); + +'use strict'; + +// Update superscripts to not suffer misinterpretation when copied and pasted as plain text. +// For example, +// * Replace `103` with +// `103` +// so it gets pasted as `10**3` rather than `103`. +// * Replace `10-x` with +// `10-x` +// so it gets pasted as `10**-x` rather than `10-x`. +// * Replace `2a + 1` with +// `2**(a + 1)` +// so it gets pasted as `2**(a + 1)` rather than `2a + 1`. + +function makeExponentPlainTextSafe(sup) { + // Change a only if it appears to be an exponent: + // * text-only and contains only mathematical content (not e.g. `1st`) + // * contains only s and internal links (e.g. + // `2(_y_)`) + const isText = [...sup.childNodes].every(node => node.nodeType === 3); + const text = sup.textContent; + if (isText) { + if (!/^[0-9. 𝔽ℝℤ()=*×/÷±+\u2212-]+$/u.test(text)) { + return; + } + } else { + if (sup.querySelector('*:not(var, emu-xref, :scope emu-xref a)')) { + return; + } + } + + let prefix = '**'; + let suffix = ''; + + // Add wrapping parentheses unless they are already present + // or this is a simple (possibly signed) integer or single-variable exponent. + const skipParens = + /^[±+\u2212-]?(?:[0-9]+|\p{ID_Start}\p{ID_Continue}*)$/u.test(text) || + // Split on parentheses and remember them; the resulting parts must + // start and end empty (i.e., with open/close parentheses) + // and increase depth to 1 only at the first parenthesis + // to e.g. wrap `(a+1)*(b+1)` but not `((a+1)*(b+1))`. + text + .trim() + .split(/([()])/g) + .reduce((depth, s, i, parts) => { + if (s === '(') { + return depth > 0 || i === 1 ? depth + 1 : NaN; + } else if (s === ')') { + return depth > 0 ? depth - 1 : NaN; + } else if (s === '' || (i > 0 && i < parts.length - 1)) { + return depth; + } + return NaN; + }, 0) === 0; + if (!skipParens) { + prefix += '('; + suffix += ')'; + } + + sup.insertAdjacentHTML('beforebegin', ``); + if (suffix) { + sup.insertAdjacentHTML('afterend', ``); + } +} + +document.addEventListener('DOMContentLoaded', () => { + document.querySelectorAll('sup:not(.text)').forEach(sup => { + makeExponentPlainTextSafe(sup); + }); +}); + +let sdoMap = JSON.parse(`{}`); +let biblio = JSON.parse(`{"refsByClause":{"useful-options":["_ref_0","_ref_1"],"editorial-conventions":["_ref_2","_ref_3","_ref_4","_ref_5","_ref_6","_ref_7","_ref_8","_ref_9","_ref_10","_ref_11","_ref_12","_ref_13","_ref_14"],"metadata":["_ref_15","_ref_16"],"emu-alg":["_ref_17","_ref_18","_ref_19","_ref_20","_ref_21","_ref_22"],"labeled-steps":["_ref_23"],"emu-xref":["_ref_24","_ref_25","_ref_26","_ref_27"],"emu-figure":["_ref_28"],"emu-table":["_ref_29"],"emu-example":["_ref_30"],"emu-production":["_ref_31","_ref_32","_ref_33"],"emu-rhs":["_ref_34","_ref_35","_ref_36","_ref_48","_ref_49"],"emu-nt":["_ref_37","_ref_50"],"emu-gmod":["_ref_38"],"emu-gann":["_ref_39"],"emu-gprose":["_ref_40"],"emu-clause-legacy-attributes":["_ref_41","_ref_42"],"definitions":["_ref_43","_ref_44"],"sec-transitive-effect":["_ref_45"],"emu-grammar":["_ref_46","_ref_47"],"emu-prodref":["_ref_51","_ref_52","_ref_53"],"emu-not-ref":["_ref_54"],"grammar":["_ref_55"]},"entries":[{"type":"clause","id":"intro","titleHTML":"Ecmarkup","number":""},{"type":"clause","id":"getting-started","titleHTML":"Getting Started","number":"1"},{"type":"table","id":"build-options","number":1,"caption":"Table 1: Build options","referencingIds":["_ref_15"]},{"type":"table","id":"document-options","number":2,"caption":"Table 2: Document options","referencingIds":["_ref_16"]},{"type":"clause","id":"useful-options","titleHTML":"Options","number":"2"},{"type":"clause","id":"stylesheets-and-scripts","titleHTML":"Stylesheets and Scripts","number":"3"},{"type":"table","id":"emd-overview","number":4,"caption":"Table 4: Inline styles/conventions","referencingIds":["_ref_6"]},{"type":"clause","id":"editorial-conventions","titleHTML":"Editorial Conventions","number":"4"},{"type":"clause","id":"metadata","titleHTML":"Metadata","number":"5","referencingIds":["_ref_0"]},{"type":"clause","id":"emu-intro","titleHTML":"emu-intro","number":"6.1","referencingIds":["_ref_3"]},{"type":"clause","id":"emu-clause-legacy-attributes","titleHTML":"Legacy Attributes","number":"6.2.1"},{"type":"term","term":"structured header","refId":"emu-clause-structured-headers"},{"type":"clause","id":"emu-clause-structured-headers","titleHTML":"Structured Headers","number":"6.2.2","referencingIds":["_ref_41","_ref_42"]},{"type":"op","aoid":"ExampleOperation","refId":"example-normative-optional"},{"type":"clause","id":"example-normative-optional","title":"ExampleOperation ( S [ , length ] )","titleHTML":"ExampleOperation ( S [ , length ] )","number":"6.2.3"},{"type":"clause","id":"emu-clause","titleHTML":"emu-clause","number":"6.2","referencingIds":["_ref_2","_ref_24"]},{"type":"clause","id":"emu-annex","titleHTML":"emu-annex","number":"6.3","referencingIds":["_ref_1","_ref_4"]},{"type":"clause","id":"clauses","titleHTML":"Clauses","number":"6"},{"type":"term","term":"swordfish","id":"swordfish","referencingIds":["_ref_43","_ref_44"]},{"type":"clause","id":"definitions","titleHTML":"Definitions","number":"7"},{"type":"op","aoid":"Effect","refId":"sec-effect"},{"type":"clause","id":"sec-effect","titleHTML":"Effect ( )","number":"8.1","referencingIds":["_ref_45"]},{"type":"op","aoid":"TransitiveEffect","refId":"sec-transitive-effect"},{"type":"clause","id":"sec-transitive-effect","titleHTML":"TransitiveEffect ( )","number":"8.2"},{"type":"clause","id":"effects","titleHTML":"Effects","number":"8","referencingIds":["_ref_22"]},{"type":"step","id":"step-example-label","referencingIds":["_ref_19"]},{"type":"table","id":"algorithm-step-annotation","number":5,"caption":"Table 5: Algorithm Step Annotation","referencingIds":["_ref_17"]},{"type":"clause","id":"labeled-steps","titleHTML":"Step Labels","number":"9.1","referencingIds":["_ref_18","_ref_20"]},{"type":"clause","id":"variable-declarations","titleHTML":"Variable Declarations","number":"9.2","referencingIds":["_ref_21"]},{"type":"clause","id":"emu-alg","titleHTML":"emu-alg","number":"9","referencingIds":["_ref_5"]},{"type":"clause","id":"emu-eqn","titleHTML":"emu-eqn","number":"10","referencingIds":["_ref_7"]},{"type":"clause","id":"emu-note","titleHTML":"emu-note","number":"11","referencingIds":["_ref_8","_ref_25"]},{"type":"step","id":"step-label-example","referencingIds":["_ref_27"]},{"type":"clause","id":"emu-xref","titleHTML":"emu-xref","number":"12","referencingIds":["_ref_14","_ref_23","_ref_28","_ref_29","_ref_30"]},{"type":"clause","id":"emu-not-ref","titleHTML":"emu-not-ref","number":"13","referencingIds":["_ref_54"]},{"type":"clause","id":"emu-figure","titleHTML":"emu-figure","number":"14","referencingIds":["_ref_9"]},{"type":"clause","id":"emu-table","titleHTML":"emu-table","number":"15","referencingIds":["_ref_10"]},{"type":"clause","id":"emu-example","titleHTML":"emu-example","number":"16","referencingIds":["_ref_11"]},{"type":"clause","id":"emu-biblio","titleHTML":"emu-biblio","number":"17","referencingIds":["_ref_26"]},{"type":"production","id":"prod-HexIntegerLiteral","name":"HexIntegerLiteral"},{"type":"production","id":"prod-HexIntegerLiteral","name":"HexIntegerLiteral"},{"type":"clause","id":"emu-grammar","titleHTML":"emu-grammar","number":"18","referencingIds":["_ref_12"]},{"type":"production","id":"prod-DecimalDigit","name":"DecimalDigit"},{"type":"production","id":"prod-Identifier","name":"Identifier"},{"type":"production","id":"prod-SourceCharacter","name":"SourceCharacter"},{"type":"production","id":"prod-ExpressionStatement","name":"ExpressionStatement","referencingIds":["_ref_55"]},{"type":"production","id":"prod-IterationStatement","name":"IterationStatement"},{"type":"production","id":"prod-StatementList","name":"StatementList"},{"type":"term","term":"LHS","refId":"emu-production"},{"type":"term","term":"RHS","refId":"emu-production"},{"type":"clause","id":"emu-production","titleHTML":"emu-production","number":"19.1","referencingIds":["_ref_13","_ref_36","_ref_37","_ref_46","_ref_47","_ref_48","_ref_49","_ref_50","_ref_51","_ref_52","_ref_53"]},{"type":"clause","id":"emu-rhs","titleHTML":"emu-rhs","number":"19.2","referencingIds":["_ref_32"]},{"type":"clause","id":"emu-nt","titleHTML":"emu-nt","number":"19.3","referencingIds":["_ref_31"]},{"type":"clause","id":"emu-t","titleHTML":"emu-t","number":"19.4"},{"type":"clause","id":"emu-gmod","titleHTML":"emu-gmod","number":"19.5"},{"type":"clause","id":"emu-gann","titleHTML":"emu-gann","number":"19.6"},{"type":"clause","id":"emu-gprose","titleHTML":"emu-gprose","number":"19.7"},{"type":"clause","id":"emu-prodref","titleHTML":"emu-prodref","number":"19.8","referencingIds":["_ref_35"]},{"type":"clause","id":"grammar","titleHTML":"Directly Specifying Grammar","number":"19","referencingIds":["_ref_33","_ref_34","_ref_38","_ref_39","_ref_40"]},{"type":"clause","id":"emu-import","titleHTML":"emu-import","number":"20"},{"type":"clause","id":"oldids","titleHTML":"Old IDs","number":"21.1"},{"type":"clause","id":"ins-del","titleHTML":"Indicating Changes","number":"21.2"},{"type":"clause","id":"pre-code","titleHTML":"Code Listings","number":"21.3"},{"type":"clause","id":"miscellaneous","title":"Other Styles & Conventions","titleHTML":"Other Styles & Conventions","number":"21"},{"type":"term","term":"example","refId":"emu-not-ref"}]}`); +;let usesMultipage = false \ No newline at end of file diff --git a/index.html b/index.html new file mode 100644 index 00000000..1ae5c856 --- /dev/null +++ b/index.html @@ -0,0 +1,894 @@ + + +Ecmarkup +
    +
      +
    • Toggle shortcuts help?
    • +
    • Toggle "can call user code" annotationsu
    • + +
    • Jump to search box/
    • +
    • Toggle pinning of the current clausep
    • +
    • Jump to nth pin1-9
    • +
    + + + + +

    Ecmarkup

    +

    Ecmarkup is a number of custom elements and a toolchain suitable for specifying semantics for ECMAScript and other programming languages.

    +

    Ecmarkup offers the following conveniences (among many others):

    +
      +
    • Ecmarkdown syntax for paragraphs and algorithms
    • +
    • Grammarkdown for specifying grammar
    • +
    • Terse markup for the authoring conventions used in ECMAScript specifications including clauses, notes, examples, figures, tables, and more.
    • +
    • Easy cross-references by element ID to clauses, abstract operations, examples, figures, tables, and etc. in the current document, ES6, and other specs.
    • +
    • Generates table of contents
    • +
    • Generates clause ids based on document position
    • +
    • Auto-links abstract operations and terms based on name with support for external bibliographies for cross-referencing between specs
    • +
    • Source code syntax highlighting inside pre code blocks
    • +
    +

    This document is itself written using Ecmarkup. Its source can be viewed on github.

    +
    + + +

    1 Getting Started

    +
    $ npm install -g ecmarkup
    +$ ecmarkup --help
    +$ ecmarkup spec.html out.html
    +
    + + +

    2 Options

    +

    Build options and document options can be passed in via the command line or specified in the front-matter of your document. There is no bright line separating the two categories, but for purposes of organization, document options are defined as those affecting the observable rendered output content.

    +
    Table 1: Build options
    + + + + + + + + + + + + +
    Command Line OptionFront-Matter KeyDescription
    --watchRebuild when files change.
    --verbosePrint verbose logging info.
    --write-biblioEmit a biblio file to the specified path.
    --assetsassets"none" for no CSS/JS/etc, "inline" for inline CSS/JS/etc, "external" for external CSS/JS/etc.
    --assets-dirassetsDirDirectory in which to place assets when using --assets=external.
    --lint-speclintSpecEnforce some style and correctness checks.
    --error-formatterThe eslint formatter to be used for printing warnings and errors when using --verbose. Either the name of a built-in eslint formatter or the package name of an installed eslint compatible formatter.
    --strictExit with an error if there are warnings. Cannot be used with --watch.
    --multipageEmit a distinct page for each top-level clause.
    +
    + +
    Table 2: Document options
    + + + + + + + + + + + + + + + + + + +
    Command Line OptionFront-Matter KeyDescription
    titleDocument title, for example "ECMAScript 2016" or "Async Functions".
    statusDocument status. Can be "proposal", "draft", or "standard". Defaults to "proposal".
    stageTC39 proposal stage. If present and status is "proposal", version defaults to "Stage stage Draft".
    versionDocument version, for example "6<sup>th</sup> Edition" (which renders like "6th Edition") or "Draft 1".
    dateTimestamp of document rendering, used for various pieces of boilerplate. Defaults to the value of the SOURCE_DATE_EPOCH environment variable (as a number of second since the POSIX epoch) if it exists, otherwise to the current date and time.
    shortnameDocument shortname, for example "ECMA-262". If present and status is "draft", version defaults to "Draft shortname".
    descriptionBrief description to be used for link previews in social media and the like.
    locationURL of this document. Used in conjunction with the biblio file to support inbound references from other documents.
    copyrightEmit copyright and software license information. Boolean, default true.
    contributorsContributors to this specification, i.e. those who own the copyright. If your proposal includes text from any Ecma specification, this should include "Ecma International".
    --no-toctocEmit table of contents. Boolean, default true.
    --old-tocoldTocEmit the table of contents at the top of the document rather than as a side pane. Boolean, default false.
    --load-biblioextraBibliosExtra biblio.json file(s) to load. This should contain either an object as exported by --write-biblio or an array of such objects.
    boilerplateAn object with address and/or copyright and/or license fields containing paths to files containing the corresponding content for populating an element with class "copyright-and-software-license" (or if not found, an appended emu-annex with that id) when copyright is true. Absent fields are assumed to reference files in a "boilerplate" directory sibling of the directory containing the ecmarkup executable.
    --mark-effectsmarkEffectsPropagate and render effects like "user code".
    +
    +
    + + +

    3 Stylesheets and Scripts

    +

    Ecmarkup requires CSS styles and, if you're using the sidebar table of contents, javascript as well. By default CSS and JS dependencies are inlined into the document. You can override this by setting assets to "none" (for example if you want to manually link to external assets). Passing cssOut and jsOut will write the respective files to disk at the given path.

    +
    + + +

    4 Editorial Conventions

    +

    There are a large number of features in Ecmarkup. Detailed documentation can be found in later sections. This section provides a high-level overview of what capabilities are available and when to use them.

    + +
    Table 3: Structuring your Ecmarkup document
    + + + + + + + + + + + + +
    Notational ConventionEcmarkup Functionality
    Clausesemu-clause, emu-intro, and emu-annex
    Algorithmsemu-alg with Ecmarkdown (see also Table 4)
    Mathematical expressionsemu-eqn
    Notesemu-note
    Figures (such as images)emu-figure
    Tablesemu-table
    Examplesemu-example
    Grammaremu-grammar using Grammarkdown syntax or emu-production and child elements
    Cross-referencesemu-xref (can reference clauses, productions, abstract operations, tables, figuers, and examples)
    +
    + +

    There are also a number of elements and corresponding Ecmarkdown syntax to give inline styles to various parts of your document in accordance with ECMAScript conventions. More detail on Ecmarkdown syntax can be found on its readme.

    +
    Table 4: Inline styles/conventions
    + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Notational ConventionEcmarkdown SyntaxEcmarkup Functionality
    Variables, such as those in defined in algorithms, parameters in abstract operations, etc._variable_<var>variable</var>
    ECMAScript Language Type Values, eg. undefined, true, 1𝔽, "string", TypeError, etc.*TypeError*<emu-val>TypeError</emu-val>
    Specification types and their instances, eg. throw and empty.~throw~<emu-const>throw<emu-const>
    Source Code, eg. Function.prototype.`code`<code>code</code>
    Non-terminal references, eg. as FunctionDeclaration and FunctionExpression.|FunctionExpression|<emu-nt>FunctionExpression</emu-nt>
    +
    +
    + + +

    5 Metadata

    +

    There are a number of settings that allow customizations or enable generation of boilerplate. To add metadata to your document, use yaml syntax inside a <pre class="metadata"> element somewhere in the root of your document. See Table 1 and Table 2 for a list (or consult --help).

    +

    Some metadata can also be passed on the command line, for example --no-toc. Such options take precedence over embedded metadata.

    + +

    Example

    +
    <pre class="metadata">
    +title: Document 1
    +toc: false
    +</pre>
    +
    + + +

    6 Clauses

    +

    Clauses are referenced using their id and are numbered automatically based on document position. Ecmarkdown syntax can be used in descendent text nodes as well. Text nodes are parsed as Ecmarkdown Fragments.

    + + +

    6.1 emu-intro

    +

    Non-normative introductory information.

    + +

    Attributes

    +

    id: Clause id. Must be unique.

    +

    aoid: Abstract operation ID. A unique name identifying this clause as an abstract operation. Algorithm steps will auto-link calls to this abstract operation to this clause. If left blank, the aoid will be set to the id of this clause.

    +
    + + +

    6.2 emu-clause

    +

    Normative clause.

    + +

    Attributes

    +

    id: Clause id. Must be unique.

    +

    example: If present, the clause is an example.

    +

    legacy: If present, the clause is Legacy.

    +

    normative-optional: If present, the clause is Normative Optional.

    +

    number: Optional: An explicit clause number, overriding the default auto-incrementing number. Can be a nested number, as in number="2.1".

    +

    type: Optional: Type of feature described by the clause.

    +
      +
    • "abstract operation"
    • +
    • "concrete method"
    • +
    • "host-defined abstract operation"
    • +
    • "implementation-defined abstract operation"
    • +
    • "internal method"
    • +
    • "numeric method"
    • +
    • "sdo" or "syntax-directed operation"
    • +
    + +
    Legacy
    +

    6.2.1 Legacy Attributes

    +

    aoid: Abstract operation ID. A unique name identifying this clause as an abstract operation. Algorithm steps will auto-link calls to this abstract operation to this clause. If left blank, the aoid will be set to the id of this clause. When a structured header is present the aoid will be inferred, but this attribute may be set explicitly when not using a structured header.

    +
    + + +

    6.2.2 Structured Headers

    +

    A clause describing an operation or method may have a structured header to document its signature and metadata. Such a structure consists of:

    +
      +
    1. an <h1> element containing the name of the operation followed by a parenthesized list of parameters, each appearing on a comma-terminated line of its own and consisting of a name and value constraints separated by a colon and optionally preceded with "optional"
    2. +
    3. + a <dl> element with class "header" and a collection of optional key-value pairs expressed using <dt> and <dd> elements: +
        +
      • description: An explanation of the operation's behaviour.
      • +
      • effects: A list of "effects" associated with the clause (and transitively with those referencing it), separated by commas with optional whitespace. The only currently-known effect is "user-code", which indicates that the operation or method can evaluate non-implementation code (such as custom getters).
      • +
      • for: The type of value to which a clause of type "concrete method" or "internal method" applies.
      • +
      • redefinition: If "true", the name of the operation will not automatically link (i.e., it will not automatically be given an aoid).
      • +
      • skip global checks: If "true", disables consistency checks for this AO which require knowing every callsite.
      • +
      • skip return checks: If "true", disables checking that the returned values from this AO correspond to its declared return type. Adding this to an AO which does not require it will produce a warning.
      • +
      +
    4. +
    +
    + +

    Example

    +
    <emu-clause id="example-normative-optional" type="abstract operation" normative-optional example>
    +  <h1>
    +    ExampleOperation (
    +      _S_: a String,
    +      optional _length_: a non-negative integer,
    +    )
    +  </h1>
    +  <dl class="header">
    +    <dt>description</dt>
    +    <dd>It returns a List of the code points of the first _length_ grapheme clusters of _S_ using the host environment's current locale, or all code points when _length_ is not present.</dd>
    +  </dl>
    +  <p>This clause is normative optional.</p>
    +</emu-clause>
    + +

    Result

    + +
    + + +

    6.3 emu-annex

    +

    Annex clause.

    + +

    Attributes

    +

    normative: If present, annex is normative. Default is non-normative.

    +

    id: Clause id. Must be unique.

    +

    aoid: Abstract operation ID. A unique name identifying this clause as an abstract operation. Algorithm steps will auto-link calls to this abstract operation to this clause. If left blank, the aoid will be set to the id of this clause.

    +
    +
    + + +

    7 Definitions

    +

    Terms can be defined using the <dfn> element. Any uses of that term will be automatically linked to the clause containing the definition, or, if the <dfn> element has an id, to the <dfn> itself. This can be suppressed with the emu-not-ref element.

    +

    When the term starts with a lowercase English letter, usages of the term with the first letter capitalized will also link.

    + +

    Attributes

    +

    variants: If present, specifies other variants of the term (such as plural forms). Multiple variants are separated by commas with optional whitespace.

    + +

    Example

    +
    <p>A <dfn id="swordfish" variants="swordfishes,broadbill,broadbills,Xiphias gladius">swordfish</dfn> is a large fish characterized by a long, pointed bill.</p>
    +<p>The Latin name for swordfishes, <em>Xiphias gladius</em>, comes from the Greek word "ξίφος" (xiphos, "sword") and from the Latin word "gladius" ("sword").</p>
    + +

    Result

    + +
    + + +

    8 Effects

    +

    Abstract operations can be marked as having effects that propagate to their invocation sites. Subject to the following rules, calls to such operations are given a class of the effect name prefixed with “e-”. When using markEffects, the CSS includes styling for the "user-code" effect via class name e-user-code that is off by default but can be togged by pressing u.

    + +

    Within an algorithm, an emu-meta element enclosing an operation invocation can be used to indicate either the origination of effects or the absence of effects using a list of effect names separated by commas with optional whitespace in an effects or suppress-effects attribute (respectively).

    +
    1. Let _value_ be ? <emu-meta effects="user-code">_O_.[[Get]]</emu-meta>(_P_, _O_).
    +1. Let _primValue_ be ? ToPrimitive(_value_, ~number~).
    +1. Let _stringValue_ be ? <emu-meta suppress-effects="user-code">ToString(_primValue_)</emu-meta>.
    + +

    Effects do not normally propagate to callsites marked as infallibly non-abrupt with !. Such sites must be manually marked with emu-meta.

    + +

    When a step or its substeps have effects but those effects should not imply that the containing algorithm has the same effects, propagation may be prevented using a [fence-effects="user-code"] annotation at the begining of the step. This implicitly applies to steps which define abstract closures.

    + +

    It is also possible to associate effects at the level of an entire operation (for example, because it is not defined with algorithm steps) by adding to its structured header an effects entry.

    + +

    Example

    +
    <emu-clause id="sec-effect" type="abstract operation">
    +  <h1>Effect ()</h1>
    +  <dl class="header"></dl>
    +  <emu-alg>
    +    1. <emu-meta effects="user-code">Call user code.</emu-meta>
    +  </emu-alg>
    +</emu-clause>
    +
    +<emu-clause id="sec-transitive-effect" type="abstract operation">
    +  <h1>TransitiveEffect ()</h1>
    +  <dl class="header"></dl>
    +  <emu-alg>
    +    1. Perform Effect().
    +  </emu-alg>
    +</emu-clause>
    + +

    Result

    + Press u to toggle annotation visibility. + +
    + + +

    9 emu-alg

    +

    Algorithm steps. Should not contain any HTML. The node's textContent is parsed as an Ecmarkdown document. Additionally, calls to abstract operations inside algorithm steps are automatically linked to their definitions by first checking for any clauses or algorithms with the appropriate aoid in the current spec, and afterwards checking any linked bibliography files.

    +

    Algorithm steps can be annotated with additional metadata as summarized in Table 5.

    + +

    Attributes

    +

    example: If present, the element is an example.

    +

    replaces-step: If present, references a step to replace by its id (whose numbering the algorithm adopts).

    + +

    Example

    +
    <emu-alg>
    +  1. Let _length_ be 0.
    +  1. Let _node_ be this |ListNode|.
    +  1. Repeat, while _node_ is not *undefined*,
    +    1. Set _node_ to NextNode of _node_.
    +    1. [id="step-example-label"] Set _length_ to _length_ + 1.
    +  1. Return _length_.
    +</emu-alg>
    +<p>The following is an alernative definition of step <emu-xref href="#step-example-label"></emu-xref>.</p>
    +<emu-alg replaces-step="step-example-label">
    +  1. Set _length_ to _length_ + 1.
    +  1. Let _weight_ be GetWeight of _node_.
    +  1. If _length_ = 10<sup>9</sup> or _weight_ > 2<sup>_length_ + 1</sup>, then
    +    1. NOTE: This is an out-of-bounds state.
    +    1. If _ignoreErrors_ is not *true*, then
    +      1. Throw a *RangeError* exception.
    +    1. Set _hadError_ to *true*.
    +</emu-alg>
    + +

    Result

    + + +
    Table 5: Algorithm Step Annotation
    + + + + + + + + + + + + + + + + + + + + + + + +
    DataDescriptionExample Syntax
    Step labelsProvide an id for referencing an individual step.[id="step-label"]
    Variable DeclarationsInform the linter of the existence of a variable.[declared="x"]
    Effect propagation controlOriginate, suppress, or constrain the scope of effects.<emu-meta>, [fence-effects="user-code"]
    Step optionalityMark a step as deprecated, normative-optional, or legacy.[normative-optional]
    +
    + + +

    9.1 Step Labels

    +

    Each step can be given an id for reference by emu-xref or replaces-step.

    + +

    Example

    +
    <emu-alg>
    +  1. [id="step-example-internal"] Let _obj_ be OrdinaryObjectCreate(*null*).
    +</emu-alg>
    +
    + + +

    9.2 Variable Declarations

    +

    The linter checks that each used variable in an algorithm step is declared earlier. It recognizes most common forms of declaration automatically, but in case there is a declaration which is not automatically inferred, a variable can be marked as declared for the purposes of this analysis by adding a [declared="v"] attribute at the start of the step. Multiple variables can be separated by commas with optional whitespace.

    +

    Note that the [declared="v"] attribute must be at the beginning of the step, including before any <ins> tags.

    + +

    Example

    +
    <emu-alg>
    +  1. [declared="x"] Suppose the existence of _x_.
    +</emu-alg>
    +
    +
    + + +

    10 emu-eqn

    +

    An equation, similar to those found in ES6 Year Number.

    + +

    Attributes

    +

    aoid: If present, an abstract operation id defined by this equation.

    +

    id: If present, links will go directly to the eqn definition. Otherwise, links go to the parent clause.

    +
    + + +

    11 emu-note

    +

    Non-normative explanatory text. Comes in two types - regular notes and Editor's notes. Regular notes are intended for the implementers and end users of this specification. Editor's notes are notes to and from the Editors and will generally be removed prior to a specification being finalized and ratified.

    + +

    Attributes

    +

    type: The type of note, either blank or "editor".

    + +

    Example

    +
    <p>For authentication only, servers and clients MUST support SASL Salted Challenge Response Authentication Mechanism [SCRAM].</p>
    +<emu-note>But we know you won't.</emu-note>
    +<emu-note type="editor">This joke brought to you by RFC6919.</emu-note>
    + +

    Result

    + +
    + + +

    12 emu-xref

    +

    Cross-reference another clause, production, note, example, abstract operation, or labeled step. If the text content of this element is empty, a suitable default is used. The title attribute controls this default - when present, clauses are referred to using their title rather than number. This also applies to examples which are indexed first by their containing clause and then their example number.

    +

    Cross-references to an id check for clauses, productions, examples, and steps in this order. For each type, the local document is consulted before looking for external sources.

    + +

    Attributes

    +

    href: Optional: URL of the target clause, production, or example to cross-reference, interpreted relative to the containing document.

    +

    title: Optional: If present, xref will be filled in with the referenced clause's title. Otherwise, the clause's section number is used.

    +

    aoid: Optional: aoid of an abstract operation to reference.

    + +

    One of aoid or href must be specified.

    + +

    Example

    +
    <p>The clause element is specified in <emu-xref href="#emu-clause"></emu-xref>.</p>
    +<p>See <emu-xref href="#emu-note" title></emu-xref> for information on the emu-note element.</p>
    +<p>The <emu-xref href="#emu-biblio">biblio element</emu-xref> supports xref to external specs.</p>
    +<p><emu-xref aoid="Get"></emu-xref> is an abstract operation from ES6</p>
    +<p><emu-alg>1. [id="step-label-example"] Example labeled step.</emu-alg></p>
    +<p>You can reference step <emu-xref href="#step-label-example"></emu-xref> like this</p>
    + +

    Result

    + +
    + + +

    13 emu-not-ref

    +

    Suppresses automatic linking to definitions.

    + +

    Example

    +
    <div id="not-ref-section-1">
    +  <p>An <dfn>example</dfn> is used for illustrative purposes.</p>
    +</div>
    +<div id="not-ref-section-2">
    +  <p>When a word defined in another section (or algorithm step) is used, as in this example, it is normally automatically linked to the section containing the definition.</p>
    +  <p>When such a word should not be automatically linked, for <emu-not-ref>example</emu-not-ref> when using its colloquial definition, it can be wrapped with this tag to surpress the automatic linking.</p>
    +</div>
    + +

    Result

    + +
    + + +

    14 emu-figure

    +

    Creates a figure that can be referenced by emu-xref. Add a caption using a child emu-caption element.

    + +

    Attributes

    +

    informative: Optional: If present, the figure is informative. Otherwise it is normative.

    + +

    Example

    +
    <emu-figure>
    +  <emu-caption>Example figure<emu-caption>
    +  [[insert some awesome graphic here, maybe something like figure 2]]
    +</emu-figure>
    + +

    Result

    + +
    + + +

    15 emu-table

    +

    Creates a table that can be referenced by emu-xref. Add a caption using a child emu-caption element.

    + +

    Attributes

    +

    caption: Optional: Caption for the example

    +

    informative: Optional: If present, the table is informative. Otherwise it is normative.

    + +

    Example

    +
    <emu-table>
    +  <emu-caption>Example table<emu-caption>
    +  <table>
    +    <tr><th>Column 1</th><th>Column 2</th></tr>
    +    <tr><td>Value</td><td>Value 2</td></tr>
    +    <tr><td>Value</td><td>Value 2</td></tr>
    +    <tr><td>Value</td><td>Value 2</td></tr>
    +    <tr><td>Value</td><td>Value 2</td></tr>
    +  </table>
    +</emu-table>
    + +

    Result

    + +
    + + +

    16 emu-example

    +

    Creates an informative example that can be referenced by emu-xref. Examples are numbered based on how many are present in the example's containing clause.

    + +

    Attributes

    +

    caption: Optional: Caption for the example

    + +

    Example

    +
    <emu-example caption="Example Example">
    +  This is an example.
    +</emu-example>
    +
    +<emu-example caption="Another Example Example">
    +  This is also an example.
    +</emu-example>
    + +

    Result

    + +
    + + +

    17 emu-biblio

    +

    Links a bibliography file. The bibliography file is a JSON document containing URLs for referenced documents along with any algorithms they define.

    + +

    Attributes

    +

    href: Required: URL of the biblio file, interpreted relative to the containing document.

    + +

    Example

    + biblio.json +
    {
    +  "https://tc39.github.io/ecma262/": [
    +    {
    +      "type": "op",
    +      "id": "sec-returnifabrupt",
    +      "aoid": "ReturnIfAbrupt"
    +    },
    +    {
    +      "type": "op",
    +      "id": "sec-get-o-p",
    +      "aoid": "Get"
    +    }
    +  ]
    +}
    + spec.emu +
    <emu-biblio href="./biblio.json"></emu-biblio>
    +<emu-alg>
    +1. Let _res_ be Get(_obj_, _key_).
    +1. ReturnIfAbrupt(_res_).
    +</emu-alg>
    + +

    Result

    + +
    + + +

    18 emu-grammar

    +

    Text inside emu-grammar elements is parsed using Grammarkdown. The syntax is essentially identical to the notational conventions in ECMAScript (minus formatting). See the Grammarkdown readme.

    +

    Grammar will be displayed as an inline element flowing with text unless it is the immediate child of an emu-clause-like element.

    + +

    Attributes

    +

    collapsed: If present, productions are displayed in collapsed format with LHS and RHS on the same line.

    +

    example: If present, the element is an example.

    +

    primary: Optional: Deprecated in favor of type="definition".

    +

    type: Optional: Disposition of the grammar. If absent, it is considered to reference productions defined elsewhere.

    +
      +
    • "definition": The grammar is an authoritative source for productions which should be the target of references.
    • +
    • "example": Deprecated in favor of the example attribute.
    • +
    + +

    Example

    +
    <p>
    +  <emu-grammar type="definition" collapsed example>
    +    HexIntegerLiteral :: `0x` HexDigits
    +  </emu-grammar>
    +  is a shorthand for:
    +</p>
    +<emu-grammar type="definition" example>
    +  HexIntegerLiteral :: `0` `x` HexDigits
    +</emu-grammar>
    + +

    Result

    + +
    + + +

    19 Directly Specifying Grammar

    +

    Although discouraged, it is possible to directly specify grammar using the emu-production element and related elements emitted by Grammarkdown.

    + +

    Lexical Grammar Examples

    +
    <emu-production name="DecimalDigit" type="lexical" oneof>
    +  <emu-rhs>0 1 2 3 4 5 6 7 8 9</emu-rhs>
    +</emu-production>
    +
    +<emu-production name="Identifier" type="lexical">
    +  <emu-rhs><emu-nt>IdentifierName</emu-nt> <emu-gmod>but not
    +      <emu-nt>ReservedWord</emu-nt></emu-gmod></emu-rhs>
    +</emu-production>
    +
    +<emu-production name="SourceCharacter" type="lexical">
    +  <emu-rhs><emu-gprose>any Unicode code point</emu-gprose></emu-rhs>
    +</emu-production>
    + +

    Result

    + + +

    Syntactic Grammar Examples

    +
    <emu-production name="ExpressionStatement" params="Yield">
    +  <emu-rhs>
    +    <emu-gann>lookahead ∉ { <emu-t>{</emu-t>, <emu-t>function</emu-t>,
    +        <emu-t>class</emu-t>, <emu-t>let [</emu-t> }</emu-gann>
    +  </emu-rhs>
    +</emu-production>
    +
    +<emu-production name="IterationStatement">
    +  <emu-rhs>for ( <emu-nt>LexicalDeclaration</emu-nt> ; <emu-nt optional>Expression</emu-nt> ;
    +  <emu-nt optional>Expression</emu-nt> ) <emu-nt>Statement</emu-nt></emu-rhs>
    +</emu-production>
    +
    +<emu-production name="StatementList" params="Return, In">
    +  <emu-rhs constraints="~Return"><emu-nt>ReturnStatement</emu-nt></emu-rhs>
    +  <emu-rhs><emu-nt>ExpressionStatement</emu-nt></emu-rhs>
    +</emu-production>
    + +

    Result

    + + + +

    19.1 emu-production

    +

    This is the top level element that contains a grammar production associating a shared LHS (left-hand side) emu-nt non-terminal symbol with one or more emu-rhs elements containing arbitrary RHS (right-hand side) sequences of terminal and non-terminal symbols.

    +

    The production will be displayed as an inline element flowing with text unless it is the immediate child of an emu-clause-like element or if its containing emu-grammar element is an immediate child of an emu-clause-like element.

    + +

    Attributes

    +

    name: Required: Name of the production (i.e. the LHS). Should conform with the syntax of Unicode Standard Annex #31, UAX31-D1 Default Identifier Syntax and not include any underscore (U+005F LOW LINE “_”).

    +

    collapsed: If present, the production is displayed in collapsed format with LHS and RHS on the same line.

    +

    oneof: If present, the production is a one-of production. See the DecimalDigit example above.

    + +

    optional: If present, the LHS is marked as optional.

    +

    params: Parameters for this production. Multiple parameters are separated by commas with optional whitespace.

    +

    primary: Optional: The production is authoritative and should be the target of references.

    +

    type: Optional: Type of production. If absent, the production is considered syntactic.

    +
      +
    • "lexical": The production is part of a lexical grammar, and should render with the LHS and RHS separated by two colons “::”.
    • +
    • "regexp": Deprecated because the ECMAScript regular expression grammar is considered to be lexical. Productions with this type render with the LHS and RHS separated by three colons “:::”.
    • +
    +
    + + +

    19.2 emu-rhs

    +

    Describes one right-hand-side alternative of a production. Interior text nodes are split on each space and turned into terminal symbols. For example, the following elements are semantically equivalent:

    +
    <emu-rhs>a b c</emu-rhs>
    +<emu-rhs><emu-t>a</emu-t> <emu-t>b</emu-t> <emu-t>c</emu-t></emu-rhs>
    + +

    Attributes

    +

    constraints: any constraints for this RHS. Multiple constraints are separated by commas with optional whitespace. See the StatementList example above.

    +

    a: Optional: alternative id used to reference a particular RHS from an emu-prodref. Must be unique for all emu-rhs elements inside an emu-production.

    +
    + + +

    19.3 emu-nt

    +

    References a non-terminal symbol by the name attribute of its emu-production.

    + +

    Attributes

    +

    example: If present, the element is an example.

    +

    optional: If present, the non-terminal is optional in its containing RHS.

    +

    params: Parameters for this non-terminal. Multiple parameters are separated by commas with optional whitespace.

    +
    + + +

    19.4 emu-t

    +

    References a terminal symbol. No attributes are available. Manual creation of these elements should be unnecessary as they are created automatically inside emu-rhs elements.

    +
    + + +

    19.5 emu-gmod

    + +

    Contains well-known modifiers to a right-hand side of a production. The only well-known modifier at present is the "but not" modifier. See the Identifier example above.

    +
    + + +

    19.6 emu-gann

    + +

    Contains well-known annotations to to a right-hand side of a production. The only well-known modifiers at present are "lookahead" and "empty". See the ExpressionStatement example above. Any text inside a gann element is wrapped in square brackets.

    +
    + + +

    19.7 emu-gprose

    +

    Contains any prose text that describes a production. See the SourceCharacter example above.

    +
    + + +

    19.8 emu-prodref

    +

    References a production or RHS defined elsewhere. Ecmarkup will insert either the entire production or a particular RHS depending on attributes.

    + +

    Attributes

    +

    name: Required: Name of the production to reference.

    +

    a: Optional. If present, references a particular RHS with the specified alternative id in its a attribute.

    +
    +
    + + +

    20 emu-import

    +

    Imports are treated specially in an ecmarkup document. Every import is inlined into its parent document at the location of the import tag. This is useful for breaking your spec document up into many smaller pieces.

    + +

    Attributes

    +

    href: Required: URL of the document to import, interpreted relative to the document containing the emu-import.

    +
    + + +

    21 Other Styles & Conventions

    + + +

    21.1 Old IDs

    +

    Old IDs for any element can be stored separated by commas with optional whitespace inside an oldids attribute, allowing links using them to continue working as expected.

    +
    + + +

    21.2 Indicating Changes

    +

    The ins and del HTML elements can be used to mark insertions and deletions respectively. When adding or removing block content (such as entire list items, paragrpahs, clauses, grammar RHSes, etc.), use a class of "block".

    + +

    Inline Example

    +
    ECMAScript <del>6</del><ins>2015</ins> <del>will be ratified</del><ins>was ratified</ins> in June, 2015.
    + +

    Result

    + + +

    Block Example

    +
    <emu-production name="HoistableDeclaration">
    +  <emu-rhs><emu-nt>FunctionDeclaration</emu-nt></emu-rhs>
    +  <emu-rhs><emu-nt>GeneratorDeclaration</emu-nt></emu-rhs>
    +  <ins class="block"><emu-rhs><emu-nt>AsyncFunctionDeclaration</emu-nt></emu-rhs></ins>
    +</emu-production>
    + +

    Result

    + +
    + + +

    21.3 Code Listings

    +

    Create a code listing using <pre><code>. The code element takes a class of javascript, html, or any other language provided by highlightjs. Ecmarkup will trim any leading blank lines and also normalize the indentation based on the indentation of the first line.

    +
    +
    +
    \ No newline at end of file diff --git a/print.css b/print.css new file mode 100644 index 00000000..06709094 --- /dev/null +++ b/print.css @@ -0,0 +1,202 @@ +body { + font-family: Arial, Helvetica, sans-serif; + font-size: 10pt; + background: #fff; + color: #000; +} + +.title { + font-family: Verdana; +} + +p { + text-align: justify; + text-rendering: optimizeLegibility; + text-wrap: pretty; + overflow-wrap: break-word; + hyphens: auto; + orphans: 2; + widows: 2; +} + +h1 { + text-wrap: balance; + line-height: 1.4; +} + +emu-note p, +emu-table td p { + text-align: left; + hyphens: manual; + overflow: hidden; +} + +emu-table td, +emu-table th { + overflow-wrap: break-word; +} + +emu-table table { + table-layout: auto; + width: 100%; +} + +emu-figure img { + max-width: 100%; + height: auto; +} + +#spec-container { + max-width: none; +} + +#toc a, +#toc var { + color: #000; +} + +#toc a[href] { + background: #fff; + padding-right: 0.5em; +} +#toc a[href]::after { + content: /* leader(dotted) */ target-counter(attr(href), page); + float: right; + padding-left: 0.5em; + background: #fff; +} +/* NOTE: hacks because Paged.js doesn't support leader() in content directives */ +#toc ol { + overflow-x: hidden; +} +#toc ol li:before { + float: left; + width: 0; + white-space: nowrap; + content: '. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . ' + '. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . ' + '. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .'; +} + +/* skip the Introduction since it's before the first emu-clause (and therefore doesn't have a proper page number) */ +#toc > ol > li:first-child { + display: none; +} + +#toc > ol > li { + margin-top: 1ex; +} + +#toc, +#spec-container > emu-intro, +#spec-container > emu-annex { + break-before: recto; + break-after: always; +} + +/* according to Ecma guidelines, we're actually not supposed to break before every clause (only the first), but Paged.js fails if we do that */ +/* so instead, just break before any of the clauses that have sub-clauses */ +#spec-container > emu-clause:has(emu-clause:not([example])) { + break-before: always; +} + +#spec-container > emu-clause:first-of-type { + break-before: recto; +} + +emu-note, +emu-note p, +emu-table tr, +emu-table th, +emu-table td, +emu-alg li, +pre, +h1, +#metadata-block { + break-inside: avoid; +} + +emu-table thead, +h1, +figcaption, +emu-alg > ol > li:first-child { + break-after: avoid; +} + +emu-grammar + emu-alg, +figcaption + emu-table, +figcaption + span[id] + emu-table, +emu-alg > ol > li:last-child { + break-before: avoid; +} + +a[data-print-href]::after { + content: ' <' attr(href) '>'; + color: initial; +} + +emu-table thead { + display: table-header-group; +} +emu-table tfoot { + display: table-footer-group; +} + +@page { + size: A4; +} + +@page { + @top-center { + content: url(./ecma-header.png); + } +} +@page :first { + @top-center { + content: none; + } +} + +:root { + --page-number-style: decimal; +} + +#toc { + page: toc; +} +@page toc { + --page-number-style: lower-roman; +} +emu-intro { + page: intro; +} +@page intro { + --page-number-style: lower-roman; +} + +#toc { + counter-reset: page 1; +} +#spec-container > emu-clause:first-of-type { + counter-reset: page 1; +} + +@page :left { + @bottom-left { + content: counter(page, var(--page-number-style)); + } +} +@page :right { + @bottom-right { + content: counter(page, var(--page-number-style)); + } +} + +@page :first { + @bottom-left { + content: ''; + } + @bottom-right { + content: ''; + } +}