Spaces:
Sleeping
Sleeping
k-l-lambda
commited on
Commit
·
d605f27
1
Parent(s):
4cadbaf
commit lotus dist.
Browse filesThis view is limited to 50 files because it contains too many changes.
See raw diff
- .gitignore +1 -0
- Dockerfile +15 -0
- backend/advancedEngraver.ts +169 -0
- backend/canvas.ts +26 -0
- backend/constants.ts +13 -0
- backend/dirServer.ts +41 -0
- backend/index.ts +155 -0
- backend/lilyAddon.ts +104 -0
- backend/lilyCommands.ts +438 -0
- backend/loadJisonParserNode.ts +22 -0
- backend/loadLilyParserNode.ts +8 -0
- backend/scoreMaker.ts +359 -0
- backend/scoreMaker.ts.old +665 -0
- backend/statStorage.ts +34 -0
- backend/walkDir.ts +35 -0
- backend/xmlTools.ts +214 -0
- dist/e5c1442134f1e7dfb9dd.worker.js +0 -0
- dist/favicon.ico +0 -0
- dist/index.html +1 -0
- dist/js/chunk-04395031.dbff2f6b.js +0 -0
- dist/js/chunk-0c4e36c8.95d70738.js +16 -0
- dist/js/chunk-0cbfe13e.73856287.js +2 -0
- dist/js/chunk-117382e0.d47336d3.js +2 -0
- dist/js/chunk-2d0c53c7.d24941b8.js +2 -0
- dist/js/chunk-2d0db258.a4804a7a.js +2 -0
- dist/js/chunk-40965e1a.74707226.js +2 -0
- dist/js/chunk-48b5b2a0.3db5a0aa.js +0 -0
- dist/js/chunk-a06ef50c.1caef24f.js +2 -0
- dist/js/chunk-ae402692.003457bc.js +0 -0
- dist/js/chunk-vendors.20f7f886.js +8 -0
- dist/js/index.cbb15892.js +2 -0
- dist/soundfont/acoustic_grand_piano-mp3.js +0 -0
- dist/soundfont/acoustic_grand_piano-ogg.js +0 -0
- inc/DictArray.ts +35 -0
- inc/asyncCall.ts +11 -0
- inc/constants.ts +38 -0
- inc/domUtils.ts +49 -0
- inc/jisonWrapper.ts +34 -0
- inc/jsonRecovery.ts +41 -0
- inc/lilyNotation/fuzzyMatch.ts +125 -0
- inc/lilyNotation/implicitType.ts +16 -0
- inc/lilyNotation/index.ts +27 -0
- inc/lilyNotation/matcher.ts +191 -0
- inc/lilyNotation/notation.ts +675 -0
- inc/lilyNotation/scheduler.ts +167 -0
- inc/lilyParser/idioms.ts +51 -0
- inc/lilyParser/index.ts +28 -0
- inc/lilyParser/lilyDocument.ts +1062 -0
- inc/lilyParser/lilyInterpreter.ts +1476 -0
- inc/lilyParser/lilyTerms.ts +2991 -0
.gitignore
ADDED
@@ -0,0 +1 @@
|
|
|
|
|
1 |
+
*.map
|
Dockerfile
CHANGED
@@ -24,6 +24,21 @@ WORKDIR /app
|
|
24 |
# Copy the entire project including node_modules
|
25 |
COPY . .
|
26 |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
27 |
# Expose port
|
28 |
EXPOSE 7860
|
29 |
|
|
|
24 |
# Copy the entire project including node_modules
|
25 |
COPY . .
|
26 |
|
27 |
+
# Remove .env.local if exists
|
28 |
+
RUN rm -f .env.local
|
29 |
+
|
30 |
+
# Set all environment variables
|
31 |
+
ENV PORT=7860
|
32 |
+
ENV HOST=0.0.0.0
|
33 |
+
ENV LILYPOND_DATADIR=/app/node-addon-lilypond/output/share/lilypond/current/
|
34 |
+
ENV GUILE_LOAD_PATH=/app/node-addon-lilypond/output/share/guile/1.8
|
35 |
+
ENV LD_LIBRARY_PATH=/app/node-addon-lilypond/output
|
36 |
+
ENV LILYPOND_ADDON=/app/node-addon-lilypond/output/lilypond.node
|
37 |
+
|
38 |
+
ENV TEMP_DIR=./temp/
|
39 |
+
ENV LILYPOND_DIR=""
|
40 |
+
ENV MIDI_FILE_EXTEND=midi
|
41 |
+
|
42 |
# Expose port
|
43 |
EXPOSE 7860
|
44 |
|
backend/advancedEngraver.ts
ADDED
@@ -0,0 +1,169 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
|
2 |
+
import {Writable} from "stream";
|
3 |
+
import {MusicNotation} from "@k-l-lambda/music-widgets";
|
4 |
+
import {DOMParser} from "xmldom";
|
5 |
+
|
6 |
+
import {LilyDocument, docLocationSet} from "../inc/lilyParser";
|
7 |
+
import {engraveSvg} from "./lilyCommands";
|
8 |
+
import {SingleLock} from "../inc/mutex";
|
9 |
+
import * as staffSvg from "../inc/staffSvg";
|
10 |
+
import * as LilyNotation from "../inc/lilyNotation";
|
11 |
+
import LogRecorder from "../inc/logRecorder";
|
12 |
+
|
13 |
+
|
14 |
+
|
15 |
+
type StaffArguments = {attributes: staffSvg.StaffAttributes, tieLocations?: Set<string>, briefChordLocations?: Set<string>, lyricLocations?: Set<string>};
|
16 |
+
|
17 |
+
|
18 |
+
interface EngraverOptions {
|
19 |
+
streamSeparator: string;
|
20 |
+
|
21 |
+
includeFolders: string[];
|
22 |
+
withMIDI: boolean;
|
23 |
+
withNotation: boolean;
|
24 |
+
withLilyDoc: boolean;
|
25 |
+
withLilyNotation: boolean;
|
26 |
+
logger: LogRecorder;
|
27 |
+
|
28 |
+
lilyNotation: LilyNotation.Notation;
|
29 |
+
staffArgs: StaffArguments;
|
30 |
+
};
|
31 |
+
|
32 |
+
|
33 |
+
interface GrammarParser {
|
34 |
+
parse (source: string): any;
|
35 |
+
};
|
36 |
+
|
37 |
+
|
38 |
+
const STREAM_SEPARATOR = "\n\n\n\n";
|
39 |
+
|
40 |
+
|
41 |
+
const advancedEngrave = async (source: string, lilyParser: GrammarParser, output: Writable, options: Partial<EngraverOptions> = {}) => {
|
42 |
+
const {streamSeparator = STREAM_SEPARATOR} = options;
|
43 |
+
|
44 |
+
const outputJSON = data => setImmediate(() => {
|
45 |
+
output.write(JSON.stringify(data));
|
46 |
+
output.write(streamSeparator);
|
47 |
+
});
|
48 |
+
|
49 |
+
const t0 = Date.now();
|
50 |
+
|
51 |
+
const notatioinGen = new SingleLock<LilyNotation.Notation>(true);
|
52 |
+
const argsGen = new SingleLock<StaffArguments>(true);
|
53 |
+
|
54 |
+
const hashKeys = new Set<string>();
|
55 |
+
|
56 |
+
const engraving = await engraveSvg(source, {
|
57 |
+
includeFolders: options.includeFolders,
|
58 |
+
|
59 |
+
// do some work during lilypond process running to save time
|
60 |
+
onProcStart: () => {
|
61 |
+
//console.log("tp.0:", Date.now() - t0);
|
62 |
+
if (options.staffArgs)
|
63 |
+
argsGen.release(options.staffArgs);
|
64 |
+
|
65 |
+
if (!options.withLilyNotation && !options.withLilyDoc && options.staffArgs)
|
66 |
+
return;
|
67 |
+
|
68 |
+
const lilyDocument = new LilyDocument(lilyParser.parse(source));
|
69 |
+
|
70 |
+
if (!options.lilyNotation) {
|
71 |
+
const interpreter = lilyDocument.interpret();
|
72 |
+
options.lilyNotation = interpreter.getNotation();
|
73 |
+
|
74 |
+
notatioinGen.release(options.lilyNotation);
|
75 |
+
//console.log("tp.1:", Date.now() - t0);
|
76 |
+
}
|
77 |
+
|
78 |
+
if (options.withLilyDoc)
|
79 |
+
outputJSON({lilyDocument: lilyDocument.root});
|
80 |
+
|
81 |
+
if (!options.staffArgs) {
|
82 |
+
const attributes = lilyDocument.globalAttributes({readonly: true}) as staffSvg.StaffAttributes;
|
83 |
+
|
84 |
+
const tieLocations = docLocationSet(lilyDocument.getTiedNoteLocations2());
|
85 |
+
const briefChordLocations = docLocationSet(lilyDocument.getBriefChordLocations());
|
86 |
+
const lyricLocations = docLocationSet(lilyDocument.getLyricLocations());
|
87 |
+
|
88 |
+
//console.log("tp.2:", Date.now() - t0);
|
89 |
+
|
90 |
+
options.staffArgs = {attributes, tieLocations, briefChordLocations, lyricLocations};
|
91 |
+
argsGen.release(options.staffArgs);
|
92 |
+
//console.log("tp.3:", Date.now() - t0);
|
93 |
+
}
|
94 |
+
},
|
95 |
+
onMidiRead: async midi => {
|
96 |
+
//console.log("tm.0:", Date.now() - t0);
|
97 |
+
if (options.withMIDI)
|
98 |
+
outputJSON({midi});
|
99 |
+
|
100 |
+
if (options.withNotation && midi) {
|
101 |
+
const midiNotation = MusicNotation.Notation.parseMidi(midi);
|
102 |
+
outputJSON({midiNotation});
|
103 |
+
}
|
104 |
+
|
105 |
+
if (options.withLilyNotation && midi) {
|
106 |
+
const lilyNotation = options.lilyNotation || await notatioinGen.wait();
|
107 |
+
//console.log("tm.2:", Date.now() - t0);
|
108 |
+
await LilyNotation.matchWithExactMIDI(lilyNotation, midi);
|
109 |
+
//console.log("tm.3:", Date.now() - t0);
|
110 |
+
|
111 |
+
outputJSON({lilyNotation});
|
112 |
+
}
|
113 |
+
|
114 |
+
//console.log("tm.4:", Date.now() - t0);
|
115 |
+
},
|
116 |
+
onSvgRead: async (index, svg) => {
|
117 |
+
//console.log("ts.0:", index, Date.now() - t0);
|
118 |
+
const args = options.staffArgs || await argsGen.wait();
|
119 |
+
//console.log("ts.1:", index, Date.now() - t0);
|
120 |
+
const page = staffSvg.parseSvgPage(svg, source, {DOMParser, logger: options.logger, ...args});
|
121 |
+
|
122 |
+
// select incremental keys to send
|
123 |
+
const hashTable = {};
|
124 |
+
Object.entries(page.hashTable).forEach(([key, elem]) => {
|
125 |
+
if (!hashKeys.has(key))
|
126 |
+
hashTable[key] = elem;
|
127 |
+
});
|
128 |
+
|
129 |
+
// rectify page data by lilyNotation
|
130 |
+
const lilyNotation = options.lilyNotation || await notatioinGen.wait();
|
131 |
+
//console.log("ts.2:", index, Date.now() - t0);
|
132 |
+
if (lilyNotation) {
|
133 |
+
const sheetDocument = new staffSvg.SheetDocument({pages: [page.structure]}, {initialize: true});
|
134 |
+
|
135 |
+
sheetDocument.alignTokensWithNotation(options.lilyNotation, {partial: true});
|
136 |
+
sheetDocument.updateMatchedTokens(options.lilyNotation.idSet);
|
137 |
+
}
|
138 |
+
|
139 |
+
outputJSON({
|
140 |
+
page: index,
|
141 |
+
structure: page.structure,
|
142 |
+
hashTable,
|
143 |
+
});
|
144 |
+
|
145 |
+
Object.keys(hashTable).forEach(key => hashKeys.add(key));
|
146 |
+
|
147 |
+
//console.log("ts.3:", index, Date.now() - t0);
|
148 |
+
},
|
149 |
+
});
|
150 |
+
|
151 |
+
const tn = Date.now();
|
152 |
+
//console.log("tn:", tn - t0);
|
153 |
+
|
154 |
+
options.logger.append("advancedEngraver.profile.engraving", {cost: tn - t0});
|
155 |
+
|
156 |
+
await new Promise(resolve => setImmediate(resolve));
|
157 |
+
|
158 |
+
output.write(JSON.stringify({
|
159 |
+
logs: engraving.logs,
|
160 |
+
logger: options.logger,
|
161 |
+
errorLevel: engraving.errorLevel,
|
162 |
+
}));
|
163 |
+
};
|
164 |
+
|
165 |
+
|
166 |
+
|
167 |
+
export {
|
168 |
+
advancedEngrave,
|
169 |
+
};
|
backend/canvas.ts
ADDED
@@ -0,0 +1,26 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
|
2 |
+
//import {loadImage, createCanvas} from "canvas";
|
3 |
+
import {PNGStream} from "canvas";
|
4 |
+
|
5 |
+
const {loadImage, createCanvas}: any = !process.env.MOBILE_MODE ? require("canvas") : {};
|
6 |
+
|
7 |
+
|
8 |
+
|
9 |
+
const svgToPng = async (sourceURL: string|Buffer): Promise<PNGStream> => {
|
10 |
+
if (process.env.MOBILE_MODE)
|
11 |
+
return;
|
12 |
+
|
13 |
+
const image = await loadImage(sourceURL);
|
14 |
+
|
15 |
+
const canvas = createCanvas(image.width, image.height);
|
16 |
+
const ctx = canvas.getContext("2d");
|
17 |
+
ctx.drawImage(image, 0, 0, image.width, image.height);
|
18 |
+
|
19 |
+
return canvas.createPNGStream();
|
20 |
+
};
|
21 |
+
|
22 |
+
|
23 |
+
|
24 |
+
export {
|
25 |
+
svgToPng,
|
26 |
+
};
|
backend/constants.ts
ADDED
@@ -0,0 +1,13 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
|
2 |
+
import path from "path";
|
3 |
+
|
4 |
+
|
5 |
+
|
6 |
+
// @ts-ignore
|
7 |
+
const LY_INCLUDE_FOLDERS = ["../ly"].map(folder => path.resolve(process.env.TEMP_DIR, folder));
|
8 |
+
|
9 |
+
|
10 |
+
|
11 |
+
export {
|
12 |
+
LY_INCLUDE_FOLDERS,
|
13 |
+
};
|
backend/dirServer.ts
ADDED
@@ -0,0 +1,41 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
|
2 |
+
import path from "path";
|
3 |
+
import serveHandler from "serve-handler";
|
4 |
+
|
5 |
+
|
6 |
+
|
7 |
+
const createHandler = (publicDir: string, apiPath: string) => (req, res) => {
|
8 |
+
req.url = "/" + path.relative(`${apiPath}/`, req.baseUrl);
|
9 |
+
|
10 |
+
// modify link URL origin
|
11 |
+
const responseProxy = new Proxy(res, {
|
12 |
+
get (res, prop) {
|
13 |
+
//console.log("proxy.get:", prop);
|
14 |
+
switch (prop) {
|
15 |
+
case "end":
|
16 |
+
return (content) => {
|
17 |
+
//console.log("end with:", content);
|
18 |
+
const html = content && content.replace(/href="/g, `href="${apiPath}`);
|
19 |
+
|
20 |
+
res.end(html);
|
21 |
+
};
|
22 |
+
}
|
23 |
+
|
24 |
+
if (typeof res[prop] === "function")
|
25 |
+
return res[prop].bind(res);
|
26 |
+
|
27 |
+
return res[prop];
|
28 |
+
},
|
29 |
+
});
|
30 |
+
|
31 |
+
serveHandler(req, responseProxy, {
|
32 |
+
public: publicDir,
|
33 |
+
//symlinks: true,
|
34 |
+
});
|
35 |
+
};
|
36 |
+
|
37 |
+
|
38 |
+
|
39 |
+
export {
|
40 |
+
createHandler,
|
41 |
+
};
|
backend/index.ts
ADDED
@@ -0,0 +1,155 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
|
2 |
+
import formidable from "formidable";
|
3 |
+
import {DOMParser} from "xmldom";
|
4 |
+
import {MIDI} from "@k-l-lambda/music-widgets";
|
5 |
+
|
6 |
+
import * as lilyCommands from "./lilyCommands";
|
7 |
+
import * as staffSvg from "../inc/staffSvg";
|
8 |
+
import loadLilyParser from "./loadLilyParserNode";
|
9 |
+
import {LilyDocument} from "../inc/lilyParser";
|
10 |
+
import LogRecorder from "../inc/logRecorder";
|
11 |
+
import * as ScoreMaker from "./scoreMaker";
|
12 |
+
import * as constants from "./constants";
|
13 |
+
import {advancedEngrave} from "./advancedEngraver";
|
14 |
+
|
15 |
+
|
16 |
+
|
17 |
+
const formidableHandle = (name, req, res, handle) =>
|
18 |
+
new formidable.IncomingForm().parse(req, async (err, fields, files) => {
|
19 |
+
try {
|
20 |
+
if (err)
|
21 |
+
throw err;
|
22 |
+
|
23 |
+
const result = await handle(fields, files);
|
24 |
+
|
25 |
+
const resultObj = (typeof result === "string" || result instanceof Buffer) ? {body: result} : result;
|
26 |
+
|
27 |
+
res.writeHead(resultObj.body ? 200 : 404, resultObj.header);
|
28 |
+
res.write(resultObj.body);
|
29 |
+
res.end();
|
30 |
+
}
|
31 |
+
catch (error) {
|
32 |
+
console.error(`${name} error:`, error);
|
33 |
+
|
34 |
+
res.writeHead(500);
|
35 |
+
res.write(error.toString());
|
36 |
+
res.end();
|
37 |
+
}
|
38 |
+
});
|
39 |
+
|
40 |
+
|
41 |
+
const service = {
|
42 |
+
"/musicxml2ly": {
|
43 |
+
post: (req, res) => formidableHandle("musicxml2ly", req, res,
|
44 |
+
({xml, options}) => lilyCommands.xml2ly(xml, options && JSON.parse(options))),
|
45 |
+
},
|
46 |
+
|
47 |
+
|
48 |
+
"/midi2ly": {
|
49 |
+
post: (req, res) => formidableHandle("midi2ly", req, res,
|
50 |
+
({options}, {midi}) => lilyCommands.midi2ly(midi, options && JSON.parse(options))),
|
51 |
+
},
|
52 |
+
|
53 |
+
|
54 |
+
"/engrave": {
|
55 |
+
post: (req, res) => formidableHandle("engrave", req, res,
|
56 |
+
async ({source, tokenize = false, log = false}) => {
|
57 |
+
const result = await lilyCommands.engraveSvg(source, {includeFolders: constants.LY_INCLUDE_FOLDERS});
|
58 |
+
if (!tokenize)
|
59 |
+
return JSON.stringify(result);
|
60 |
+
|
61 |
+
const lilyParser = await loadLilyParser();
|
62 |
+
const lilyDocument = new LilyDocument(lilyParser.parse(source));
|
63 |
+
//const attributes = lilyDocument.globalAttributes({readonly: true});
|
64 |
+
//console.log("attributes:", attributes);
|
65 |
+
|
66 |
+
lilyDocument.interpret();
|
67 |
+
|
68 |
+
const logger = new LogRecorder({enabled: log});
|
69 |
+
|
70 |
+
const {doc, hashTable} = staffSvg.createSheetDocumentFromSvgs(result.svgs, source, lilyDocument, {logger, DOMParser});
|
71 |
+
|
72 |
+
return JSON.stringify({
|
73 |
+
...result,
|
74 |
+
doc,
|
75 |
+
hashTable,
|
76 |
+
logger,
|
77 |
+
});
|
78 |
+
}),
|
79 |
+
},
|
80 |
+
|
81 |
+
|
82 |
+
"/engraveScm": {
|
83 |
+
post: (req, res) => formidableHandle("engraveScm", req, res,
|
84 |
+
async ({source}) => {
|
85 |
+
const result = await lilyCommands.engraveScm(source, {includeFolders: constants.LY_INCLUDE_FOLDERS});
|
86 |
+
return JSON.stringify(result);
|
87 |
+
}),
|
88 |
+
},
|
89 |
+
|
90 |
+
|
91 |
+
"/engraveMIDI": {
|
92 |
+
post: (req, res) => formidableHandle("engraveMIDI", req, res,
|
93 |
+
async ({source, articulate = false}) => {
|
94 |
+
const lilyParser = await loadLilyParser();
|
95 |
+
const midi = await (articulate ? ScoreMaker.makeArticulatedMIDI(source, lilyParser, {includeFolders: constants.LY_INCLUDE_FOLDERS})
|
96 |
+
: ScoreMaker.makeMIDI(source, lilyParser, {includeFolders: constants.LY_INCLUDE_FOLDERS}));
|
97 |
+
const buffer = Buffer.from(MIDI.encodeMidiFile(midi));
|
98 |
+
|
99 |
+
return {
|
100 |
+
header: {
|
101 |
+
"Content-Type": "audio/midi",
|
102 |
+
},
|
103 |
+
body: buffer,
|
104 |
+
};
|
105 |
+
}),
|
106 |
+
},
|
107 |
+
|
108 |
+
|
109 |
+
"/advanced-engrave": {
|
110 |
+
post: (req, res) => new formidable.IncomingForm().parse(req, async (err, fields) => {
|
111 |
+
try {
|
112 |
+
if (err)
|
113 |
+
throw err;
|
114 |
+
|
115 |
+
const logger = new LogRecorder({enabled: !!fields.log});
|
116 |
+
|
117 |
+
const lilyParser = await loadLilyParser();
|
118 |
+
|
119 |
+
const task = advancedEngrave(fields.source, lilyParser, res, {
|
120 |
+
includeFolders: constants.LY_INCLUDE_FOLDERS,
|
121 |
+
withMIDI: !!fields.withMIDI,
|
122 |
+
withNotation: !!fields.withNotation,
|
123 |
+
withLilyDoc: !!fields.withLilyDoc,
|
124 |
+
withLilyNotation: !!fields.withLilyNotation,
|
125 |
+
logger,
|
126 |
+
});
|
127 |
+
|
128 |
+
res.writeHead(200);
|
129 |
+
|
130 |
+
await task;
|
131 |
+
res.end();
|
132 |
+
}
|
133 |
+
catch (err) {
|
134 |
+
console.warn("advanced-engrave error:", err);
|
135 |
+
|
136 |
+
res.writeHead(500);
|
137 |
+
res.write(err.toString());
|
138 |
+
res.end();
|
139 |
+
}
|
140 |
+
}),
|
141 |
+
},
|
142 |
+
};
|
143 |
+
|
144 |
+
|
145 |
+
const setEnvironment = env => lilyCommands.setEnvironment(env);
|
146 |
+
|
147 |
+
|
148 |
+
|
149 |
+
export {
|
150 |
+
setEnvironment,
|
151 |
+
service,
|
152 |
+
lilyCommands,
|
153 |
+
ScoreMaker,
|
154 |
+
advancedEngrave,
|
155 |
+
};
|
backend/lilyAddon.ts
ADDED
@@ -0,0 +1,104 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
|
2 |
+
import {Writable} from "stream";
|
3 |
+
import {MIDI} from "@k-l-lambda/music-widgets";
|
4 |
+
|
5 |
+
import {EngraverOptions, EngraverResult, postProcessSvg} from "./lilyCommands";
|
6 |
+
|
7 |
+
|
8 |
+
|
9 |
+
let lilypondEx = null;
|
10 |
+
|
11 |
+
const loadAddon = (path: string): boolean => {
|
12 |
+
lilypondEx = require(path);
|
13 |
+
|
14 |
+
return !!lilypondEx;
|
15 |
+
};
|
16 |
+
|
17 |
+
|
18 |
+
const engraveSvg = async (source: string,
|
19 |
+
{onProcStart, onMidiRead, onSvgRead, includeFolders = []}: EngraverOptions = {}): Promise<Partial<EngraverResult>> => {
|
20 |
+
if (!lilypondEx)
|
21 |
+
throw new Error("Lilypond addon not loaded.");
|
22 |
+
|
23 |
+
let logs;
|
24 |
+
const svgs = [];
|
25 |
+
let midi;
|
26 |
+
|
27 |
+
const engravePromise = lilypondEx.engrave(source, {
|
28 |
+
includeFolders,
|
29 |
+
log (message) {
|
30 |
+
logs = message;
|
31 |
+
},
|
32 |
+
onSVG (filename, content) {
|
33 |
+
const captures = filename.match(/-(\d+)\.svg$/);
|
34 |
+
const index = captures ? Number(captures[1]) - 1 : 0;
|
35 |
+
|
36 |
+
const svg = postProcessSvg(content);
|
37 |
+
svgs[index] = svg;
|
38 |
+
|
39 |
+
onSvgRead && onSvgRead(index, svg);
|
40 |
+
},
|
41 |
+
onMIDI (_, buffer) {
|
42 |
+
midi = MIDI.parseMidiData(buffer);
|
43 |
+
|
44 |
+
onMidiRead && onMidiRead(midi);
|
45 |
+
},
|
46 |
+
});
|
47 |
+
|
48 |
+
await onProcStart && onProcStart();
|
49 |
+
|
50 |
+
const errorLevel = await engravePromise;
|
51 |
+
|
52 |
+
return {
|
53 |
+
logs,
|
54 |
+
svgs,
|
55 |
+
midi,
|
56 |
+
errorLevel,
|
57 |
+
};
|
58 |
+
};
|
59 |
+
|
60 |
+
|
61 |
+
const engraveSvgWithStream = async (source: string,
|
62 |
+
output: Writable, {includeFolders = []}: {includeFolders?: string[]} = {},
|
63 |
+
): Promise<Partial<EngraverResult>> => {
|
64 |
+
let logs;
|
65 |
+
//let midi;
|
66 |
+
|
67 |
+
//const tasks = [];
|
68 |
+
|
69 |
+
const errorLevel = await lilypondEx.engrave(source, {
|
70 |
+
includeFolders,
|
71 |
+
log (message) {
|
72 |
+
logs = message;
|
73 |
+
},
|
74 |
+
onSVG (_, content) {
|
75 |
+
// Write stream in non-main thread may result in blocking, so post a task to event looping.
|
76 |
+
//tasks.push(new Promise(resolve => ));
|
77 |
+
setImmediate(() => {
|
78 |
+
output.write(content);
|
79 |
+
output.write("\n\n\n\n");
|
80 |
+
});
|
81 |
+
},
|
82 |
+
//onMIDI (_, buffer) {
|
83 |
+
// midi = MIDI.parseMidiData(buffer);
|
84 |
+
//},
|
85 |
+
});
|
86 |
+
|
87 |
+
//await Promise.all(tasks);
|
88 |
+
await new Promise(resolve => setImmediate(resolve));
|
89 |
+
|
90 |
+
return {
|
91 |
+
logs,
|
92 |
+
//midi,
|
93 |
+
errorLevel,
|
94 |
+
};
|
95 |
+
};
|
96 |
+
|
97 |
+
|
98 |
+
|
99 |
+
|
100 |
+
export {
|
101 |
+
loadAddon,
|
102 |
+
engraveSvg,
|
103 |
+
engraveSvgWithStream,
|
104 |
+
};
|
backend/lilyCommands.ts
ADDED
@@ -0,0 +1,438 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
|
2 |
+
|
3 |
+
import fs from "fs";
|
4 |
+
import path from "path";
|
5 |
+
import child_process from "child-process-promise";
|
6 |
+
import {MIDI} from "@k-l-lambda/music-widgets";
|
7 |
+
import {Writable} from "stream";
|
8 |
+
|
9 |
+
import asyncCall from "../inc/asyncCall";
|
10 |
+
import {SingleLock} from "../inc/mutex";
|
11 |
+
import {preprocessXml} from "./xmlTools";
|
12 |
+
import * as lilyAddon from "./lilyAddon";
|
13 |
+
|
14 |
+
|
15 |
+
|
16 |
+
let MUSICXML2LY_PATH;
|
17 |
+
let MIDI2LY_PATH;
|
18 |
+
let LILYPOND_PATH;
|
19 |
+
|
20 |
+
|
21 |
+
let env; // setEnvironment must be called before use commands
|
22 |
+
const setEnvironment = e => {
|
23 |
+
env = e;
|
24 |
+
|
25 |
+
MUSICXML2LY_PATH = filePathResolve(env.LILYPOND_DIR, "musicxml2ly");
|
26 |
+
MIDI2LY_PATH = filePathResolve(env.LILYPOND_DIR, "midi2ly");
|
27 |
+
LILYPOND_PATH = filePathResolve(env.LILYPOND_DIR, "lilypond");
|
28 |
+
|
29 |
+
emptyCache();
|
30 |
+
|
31 |
+
if (env.LILYPOND_ADDON) {
|
32 |
+
if (env.LILYPOND_ADDON_ASSETS_DIR) {
|
33 |
+
process.env.LILYPOND_PATH = path.join(env.LILYPOND_ADDON_ASSETS_DIR, "bin/lilypond");
|
34 |
+
process.env.GUILE_LOAD_PATH = path.join(env.LILYPOND_ADDON_ASSETS_DIR, "share/guile/1.8");
|
35 |
+
process.env.FONTCONFIG_PATH = path.join(env.LILYPOND_ADDON_ASSETS_DIR, "share/fonts");
|
36 |
+
}
|
37 |
+
|
38 |
+
lilyAddon.loadAddon(env.LILYPOND_ADDON);
|
39 |
+
}
|
40 |
+
};
|
41 |
+
|
42 |
+
|
43 |
+
const _WINDOWS = process.platform === "win32";
|
44 |
+
|
45 |
+
|
46 |
+
const genHashString = (len = 8) => Buffer.from(Math.random().toString()).toString("base64").substr(3, 3 + len);
|
47 |
+
|
48 |
+
|
49 |
+
const filePathResolve = (...parts: string[]): string => {
|
50 |
+
const result = path.join(...parts);
|
51 |
+
return _WINDOWS ? `"${result}"` : result;
|
52 |
+
};
|
53 |
+
|
54 |
+
|
55 |
+
const emptyCache = async () => {
|
56 |
+
// empty temporary directory
|
57 |
+
try {
|
58 |
+
if (env.TEMP_DIR) {
|
59 |
+
if (_WINDOWS)
|
60 |
+
await child_process.exec(`del /q "${env.TEMP_DIR}*"`);
|
61 |
+
else
|
62 |
+
await child_process.exec(`rm ${env.TEMP_DIR}*`);
|
63 |
+
console.log("Temporary directory clear.");
|
64 |
+
}
|
65 |
+
}
|
66 |
+
catch (err) {
|
67 |
+
if (_WINDOWS)
|
68 |
+
console.log("emptyCache error:", err);
|
69 |
+
}
|
70 |
+
};
|
71 |
+
|
72 |
+
|
73 |
+
export interface LilyProcessOptions {
|
74 |
+
// xml
|
75 |
+
language?: string;
|
76 |
+
removeMeasureImplicit?: boolean;
|
77 |
+
replaceEncoding?: boolean;
|
78 |
+
removeNullDynamics?: boolean;
|
79 |
+
fixHeadMarkup?: boolean;
|
80 |
+
fixBackSlashes?: boolean;
|
81 |
+
roundTempo?: boolean;
|
82 |
+
escapedWordsDoubleQuotation?: boolean;
|
83 |
+
removeTrivialRests?: boolean;
|
84 |
+
removeBadMetronome?: boolean;
|
85 |
+
removeInvalidHarmonies?: boolean;
|
86 |
+
removeAllHarmonies?: boolean;
|
87 |
+
fixChordVoice?: boolean;
|
88 |
+
fixBarlines?: boolean;
|
89 |
+
removeInvalidClef?: boolean;
|
90 |
+
|
91 |
+
// lilypond
|
92 |
+
pointClick?: boolean;
|
93 |
+
midi?: boolean;
|
94 |
+
removeBreak?: boolean;
|
95 |
+
removePageBreak?: boolean;
|
96 |
+
removeInstrumentName?: boolean;
|
97 |
+
removeTempo?: boolean;
|
98 |
+
tupletReplace?: boolean;
|
99 |
+
};
|
100 |
+
|
101 |
+
|
102 |
+
|
103 |
+
|
104 |
+
const postProcessLy = (ly: string, {
|
105 |
+
pointClick = true,
|
106 |
+
midi = true,
|
107 |
+
removeBreak = false,
|
108 |
+
removePageBreak = false,
|
109 |
+
removeInstrumentName = false,
|
110 |
+
removeTempo = false,
|
111 |
+
tupletReplace = false,
|
112 |
+
} = {}): string => {
|
113 |
+
let result = ly;
|
114 |
+
|
115 |
+
if (pointClick)
|
116 |
+
result = result.replace(/\\pointAndClickOff\n/g, "");
|
117 |
+
|
118 |
+
if (midi)
|
119 |
+
result = result.replace(/% \\midi/g, "\\midi");
|
120 |
+
|
121 |
+
if (removeBreak)
|
122 |
+
result = result.replace(/\s\\break\s/g, " ");
|
123 |
+
|
124 |
+
if (removePageBreak)
|
125 |
+
result = result.replace(/\s\\pageBreak\s/g, " ");
|
126 |
+
|
127 |
+
if (removeInstrumentName)
|
128 |
+
result = result.replace(/\\set Staff\.instrumentName/g, "% \\set Staff.instrumentName");
|
129 |
+
|
130 |
+
if (removeTempo)
|
131 |
+
result = result.replace(/\\tempo /g, "% \\tempo ");
|
132 |
+
|
133 |
+
if (tupletReplace) {
|
134 |
+
result = result
|
135 |
+
.replace(/4\*128\/384/g, "8*2/3")
|
136 |
+
.replace(/4\*64\/384/g, "16*2/3");
|
137 |
+
}
|
138 |
+
|
139 |
+
return result;
|
140 |
+
};
|
141 |
+
|
142 |
+
|
143 |
+
const xml2ly = async (xml: string, {language = "english", ...options}: LilyProcessOptions): Promise<string> => {
|
144 |
+
xml = preprocessXml(xml, options);
|
145 |
+
//console.log("xml:", options, xml.substr(0, 100));
|
146 |
+
|
147 |
+
const hash = genHashString();
|
148 |
+
const xmlFileName = `${env.TEMP_DIR}xml2ly-${hash}.xml`;
|
149 |
+
await asyncCall(fs.writeFile, xmlFileName, xml);
|
150 |
+
|
151 |
+
const lyFileName = `${env.TEMP_DIR}xml2ly-${hash}.ly`;
|
152 |
+
|
153 |
+
if (env.MUSICXML2LY_BY_PYTHON) {
|
154 |
+
await child_process.spawn(path.resolve(env.LILYPOND_DIR, "python"), [
|
155 |
+
path.resolve(env.LILYPOND_DIR, "musicxml2ly.py"), xmlFileName, "-o", lyFileName,
|
156 |
+
...(language ? ["-l", language] : []),
|
157 |
+
]);
|
158 |
+
}
|
159 |
+
else
|
160 |
+
await child_process.exec(`${MUSICXML2LY_PATH} ${xmlFileName} -o ${lyFileName} ${language ? "-l " + language : ""}`, {maxBuffer: 0x80000});
|
161 |
+
//console.log("musicxml2ly:", result.stdout, result.stderr);
|
162 |
+
|
163 |
+
const ly = await asyncCall(fs.readFile, lyFileName);
|
164 |
+
|
165 |
+
return postProcessLy(ly.toString(), options);
|
166 |
+
};
|
167 |
+
|
168 |
+
|
169 |
+
const midi2ly = async (midi, options: LilyProcessOptions): Promise<string> => {
|
170 |
+
const hash = genHashString();
|
171 |
+
//const midiFileName = `${env.TEMP_DIR}midi2ly-${hash}.midi`;
|
172 |
+
//await asyncCall(fs.writeFile, midiFileName, midi);
|
173 |
+
|
174 |
+
const lyFileName = `${env.TEMP_DIR}midi2ly-${hash}-midi.ly`;
|
175 |
+
|
176 |
+
let result;
|
177 |
+
if (env.MUSICXML2LY_BY_PYTHON) {
|
178 |
+
result = await child_process.spawn(path.resolve(env.LILYPOND_DIR, "python"),
|
179 |
+
[path.resolve(env.LILYPOND_DIR, "midi2ly.py"), midi.path, "-o", lyFileName]);
|
180 |
+
}
|
181 |
+
else
|
182 |
+
result = await child_process.exec(`${MIDI2LY_PATH} ${midi.path} -o ${lyFileName}`);
|
183 |
+
console.log("midi2ly:", result.stdout, result.stderr);
|
184 |
+
|
185 |
+
const ly = await asyncCall(fs.readFile, lyFileName);
|
186 |
+
|
187 |
+
return postProcessLy(ly.toString(), options);
|
188 |
+
};
|
189 |
+
|
190 |
+
|
191 |
+
const postProcessSvg = (svg: string): string => {
|
192 |
+
return svg.replace(/textedit:[^"]+:(\d+:\d+:\d+)/g, "textedit:$1");
|
193 |
+
};
|
194 |
+
|
195 |
+
|
196 |
+
//const nameNumber = name => Number(name.match(/-(\d+)\./)[1]);
|
197 |
+
|
198 |
+
|
199 |
+
// lilypond command line output file pattern
|
200 |
+
const FILE_BORN_OUPUT_PATTERN = /output\sto\s`(.+)'/;
|
201 |
+
|
202 |
+
|
203 |
+
export interface EngraverOptions {
|
204 |
+
onProcStart?: () => void|Promise<void>;
|
205 |
+
onMidiRead?: (content: MIDI.MidiData, options?: {filePath: string}) => void|Promise<void>;
|
206 |
+
onSvgRead?: (index: number, content: string, options?: {filePath: string}) => void|Promise<void>;
|
207 |
+
includeFolders?: string[]; // include folder path should be relative to TEMP_DIR
|
208 |
+
};
|
209 |
+
|
210 |
+
export interface EngraverResult {
|
211 |
+
logs: string;
|
212 |
+
svgs: string[];
|
213 |
+
midi: MIDI.MidiData;
|
214 |
+
errorLevel: number;
|
215 |
+
};
|
216 |
+
|
217 |
+
|
218 |
+
const engraveSvgCli = async (source: string,
|
219 |
+
{onProcStart, onMidiRead, onSvgRead, includeFolders = []}: EngraverOptions = {},
|
220 |
+
): Promise<Partial<EngraverResult>> => {
|
221 |
+
const hash = genHashString();
|
222 |
+
const sourceFilename = `${env.TEMP_DIR}engrave-${hash}.ly`;
|
223 |
+
|
224 |
+
await asyncCall(fs.writeFile, sourceFilename, source);
|
225 |
+
//console.log("ly source written:", sourceFilename);
|
226 |
+
|
227 |
+
let midi = null;
|
228 |
+
const svgs = [];
|
229 |
+
|
230 |
+
const fileReady = new SingleLock<string>();
|
231 |
+
|
232 |
+
const loadFile = async filename => {
|
233 |
+
//console.log("loadFile:", filename);
|
234 |
+
|
235 |
+
// wait for file writing finished
|
236 |
+
//await new Promise(resolve => setTimeout(resolve, 100));
|
237 |
+
|
238 |
+
const captures = filename.match(/\.(\w+)$/);
|
239 |
+
if (!captures) {
|
240 |
+
console.warn("invalid filename:", filename);
|
241 |
+
return;
|
242 |
+
}
|
243 |
+
const [_, ext] = captures;
|
244 |
+
|
245 |
+
const filePath = path.resolve(env.TEMP_DIR, filename);
|
246 |
+
//console.log("file output:", filePath, ext);
|
247 |
+
|
248 |
+
switch (ext) {
|
249 |
+
case env.MIDI_FILE_EXTEND: {
|
250 |
+
// wait extra time for MIDI file
|
251 |
+
//await fileReady.lock();
|
252 |
+
await new Promise(resolve => setTimeout(resolve, 100));
|
253 |
+
|
254 |
+
const buffer = await asyncCall(fs.readFile, filePath);
|
255 |
+
if (!buffer.length)
|
256 |
+
console.warn("empty MIDI buffer:", filename);
|
257 |
+
|
258 |
+
midi = MIDI.parseMidiData(buffer);
|
259 |
+
|
260 |
+
await onMidiRead && onMidiRead(midi, {filePath});
|
261 |
+
}
|
262 |
+
|
263 |
+
break;
|
264 |
+
case "svg": {
|
265 |
+
// eslint-disable-next-line
|
266 |
+
const captures = filename.match(/(\d+)\.svg$/);
|
267 |
+
const index = captures ? Number(captures[1]) - 1 : 0;
|
268 |
+
|
269 |
+
const buffer = await asyncCall(fs.readFile, filePath);
|
270 |
+
if (!buffer.length)
|
271 |
+
console.warn("empty SVG buffer:", filename);
|
272 |
+
|
273 |
+
const svg = postProcessSvg(buffer.toString());
|
274 |
+
svgs[index] = svg;
|
275 |
+
|
276 |
+
//console.log("svg load:", filePath);
|
277 |
+
await onSvgRead && onSvgRead(index, svg, {filePath});
|
278 |
+
}
|
279 |
+
|
280 |
+
break;
|
281 |
+
}
|
282 |
+
};
|
283 |
+
|
284 |
+
const loadProcs: Promise<void>[] = [];
|
285 |
+
|
286 |
+
const checkFile = async (line: string) => {
|
287 |
+
fileReady.release(line);
|
288 |
+
|
289 |
+
// skip error messages in command line output
|
290 |
+
if (FILE_BORN_OUPUT_PATTERN.test(line)) {
|
291 |
+
let newLine = await fileReady.lock();
|
292 |
+
while (/error|warning/.test(newLine))
|
293 |
+
newLine = await fileReady.lock();
|
294 |
+
|
295 |
+
let captures;
|
296 |
+
const exp = new RegExp(FILE_BORN_OUPUT_PATTERN.source, "g");
|
297 |
+
while (captures = exp.exec(line))
|
298 |
+
loadProcs.push(loadFile(captures[1]));
|
299 |
+
}
|
300 |
+
};
|
301 |
+
|
302 |
+
const includeParameters = includeFolders.map(folder => `--include=${folder}`).join(" ");
|
303 |
+
|
304 |
+
const proc: any = child_process.exec(`${LILYPOND_PATH} -dbackend=svg -o ${env.TEMP_DIR} ${includeParameters} ${sourceFilename}`,
|
305 |
+
{maxBuffer: 0x100000});
|
306 |
+
//const proc = child_process.spawn(LILYPOND_PATH, ["-dbackend=svg", `-o ${env.TEMP_DIR}`, sourceFilename]);
|
307 |
+
proc.childProcess.stdout.on("data", checkFile);
|
308 |
+
proc.childProcess.stderr.on("data", checkFile);
|
309 |
+
|
310 |
+
const startPromise = onProcStart && onProcStart();
|
311 |
+
if (startPromise)
|
312 |
+
loadProcs.push(startPromise);
|
313 |
+
|
314 |
+
const result = await proc;
|
315 |
+
|
316 |
+
fileReady.release();
|
317 |
+
|
318 |
+
await Promise.all(loadProcs);
|
319 |
+
|
320 |
+
//console.log("svgs:", svgs.length);
|
321 |
+
|
322 |
+
const validCount = svgs.filter(svg => svg).length;
|
323 |
+
if (validCount < svgs.length)
|
324 |
+
console.warn("svg loading incompleted: ", validCount, svgs.length);
|
325 |
+
|
326 |
+
return {
|
327 |
+
logs: result.stderr,
|
328 |
+
svgs,
|
329 |
+
midi,
|
330 |
+
};
|
331 |
+
};
|
332 |
+
|
333 |
+
|
334 |
+
const engraveSvgWithStreamCli = async (source: string, output: Writable, {includeFolders = []}: {includeFolders?: string[]} = {}) => {
|
335 |
+
const hash = genHashString();
|
336 |
+
const sourceFilename = `${env.TEMP_DIR}engrave-${hash}.ly`;
|
337 |
+
|
338 |
+
await asyncCall(fs.writeFile, sourceFilename, source);
|
339 |
+
|
340 |
+
const writing = new SingleLock();
|
341 |
+
const fileReady = new SingleLock();
|
342 |
+
|
343 |
+
const loadFile = async line => {
|
344 |
+
const [_, filename] = line.match(FILE_BORN_OUPUT_PATTERN);
|
345 |
+
const [__, ext] = filename.match(/\.(\w+)$/);
|
346 |
+
|
347 |
+
switch (ext) {
|
348 |
+
case "svg": {
|
349 |
+
await fileReady.lock();
|
350 |
+
await writing.wait();
|
351 |
+
|
352 |
+
writing.lock();
|
353 |
+
|
354 |
+
const filePath = path.resolve(env.TEMP_DIR, filename);
|
355 |
+
const fileStream = fs.createReadStream(filePath);
|
356 |
+
fileStream.pipe(output, {end: false});
|
357 |
+
|
358 |
+
fileStream.on("close", () => output.write("\n\n\n\n"));
|
359 |
+
|
360 |
+
writing.release();
|
361 |
+
}
|
362 |
+
|
363 |
+
break;
|
364 |
+
}
|
365 |
+
};
|
366 |
+
|
367 |
+
const checkFile = line => {
|
368 |
+
fileReady.release();
|
369 |
+
|
370 |
+
if (FILE_BORN_OUPUT_PATTERN.test(line))
|
371 |
+
loadFile(line);
|
372 |
+
};
|
373 |
+
|
374 |
+
const includeParameters = includeFolders.map(folder => `--include=${folder}`).join(" ");
|
375 |
+
|
376 |
+
const proc: any = child_process.exec(`${LILYPOND_PATH} -dbackend=svg -o ${env.TEMP_DIR} ${includeParameters} ${sourceFilename}`);
|
377 |
+
proc.childProcess.stdout.on("data", checkFile);
|
378 |
+
proc.childProcess.stderr.on("data", checkFile);
|
379 |
+
|
380 |
+
proc.then(() => fileReady.release());
|
381 |
+
|
382 |
+
return proc;
|
383 |
+
};
|
384 |
+
|
385 |
+
|
386 |
+
const engraveScm = async (source: string, {onProcStart, includeFolders = []}: {
|
387 |
+
onProcStart?: () => void|Promise<void>,
|
388 |
+
includeFolders?: string[], // include folder path should be relative to TEMP_DIR
|
389 |
+
} = {}): Promise<{
|
390 |
+
logs: string,
|
391 |
+
scm: string,
|
392 |
+
}> => {
|
393 |
+
const hash = genHashString();
|
394 |
+
const sourceFilename = `${env.TEMP_DIR}engrave-${hash}.ly`;
|
395 |
+
const targetFilename = `${env.TEMP_DIR}engrave-${hash}.scm`;
|
396 |
+
|
397 |
+
await fs.promises.writeFile(sourceFilename, source);
|
398 |
+
//console.log("ly source written:", sourceFilename);
|
399 |
+
|
400 |
+
const includeParameters = includeFolders.map(folder => `--include=${folder}`).join(" ");
|
401 |
+
|
402 |
+
const proc = child_process.exec(`${LILYPOND_PATH} -dbackend=scm -o ${env.TEMP_DIR} ${includeParameters} ${sourceFilename}`,
|
403 |
+
{maxBuffer: 0x100000});
|
404 |
+
|
405 |
+
await onProcStart && onProcStart();
|
406 |
+
|
407 |
+
const result = await proc;
|
408 |
+
|
409 |
+
const scmBuffer = await fs.promises.readFile(targetFilename);
|
410 |
+
|
411 |
+
return {
|
412 |
+
logs: result.stderr,
|
413 |
+
scm: scmBuffer.toString(),
|
414 |
+
};
|
415 |
+
};
|
416 |
+
|
417 |
+
|
418 |
+
const engraveSvg = async (source: string, options: EngraverOptions = {}): Promise<Partial<EngraverResult>> =>
|
419 |
+
(env.LILYPOND_ADDON ? lilyAddon.engraveSvg : engraveSvgCli)(source, options);
|
420 |
+
|
421 |
+
|
422 |
+
const engraveSvgWithStream = async (source: string, output: Writable, options: {includeFolders?: string[]} = {}) =>
|
423 |
+
(env.LILYPOND_ADDON ? lilyAddon.engraveSvgWithStream : engraveSvgWithStreamCli)(source, output, options);
|
424 |
+
|
425 |
+
|
426 |
+
|
427 |
+
export {
|
428 |
+
xml2ly,
|
429 |
+
midi2ly,
|
430 |
+
engraveSvg,
|
431 |
+
engraveSvgCli,
|
432 |
+
engraveSvgWithStream,
|
433 |
+
engraveSvgWithStreamCli,
|
434 |
+
engraveScm,
|
435 |
+
setEnvironment,
|
436 |
+
emptyCache,
|
437 |
+
postProcessSvg,
|
438 |
+
};
|
backend/loadJisonParserNode.ts
ADDED
@@ -0,0 +1,22 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
|
2 |
+
import fs from "fs";
|
3 |
+
|
4 |
+
import {Parser} from "../inc/jisonWrapper";
|
5 |
+
|
6 |
+
|
7 |
+
|
8 |
+
const parsers = new Map<string, Parser>();
|
9 |
+
|
10 |
+
|
11 |
+
|
12 |
+
export default async function load (jison): Promise<Parser> {
|
13 |
+
if (!parsers.get(jison)) {
|
14 |
+
const grammar = (await fs.promises.readFile(jison)).toString();
|
15 |
+
|
16 |
+
//console.log("grammar:", grammar);
|
17 |
+
|
18 |
+
parsers.set(jison, new Parser(grammar)) ;
|
19 |
+
}
|
20 |
+
|
21 |
+
return parsers.get(jison);
|
22 |
+
};
|
backend/loadLilyParserNode.ts
ADDED
@@ -0,0 +1,8 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
|
2 |
+
import loadJisonParser from "./loadJisonParserNode";
|
3 |
+
|
4 |
+
import {Parser} from "../inc/jisonWrapper";
|
5 |
+
|
6 |
+
|
7 |
+
|
8 |
+
export default (): Promise<Parser> => loadJisonParser("./jison/lilypond.jison");
|
backend/scoreMaker.ts
ADDED
@@ -0,0 +1,359 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
|
2 |
+
import {DOMParser, XMLSerializer} from "xmldom";
|
3 |
+
import {MusicNotation} from "@k-l-lambda/music-widgets";
|
4 |
+
import {MIDI} from "@k-l-lambda/music-widgets";
|
5 |
+
import {Readable} from "stream";
|
6 |
+
import CRC32 from "crc-32";
|
7 |
+
|
8 |
+
import npmPackage from "../package.json";
|
9 |
+
import {xml2ly, engraveSvg, LilyProcessOptions} from "./lilyCommands";
|
10 |
+
import {LilyDocument, LilyTerms, docLocationSet} from "../inc/lilyParser";
|
11 |
+
import * as staffSvg from "../inc/staffSvg";
|
12 |
+
import {SingleLock} from "../inc/mutex";
|
13 |
+
import * as LilyNotation from "../inc/lilyNotation";
|
14 |
+
import {svgToPng} from "./canvas";
|
15 |
+
import LogRecorder from "../inc/logRecorder";
|
16 |
+
import ScoreJSON from "../inc/scoreJSON";
|
17 |
+
import {LilyDocumentAttributeReadOnly} from "../inc/lilyParser/lilyDocument";
|
18 |
+
import {Block} from "../inc/lilyParser/lilyTerms";
|
19 |
+
|
20 |
+
|
21 |
+
|
22 |
+
interface GrammarParser {
|
23 |
+
parse (source: string): any;
|
24 |
+
};
|
25 |
+
|
26 |
+
|
27 |
+
const markupLily = (source: string, markup: string, lilyParser: GrammarParser): string => {
|
28 |
+
const docMarkup = new LilyDocument(lilyParser.parse(markup));
|
29 |
+
const docSource = new LilyDocument(lilyParser.parse(source));
|
30 |
+
|
31 |
+
/*// copy attributes
|
32 |
+
const attrS = docSource.globalAttributes() as LilyDocumentAttribute;
|
33 |
+
const attrM = docMarkup.globalAttributes({readonly: true}) as LilyDocumentAttributeReadOnly;
|
34 |
+
|
35 |
+
[
|
36 |
+
"staffSize", "paperWidth", "paperHeight",
|
37 |
+
"topMargin", "bottomMargin", "leftMargin", "rightMargin",
|
38 |
+
"systemSpacing", "topMarkupSpacing", "raggedLast", "raggedBottom", "raggedLastBottom",
|
39 |
+
"printPageNumber",
|
40 |
+
].forEach(field => {
|
41 |
+
if (attrM[field] !== undefined) {
|
42 |
+
if (typeof attrS[field].value === "object" && attrS[field].value && (attrS[field].value as any).set)
|
43 |
+
(attrS[field].value as any).set(attrM[field]);
|
44 |
+
else
|
45 |
+
attrS[field].value = attrM[field];
|
46 |
+
}
|
47 |
+
});
|
48 |
+
|
49 |
+
// execute commands list
|
50 |
+
const commands = docMarkup.root.getField("LotusCommands");
|
51 |
+
const cmdList = commands && commands.value && commands.value.args && commands.value.args[0].body;
|
52 |
+
if (cmdList && Array.isArray(cmdList)) {
|
53 |
+
for (const command of cmdList) {
|
54 |
+
if (command.exp && docSource[command.exp])
|
55 |
+
docSource[command.exp]();
|
56 |
+
else
|
57 |
+
console.warn("unexpected markup command:", command);
|
58 |
+
}
|
59 |
+
}
|
60 |
+
|
61 |
+
// copy LotusOption assignments
|
62 |
+
const assignments = docMarkup.root.entries.filter(term => term instanceof LilyTerms.Assignment && /^LotusOption\..+/.test(term.key.toString()));
|
63 |
+
assignments.forEach(assignment => docSource.root.sections.push(assignment.clone()));*/
|
64 |
+
docSource.markup(docMarkup);
|
65 |
+
|
66 |
+
return docSource.toString();
|
67 |
+
};
|
68 |
+
|
69 |
+
|
70 |
+
const xmlBufferToLy = async (xml: Buffer, options: LilyProcessOptions = {}): Promise<string> => {
|
71 |
+
const bom = (xml[0] << 8 | xml[1]);
|
72 |
+
const utf16 = bom === 0xfffe;
|
73 |
+
const content = xml.toString(utf16 ? "utf16le" : "utf8");
|
74 |
+
|
75 |
+
return await xml2ly(content, {replaceEncoding: utf16, ...options});
|
76 |
+
};
|
77 |
+
|
78 |
+
|
79 |
+
const unescapeStringExp = exp => exp && exp.toString();
|
80 |
+
|
81 |
+
|
82 |
+
interface SheetNotationResult extends Partial<ScoreJSON> {
|
83 |
+
midi: MIDI.MidiData;
|
84 |
+
midiNotation: MusicNotation.NotationData;
|
85 |
+
//sheetNotation: staffSvg.StaffNotation.SheetNotation;
|
86 |
+
lilyDocument: LilyDocument;
|
87 |
+
bakingImages?: Readable[];
|
88 |
+
};
|
89 |
+
|
90 |
+
|
91 |
+
const makeSheetNotation = async (source: string, lilyParser: GrammarParser, {withNotation = false, logger, lilyDocument, includeFolders, baking}: {
|
92 |
+
withNotation?: boolean,
|
93 |
+
logger?: LogRecorder,
|
94 |
+
lilyDocument?: LilyDocument,
|
95 |
+
includeFolders?: string[],
|
96 |
+
baking?: boolean,
|
97 |
+
} = {}): Promise<SheetNotationResult> => {
|
98 |
+
let midi = null;
|
99 |
+
let midiNotation = null;
|
100 |
+
|
101 |
+
const pages = [];
|
102 |
+
const hashTable = {};
|
103 |
+
const bakingImages = [];
|
104 |
+
|
105 |
+
const t0 = Date.now();
|
106 |
+
|
107 |
+
type ParserArguments = {
|
108 |
+
attributes: LilyDocumentAttributeReadOnly,
|
109 |
+
tieLocations: Set<string>,
|
110 |
+
briefChordLocations: Set<string>,
|
111 |
+
lyricLocations: Set<string>,
|
112 |
+
};
|
113 |
+
|
114 |
+
const argsGen = new SingleLock<ParserArguments>(true);
|
115 |
+
|
116 |
+
const engraving = await engraveSvg(source, {
|
117 |
+
includeFolders,
|
118 |
+
// do some work during lilypond process running to save time
|
119 |
+
onProcStart: () => {
|
120 |
+
//console.log("tp.0:", Date.now() - t0);
|
121 |
+
if (!lilyDocument) {
|
122 |
+
lilyDocument = new LilyDocument(lilyParser.parse(source));
|
123 |
+
lilyDocument.interpret();
|
124 |
+
}
|
125 |
+
|
126 |
+
const attributes = lilyDocument.globalAttributes({readonly: true}) as LilyDocumentAttributeReadOnly;
|
127 |
+
|
128 |
+
const tieLocations = docLocationSet(lilyDocument.getTiedNoteLocations2());
|
129 |
+
const briefChordLocations = docLocationSet(lilyDocument.getBriefChordLocations());
|
130 |
+
const lyricLocations = docLocationSet(lilyDocument.getLyricLocations());
|
131 |
+
|
132 |
+
argsGen.release({attributes, tieLocations, briefChordLocations, lyricLocations});
|
133 |
+
//console.log("tp.1:", Date.now() - t0);
|
134 |
+
},
|
135 |
+
onMidiRead: midi_ => {
|
136 |
+
//console.log("tm.0:", Date.now() - t0);
|
137 |
+
midi = midi_;
|
138 |
+
if (withNotation)
|
139 |
+
midiNotation = midi && MusicNotation.Notation.parseMidi(midi);
|
140 |
+
//console.log("tm.1:", Date.now() - t0);
|
141 |
+
},
|
142 |
+
onSvgRead: async (index, svg) => {
|
143 |
+
//console.log("ts.0:", Date.now() - t0);
|
144 |
+
const args = await argsGen.wait();
|
145 |
+
const page = staffSvg.parseSvgPage(svg, source, {DOMParser, logger, ...args});
|
146 |
+
pages[index] = page.structure;
|
147 |
+
Object.assign(hashTable, page.hashTable);
|
148 |
+
//console.log("ts.1:", Date.now() - t0);
|
149 |
+
},
|
150 |
+
});
|
151 |
+
|
152 |
+
logger.append("scoreMaker.profile.engraving", {cost: Date.now() - t0});
|
153 |
+
logger.append("lilypond.log", engraving.logs);
|
154 |
+
|
155 |
+
const doc = new staffSvg.SheetDocument({pages});
|
156 |
+
|
157 |
+
staffSvg.postProcessSheetDocument(doc, lilyDocument);
|
158 |
+
|
159 |
+
if (baking) {
|
160 |
+
await Promise.all(engraving.svgs.map(async (svg, index) => {
|
161 |
+
const svgText = staffSvg.turnRawSvgWithSheetDocument(svg, pages[index], {DOMParser, XMLSerializer});
|
162 |
+
bakingImages[index] = await svgToPng(Buffer.from(svgText));
|
163 |
+
}));
|
164 |
+
}
|
165 |
+
|
166 |
+
const midiMusic = lilyDocument.interpret().midiMusic;
|
167 |
+
|
168 |
+
const {attributes} = await argsGen.wait();
|
169 |
+
const meta = {
|
170 |
+
title: unescapeStringExp(attributes.title),
|
171 |
+
composer: unescapeStringExp(attributes.composer),
|
172 |
+
pageSize: doc.pageSize,
|
173 |
+
pageCount: doc.pages.length,
|
174 |
+
staffSize: attributes.staffSize as number,
|
175 |
+
trackInfos: midiMusic && midiMusic.trackContextDicts,
|
176 |
+
};
|
177 |
+
|
178 |
+
/*const t00 = Date.now();
|
179 |
+
const sheetNotation = staffSvg.StaffNotation.parseNotationFromSheetDocument(doc, {logger});
|
180 |
+
|
181 |
+
// correct notation time by location-tick table from lily document
|
182 |
+
const tickTable = lilyDocument.getLocationTickTable();
|
183 |
+
staffSvg.StaffNotation.assignTickByLocationTable(sheetNotation, tickTable);
|
184 |
+
console.log("parseNotationFromSheetDocument cost:", Date.now() - t00);*/
|
185 |
+
|
186 |
+
return {
|
187 |
+
midi,
|
188 |
+
bakingImages: baking ? bakingImages : null,
|
189 |
+
midiNotation,
|
190 |
+
//sheetNotation,
|
191 |
+
meta,
|
192 |
+
doc,
|
193 |
+
hashTable,
|
194 |
+
lilyDocument,
|
195 |
+
};
|
196 |
+
};
|
197 |
+
|
198 |
+
|
199 |
+
interface MakerOptions {
|
200 |
+
midi: MIDI.MidiData;
|
201 |
+
logger: LogRecorder;
|
202 |
+
includeFolders: string[];
|
203 |
+
baking: boolean;
|
204 |
+
ignoreNotation: boolean;
|
205 |
+
};
|
206 |
+
|
207 |
+
|
208 |
+
interface MakerResult {
|
209 |
+
bakingImages?: Readable[],
|
210 |
+
score: Partial<ScoreJSON>,
|
211 |
+
};
|
212 |
+
|
213 |
+
|
214 |
+
// non-negative crc-32
|
215 |
+
const hashString = (str: string): number => {
|
216 |
+
const value = CRC32.str(str);
|
217 |
+
return value < 0 ? 0x100000000 + value : value;
|
218 |
+
};
|
219 |
+
|
220 |
+
|
221 |
+
const makeScore = async (
|
222 |
+
source: string,
|
223 |
+
lilyParser: GrammarParser,
|
224 |
+
{midi, logger, baking = false, includeFolders, ignoreNotation = false}: Partial<MakerOptions> = {},
|
225 |
+
): Promise<MakerResult> => {
|
226 |
+
const t0 = Date.now();
|
227 |
+
|
228 |
+
const hash = hashString(source);
|
229 |
+
|
230 |
+
const foldData = await makeSheetNotation(source, lilyParser, {logger, includeFolders, baking});
|
231 |
+
const {meta, doc, hashTable, bakingImages, lilyDocument} = foldData;
|
232 |
+
|
233 |
+
midi = midi || foldData.midi;
|
234 |
+
const lilyNotation = !ignoreNotation && lilyDocument.interpret().getNotation();
|
235 |
+
|
236 |
+
if (ignoreNotation || !midi || !lilyNotation) {
|
237 |
+
if (!ignoreNotation) {
|
238 |
+
if (!midi)
|
239 |
+
console.warn("Neither lilypond or external arguments did not offer MIDI data, score maker finished incompletely.");
|
240 |
+
|
241 |
+
if (!lilyNotation)
|
242 |
+
console.warn("lilyNotation parsing failed, score maker finished incompletely.");
|
243 |
+
}
|
244 |
+
|
245 |
+
return {
|
246 |
+
score: {
|
247 |
+
version: npmPackage.version,
|
248 |
+
hash,
|
249 |
+
meta,
|
250 |
+
doc,
|
251 |
+
hashTable,
|
252 |
+
},
|
253 |
+
};
|
254 |
+
}
|
255 |
+
|
256 |
+
const t5 = Date.now();
|
257 |
+
|
258 |
+
const matcher = await LilyNotation.matchWithExactMIDI(lilyNotation, midi);
|
259 |
+
|
260 |
+
logger.append("scoreMaker.profile.matching", {cost: Date.now() - t5});
|
261 |
+
|
262 |
+
if (logger && logger.enabled) {
|
263 |
+
const cis = new Set(Array(matcher.criterion.notes.length).keys());
|
264 |
+
matcher.path.forEach(ci => cis.delete(ci));
|
265 |
+
|
266 |
+
const omitC = cis.size;
|
267 |
+
const omitS = matcher.path.filter(ci => ci < 0).length;
|
268 |
+
|
269 |
+
const coverage = ((matcher.criterion.notes.length - omitC) / matcher.criterion.notes.length)
|
270 |
+
* ((matcher.sample.notes.length - omitS) / matcher.sample.notes.length);
|
271 |
+
|
272 |
+
logger.append("makeScore.match", {coverage, omitC, omitS, path: matcher.path});
|
273 |
+
}
|
274 |
+
|
275 |
+
doc.updateMatchedTokens(lilyNotation.idSet);
|
276 |
+
|
277 |
+
// idTrackMap is useless in bundled score
|
278 |
+
delete lilyNotation.idTrackMap;
|
279 |
+
|
280 |
+
if (baking)
|
281 |
+
doc.pruneForBakingMode();
|
282 |
+
|
283 |
+
logger.append("scoreMaker.profile.full", {cost: Date.now() - t0});
|
284 |
+
|
285 |
+
return {
|
286 |
+
bakingImages,
|
287 |
+
score: {
|
288 |
+
version: npmPackage.version,
|
289 |
+
hash,
|
290 |
+
meta,
|
291 |
+
doc,
|
292 |
+
hashTable: !baking ? hashTable : null,
|
293 |
+
lilyNotation,
|
294 |
+
},
|
295 |
+
};
|
296 |
+
};
|
297 |
+
|
298 |
+
|
299 |
+
const makeMIDI = async (source: string, lilyParser: GrammarParser, {unfoldRepeats = false, fixNestedRepeat = false, includeFolders = undefined} = {}): Promise<MIDI.MidiData> => {
|
300 |
+
const lilyDocument = new LilyDocument(lilyParser.parse(source));
|
301 |
+
|
302 |
+
if (fixNestedRepeat)
|
303 |
+
lilyDocument.fixNestedRepeat();
|
304 |
+
|
305 |
+
if (unfoldRepeats)
|
306 |
+
lilyDocument.unfoldRepeats();
|
307 |
+
|
308 |
+
const score = lilyDocument.root.getBlock("score");
|
309 |
+
if (score) {
|
310 |
+
// remove layout block to save time
|
311 |
+
score.body = score.body.filter(term => !(term instanceof LilyTerms.Block && term.head === "\\layout"));
|
312 |
+
|
313 |
+
// remove invalid tempo
|
314 |
+
const midi: any = score.body.find(term => term instanceof LilyTerms.Block && term.head === "\\midi");
|
315 |
+
if (midi)
|
316 |
+
midi.body = midi.body.filter(term => !(term instanceof LilyTerms.Tempo && term.beatsPerMinute > 200));
|
317 |
+
}
|
318 |
+
|
319 |
+
const markupSource = lilyDocument.toString();
|
320 |
+
//console.log("markupSource:", markupSource);
|
321 |
+
|
322 |
+
return new Promise((resolve, reject) => engraveSvg(markupSource, {
|
323 |
+
includeFolders,
|
324 |
+
onMidiRead: resolve,
|
325 |
+
}).catch(reject));
|
326 |
+
};
|
327 |
+
|
328 |
+
|
329 |
+
const makeArticulatedMIDI = async (source: string, lilyParser: GrammarParser, {ignoreRepeats = true, includeFolders = undefined} = {}): Promise<MIDI.MidiData> => {
|
330 |
+
const lilyDocument = new LilyDocument(lilyParser.parse(source));
|
331 |
+
|
332 |
+
if (ignoreRepeats)
|
333 |
+
lilyDocument.removeRepeats();
|
334 |
+
|
335 |
+
lilyDocument.articulateMIDIOutput();
|
336 |
+
|
337 |
+
// remove layout block to save time
|
338 |
+
lilyDocument.root.sections = lilyDocument.root.sections.filter(section => !(section instanceof Block)
|
339 |
+
|| !(section.head === "\\score")
|
340 |
+
|| section.isMIDIDedicated);
|
341 |
+
|
342 |
+
const markupSource = lilyDocument.toString();
|
343 |
+
//console.log("markupSource:", markupSource);
|
344 |
+
|
345 |
+
return new Promise((resolve, reject) => engraveSvg(markupSource, {
|
346 |
+
includeFolders,
|
347 |
+
onMidiRead: resolve,
|
348 |
+
}).catch(reject));
|
349 |
+
};
|
350 |
+
|
351 |
+
|
352 |
+
|
353 |
+
export {
|
354 |
+
markupLily,
|
355 |
+
xmlBufferToLy,
|
356 |
+
makeScore,
|
357 |
+
makeMIDI,
|
358 |
+
makeArticulatedMIDI,
|
359 |
+
};
|
backend/scoreMaker.ts.old
ADDED
@@ -0,0 +1,665 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
//
|
2 |
+
// OBSOLETE code, only for archive
|
3 |
+
//
|
4 |
+
|
5 |
+
import _ from "lodash";
|
6 |
+
import {DOMParser, XMLSerializer} from "xmldom";
|
7 |
+
import {MusicNotation} from "@k-l-lambda/music-widgets";
|
8 |
+
import {MIDI} from "@k-l-lambda/music-widgets";
|
9 |
+
import {Readable} from "stream";
|
10 |
+
|
11 |
+
import npmPackage from "../package.json";
|
12 |
+
import {xml2ly, engraveSvg, LilyProcessOptions} from "./lilyCommands";
|
13 |
+
import {LilyDocument, replaceSourceToken, LilyTerms} from "../inc/lilyParser";
|
14 |
+
import * as staffSvg from "../inc/staffSvg";
|
15 |
+
import {SingleLock} from "../inc/mutex";
|
16 |
+
import {PitchContextTable} from "../inc/pitchContext";
|
17 |
+
import * as LilyNotation from "../inc/lilyNotation";
|
18 |
+
import {svgToPng} from "./canvas";
|
19 |
+
import LogRecorder from "../inc/logRecorder";
|
20 |
+
import ScoreJSON, {NoteLinking} from "../inc/scoreJSON";
|
21 |
+
import {LilyDocumentAttribute, LilyDocumentAttributeReadOnly} from "../inc/lilyParser/lilyDocument";
|
22 |
+
import {Block} from "../inc/lilyParser/lilyTerms";
|
23 |
+
|
24 |
+
|
25 |
+
|
26 |
+
interface GrammarParser {
|
27 |
+
parse (source: string): any;
|
28 |
+
};
|
29 |
+
|
30 |
+
|
31 |
+
const markupLily = (source: string, markup: string, lilyParser: GrammarParser): string => {
|
32 |
+
const docMarkup = new LilyDocument(lilyParser.parse(markup));
|
33 |
+
const docSource = new LilyDocument(lilyParser.parse(source));
|
34 |
+
|
35 |
+
// copy attributes
|
36 |
+
const attrS = docSource.globalAttributes() as LilyDocumentAttribute;
|
37 |
+
const attrM = docMarkup.globalAttributes({readonly: true}) as LilyDocumentAttributeReadOnly;
|
38 |
+
|
39 |
+
[
|
40 |
+
"staffSize", "paperWidth", "paperHeight",
|
41 |
+
"topMargin", "bottomMargin", "leftMargin", "rightMargin",
|
42 |
+
"systemSpacing", "topMarkupSpacing", "raggedLast", "raggedBottom", "raggedLastBottom",
|
43 |
+
"printPageNumber",
|
44 |
+
].forEach(field => {
|
45 |
+
if (attrM[field] !== undefined) {
|
46 |
+
if (typeof attrS[field].value === "object" && attrS[field].value && (attrS[field].value as any).set)
|
47 |
+
(attrS[field].value as any).set(attrM[field]);
|
48 |
+
else
|
49 |
+
attrS[field].value = attrM[field];
|
50 |
+
}
|
51 |
+
});
|
52 |
+
|
53 |
+
// execute commands list
|
54 |
+
const commands = docMarkup.root.getField("LotusCommands");
|
55 |
+
const cmdList = commands && commands.value && commands.value.args && commands.value.args[0].body;
|
56 |
+
if (cmdList && Array.isArray(cmdList)) {
|
57 |
+
for (const command of cmdList) {
|
58 |
+
if (command.exp && docSource[command.exp])
|
59 |
+
docSource[command.exp]();
|
60 |
+
else
|
61 |
+
console.warn("unexpected markup command:", command);
|
62 |
+
}
|
63 |
+
}
|
64 |
+
|
65 |
+
// copy LotusOption assignments
|
66 |
+
const assignments = docMarkup.root.entries.filter(term => term instanceof LilyTerms.Assignment && /^LotusOption\..+/.test(term.key.toString()));
|
67 |
+
assignments.forEach(assignment => docSource.root.sections.push(assignment.clone()));
|
68 |
+
|
69 |
+
return docSource.toString();
|
70 |
+
};
|
71 |
+
|
72 |
+
|
73 |
+
const xmlBufferToLy = async (xml: Buffer, options: LilyProcessOptions = {}): Promise<string> => {
|
74 |
+
const bom = (xml[0] << 8 | xml[1]);
|
75 |
+
const utf16 = bom === 0xfffe;
|
76 |
+
const content = xml.toString(utf16 ? "utf16le" : "utf8");
|
77 |
+
|
78 |
+
return await xml2ly(content, {replaceEncoding: utf16, ...options});
|
79 |
+
};
|
80 |
+
|
81 |
+
|
82 |
+
const unescapeStringExp = exp => exp && exp.toString();
|
83 |
+
|
84 |
+
|
85 |
+
const makeScoreV1 = async (source: string, lilyParser: GrammarParser, {midi, logger}: {midi?: MIDI.MidiData, logger?: LogRecorder} = {}): Promise<ScoreJSON> => {
|
86 |
+
const t0 = Date.now();
|
87 |
+
|
88 |
+
const engraving = await engraveSvg(source);
|
89 |
+
|
90 |
+
logger.append("scoreMaker.profile.engraving", {cost: Date.now() - t0});
|
91 |
+
logger.append("lilypond.log", engraving.logs);
|
92 |
+
|
93 |
+
const lilyDocument = new LilyDocument(lilyParser.parse(source));
|
94 |
+
const {doc, hashTable} = staffSvg.createSheetDocumentFromSvgs(engraving.svgs, source, lilyDocument, {logger, DOMParser});
|
95 |
+
|
96 |
+
const sheetNotation = staffSvg.StaffNotation.parseNotationFromSheetDocument(doc, {logger});
|
97 |
+
|
98 |
+
const attributes = lilyDocument.globalAttributes({readonly: true});
|
99 |
+
|
100 |
+
const meta = {
|
101 |
+
title: unescapeStringExp(attributes.title),
|
102 |
+
composer: unescapeStringExp(attributes.composer),
|
103 |
+
pageSize: doc.pageSize,
|
104 |
+
pageCount: doc.pages.length,
|
105 |
+
staffSize: attributes.staffSize as number,
|
106 |
+
};
|
107 |
+
|
108 |
+
midi = midi || engraving.midi;
|
109 |
+
const midiNotation = MusicNotation.Notation.parseMidi(midi);
|
110 |
+
|
111 |
+
const t5 = Date.now();
|
112 |
+
|
113 |
+
const matcher = await staffSvg.StaffNotation.matchNotations(midiNotation, sheetNotation);
|
114 |
+
|
115 |
+
logger.append("scoreMaker.profile.matching", {cost: Date.now() - t5});
|
116 |
+
|
117 |
+
if (logger) {
|
118 |
+
const cis = new Set(Array(matcher.criterion.notes.length).keys());
|
119 |
+
matcher.path.forEach(ci => cis.delete(ci));
|
120 |
+
|
121 |
+
const omitC = cis.size;
|
122 |
+
const omitS = matcher.path.filter(ci => ci < 0).length;
|
123 |
+
|
124 |
+
const coverage = ((matcher.criterion.notes.length - omitC) / matcher.criterion.notes.length)
|
125 |
+
* ((matcher.sample.notes.length - omitS) / matcher.sample.notes.length);
|
126 |
+
|
127 |
+
logger.append("makeScore.match", {coverage, omitC, omitS, path: matcher.path});
|
128 |
+
}
|
129 |
+
|
130 |
+
const matchedIds: Set<string> = new Set();
|
131 |
+
midiNotation.notes.forEach(note => note.ids && note.ids.forEach(id => matchedIds.add(id)));
|
132 |
+
|
133 |
+
doc.updateMatchedTokens(matchedIds);
|
134 |
+
|
135 |
+
const pitchContextGroup = PitchContextTable.createPitchContextGroup(sheetNotation.pitchContexts, midiNotation);
|
136 |
+
|
137 |
+
const noteLinkings = midiNotation.notes.map(note => _.pick(note, ["ids", "staffTrack", "contextIndex"]) as NoteLinking);
|
138 |
+
|
139 |
+
logger.append("scoreMaker.profile.full", {cost: Date.now() - t0});
|
140 |
+
|
141 |
+
return {
|
142 |
+
meta,
|
143 |
+
doc,
|
144 |
+
midi,
|
145 |
+
hashTable,
|
146 |
+
noteLinkings,
|
147 |
+
pitchContextGroup,
|
148 |
+
};
|
149 |
+
};
|
150 |
+
|
151 |
+
|
152 |
+
interface IncompleteScoreJSON {
|
153 |
+
meta?: any,
|
154 |
+
doc?: any;
|
155 |
+
hashTable?: {[key: string]: any};
|
156 |
+
midi?: MIDI.MidiData;
|
157 |
+
noteLinkings?: NoteLinking;
|
158 |
+
pitchContextGroup?: any;
|
159 |
+
};
|
160 |
+
|
161 |
+
|
162 |
+
interface SheetNotationResult extends IncompleteScoreJSON {
|
163 |
+
midiNotation: MusicNotation.NotationData;
|
164 |
+
sheetNotation: staffSvg.StaffNotation.SheetNotation;
|
165 |
+
lilyDocument: LilyDocument;
|
166 |
+
bakingImages?: Readable[];
|
167 |
+
};
|
168 |
+
|
169 |
+
|
170 |
+
const makeScoreV2 = async (source: string, lilyParser: GrammarParser, {midi, logger}: {midi?: MIDI.MidiData, logger?: LogRecorder} = {}): Promise<ScoreJSON | IncompleteScoreJSON> => {
|
171 |
+
let midiNotation = null;
|
172 |
+
|
173 |
+
const pages = [];
|
174 |
+
const hashTable = {};
|
175 |
+
|
176 |
+
const t0 = Date.now();
|
177 |
+
|
178 |
+
const attrGen = new SingleLock<LilyDocumentAttributeReadOnly>(true);
|
179 |
+
|
180 |
+
const engraving = await engraveSvg(source, {
|
181 |
+
// do some work during lilypond process running to save time
|
182 |
+
onProcStart: () => {
|
183 |
+
//console.log("tp.0:", Date.now() - t0);
|
184 |
+
const lilyDocument = new LilyDocument(lilyParser.parse(source));
|
185 |
+
attrGen.release(lilyDocument.globalAttributes({readonly: true}) as LilyDocumentAttributeReadOnly);
|
186 |
+
//console.log("tp.1:", Date.now() - t0);
|
187 |
+
},
|
188 |
+
onMidiRead: midi_ => {
|
189 |
+
//console.log("tm.0:", Date.now() - t0);
|
190 |
+
if (!midi) {
|
191 |
+
midi = midi_;
|
192 |
+
midiNotation = midi && MusicNotation.Notation.parseMidi(midi);
|
193 |
+
}
|
194 |
+
//console.log("tm.1:", Date.now() - t0);
|
195 |
+
},
|
196 |
+
onSvgRead: async svg => {
|
197 |
+
//console.log("ts.0:", Date.now() - t0);
|
198 |
+
const attributes = await attrGen.wait();
|
199 |
+
const page = staffSvg.parseSvgPage(svg, source, {DOMParser, logger, attributes});
|
200 |
+
pages.push(page.structure);
|
201 |
+
Object.assign(hashTable, page.hashTable);
|
202 |
+
//console.log("ts.1:", Date.now() - t0);
|
203 |
+
},
|
204 |
+
});
|
205 |
+
|
206 |
+
//console.log("t2:", Date.now() - t0);
|
207 |
+
|
208 |
+
logger.append("scoreMaker.profile.engraving", {cost: Date.now() - t0});
|
209 |
+
logger.append("lilypond.log", engraving.logs);
|
210 |
+
|
211 |
+
const doc = new staffSvg.SheetDocument({pages});
|
212 |
+
|
213 |
+
//console.log("t3:", Date.now() - t0);
|
214 |
+
|
215 |
+
const attributes = await attrGen.wait();
|
216 |
+
const meta = {
|
217 |
+
title: unescapeStringExp(attributes.title),
|
218 |
+
composer: unescapeStringExp(attributes.composer),
|
219 |
+
pageSize: doc.pageSize,
|
220 |
+
pageCount: doc.pages.length,
|
221 |
+
staffSize: attributes.staffSize,
|
222 |
+
};
|
223 |
+
|
224 |
+
if (!midiNotation) {
|
225 |
+
console.warn("Neither lilypond or external arguments did not offer MIDI data, score maker finish incompletely.");
|
226 |
+
return {
|
227 |
+
meta,
|
228 |
+
doc,
|
229 |
+
midi,
|
230 |
+
hashTable,
|
231 |
+
};
|
232 |
+
}
|
233 |
+
|
234 |
+
//console.log("t4:", Date.now() - t0);
|
235 |
+
|
236 |
+
const sheetNotation = staffSvg.StaffNotation.parseNotationFromSheetDocument(doc, {logger});
|
237 |
+
|
238 |
+
//console.log("t5:", Date.now() - t0);
|
239 |
+
const t5 = Date.now();
|
240 |
+
|
241 |
+
const matcher = await staffSvg.StaffNotation.matchNotations(midiNotation, sheetNotation);
|
242 |
+
|
243 |
+
//console.log("t6:", Date.now() - t0);
|
244 |
+
logger.append("scoreMaker.profile.matching", {cost: Date.now() - t5});
|
245 |
+
|
246 |
+
if (logger) {
|
247 |
+
const cis = new Set(Array(matcher.criterion.notes.length).keys());
|
248 |
+
matcher.path.forEach(ci => cis.delete(ci));
|
249 |
+
|
250 |
+
const omitC = cis.size;
|
251 |
+
const omitS = matcher.path.filter(ci => ci < 0).length;
|
252 |
+
|
253 |
+
const coverage = ((matcher.criterion.notes.length - omitC) / matcher.criterion.notes.length)
|
254 |
+
* ((matcher.sample.notes.length - omitS) / matcher.sample.notes.length);
|
255 |
+
|
256 |
+
logger.append("markScore.match", {coverage, omitC, omitS, path: matcher.path});
|
257 |
+
}
|
258 |
+
|
259 |
+
//console.log("t7:", Date.now() - t0);
|
260 |
+
|
261 |
+
const matchedIds: Set<string> = new Set();
|
262 |
+
midiNotation.notes.forEach(note => note.ids && note.ids.forEach(id => matchedIds.add(id)));
|
263 |
+
|
264 |
+
//console.log("t8:", Date.now() - t0);
|
265 |
+
|
266 |
+
doc.updateMatchedTokens(matchedIds);
|
267 |
+
|
268 |
+
//console.log("t9:", Date.now() - t0);
|
269 |
+
|
270 |
+
const pitchContextGroup = PitchContextTable.createPitchContextGroup(sheetNotation.pitchContexts, midiNotation);
|
271 |
+
|
272 |
+
//console.log("t10:", Date.now() - t0);
|
273 |
+
|
274 |
+
const noteLinkings = midiNotation.notes.map(note => _.pick(note, ["ids", "staffTrack", "contextIndex"]));
|
275 |
+
|
276 |
+
//console.log("t11:", Date.now() - t0);
|
277 |
+
logger.append("scoreMaker.profile.full", {cost: Date.now() - t0});
|
278 |
+
|
279 |
+
return {
|
280 |
+
meta,
|
281 |
+
doc,
|
282 |
+
midi,
|
283 |
+
hashTable,
|
284 |
+
noteLinkings,
|
285 |
+
pitchContextGroup,
|
286 |
+
};
|
287 |
+
};
|
288 |
+
|
289 |
+
|
290 |
+
const makeSheetNotation = async (source: string, lilyParser: GrammarParser, {withNotation = true, logger, lilyDocument, includeFolders, baking}: {
|
291 |
+
withNotation?: boolean,
|
292 |
+
logger?: LogRecorder,
|
293 |
+
lilyDocument?: LilyDocument,
|
294 |
+
includeFolders?: string[],
|
295 |
+
baking?: boolean,
|
296 |
+
} = {}): Promise<SheetNotationResult> => {
|
297 |
+
let midi = null;
|
298 |
+
let midiNotation = null;
|
299 |
+
|
300 |
+
const pages = [];
|
301 |
+
const hashTable = {};
|
302 |
+
const bakingImages = [];
|
303 |
+
|
304 |
+
const t0 = Date.now();
|
305 |
+
|
306 |
+
type ParserArguments = {attributes: LilyDocumentAttributeReadOnly, tieLocations: {[key: string]: boolean}};
|
307 |
+
|
308 |
+
const argsGen = new SingleLock<ParserArguments>(true);
|
309 |
+
|
310 |
+
const engraving = await engraveSvg(source, {
|
311 |
+
includeFolders,
|
312 |
+
// do some work during lilypond process running to save time
|
313 |
+
onProcStart: () => {
|
314 |
+
//console.log("tp.0:", Date.now() - t0);
|
315 |
+
if (!lilyDocument) {
|
316 |
+
lilyDocument = new LilyDocument(lilyParser.parse(source));
|
317 |
+
lilyDocument.interpret();
|
318 |
+
}
|
319 |
+
|
320 |
+
const attributes = lilyDocument.globalAttributes({readonly: true}) as LilyDocumentAttributeReadOnly;
|
321 |
+
|
322 |
+
//const tieLocations = lilyDocument.getTiedNoteLocations(text)
|
323 |
+
const tieLocations = lilyDocument.getTiedNoteLocations2()
|
324 |
+
.reduce((table, loc) => ((table[`${loc[0]}:${loc[1]}`] = true), table), {});
|
325 |
+
|
326 |
+
argsGen.release({attributes, tieLocations});
|
327 |
+
//console.log("tp.1:", Date.now() - t0);
|
328 |
+
},
|
329 |
+
onMidiRead: withNotation && (midi_ => {
|
330 |
+
//console.log("tm.0:", Date.now() - t0);
|
331 |
+
midi = midi_;
|
332 |
+
midiNotation = midi && MusicNotation.Notation.parseMidi(midi);
|
333 |
+
//console.log("tm.1:", Date.now() - t0);
|
334 |
+
}),
|
335 |
+
onSvgRead: async (index, svg) => {
|
336 |
+
//console.log("ts.0:", Date.now() - t0);
|
337 |
+
const args = await argsGen.wait();
|
338 |
+
const page = staffSvg.parseSvgPage(svg, source, {DOMParser, logger, ...args});
|
339 |
+
pages[index] = page.structure;
|
340 |
+
Object.assign(hashTable, page.hashTable);
|
341 |
+
//console.log("ts.1:", Date.now() - t0);
|
342 |
+
},
|
343 |
+
});
|
344 |
+
|
345 |
+
logger.append("scoreMaker.profile.engraving", {cost: Date.now() - t0});
|
346 |
+
logger.append("lilypond.log", engraving.logs);
|
347 |
+
|
348 |
+
const doc = new staffSvg.SheetDocument({pages});
|
349 |
+
|
350 |
+
staffSvg.postProcessSheetDocument(doc, lilyDocument);
|
351 |
+
|
352 |
+
if (baking) {
|
353 |
+
await Promise.all(engraving.svgs.map(async (svg, index) => {
|
354 |
+
const svgText = staffSvg.turnRawSvgWithSheetDocument(svg, pages[index], {DOMParser, XMLSerializer});
|
355 |
+
bakingImages[index] = await svgToPng(Buffer.from(svgText));
|
356 |
+
}));
|
357 |
+
}
|
358 |
+
|
359 |
+
const {attributes} = await argsGen.wait();
|
360 |
+
const meta = {
|
361 |
+
title: unescapeStringExp(attributes.title),
|
362 |
+
composer: unescapeStringExp(attributes.composer),
|
363 |
+
pageSize: doc.pageSize,
|
364 |
+
pageCount: doc.pages.length,
|
365 |
+
staffSize: attributes.staffSize,
|
366 |
+
};
|
367 |
+
|
368 |
+
const sheetNotation = staffSvg.StaffNotation.parseNotationFromSheetDocument(doc, {logger});
|
369 |
+
|
370 |
+
// correct notation time by location-tick table from lily document
|
371 |
+
const tickTable = lilyDocument.getLocationTickTable();
|
372 |
+
staffSvg.StaffNotation.assignTickByLocationTable(sheetNotation, tickTable);
|
373 |
+
|
374 |
+
return {
|
375 |
+
midi,
|
376 |
+
bakingImages: baking ? bakingImages : null,
|
377 |
+
midiNotation,
|
378 |
+
sheetNotation,
|
379 |
+
meta,
|
380 |
+
doc,
|
381 |
+
hashTable,
|
382 |
+
lilyDocument,
|
383 |
+
};
|
384 |
+
};
|
385 |
+
|
386 |
+
|
387 |
+
const makeScoreV3 = async (source: string, lilyParser: GrammarParser, {midi, logger, unfoldRepeats = false, includeFolders}: {
|
388 |
+
midi?: MIDI.MidiData,
|
389 |
+
logger?: LogRecorder,
|
390 |
+
unfoldRepeats?: boolean,
|
391 |
+
includeFolders?: string[],
|
392 |
+
} = {}): Promise<ScoreJSON | IncompleteScoreJSON> => {
|
393 |
+
const t0 = Date.now();
|
394 |
+
|
395 |
+
let lilyDocument = null;
|
396 |
+
let unfoldSource = null;
|
397 |
+
|
398 |
+
if (unfoldRepeats) {
|
399 |
+
lilyDocument = new LilyDocument(lilyParser.parse(source));
|
400 |
+
lilyDocument.interpret();
|
401 |
+
if (lilyDocument.containsRepeat()) {
|
402 |
+
lilyDocument.unfoldRepeats();
|
403 |
+
unfoldSource = lilyDocument.toString();
|
404 |
+
|
405 |
+
// keep 2 version lilypond source note href uniform
|
406 |
+
source = replaceSourceToken(unfoldSource, "\\unfoldRepeats");
|
407 |
+
}
|
408 |
+
}
|
409 |
+
|
410 |
+
const foldData = await makeSheetNotation(source, lilyParser, {logger, lilyDocument, withNotation: !midi && !unfoldSource, includeFolders});
|
411 |
+
const {meta, doc, hashTable} = foldData;
|
412 |
+
|
413 |
+
lilyDocument = lilyDocument || foldData.lilyDocument;
|
414 |
+
let midiNotation = foldData.midiNotation;
|
415 |
+
let sheetNotation = foldData.sheetNotation;
|
416 |
+
|
417 |
+
if (midi)
|
418 |
+
midiNotation = MusicNotation.Notation.parseMidi(midi);
|
419 |
+
|
420 |
+
if (unfoldSource) {
|
421 |
+
const unfoldData = await makeSheetNotation(unfoldSource, lilyParser, {logger, lilyDocument, withNotation: !midi, includeFolders});
|
422 |
+
|
423 |
+
midi = midi || unfoldData.midi;
|
424 |
+
midiNotation = unfoldData.midiNotation;
|
425 |
+
sheetNotation = unfoldData.sheetNotation;
|
426 |
+
}
|
427 |
+
|
428 |
+
midi = midi || foldData.midi;
|
429 |
+
if (!midi || !sheetNotation) {
|
430 |
+
if (!midi)
|
431 |
+
console.warn("Neither lilypond or external arguments did not offer MIDI data, score maker finished incompletely.");
|
432 |
+
|
433 |
+
if (!sheetNotation)
|
434 |
+
console.warn("sheetNotation parsing failed, score maker finished incompletely.");
|
435 |
+
|
436 |
+
return {
|
437 |
+
meta,
|
438 |
+
doc,
|
439 |
+
midi,
|
440 |
+
hashTable,
|
441 |
+
};
|
442 |
+
}
|
443 |
+
|
444 |
+
const t5 = Date.now();
|
445 |
+
|
446 |
+
const matcher = await staffSvg.StaffNotation.matchNotations(midiNotation, sheetNotation);
|
447 |
+
|
448 |
+
logger.append("scoreMaker.profile.matching", {cost: Date.now() - t5});
|
449 |
+
|
450 |
+
if (logger && logger.enabled) {
|
451 |
+
const cis = new Set(Array(matcher.criterion.notes.length).keys());
|
452 |
+
matcher.path.forEach(ci => cis.delete(ci));
|
453 |
+
|
454 |
+
const omitC = cis.size;
|
455 |
+
const omitS = matcher.path.filter(ci => ci < 0).length;
|
456 |
+
|
457 |
+
const coverage = ((matcher.criterion.notes.length - omitC) / matcher.criterion.notes.length)
|
458 |
+
* ((matcher.sample.notes.length - omitS) / matcher.sample.notes.length);
|
459 |
+
|
460 |
+
logger.append("makeScore.match", {coverage, omitC, omitS, path: matcher.path});
|
461 |
+
}
|
462 |
+
|
463 |
+
const matchedIds: Set<string> = new Set();
|
464 |
+
midiNotation.notes.forEach(note => note.ids && note.ids.forEach(id => matchedIds.add(id)));
|
465 |
+
|
466 |
+
doc.updateMatchedTokens(matchedIds);
|
467 |
+
|
468 |
+
const pitchContextGroup = PitchContextTable.createPitchContextGroup(sheetNotation.pitchContexts, midiNotation);
|
469 |
+
|
470 |
+
const noteLinkings = midiNotation.notes.map(note => _.pick(note, ["ids", "staffTrack", "contextIndex"]) as NoteLinking);
|
471 |
+
|
472 |
+
logger.append("scoreMaker.profile.full", {cost: Date.now() - t0});
|
473 |
+
|
474 |
+
return {
|
475 |
+
meta,
|
476 |
+
doc,
|
477 |
+
midi,
|
478 |
+
hashTable,
|
479 |
+
noteLinkings,
|
480 |
+
pitchContextGroup,
|
481 |
+
};
|
482 |
+
};
|
483 |
+
|
484 |
+
|
485 |
+
const makeScoreV4 = async (source: string, lilyParser: GrammarParser, {midi, logger, unfoldRepeats = false, baking = false, includeFolders}: {
|
486 |
+
midi?: MIDI.MidiData,
|
487 |
+
logger?: LogRecorder,
|
488 |
+
unfoldRepeats?: boolean,
|
489 |
+
includeFolders?: string[],
|
490 |
+
baking?: boolean,
|
491 |
+
} = {}): Promise<{
|
492 |
+
bakingImages?: Readable[],
|
493 |
+
score: ScoreJSON | IncompleteScoreJSON,
|
494 |
+
}> => {
|
495 |
+
const t0 = Date.now();
|
496 |
+
|
497 |
+
let lilyDocument = null;
|
498 |
+
let unfoldSource = null;
|
499 |
+
|
500 |
+
if (unfoldRepeats) {
|
501 |
+
lilyDocument = new LilyDocument(lilyParser.parse(source));
|
502 |
+
lilyDocument.interpret();
|
503 |
+
if (lilyDocument.containsRepeat()) {
|
504 |
+
lilyDocument.unfoldRepeats();
|
505 |
+
unfoldSource = lilyDocument.toString();
|
506 |
+
|
507 |
+
// keep 2 version lilypond source note href uniform
|
508 |
+
source = replaceSourceToken(unfoldSource, "\\unfoldRepeats");
|
509 |
+
}
|
510 |
+
}
|
511 |
+
|
512 |
+
const foldData = await makeSheetNotation(source, lilyParser, {logger, lilyDocument, withNotation: !midi && !unfoldSource, includeFolders, baking});
|
513 |
+
const {meta, doc, hashTable, bakingImages} = foldData;
|
514 |
+
|
515 |
+
lilyDocument = lilyDocument || foldData.lilyDocument;
|
516 |
+
//let sheetNotation = foldData.sheetNotation;
|
517 |
+
|
518 |
+
const matchingMidi = midi || foldData.midi;
|
519 |
+
|
520 |
+
if (unfoldSource) {
|
521 |
+
const unfoldData = await makeSheetNotation(unfoldSource, lilyParser, {logger, lilyDocument, withNotation: !midi, includeFolders});
|
522 |
+
|
523 |
+
midi = midi || unfoldData.midi;
|
524 |
+
//sheetNotation = unfoldData.sheetNotation;
|
525 |
+
}
|
526 |
+
|
527 |
+
midi = midi || foldData.midi;
|
528 |
+
const lilyNotation = lilyDocument.interpret().getNotation();
|
529 |
+
|
530 |
+
if (!midi || !lilyNotation) {
|
531 |
+
if (!midi)
|
532 |
+
console.warn("Neither lilypond or external arguments did not offer MIDI data, score maker finished incompletely.");
|
533 |
+
|
534 |
+
if (!lilyNotation)
|
535 |
+
console.warn("sheetNotation parsing failed, score maker finished incompletely.");
|
536 |
+
|
537 |
+
return {
|
538 |
+
score: {
|
539 |
+
version: npmPackage.version,
|
540 |
+
meta,
|
541 |
+
doc,
|
542 |
+
midi,
|
543 |
+
hashTable,
|
544 |
+
},
|
545 |
+
};
|
546 |
+
}
|
547 |
+
|
548 |
+
const t5 = Date.now();
|
549 |
+
|
550 |
+
const matcher = await LilyNotation.matchWithMIDI(lilyNotation, matchingMidi);
|
551 |
+
|
552 |
+
logger.append("scoreMaker.profile.matching", {cost: Date.now() - t5});
|
553 |
+
|
554 |
+
if (logger && logger.enabled) {
|
555 |
+
const cis = new Set(Array(matcher.criterion.notes.length).keys());
|
556 |
+
matcher.path.forEach(ci => cis.delete(ci));
|
557 |
+
|
558 |
+
const omitC = cis.size;
|
559 |
+
const omitS = matcher.path.filter(ci => ci < 0).length;
|
560 |
+
|
561 |
+
const coverage = ((matcher.criterion.notes.length - omitC) / matcher.criterion.notes.length)
|
562 |
+
* ((matcher.sample.notes.length - omitS) / matcher.sample.notes.length);
|
563 |
+
|
564 |
+
logger.append("makeScore.match", {coverage, omitC, omitS, path: matcher.path});
|
565 |
+
}
|
566 |
+
|
567 |
+
const midiNotation = matcher.sample;
|
568 |
+
|
569 |
+
const matchedIds: Set<string> = new Set();
|
570 |
+
midiNotation.notes.forEach(note => note.ids && note.ids.forEach(id => matchedIds.add(id)));
|
571 |
+
|
572 |
+
doc.updateMatchedTokens(matchedIds);
|
573 |
+
|
574 |
+
if (baking)
|
575 |
+
doc.pruneForBakingMode();
|
576 |
+
|
577 |
+
const pitchContextGroup = PitchContextTable.createPitchContextGroup(
|
578 |
+
lilyNotation.pitchContextGroup.map(table => table.items.map(item => item.context)), midiNotation);
|
579 |
+
|
580 |
+
const noteLinkings = midiNotation.notes.map(note => _.pick(note, ["ids", "staffTrack", "contextIndex"]) as NoteLinking);
|
581 |
+
|
582 |
+
logger.append("scoreMaker.profile.full", {cost: Date.now() - t0});
|
583 |
+
|
584 |
+
return {
|
585 |
+
bakingImages,
|
586 |
+
score: {
|
587 |
+
version: npmPackage.version,
|
588 |
+
meta,
|
589 |
+
doc,
|
590 |
+
midi,
|
591 |
+
hashTable: !baking ? hashTable : null,
|
592 |
+
noteLinkings,
|
593 |
+
pitchContextGroup,
|
594 |
+
},
|
595 |
+
};
|
596 |
+
};
|
597 |
+
|
598 |
+
|
599 |
+
void makeScoreV1;
|
600 |
+
void makeScoreV2;
|
601 |
+
void makeScoreV3;
|
602 |
+
const makeScore = makeScoreV4;
|
603 |
+
|
604 |
+
|
605 |
+
const makeMIDI = async (source: string, lilyParser: GrammarParser, {unfoldRepeats = false, fixNestedRepeat = false, includeFolders = undefined} = {}): Promise<MIDI.MidiData> => {
|
606 |
+
const lilyDocument = new LilyDocument(lilyParser.parse(source));
|
607 |
+
|
608 |
+
if (fixNestedRepeat)
|
609 |
+
lilyDocument.fixNestedRepeat();
|
610 |
+
|
611 |
+
if (unfoldRepeats)
|
612 |
+
lilyDocument.unfoldRepeats();
|
613 |
+
|
614 |
+
const score = lilyDocument.root.getBlock("score");
|
615 |
+
if (score) {
|
616 |
+
// remove layout block to save time
|
617 |
+
score.body = score.body.filter(term => !(term instanceof LilyTerms.Block && term.head === "\\layout"));
|
618 |
+
|
619 |
+
// remove invalid tempo
|
620 |
+
const midi: any = score.body.find(term => term instanceof LilyTerms.Block && term.head === "\\midi");
|
621 |
+
if (midi)
|
622 |
+
midi.body = midi.body.filter(term => !(term instanceof LilyTerms.Tempo && term.beatsPerMinute > 200));
|
623 |
+
}
|
624 |
+
|
625 |
+
const markupSource = lilyDocument.toString();
|
626 |
+
//console.log("markupSource:", markupSource);
|
627 |
+
|
628 |
+
return new Promise((resolve, reject) => engraveSvg(markupSource, {
|
629 |
+
includeFolders,
|
630 |
+
onMidiRead: resolve,
|
631 |
+
}).catch(reject));
|
632 |
+
};
|
633 |
+
|
634 |
+
|
635 |
+
const makeArticulatedMIDI = async (source: string, lilyParser: GrammarParser, {ignoreRepeats = true, includeFolders = undefined} = {}): Promise<MIDI.MidiData> => {
|
636 |
+
const lilyDocument = new LilyDocument(lilyParser.parse(source));
|
637 |
+
|
638 |
+
if (ignoreRepeats)
|
639 |
+
lilyDocument.removeRepeats();
|
640 |
+
|
641 |
+
lilyDocument.articulateMIDIOutput();
|
642 |
+
|
643 |
+
// remove layout block to save time
|
644 |
+
lilyDocument.root.sections = lilyDocument.root.sections.filter(section => !(section instanceof Block)
|
645 |
+
|| !(section.head === "\\score")
|
646 |
+
|| section.isMIDIDedicated);
|
647 |
+
|
648 |
+
const markupSource = lilyDocument.toString();
|
649 |
+
//console.log("markupSource:", markupSource);
|
650 |
+
|
651 |
+
return new Promise((resolve, reject) => engraveSvg(markupSource, {
|
652 |
+
includeFolders,
|
653 |
+
onMidiRead: resolve,
|
654 |
+
}).catch(reject));
|
655 |
+
};
|
656 |
+
|
657 |
+
|
658 |
+
|
659 |
+
export {
|
660 |
+
markupLily,
|
661 |
+
xmlBufferToLy,
|
662 |
+
makeScore,
|
663 |
+
makeMIDI,
|
664 |
+
makeArticulatedMIDI,
|
665 |
+
};
|
backend/statStorage.ts
ADDED
@@ -0,0 +1,34 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
|
2 |
+
import fs from "fs";
|
3 |
+
import path from "path";
|
4 |
+
import YAML from "yaml";
|
5 |
+
|
6 |
+
import asyncCall from "../inc/asyncCall";
|
7 |
+
|
8 |
+
|
9 |
+
|
10 |
+
const exists = async (path: string) => new Promise(resolve => fs.access(path, err => resolve(!err)));
|
11 |
+
|
12 |
+
|
13 |
+
const appendData = async (filePath: string, data: object) => {
|
14 |
+
const dir = path.resolve(filePath, "..");
|
15 |
+
const filename = path.relative(dir, filePath);
|
16 |
+
const statPath = path.resolve(dir, ".lotus_stat.yaml");
|
17 |
+
|
18 |
+
let stat = {};
|
19 |
+
if (await exists(statPath)) {
|
20 |
+
const buffer = await asyncCall(fs.readFile, statPath);
|
21 |
+
stat = YAML.parse(buffer.toString()) || stat;
|
22 |
+
}
|
23 |
+
|
24 |
+
stat[filename] = stat[filename] || {};
|
25 |
+
Object.assign(stat[filename], data);
|
26 |
+
|
27 |
+
await asyncCall(fs.writeFile, statPath, YAML.stringify(stat));
|
28 |
+
};
|
29 |
+
|
30 |
+
|
31 |
+
|
32 |
+
export {
|
33 |
+
appendData,
|
34 |
+
};
|
backend/walkDir.ts
ADDED
@@ -0,0 +1,35 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
|
2 |
+
import fs from "fs";
|
3 |
+
import path from "path";
|
4 |
+
|
5 |
+
|
6 |
+
|
7 |
+
const walkDir = (dir, pattern, {recursive = false} = {}) => {
|
8 |
+
const list = fs.readdirSync(dir);
|
9 |
+
//console.log("files:", files);
|
10 |
+
|
11 |
+
const subdirs = [];
|
12 |
+
|
13 |
+
const files = list.map(filename => {
|
14 |
+
const file = path.resolve(dir, filename);
|
15 |
+
const stat = fs.statSync(file);
|
16 |
+
|
17 |
+
if (stat.isDirectory())
|
18 |
+
subdirs.push(file);
|
19 |
+
|
20 |
+
return {file, filename, stat};
|
21 |
+
}).filter(({filename, stat}) => !stat.isDirectory() && (!pattern || pattern.test(filename)))
|
22 |
+
.map(({file}) => file);
|
23 |
+
|
24 |
+
if (recursive) {
|
25 |
+
subdirs.forEach(subdir => {
|
26 |
+
files.push(...walkDir(subdir, pattern, {recursive}));
|
27 |
+
});
|
28 |
+
}
|
29 |
+
|
30 |
+
return files;
|
31 |
+
};
|
32 |
+
|
33 |
+
|
34 |
+
|
35 |
+
export default walkDir;
|
backend/xmlTools.ts
ADDED
@@ -0,0 +1,214 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
|
2 |
+
import {DOMParser, XMLSerializer} from "xmldom";
|
3 |
+
|
4 |
+
import * as domUtils from "../inc/domUtils";
|
5 |
+
|
6 |
+
|
7 |
+
|
8 |
+
const isWordDirection = direction => {
|
9 |
+
const directionTypes = domUtils.childrenWithTag(direction, "direction-type");
|
10 |
+
|
11 |
+
for (const dt of directionTypes) {
|
12 |
+
if (domUtils.hasChildrenWithTag(dt, "words"))
|
13 |
+
return true;
|
14 |
+
}
|
15 |
+
|
16 |
+
return false;
|
17 |
+
};
|
18 |
+
|
19 |
+
|
20 |
+
const moveWordDirection = measure => {
|
21 |
+
//console.log("measure:", measure);
|
22 |
+
const words = [];
|
23 |
+
|
24 |
+
for (let i = 0; i < measure.childNodes.length; ++i) {
|
25 |
+
const child = measure.childNodes[i];
|
26 |
+
switch (child.tagName) {
|
27 |
+
case "direction":
|
28 |
+
if (isWordDirection(child)) {
|
29 |
+
words.push(child);
|
30 |
+
measure.removeChild(child);
|
31 |
+
}
|
32 |
+
|
33 |
+
break;
|
34 |
+
case "note":
|
35 |
+
const next = measure.childNodes[i + 1];
|
36 |
+
words.forEach(word => {
|
37 |
+
if (next)
|
38 |
+
measure.insertBefore(word, next);
|
39 |
+
else
|
40 |
+
measure.appendChild(word);
|
41 |
+
});
|
42 |
+
|
43 |
+
return;
|
44 |
+
}
|
45 |
+
}
|
46 |
+
};
|
47 |
+
|
48 |
+
|
49 |
+
const preprocessXml = (xml: string, {
|
50 |
+
removeMeasureImplicit = true,
|
51 |
+
replaceEncoding = true,
|
52 |
+
removeNullDynamics = true,
|
53 |
+
fixHeadMarkup = true,
|
54 |
+
fixBackSlashes = true,
|
55 |
+
roundTempo = true,
|
56 |
+
escapedWordsDoubleQuotation = true,
|
57 |
+
removeTrivialRests = true,
|
58 |
+
removeBadMetronome = true,
|
59 |
+
removeInvalidHarmonies = true,
|
60 |
+
removeAllHarmonies = false,
|
61 |
+
fixChordVoice = true,
|
62 |
+
fixBarlines = true,
|
63 |
+
removeInvalidClef = true,
|
64 |
+
} = {}): string => {
|
65 |
+
if (!removeMeasureImplicit && !replaceEncoding && !removeNullDynamics && !fixHeadMarkup && !fixBackSlashes && !roundTempo
|
66 |
+
&& !escapedWordsDoubleQuotation && !removeTrivialRests && !removeBadMetronome && !removeInvalidHarmonies && !removeAllHarmonies
|
67 |
+
&& !fixChordVoice && !fixBarlines)
|
68 |
+
return xml;
|
69 |
+
|
70 |
+
// @ts-ignore
|
71 |
+
const dom = new DOMParser().parseFromString(xml, "text/xml");
|
72 |
+
|
73 |
+
if (replaceEncoding) {
|
74 |
+
const headNode = Array.prototype.find.call(dom.childNodes, node => node.tagName === "xml");
|
75 |
+
if (headNode)
|
76 |
+
headNode.data = headNode.data.replace(/UTF-16/, "UTF-8");
|
77 |
+
}
|
78 |
+
|
79 |
+
const needTraverse = removeMeasureImplicit || removeNullDynamics || fixHeadMarkup || fixBackSlashes || roundTempo
|
80 |
+
|| escapedWordsDoubleQuotation || removeTrivialRests || removeBadMetronome || removeInvalidHarmonies || removeAllHarmonies
|
81 |
+
|| fixChordVoice || fixBarlines;
|
82 |
+
if (needTraverse) {
|
83 |
+
domUtils.traverse(dom, node => {
|
84 |
+
if (removeMeasureImplicit) {
|
85 |
+
if (node.tagName === "measure")
|
86 |
+
node.removeAttribute("implicit");
|
87 |
+
}
|
88 |
+
|
89 |
+
if (removeNullDynamics) {
|
90 |
+
if (node.tagName === "other-dynamics") {
|
91 |
+
const content = node.textContent;
|
92 |
+
if (!/\w/.test(content))
|
93 |
+
node.parentNode.removeChild(node);
|
94 |
+
}
|
95 |
+
}
|
96 |
+
|
97 |
+
if (fixHeadMarkup) {
|
98 |
+
if (node.tagName === "measure") {
|
99 |
+
//console.log("measure:", node);
|
100 |
+
moveWordDirection(node);
|
101 |
+
}
|
102 |
+
}
|
103 |
+
|
104 |
+
if (fixBackSlashes) {
|
105 |
+
if (["words", "credit-words", "text"].includes(node.tagName)) {
|
106 |
+
if (/^\\+/.test(node.textContent) || /\\+$/.test(node.textContent)) {
|
107 |
+
console.debug("replaced invalid text:", node.textContent);
|
108 |
+
node.textContent = node.textContent.replace(/^\\+/, "").replace(/\\+$/, "");
|
109 |
+
}
|
110 |
+
}
|
111 |
+
}
|
112 |
+
|
113 |
+
if (roundTempo) {
|
114 |
+
if (node.tagName === "sound") {
|
115 |
+
const tempo = Number(node.getAttribute("tempo"));
|
116 |
+
if (Number.isFinite(tempo) && Math.floor(tempo) !== tempo)
|
117 |
+
node.setAttribute("tempo", Math.round(tempo).toFixed(0));
|
118 |
+
}
|
119 |
+
}
|
120 |
+
|
121 |
+
if (escapedWordsDoubleQuotation) {
|
122 |
+
if (["words", "credit-words", "text"].includes(node.tagName)) {
|
123 |
+
if (node.textContent && /"/.test(node.textContent))
|
124 |
+
node.textContent = node.textContent.replace(/"/g, "'");
|
125 |
+
}
|
126 |
+
}
|
127 |
+
|
128 |
+
if (removeTrivialRests) {
|
129 |
+
if (node.tagName === "note") {
|
130 |
+
if (domUtils.hasChildrenWithTag(node, "rest") && !domUtils.hasChildrenWithTag(node, "type")) {
|
131 |
+
// append an empty tag: <type></type>
|
132 |
+
const type = dom.createElement("type");
|
133 |
+
node.appendChild(type);
|
134 |
+
}
|
135 |
+
}
|
136 |
+
}
|
137 |
+
|
138 |
+
if (removeBadMetronome) {
|
139 |
+
if (node.tagName === "metronome") {
|
140 |
+
if (!domUtils.hasChildrenWithTag(node, "per-minute")) {
|
141 |
+
console.debug("metronome without 'per-minute' removed:", node.toString());
|
142 |
+
node.parentNode.removeChild(node);
|
143 |
+
}
|
144 |
+
}
|
145 |
+
}
|
146 |
+
|
147 |
+
if (removeInvalidHarmonies || removeAllHarmonies) {
|
148 |
+
if (node.tagName === "harmony") {
|
149 |
+
if (removeAllHarmonies)
|
150 |
+
node.parentNode.removeChild(node);
|
151 |
+
else {
|
152 |
+
let next = node.nextSibling;
|
153 |
+
while (next && !next.tagName)
|
154 |
+
next = next.nextSibling;
|
155 |
+
|
156 |
+
//console.log("sibling:", next);
|
157 |
+
if (!next || next.tagName !== "note") {
|
158 |
+
node.parentNode.removeChild(node);
|
159 |
+
console.debug("invalid harmony removed:", next && next.tagName);
|
160 |
+
}
|
161 |
+
}
|
162 |
+
}
|
163 |
+
}
|
164 |
+
|
165 |
+
if (fixChordVoice) {
|
166 |
+
if (node.tagName === "note" && domUtils.hasChildrenWithTag(node, "chord") && !domUtils.hasChildrenWithTag(node, "voice")) {
|
167 |
+
//console.log("bad note:", node);
|
168 |
+
const previousNote = domUtils.findPreviousSibling(node, "note");
|
169 |
+
if (previousNote) {
|
170 |
+
const voice = domUtils.childrenWithTag(previousNote, "voice")[0];
|
171 |
+
//console.log("chord voice:", node, voice);
|
172 |
+
if (voice)
|
173 |
+
node.appendChild(voice.cloneNode(true));
|
174 |
+
}
|
175 |
+
}
|
176 |
+
}
|
177 |
+
|
178 |
+
if (fixBarlines) {
|
179 |
+
if (node.tagName === "barline") {
|
180 |
+
if ((node.getAttribute("location") === "right" || domUtils.hasChildrenWithTag(node, "bar-style"))
|
181 |
+
&& domUtils.findNextSibling(node, "backup")) {
|
182 |
+
// move the barline to the end of this measure
|
183 |
+
node.parentNode.removeChild(node);
|
184 |
+
node.parentNode.appendChild(node);
|
185 |
+
}
|
186 |
+
}
|
187 |
+
}
|
188 |
+
|
189 |
+
if(removeInvalidClef) {
|
190 |
+
if (node.tagName === "attributes") {
|
191 |
+
const clef = domUtils.childrenWithTag(node, "clef")[0];
|
192 |
+
if (clef) {
|
193 |
+
const n = Number(clef.getAttribute("number"));
|
194 |
+
if (!domUtils.findPreviousSibling(node, "backup") && n > 1) {
|
195 |
+
node.parentNode.removeChild(node);
|
196 |
+
console.debug("invalid clef removed:", n);
|
197 |
+
}
|
198 |
+
}
|
199 |
+
}
|
200 |
+
}
|
201 |
+
});
|
202 |
+
}
|
203 |
+
|
204 |
+
//console.log("dom:", dom);
|
205 |
+
|
206 |
+
// @ts-ignore
|
207 |
+
return new XMLSerializer().serializeToString(dom);
|
208 |
+
};
|
209 |
+
|
210 |
+
|
211 |
+
|
212 |
+
export {
|
213 |
+
preprocessXml,
|
214 |
+
};
|
dist/e5c1442134f1e7dfb9dd.worker.js
ADDED
The diff for this file is too large to render.
See raw diff
|
|
dist/favicon.ico
ADDED
dist/index.html
ADDED
@@ -0,0 +1 @@
|
|
|
|
|
1 |
+
<!DOCTYPE html><html><head><title>Lotus</title><link href="js/chunk-04395031.dbff2f6b.js" rel="prefetch"><link href="js/chunk-0c4e36c8.95d70738.js" rel="prefetch"><link href="js/chunk-0cbfe13e.73856287.js" rel="prefetch"><link href="js/chunk-117382e0.d47336d3.js" rel="prefetch"><link href="js/chunk-2d0c53c7.d24941b8.js" rel="prefetch"><link href="js/chunk-2d0db258.a4804a7a.js" rel="prefetch"><link href="js/chunk-40965e1a.74707226.js" rel="prefetch"><link href="js/chunk-48b5b2a0.3db5a0aa.js" rel="prefetch"><link href="js/chunk-a06ef50c.1caef24f.js" rel="prefetch"><link href="js/chunk-ae402692.003457bc.js" rel="prefetch"><link href="js/chunk-vendors.20f7f886.js" rel="preload" as="script"><link href="js/index.cbb15892.js" rel="preload" as="script"></head><body><script src="js/chunk-vendors.20f7f886.js"></script><script src="js/index.cbb15892.js"></script></body></html>
|
dist/js/chunk-04395031.dbff2f6b.js
ADDED
The diff for this file is too large to render.
See raw diff
|
|
dist/js/chunk-0c4e36c8.95d70738.js
ADDED
@@ -0,0 +1,16 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
(window["webpackJsonp"]=window["webpackJsonp"]||[]).push([["chunk-0c4e36c8"],{"010e":function(e,t,n){"use strict";n.d(t,"a",(function(){return s}));var a=n("9e5c"),i=n.n(a);const r=new Map;async function s(e){if(!r.get(e)){const t=performance.now(),{default:n}=await e,a=await(await fetch(n)).text(),s=performance.now(),o=new i.a;r.set(e,await new o.Parser(a));const l=performance.now();console.debug("Jison parser loading cost:",s-t,l-s)}return r.get(e)}},"0794":function(e,t,n){e.exports=function(){return new Worker(n.p+"e5c1442134f1e7dfb9dd.worker.js")}},"0e06":function(e,t,n){var a=n("7407");a.__esModule&&(a=a.default),"string"===typeof a&&(a=[[e.i,a,""]]),a.locals&&(e.exports=a.locals);var i=n("499e").default;i("b061a5c4",a,!0,{sourceMap:!1,shadowMode:!1})},"13c4":function(e,t,n){var a=n("b437");a.__esModule&&(a=a.default),"string"===typeof a&&(a=[[e.i,a,""]]),a.locals&&(e.exports=a.locals);var i=n("499e").default;i("155344be",a,!0,{sourceMap:!1,shadowMode:!1})},"1c08":function(e,t){(function(e){for(var t=/\((?:[^();"#\\]|\\[\s\S]|;.*(?!.)|"(?:[^"\\]|\\.)*"|#(?:\{(?:(?!#\})[\s\S])*#\}|[^{])|<expr>)*\)/.source,n=5,a=0;a<n;a++)t=t.replace(/<expr>/g,(function(){return t}));t=t.replace(/<expr>/g,/[^\s\S]/.source);var i=e.languages.lilypond={comment:/%(?:(?!\{).*|\{[\s\S]*?%\})/,"embedded-scheme":{pattern:RegExp(/(^|[=\s])#(?:"(?:[^"\\]|\\.)*"|[^\s()"]*(?:[^\s()]|<expr>))/.source.replace(/<expr>/g,(function(){return t})),"m"),lookbehind:!0,greedy:!0,inside:{scheme:{pattern:/^(#)[\s\S]+$/,lookbehind:!0,alias:"language-scheme",inside:{"embedded-lilypond":{pattern:/#\{[\s\S]*?#\}/,greedy:!0,inside:{punctuation:/^#\{|#\}$/,lilypond:{pattern:/[\s\S]+/,alias:"language-lilypond",inside:null}}},rest:e.languages.scheme}},punctuation:/#/}},string:{pattern:/"(?:[^"\\]|\\.)*"/,greedy:!0},"class-name":{pattern:/(\\new\s+)[\w-]+/,lookbehind:!0},keyword:{pattern:/\\[a-z][-\w]*/i,inside:{punctuation:/^\\/}},operator:/[=|]|<<|>>/,punctuation:{pattern:/(^|[a-z\d])(?:'+|,+|[_^]?-[_^]?(?:[-+^!>._]|(?=\d))|[_^]\.?|[.!])|[{}()[\]<>^~]|\\[()[\]<>\\!]|--|__/,lookbehind:!0},number:/\b\d+(?:\/\d+)?\b/};i["embedded-scheme"].inside["scheme"].inside["embedded-lilypond"].inside["lilypond"].inside=i,e.languages.ly=i})(Prism)},2669:function(e,t,n){var a=n("24fb");t=a(!1),t.push([e.i,".source-editor[data-v-d4e36058]{display:inline-block}.prism-editor-wrapper[data-v-d4e36058]{background-color:#f4f2f0;resize:horizontal;width:50em;height:100%;font-family:monospace;font-size:16px;line-height:19px;overflow:auto}",""]),e.exports=t},3122:function(e,t,n){"use strict";var a=function(){var e=this,t=e._self._c;return t("div",{staticClass:"source-editor",on:{click:e.onClick}},[t("PrismEditor",{attrs:{lineNumbers:!0,highlight:e.highlighter,readonly:e.disabled,tabSize:4},model:{value:e.editText,callback:function(t){e.editText=t},expression:"editText"}})],1)},i=[],r=(n("cabf"),n("13c4"),n("c197")),s=(n("1c08"),n("2b0e"));function o(){return o=Object.assign||function(e){for(var t=1;t<arguments.length;t++){var n=arguments[t];for(var a in n)Object.prototype.hasOwnProperty.call(n,a)&&(e[a]=n[a])}return e},o.apply(this,arguments)}var l=13,u=9,c=8,d=89,p=90,g=77,h=57,f=219,m=222,y=192,v=27,b=100,k=3e3,w="undefined"!==typeof window&&navigator&&/Win/i.test(navigator.platform),x="undefined"!==typeof window&&navigator&&/(Mac|iPhone|iPod|iPad)/i.test(navigator.platform),_=s["a"].extend({props:{lineNumbers:{type:Boolean,default:!1},autoStyleLineNumbers:{type:Boolean,default:!0},readonly:{type:Boolean,default:!1},value:{type:String,default:""},highlight:{type:Function,required:!0},tabSize:{type:Number,default:2},insertSpaces:{type:Boolean,default:!0},ignoreTabKey:{type:Boolean,default:!1},placeholder:{type:String,default:""}},data:function(){return{capture:!0,history:{stack:[],offset:-1},lineNumbersHeight:"20px",codeData:""}},watch:{value:{immediate:!0,handler:function(e){this.codeData=e||""}},content:{immediate:!0,handler:function(){var e=this;this.lineNumbers&&this.$nextTick((function(){e.setLineNumbersHeight()}))}},lineNumbers:function(){var e=this;this.$nextTick((function(){e.styleLineNumbers(),e.setLineNumbersHeight()}))}},computed:{isEmpty:function(){return 0===this.codeData.length},content:function(){var e=this.highlight(this.codeData)+"<br />";return e},lineNumbersCount:function(){var e=this.codeData.split(/\r\n|\n/).length;return e}},mounted:function(){this._recordCurrentState(),this.styleLineNumbers()},methods:{setLineNumbersHeight:function(){this.lineNumbersHeight=getComputedStyle(this.$refs.pre).height},styleLineNumbers:function(){if(this.lineNumbers&&this.autoStyleLineNumbers){var e=this.$refs.pre,t=this.$el.querySelector(".prism-editor__line-numbers"),n=window.getComputedStyle(e);this.$nextTick((function(){var a="border-top-left-radius",i="border-bottom-left-radius";if(t){t.style[a]=n[a],t.style[i]=n[i],e.style[a]="0",e.style[i]="0";var r=["background-color","margin-top","padding-top","font-family","font-size","line-height"];r.forEach((function(e){t.style[e]=n[e]})),t.style["margin-bottom"]="-"+n["padding-top"]}}))}},_recordCurrentState:function(){var e=this.$refs.textarea;if(e){var t=e.value,n=e.selectionStart,a=e.selectionEnd;this._recordChange({value:t,selectionStart:n,selectionEnd:a})}},_getLines:function(e,t){return e.substring(0,t).split("\n")},_applyEdits:function(e){var t=this.$refs.textarea,n=this.history.stack[this.history.offset];n&&t&&(this.history.stack[this.history.offset]=o({},n,{selectionStart:t.selectionStart,selectionEnd:t.selectionEnd})),this._recordChange(e),this._updateInput(e)},_recordChange:function(e,t){void 0===t&&(t=!1);var n=this.history,a=n.stack,i=n.offset;if(a.length&&i>-1){this.history.stack=a.slice(0,i+1);var r=this.history.stack.length;if(r>b){var s=r-b;this.history.stack=a.slice(s,r),this.history.offset=Math.max(this.history.offset-s,0)}}var l=Date.now();if(t){var u=this.history.stack[this.history.offset];if(u&&l-u.timestamp<k){var c,d,p=/[^a-z0-9]([a-z0-9]+)$/i,g=null===(c=this._getLines(u.value,u.selectionStart).pop())||void 0===c?void 0:c.match(p),h=null===(d=this._getLines(e.value,e.selectionStart).pop())||void 0===d?void 0:d.match(p);if(g&&h&&h[1].startsWith(g[1]))return void(this.history.stack[this.history.offset]=o({},e,{timestamp:l}))}}this.history.stack.push(o({},e,{timestamp:l})),this.history.offset++},_updateInput:function(e){var t=this.$refs.textarea;t&&(t.value=e.value,t.selectionStart=e.selectionStart,t.selectionEnd=e.selectionEnd,this.$emit("input",e.value))},handleChange:function(e){var t=e.target,n=t.value,a=t.selectionStart,i=t.selectionEnd;this._recordChange({value:n,selectionStart:a,selectionEnd:i},!0),this.$emit("input",n)},_undoEdit:function(){var e=this.history,t=e.stack,n=e.offset,a=t[n-1];a&&(this._updateInput(a),this.history.offset=Math.max(n-1,0))},_redoEdit:function(){var e=this.history,t=e.stack,n=e.offset,a=t[n+1];a&&(this._updateInput(a),this.history.offset=Math.min(n+1,t.length-1))},handleKeyDown:function(e){var t=this.tabSize,n=this.insertSpaces,a=this.ignoreTabKey;if(!this.$listeners.keydown||(this.$emit("keydown",e),!e.defaultPrevented)){e.keyCode===v&&(e.target.blur(),this.$emit("blur",e));var i=e.target,r=i.value,s=i.selectionStart,o=i.selectionEnd,b=(n?" ":"\t").repeat(t);if(e.keyCode===u&&!a&&this.capture)if(e.preventDefault(),e.shiftKey){var k=this._getLines(r,s),_=k.length-1,E=this._getLines(r,o).length-1,F=r.split("\n").map((function(e,t){return t>=_&&t<=E&&e.startsWith(b)?e.substring(b.length):e})).join("\n");if(r!==F){var A=k[_];this._applyEdits({value:F,selectionStart:A.startsWith(b)?s-b.length:s,selectionEnd:o-(r.length-F.length)})}}else if(s!==o){var S=this._getLines(r,s),$=S.length-1,C=this._getLines(r,o).length-1,z=S[$];this._applyEdits({value:r.split("\n").map((function(e,t){return t>=$&&t<=C?b+e:e})).join("\n"),selectionStart:/\S/.test(z)?s+b.length:s,selectionEnd:o+b.length*(C-$+1)})}else{var M=s+b.length;this._applyEdits({value:r.substring(0,s)+b+r.substring(o),selectionStart:M,selectionEnd:M})}else if(e.keyCode===c){var L=s!==o,T=r.substring(0,s);if(T.endsWith(b)&&!L){e.preventDefault();var j=s-b.length;this._applyEdits({value:r.substring(0,s-b.length)+r.substring(o),selectionStart:j,selectionEnd:j})}}else if(e.keyCode===l){if(s===o){var P=this._getLines(r,s).pop(),N=null===P||void 0===P?void 0:P.match(/^\s+/);if(N&&N[0]){e.preventDefault();var O="\n"+N[0],D=s+O.length;this._applyEdits({value:r.substring(0,s)+O+r.substring(o),selectionStart:D,selectionEnd:D})}}}else if(e.keyCode===h||e.keyCode===f||e.keyCode===m||e.keyCode===y){var R;e.keyCode===h&&e.shiftKey?R=["(",")"]:e.keyCode===f?R=e.shiftKey?["{","}"]:["[","]"]:e.keyCode===m?R=e.shiftKey?['"','"']:["'","'"]:e.keyCode!==y||e.shiftKey||(R=["`","`"]),s!==o&&R&&(e.preventDefault(),this._applyEdits({value:r.substring(0,s)+R[0]+r.substring(s,o)+R[1]+r.substring(o),selectionStart:s,selectionEnd:o+2}))}else!(x?e.metaKey&&e.keyCode===p:e.ctrlKey&&e.keyCode===p)||e.shiftKey||e.altKey?(x?e.metaKey&&e.keyCode===p&&e.shiftKey:w?e.ctrlKey&&e.keyCode===d:e.ctrlKey&&e.keyCode===p&&e.shiftKey)&&!e.altKey?(e.preventDefault(),this._redoEdit()):e.keyCode!==g||!e.ctrlKey||x&&!e.shiftKey||(e.preventDefault(),this.capture=!this.capture):(e.preventDefault(),this._undoEdit())}}},render:function(e){var t=this,n=e("div",{attrs:{class:"prism-editor__line-width-calc",style:"height: 0px; visibility: hidden; pointer-events: none;"}},"999"),a=e("div",{staticClass:"prism-editor__line-numbers",style:{"min-height":this.lineNumbersHeight},attrs:{"aria-hidden":"true"}},[n,Array.from(Array(this.lineNumbersCount).keys()).map((function(t,n){return e("div",{attrs:{class:"prism-editor__line-number token comment"}},""+ ++n)}))]),i=e("textarea",{ref:"textarea",on:{input:this.handleChange,keydown:this.handleKeyDown,click:function(e){t.$emit("click",e)},keyup:function(e){t.$emit("keyup",e)},focus:function(e){t.$emit("focus",e)},blur:function(e){t.$emit("blur",e)}},staticClass:"prism-editor__textarea",class:{"prism-editor__textarea--empty":this.isEmpty},attrs:{spellCheck:"false",autocapitalize:"off",autocomplete:"off",autocorrect:"off","data-gramm":"false",placeholder:this.placeholder,"data-testid":"textarea",readonly:this.readonly},domProps:{value:this.codeData}}),r=e("pre",{ref:"pre",staticClass:"prism-editor__editor",attrs:{"data-testid":"preview"},domProps:{innerHTML:this.content}}),s=e("div",{staticClass:"prism-editor__container"},[i,r]);return e("div",{staticClass:"prism-editor-wrapper"},[this.lineNumbers&&a,s])}}),E={name:"source-editor",props:{source:String,disabled:Boolean},components:{PrismEditor:_},data(){return{editText:this.source}},methods:{highlighter(e){return Object(r["highlight"])(e,r["languages"].lilypond)},onClick(){document.activeElement&&"TEXTAREA"===document.activeElement.tagName||this.$el.querySelector("textarea").focus()}},watch:{editText(e){this.source!==e&&this.$emit("update:source",e)},source(e){this.editText=e}}},F=E,A=(n("afdd"),n("b2c1"),n("2877")),S=Object(A["a"])(F,a,i,!1,null,"d4e36058",null);t["a"]=S.exports},"3d15":function(e,t,n){"use strict";n.r(t),n.d(t,"createEndpoint",(function(){return i})),n.d(t,"expose",(function(){return g})),n.d(t,"finalizer",(function(){return s})),n.d(t,"proxy",(function(){return $})),n.d(t,"proxyMarker",(function(){return a})),n.d(t,"releaseProxy",(function(){return r})),n.d(t,"transfer",(function(){return S})),n.d(t,"transferHandlers",(function(){return d})),n.d(t,"windowEndpoint",(function(){return C})),n.d(t,"wrap",(function(){return m}));
|
2 |
+
/**
|
3 |
+
* @license
|
4 |
+
* Copyright 2019 Google LLC
|
5 |
+
* SPDX-License-Identifier: Apache-2.0
|
6 |
+
*/
|
7 |
+
const a=Symbol("Comlink.proxy"),i=Symbol("Comlink.endpoint"),r=Symbol("Comlink.releaseProxy"),s=Symbol("Comlink.finalizer"),o=Symbol("Comlink.thrown"),l=e=>"object"===typeof e&&null!==e||"function"===typeof e,u={canHandle:e=>l(e)&&e[a],serialize(e){const{port1:t,port2:n}=new MessageChannel;return g(e,t),[n,[n]]},deserialize(e){return e.start(),m(e)}},c={canHandle:e=>l(e)&&o in e,serialize({value:e}){let t;return t=e instanceof Error?{isError:!0,value:{message:e.message,name:e.name,stack:e.stack}}:{isError:!1,value:e},[t,[]]},deserialize(e){if(e.isError)throw Object.assign(new Error(e.value.message),e.value);throw e.value}},d=new Map([["proxy",u],["throw",c]]);function p(e,t){for(const n of e){if(t===n||"*"===n)return!0;if(n instanceof RegExp&&n.test(t))return!0}return!1}function g(e,t=globalThis,n=["*"]){t.addEventListener("message",(function a(i){if(!i||!i.data)return;if(!p(n,i.origin))return void console.warn(`Invalid origin '${i.origin}' for comlink proxy`);const{id:r,type:l,path:u}=Object.assign({path:[]},i.data),c=(i.data.argumentList||[]).map(M);let d;try{const t=u.slice(0,-1).reduce((e,t)=>e[t],e),n=u.reduce((e,t)=>e[t],e);switch(l){case"GET":d=n;break;case"SET":t[u.slice(-1)[0]]=M(i.data.value),d=!0;break;case"APPLY":d=n.apply(t,c);break;case"CONSTRUCT":{const e=new n(...c);d=$(e)}break;case"ENDPOINT":{const{port1:t,port2:n}=new MessageChannel;g(e,n),d=S(t,[t])}break;case"RELEASE":d=void 0;break;default:return}}catch(h){d={value:h,[o]:0}}Promise.resolve(d).catch(e=>({value:e,[o]:0})).then(n=>{const[i,o]=z(n);t.postMessage(Object.assign(Object.assign({},i),{id:r}),o),"RELEASE"===l&&(t.removeEventListener("message",a),f(t),s in e&&"function"===typeof e[s]&&e[s]())}).catch(e=>{const[n,a]=z({value:new TypeError("Unserializable return value"),[o]:0});t.postMessage(Object.assign(Object.assign({},n),{id:r}),a)})})),t.start&&t.start()}function h(e){return"MessagePort"===e.constructor.name}function f(e){h(e)&&e.close()}function m(e,t){const n=new Map;return e.addEventListener("message",(function(e){const{data:t}=e;if(!t||!t.id)return;const a=n.get(t.id);if(a)try{a(t)}finally{n.delete(t.id)}})),_(e,n,[],t)}function y(e){if(e)throw new Error("Proxy has been released and is not useable")}function v(e){return L(e,new Map,{type:"RELEASE"}).then(()=>{f(e)})}const b=new WeakMap,k="FinalizationRegistry"in globalThis&&new FinalizationRegistry(e=>{const t=(b.get(e)||0)-1;b.set(e,t),0===t&&v(e)});function w(e,t){const n=(b.get(t)||0)+1;b.set(t,n),k&&k.register(e,t,e)}function x(e){k&&k.unregister(e)}function _(e,t,n=[],a=function(){}){let s=!1;const o=new Proxy(a,{get(a,i){if(y(s),i===r)return()=>{x(o),v(e),t.clear(),s=!0};if("then"===i){if(0===n.length)return{then:()=>o};const a=L(e,t,{type:"GET",path:n.map(e=>e.toString())}).then(M);return a.then.bind(a)}return _(e,t,[...n,i])},set(a,i,r){y(s);const[o,l]=z(r);return L(e,t,{type:"SET",path:[...n,i].map(e=>e.toString()),value:o},l).then(M)},apply(a,r,o){y(s);const l=n[n.length-1];if(l===i)return L(e,t,{type:"ENDPOINT"}).then(M);if("bind"===l)return _(e,t,n.slice(0,-1));const[u,c]=F(o);return L(e,t,{type:"APPLY",path:n.map(e=>e.toString()),argumentList:u},c).then(M)},construct(a,i){y(s);const[r,o]=F(i);return L(e,t,{type:"CONSTRUCT",path:n.map(e=>e.toString()),argumentList:r},o).then(M)}});return w(o,e),o}function E(e){return Array.prototype.concat.apply([],e)}function F(e){const t=e.map(z);return[t.map(e=>e[0]),E(t.map(e=>e[1]))]}const A=new WeakMap;function S(e,t){return A.set(e,t),e}function $(e){return Object.assign(e,{[a]:!0})}function C(e,t=globalThis,n="*"){return{postMessage:(t,a)=>e.postMessage(t,n,a),addEventListener:t.addEventListener.bind(t),removeEventListener:t.removeEventListener.bind(t)}}function z(e){for(const[t,n]of d)if(n.canHandle(e)){const[a,i]=n.serialize(e);return[{type:"HANDLER",name:t,value:a},i]}return[{type:"RAW",value:e},A.get(e)||[]]}function M(e){switch(e.type){case"HANDLER":return d.get(e.name).deserialize(e.value);case"RAW":return e.value}}function L(e,t,n,a){return new Promise(i=>{const r=T();t.set(r,i),e.start&&e.start(),e.postMessage(Object.assign({id:r},n),a)})}function T(){return new Array(4).fill(0).map(()=>Math.floor(Math.random()*Number.MAX_SAFE_INTEGER).toString(16)).join("-")}},"3f8a":function(e,t,n){"use strict";n("9e2c")},"488d":function(e,t,n){"use strict";var a=function(){var e=this;e._self._c;return e._m(0)},i=[function(){var e=this,t=e._self._c;return t("div",{staticClass:"loading-dots"},[t("div",{staticClass:"ellipsis"},[t("div"),t("div"),t("div"),t("div")])])}],r={name:"loading-dots"},s=r,o=(n("b214"),n("2877")),l=Object(o["a"])(s,a,i,!1,null,"077207e4",null);t["a"]=l.exports},"563b":function(e,t,n){var a=n("6eaa");a.__esModule&&(a=a.default),"string"===typeof a&&(a=[[e.i,a,""]]),a.locals&&(e.exports=a.locals);var i=n("499e").default;i("069f7faa",a,!0,{sourceMap:!1,shadowMode:!1})},"6eaa":function(e,t,n){var a=n("24fb");t=a(!1),t.push([e.i,".loading-dots[data-v-077207e4]{position:absolute;top:0;left:0;width:100%;height:100%;background-color:hsla(0,0%,100%,.26666666666666666)}.ellipsis[data-v-077207e4]{display:inline-block;position:absolute;top:50%;left:50%;transform:translate(-50%,-50%);width:80px;height:80px}.ellipsis div[data-v-077207e4]{position:absolute;top:33px;width:13px;height:13px;border-radius:50%;background:hsla(0,0%,40%,.8);animation-timing-function:cubic-bezier(0,1,1,0)}.ellipsis div[data-v-077207e4]:first-child{left:8px;animation:ellipsis1-077207e4 .6s infinite}.ellipsis div[data-v-077207e4]:nth-child(2){left:8px;animation:ellipsis2-077207e4 .6s infinite}.ellipsis div[data-v-077207e4]:nth-child(3){left:32px;animation:ellipsis2-077207e4 .6s infinite}.ellipsis div[data-v-077207e4]:nth-child(4){left:56px;animation:ellipsis3-077207e4 .6s infinite}@keyframes ellipsis1-077207e4{0%{transform:scale(0)}to{transform:scale(1)}}@keyframes ellipsis3-077207e4{0%{transform:scale(1)}to{transform:scale(0)}}@keyframes ellipsis2-077207e4{0%{transform:translate(0)}to{transform:translate(24px)}}",""]),e.exports=t},7407:function(e,t,n){var a=n("24fb");t=a(!1),t.push([e.i,".prism-editor-wrapper pre,.prism-editor-wrapper textarea{white-space:pre!important}.prism-editor-wrapper textarea:focus{outline:0}.prism-editor__line-numbers{-webkit-user-select:none;-moz-user-select:none;user-select:none}.prism-editor__container{overflow:unset!important;width:1000%!important}",""]),e.exports=t},8661:function(e,t,n){var a=n("2669");a.__esModule&&(a=a.default),"string"===typeof a&&(a=[[e.i,a,""]]),a.locals&&(e.exports=a.locals);var i=n("499e").default;i("769f8f1d",a,!0,{sourceMap:!1,shadowMode:!1})},"919f":function(e,t,n){var a=n("24fb");t=a(!1),t.push([e.i,".sheet[data-v-682d3448]{white-space:nowrap}.page[data-v-682d3448],.sheet[data-v-682d3448]{display:inline-block}.page[data-v-682d3448]{margin:0 1em;background-color:#f6faff}",""]),e.exports=t},"94c1":function(e,t,n){"use strict";var a=function(){var e=this,t=e._self._c;return t("div",{staticClass:"sheet"},e._l(e.documents,(function(n,a){return t("span",{key:a,staticClass:"page",domProps:{innerHTML:e._s(n)}})})),0)},i=[],r={name:"sheet-simple",props:{documents:Array},mounted(){this.bindLinks()},updated(){this.$nextTick(()=>this.bindLinks())},methods:{bindLinks(){const e=this.$el.querySelectorAll("a");for(const t of e)t.onclick=e=>this.$emit("linkClick",e,t.href.baseVal)}}},s=r,o=(n("3f8a"),n("2877")),l=Object(o["a"])(s,a,i,!1,null,"682d3448",null);t["a"]=l.exports},"9e2c":function(e,t,n){var a=n("919f");a.__esModule&&(a=a.default),"string"===typeof a&&(a=[[e.i,a,""]]),a.locals&&(e.exports=a.locals);var i=n("499e").default;i("3136d68c",a,!0,{sourceMap:!1,shadowMode:!1})},"9e5c":function(e,t,n){var a,i=n("3d15").wrap,r=n("0794");e.exports=function e(){return this instanceof e?i(r()):a||(a=i(r()))}},ac16:function(e,t,n){var a=n("24fb");t=a(!1),t.push([e.i,".prism-editor-wrapper{width:100%;height:100%;display:flex;align-items:flex-start;overflow:auto;-o-tab-size:1.5em;tab-size:1.5em;-moz-tab-size:1.5em}@media (-ms-high-contrast:active),(-ms-high-contrast:none){.prism-editor-wrapper .prism-editor__textarea{color:transparent!important}.prism-editor-wrapper .prism-editor__textarea::-moz-selection{background-color:#accef7!important;color:transparent!important}.prism-editor-wrapper .prism-editor__textarea::selection{background-color:#accef7!important;color:transparent!important}}.prism-editor-wrapper .prism-editor__container{position:relative;text-align:left;box-sizing:border-box;padding:0;overflow:hidden;width:100%}.prism-editor-wrapper .prism-editor__line-numbers{height:100%;overflow:hidden;flex-shrink:0;padding-top:4px;margin-top:0;margin-right:10px}.prism-editor-wrapper .prism-editor__line-number{text-align:right;white-space:nowrap}.prism-editor-wrapper .prism-editor__textarea{position:absolute;top:0;left:0;height:100%;width:100%;resize:none;color:inherit;overflow:hidden;-moz-osx-font-smoothing:grayscale;-webkit-font-smoothing:antialiased;-webkit-text-fill-color:transparent}.prism-editor-wrapper .prism-editor__editor,.prism-editor-wrapper .prism-editor__textarea{margin:0;border:0;background:none;box-sizing:inherit;display:inherit;font-family:inherit;font-size:inherit;font-style:inherit;font-variant-ligatures:inherit;font-weight:inherit;letter-spacing:inherit;line-height:inherit;-moz-tab-size:inherit;-o-tab-size:inherit;tab-size:inherit;text-indent:inherit;text-rendering:inherit;text-transform:inherit;white-space:pre-wrap;word-wrap:keep-all;overflow-wrap:break-word;padding:0}.prism-editor-wrapper .prism-editor__textarea--empty{-webkit-text-fill-color:inherit!important}.prism-editor-wrapper .prism-editor__editor{position:relative;pointer-events:none}",""]),e.exports=t},afdd:function(e,t,n){"use strict";n("8661")},b214:function(e,t,n){"use strict";n("563b")},b2c1:function(e,t,n){"use strict";n("0e06")},b437:function(e,t,n){var a=n("24fb");t=a(!1),t.push([e.i,"code[class*=language-],pre[class*=language-]{color:#657b83;font-family:Consolas,Monaco,Andale Mono,Ubuntu Mono,monospace;font-size:1em;text-align:left;white-space:pre;word-spacing:normal;word-break:normal;word-wrap:normal;line-height:1.5;-moz-tab-size:4;-o-tab-size:4;tab-size:4;-webkit-hyphens:none;hyphens:none}code[class*=language-]::-moz-selection,code[class*=language-] ::-moz-selection,pre[class*=language-]::-moz-selection,pre[class*=language-] ::-moz-selection{background:#073642}code[class*=language-]::selection,code[class*=language-] ::selection,pre[class*=language-]::selection,pre[class*=language-] ::selection{background:#073642}pre[class*=language-]{padding:1em;margin:.5em 0;overflow:auto;border-radius:.3em}:not(pre)>code[class*=language-],pre[class*=language-]{background-color:#fdf6e3}:not(pre)>code[class*=language-]{padding:.1em;border-radius:.3em}.token.cdata,.token.comment,.token.doctype,.token.prolog{color:#93a1a1}.token.punctuation{color:#586e75}.token.namespace{opacity:.7}.token.boolean,.token.constant,.token.deleted,.token.number,.token.property,.token.symbol,.token.tag{color:#268bd2}.token.attr-name,.token.builtin,.token.char,.token.inserted,.token.selector,.token.string,.token.url{color:#2aa198}.token.entity{color:#657b83;background:#eee8d5}.token.atrule,.token.attr-value,.token.keyword{color:#859900}.token.class-name,.token.function{color:#b58900}.token.important,.token.regex,.token.variable{color:#cb4b16}.token.bold,.token.important{font-weight:700}.token.italic{font-style:italic}.token.entity{cursor:help}",""]),e.exports=t},c197:function(e,t,n){(function(t){var n="undefined"!==typeof window?window:"undefined"!==typeof WorkerGlobalScope&&self instanceof WorkerGlobalScope?self:{},a=function(e){var t=/(?:^|\s)lang(?:uage)?-([\w-]+)(?=\s|$)/i,n=0,a={},i={manual:e.Prism&&e.Prism.manual,disableWorkerMessageHandler:e.Prism&&e.Prism.disableWorkerMessageHandler,util:{encode:function e(t){return t instanceof r?new r(t.type,e(t.content),t.alias):Array.isArray(t)?t.map(e):t.replace(/&/g,"&").replace(/</g,"<").replace(/\u00a0/g," ")},type:function(e){return Object.prototype.toString.call(e).slice(8,-1)},objId:function(e){return e["__id"]||Object.defineProperty(e,"__id",{value:++n}),e["__id"]},clone:function e(t,n){var a,r;switch(n=n||{},i.util.type(t)){case"Object":if(r=i.util.objId(t),n[r])return n[r];for(var s in a={},n[r]=a,t)t.hasOwnProperty(s)&&(a[s]=e(t[s],n));return a;case"Array":return r=i.util.objId(t),n[r]?n[r]:(a=[],n[r]=a,t.forEach((function(t,i){a[i]=e(t,n)})),a);default:return t}},getLanguage:function(e){while(e){var n=t.exec(e.className);if(n)return n[1].toLowerCase();e=e.parentElement}return"none"},setLanguage:function(e,n){e.className=e.className.replace(RegExp(t,"gi"),""),e.classList.add("language-"+n)},currentScript:function(){if("undefined"===typeof document)return null;if("currentScript"in document)return document.currentScript;try{throw new Error}catch(a){var e=(/at [^(\r\n]*\((.*):[^:]+:[^:]+\)$/i.exec(a.stack)||[])[1];if(e){var t=document.getElementsByTagName("script");for(var n in t)if(t[n].src==e)return t[n]}return null}},isActive:function(e,t,n){var a="no-"+t;while(e){var i=e.classList;if(i.contains(t))return!0;if(i.contains(a))return!1;e=e.parentElement}return!!n}},languages:{plain:a,plaintext:a,text:a,txt:a,extend:function(e,t){var n=i.util.clone(i.languages[e]);for(var a in t)n[a]=t[a];return n},insertBefore:function(e,t,n,a){a=a||i.languages;var r=a[e],s={};for(var o in r)if(r.hasOwnProperty(o)){if(o==t)for(var l in n)n.hasOwnProperty(l)&&(s[l]=n[l]);n.hasOwnProperty(o)||(s[o]=r[o])}var u=a[e];return a[e]=s,i.languages.DFS(i.languages,(function(t,n){n===u&&t!=e&&(this[t]=s)})),s},DFS:function e(t,n,a,r){r=r||{};var s=i.util.objId;for(var o in t)if(t.hasOwnProperty(o)){n.call(t,o,t[o],a||o);var l=t[o],u=i.util.type(l);"Object"!==u||r[s(l)]?"Array"!==u||r[s(l)]||(r[s(l)]=!0,e(l,n,o,r)):(r[s(l)]=!0,e(l,n,null,r))}}},plugins:{},highlightAll:function(e,t){i.highlightAllUnder(document,e,t)},highlightAllUnder:function(e,t,n){var a={callback:n,container:e,selector:'code[class*="language-"], [class*="language-"] code, code[class*="lang-"], [class*="lang-"] code'};i.hooks.run("before-highlightall",a),a.elements=Array.prototype.slice.apply(a.container.querySelectorAll(a.selector)),i.hooks.run("before-all-elements-highlight",a);for(var r,s=0;r=a.elements[s++];)i.highlightElement(r,!0===t,a.callback)},highlightElement:function(t,n,a){var r=i.util.getLanguage(t),s=i.languages[r];i.util.setLanguage(t,r);var o=t.parentElement;o&&"pre"===o.nodeName.toLowerCase()&&i.util.setLanguage(o,r);var l=t.textContent,u={element:t,language:r,grammar:s,code:l};function c(e){u.highlightedCode=e,i.hooks.run("before-insert",u),u.element.innerHTML=u.highlightedCode,i.hooks.run("after-highlight",u),i.hooks.run("complete",u),a&&a.call(u.element)}if(i.hooks.run("before-sanity-check",u),o=u.element.parentElement,o&&"pre"===o.nodeName.toLowerCase()&&!o.hasAttribute("tabindex")&&o.setAttribute("tabindex","0"),!u.code)return i.hooks.run("complete",u),void(a&&a.call(u.element));if(i.hooks.run("before-highlight",u),u.grammar)if(n&&e.Worker){var d=new Worker(i.filename);d.onmessage=function(e){c(e.data)},d.postMessage(JSON.stringify({language:u.language,code:u.code,immediateClose:!0}))}else c(i.highlight(u.code,u.grammar,u.language));else c(i.util.encode(u.code))},highlight:function(e,t,n){var a={code:e,grammar:t,language:n};if(i.hooks.run("before-tokenize",a),!a.grammar)throw new Error('The language "'+a.language+'" has no grammar.');return a.tokens=i.tokenize(a.code,a.grammar),i.hooks.run("after-tokenize",a),r.stringify(i.util.encode(a.tokens),a.language)},tokenize:function(e,t){var n=t.rest;if(n){for(var a in n)t[a]=n[a];delete t.rest}var i=new l;return u(i,i.head,e),o(e,i,t,i.head,0),d(i)},hooks:{all:{},add:function(e,t){var n=i.hooks.all;n[e]=n[e]||[],n[e].push(t)},run:function(e,t){var n=i.hooks.all[e];if(n&&n.length)for(var a,r=0;a=n[r++];)a(t)}},Token:r};function r(e,t,n,a){this.type=e,this.content=t,this.alias=n,this.length=0|(a||"").length}function s(e,t,n,a){e.lastIndex=t;var i=e.exec(n);if(i&&a&&i[1]){var r=i[1].length;i.index+=r,i[0]=i[0].slice(r)}return i}function o(e,t,n,a,l,d){for(var p in n)if(n.hasOwnProperty(p)&&n[p]){var g=n[p];g=Array.isArray(g)?g:[g];for(var h=0;h<g.length;++h){if(d&&d.cause==p+","+h)return;var f=g[h],m=f.inside,y=!!f.lookbehind,v=!!f.greedy,b=f.alias;if(v&&!f.pattern.global){var k=f.pattern.toString().match(/[imsuy]*$/)[0];f.pattern=RegExp(f.pattern.source,k+"g")}for(var w=f.pattern||f,x=a.next,_=l;x!==t.tail;_+=x.value.length,x=x.next){if(d&&_>=d.reach)break;var E=x.value;if(t.length>e.length)return;if(!(E instanceof r)){var F,A=1;if(v){if(F=s(w,_,e,y),!F||F.index>=e.length)break;var S=F.index,$=F.index+F[0].length,C=_;C+=x.value.length;while(S>=C)x=x.next,C+=x.value.length;if(C-=x.value.length,_=C,x.value instanceof r)continue;for(var z=x;z!==t.tail&&(C<$||"string"===typeof z.value);z=z.next)A++,C+=z.value.length;A--,E=e.slice(_,C),F.index-=_}else if(F=s(w,0,E,y),!F)continue;S=F.index;var M=F[0],L=E.slice(0,S),T=E.slice(S+M.length),j=_+E.length;d&&j>d.reach&&(d.reach=j);var P=x.prev;L&&(P=u(t,P,L),_+=L.length),c(t,P,A);var N=new r(p,m?i.tokenize(M,m):M,b,M);if(x=u(t,P,N),T&&u(t,x,T),A>1){var O={cause:p+","+h,reach:j};o(e,t,n,x.prev,_,O),d&&O.reach>d.reach&&(d.reach=O.reach)}}}}}}function l(){var e={value:null,prev:null,next:null},t={value:null,prev:e,next:null};e.next=t,this.head=e,this.tail=t,this.length=0}function u(e,t,n){var a=t.next,i={value:n,prev:t,next:a};return t.next=i,a.prev=i,e.length++,i}function c(e,t,n){for(var a=t.next,i=0;i<n&&a!==e.tail;i++)a=a.next;t.next=a,a.prev=t,e.length-=i}function d(e){var t=[],n=e.head.next;while(n!==e.tail)t.push(n.value),n=n.next;return t}if(e.Prism=i,r.stringify=function e(t,n){if("string"==typeof t)return t;if(Array.isArray(t)){var a="";return t.forEach((function(t){a+=e(t,n)})),a}var r={type:t.type,content:e(t.content,n),tag:"span",classes:["token",t.type],attributes:{},language:n},s=t.alias;s&&(Array.isArray(s)?Array.prototype.push.apply(r.classes,s):r.classes.push(s)),i.hooks.run("wrap",r);var o="";for(var l in r.attributes)o+=" "+l+'="'+(r.attributes[l]||"").replace(/"/g,""")+'"';return"<"+r.tag+' class="'+r.classes.join(" ")+'"'+o+">"+r.content+"</"+r.tag+">"},!e.document)return e.addEventListener?(i.disableWorkerMessageHandler||e.addEventListener("message",(function(t){var n=JSON.parse(t.data),a=n.language,r=n.code,s=n.immediateClose;e.postMessage(i.highlight(r,i.languages[a],a)),s&&e.close()}),!1),i):i;var p=i.util.currentScript();function g(){i.manual||i.highlightAll()}if(p&&(i.filename=p.src,p.hasAttribute("data-manual")&&(i.manual=!0)),!i.manual){var h=document.readyState;"loading"===h||"interactive"===h&&p&&p.defer?document.addEventListener("DOMContentLoaded",g):window.requestAnimationFrame?window.requestAnimationFrame(g):window.setTimeout(g,16)}return i}(n);
|
8 |
+
/**
|
9 |
+
* Prism: Lightweight, robust, elegant syntax highlighting
|
10 |
+
*
|
11 |
+
* @license MIT <https://opensource.org/licenses/MIT>
|
12 |
+
* @author Lea Verou <https://lea.verou.me>
|
13 |
+
* @namespace
|
14 |
+
* @public
|
15 |
+
*/e.exports&&(e.exports=a),"undefined"!==typeof t&&(t.Prism=a),a.languages.markup={comment:{pattern:/<!--(?:(?!<!--)[\s\S])*?-->/,greedy:!0},prolog:{pattern:/<\?[\s\S]+?\?>/,greedy:!0},doctype:{pattern:/<!DOCTYPE(?:[^>"'[\]]|"[^"]*"|'[^']*')+(?:\[(?:[^<"'\]]|"[^"]*"|'[^']*'|<(?!!--)|<!--(?:[^-]|-(?!->))*-->)*\]\s*)?>/i,greedy:!0,inside:{"internal-subset":{pattern:/(^[^\[]*\[)[\s\S]+(?=\]>$)/,lookbehind:!0,greedy:!0,inside:null},string:{pattern:/"[^"]*"|'[^']*'/,greedy:!0},punctuation:/^<!|>$|[[\]]/,"doctype-tag":/^DOCTYPE/i,name:/[^\s<>'"]+/}},cdata:{pattern:/<!\[CDATA\[[\s\S]*?\]\]>/i,greedy:!0},tag:{pattern:/<\/?(?!\d)[^\s>\/=$<%]+(?:\s(?:\s*[^\s>\/=]+(?:\s*=\s*(?:"[^"]*"|'[^']*'|[^\s'">=]+(?=[\s>]))|(?=[\s/>])))+)?\s*\/?>/,greedy:!0,inside:{tag:{pattern:/^<\/?[^\s>\/]+/,inside:{punctuation:/^<\/?/,namespace:/^[^\s>\/:]+:/}},"special-attr":[],"attr-value":{pattern:/=\s*(?:"[^"]*"|'[^']*'|[^\s'">=]+)/,inside:{punctuation:[{pattern:/^=/,alias:"attr-equals"},{pattern:/^(\s*)["']|["']$/,lookbehind:!0}]}},punctuation:/\/?>/,"attr-name":{pattern:/[^\s>\/]+/,inside:{namespace:/^[^\s>\/:]+:/}}}},entity:[{pattern:/&[\da-z]{1,8};/i,alias:"named-entity"},/&#x?[\da-f]{1,8};/i]},a.languages.markup["tag"].inside["attr-value"].inside["entity"]=a.languages.markup["entity"],a.languages.markup["doctype"].inside["internal-subset"].inside=a.languages.markup,a.hooks.add("wrap",(function(e){"entity"===e.type&&(e.attributes["title"]=e.content.replace(/&/,"&"))})),Object.defineProperty(a.languages.markup.tag,"addInlined",{value:function(e,t){var n={};n["language-"+t]={pattern:/(^<!\[CDATA\[)[\s\S]+?(?=\]\]>$)/i,lookbehind:!0,inside:a.languages[t]},n["cdata"]=/^<!\[CDATA\[|\]\]>$/i;var i={"included-cdata":{pattern:/<!\[CDATA\[[\s\S]*?\]\]>/i,inside:n}};i["language-"+t]={pattern:/[\s\S]+/,inside:a.languages[t]};var r={};r[e]={pattern:RegExp(/(<__[^>]*>)(?:<!\[CDATA\[(?:[^\]]|\](?!\]>))*\]\]>|(?!<!\[CDATA\[)[\s\S])*?(?=<\/__>)/.source.replace(/__/g,(function(){return e})),"i"),lookbehind:!0,greedy:!0,inside:i},a.languages.insertBefore("markup","cdata",r)}}),Object.defineProperty(a.languages.markup.tag,"addAttribute",{value:function(e,t){a.languages.markup.tag.inside["special-attr"].push({pattern:RegExp(/(^|["'\s])/.source+"(?:"+e+")"+/\s*=\s*(?:"[^"]*"|'[^']*'|[^\s'">=]+(?=[\s>]))/.source,"i"),lookbehind:!0,inside:{"attr-name":/^[^\s=]+/,"attr-value":{pattern:/=[\s\S]+/,inside:{value:{pattern:/(^=\s*(["']|(?!["'])))\S[\s\S]*(?=\2$)/,lookbehind:!0,alias:[t,"language-"+t],inside:a.languages[t]},punctuation:[{pattern:/^=/,alias:"attr-equals"},/"|'/]}}}})}}),a.languages.html=a.languages.markup,a.languages.mathml=a.languages.markup,a.languages.svg=a.languages.markup,a.languages.xml=a.languages.extend("markup",{}),a.languages.ssml=a.languages.xml,a.languages.atom=a.languages.xml,a.languages.rss=a.languages.xml,function(e){var t=/(?:"(?:\\(?:\r\n|[\s\S])|[^"\\\r\n])*"|'(?:\\(?:\r\n|[\s\S])|[^'\\\r\n])*')/;e.languages.css={comment:/\/\*[\s\S]*?\*\//,atrule:{pattern:RegExp("@[\\w-](?:"+/[^;{\s"']|\s+(?!\s)/.source+"|"+t.source+")*?"+/(?:;|(?=\s*\{))/.source),inside:{rule:/^@[\w-]+/,"selector-function-argument":{pattern:/(\bselector\s*\(\s*(?![\s)]))(?:[^()\s]|\s+(?![\s)])|\((?:[^()]|\([^()]*\))*\))+(?=\s*\))/,lookbehind:!0,alias:"selector"},keyword:{pattern:/(^|[^\w-])(?:and|not|only|or)(?![\w-])/,lookbehind:!0}}},url:{pattern:RegExp("\\burl\\((?:"+t.source+"|"+/(?:[^\\\r\n()"']|\\[\s\S])*/.source+")\\)","i"),greedy:!0,inside:{function:/^url/i,punctuation:/^\(|\)$/,string:{pattern:RegExp("^"+t.source+"$"),alias:"url"}}},selector:{pattern:RegExp("(^|[{}\\s])[^{}\\s](?:[^{};\"'\\s]|\\s+(?![\\s{])|"+t.source+")*(?=\\s*\\{)"),lookbehind:!0},string:{pattern:t,greedy:!0},property:{pattern:/(^|[^-\w\xA0-\uFFFF])(?!\s)[-_a-z\xA0-\uFFFF](?:(?!\s)[-\w\xA0-\uFFFF])*(?=\s*:)/i,lookbehind:!0},important:/!important\b/i,function:{pattern:/(^|[^-a-z0-9])[-a-z0-9]+(?=\()/i,lookbehind:!0},punctuation:/[(){};:,]/},e.languages.css["atrule"].inside.rest=e.languages.css;var n=e.languages.markup;n&&(n.tag.addInlined("style","css"),n.tag.addAttribute("style","css"))}(a),a.languages.clike={comment:[{pattern:/(^|[^\\])\/\*[\s\S]*?(?:\*\/|$)/,lookbehind:!0,greedy:!0},{pattern:/(^|[^\\:])\/\/.*/,lookbehind:!0,greedy:!0}],string:{pattern:/(["'])(?:\\(?:\r\n|[\s\S])|(?!\1)[^\\\r\n])*\1/,greedy:!0},"class-name":{pattern:/(\b(?:class|extends|implements|instanceof|interface|new|trait)\s+|\bcatch\s+\()[\w.\\]+/i,lookbehind:!0,inside:{punctuation:/[.\\]/}},keyword:/\b(?:break|catch|continue|do|else|finally|for|function|if|in|instanceof|new|null|return|throw|try|while)\b/,boolean:/\b(?:false|true)\b/,function:/\b\w+(?=\()/,number:/\b0x[\da-f]+\b|(?:\b\d+(?:\.\d*)?|\B\.\d+)(?:e[+-]?\d+)?/i,operator:/[<>]=?|[!=]=?=?|--?|\+\+?|&&?|\|\|?|[?*/~^%]/,punctuation:/[{}[\];(),.:]/},a.languages.javascript=a.languages.extend("clike",{"class-name":[a.languages.clike["class-name"],{pattern:/(^|[^$\w\xA0-\uFFFF])(?!\s)[_$A-Z\xA0-\uFFFF](?:(?!\s)[$\w\xA0-\uFFFF])*(?=\.(?:constructor|prototype))/,lookbehind:!0}],keyword:[{pattern:/((?:^|\})\s*)catch\b/,lookbehind:!0},{pattern:/(^|[^.]|\.\.\.\s*)\b(?:as|assert(?=\s*\{)|async(?=\s*(?:function\b|\(|[$\w\xA0-\uFFFF]|$))|await|break|case|class|const|continue|debugger|default|delete|do|else|enum|export|extends|finally(?=\s*(?:\{|$))|for|from(?=\s*(?:['"]|$))|function|(?:get|set)(?=\s*(?:[#\[$\w\xA0-\uFFFF]|$))|if|implements|import|in|instanceof|interface|let|new|null|of|package|private|protected|public|return|static|super|switch|this|throw|try|typeof|undefined|var|void|while|with|yield)\b/,lookbehind:!0}],function:/#?(?!\s)[_$a-zA-Z\xA0-\uFFFF](?:(?!\s)[$\w\xA0-\uFFFF])*(?=\s*(?:\.\s*(?:apply|bind|call)\s*)?\()/,number:{pattern:RegExp(/(^|[^\w$])/.source+"(?:"+/NaN|Infinity/.source+"|"+/0[bB][01]+(?:_[01]+)*n?/.source+"|"+/0[oO][0-7]+(?:_[0-7]+)*n?/.source+"|"+/0[xX][\dA-Fa-f]+(?:_[\dA-Fa-f]+)*n?/.source+"|"+/\d+(?:_\d+)*n/.source+"|"+/(?:\d+(?:_\d+)*(?:\.(?:\d+(?:_\d+)*)?)?|\.\d+(?:_\d+)*)(?:[Ee][+-]?\d+(?:_\d+)*)?/.source+")"+/(?![\w$])/.source),lookbehind:!0},operator:/--|\+\+|\*\*=?|=>|&&=?|\|\|=?|[!=]==|<<=?|>>>?=?|[-+*/%&|^!=<>]=?|\.{3}|\?\?=?|\?\.?|[~:]/}),a.languages.javascript["class-name"][0].pattern=/(\b(?:class|extends|implements|instanceof|interface|new)\s+)[\w.\\]+/,a.languages.insertBefore("javascript","keyword",{regex:{pattern:RegExp(/((?:^|[^$\w\xA0-\uFFFF."'\])\s]|\b(?:return|yield))\s*)/.source+/\//.source+"(?:"+/(?:\[(?:[^\]\\\r\n]|\\.)*\]|\\.|[^/\\\[\r\n])+\/[dgimyus]{0,7}/.source+"|"+/(?:\[(?:[^[\]\\\r\n]|\\.|\[(?:[^[\]\\\r\n]|\\.|\[(?:[^[\]\\\r\n]|\\.)*\])*\])*\]|\\.|[^/\\\[\r\n])+\/[dgimyus]{0,7}v[dgimyus]{0,7}/.source+")"+/(?=(?:\s|\/\*(?:[^*]|\*(?!\/))*\*\/)*(?:$|[\r\n,.;:})\]]|\/\/))/.source),lookbehind:!0,greedy:!0,inside:{"regex-source":{pattern:/^(\/)[\s\S]+(?=\/[a-z]*$)/,lookbehind:!0,alias:"language-regex",inside:a.languages.regex},"regex-delimiter":/^\/|\/$/,"regex-flags":/^[a-z]+$/}},"function-variable":{pattern:/#?(?!\s)[_$a-zA-Z\xA0-\uFFFF](?:(?!\s)[$\w\xA0-\uFFFF])*(?=\s*[=:]\s*(?:async\s*)?(?:\bfunction\b|(?:\((?:[^()]|\([^()]*\))*\)|(?!\s)[_$a-zA-Z\xA0-\uFFFF](?:(?!\s)[$\w\xA0-\uFFFF])*)\s*=>))/,alias:"function"},parameter:[{pattern:/(function(?:\s+(?!\s)[_$a-zA-Z\xA0-\uFFFF](?:(?!\s)[$\w\xA0-\uFFFF])*)?\s*\(\s*)(?!\s)(?:[^()\s]|\s+(?![\s)])|\([^()]*\))+(?=\s*\))/,lookbehind:!0,inside:a.languages.javascript},{pattern:/(^|[^$\w\xA0-\uFFFF])(?!\s)[_$a-z\xA0-\uFFFF](?:(?!\s)[$\w\xA0-\uFFFF])*(?=\s*=>)/i,lookbehind:!0,inside:a.languages.javascript},{pattern:/(\(\s*)(?!\s)(?:[^()\s]|\s+(?![\s)])|\([^()]*\))+(?=\s*\)\s*=>)/,lookbehind:!0,inside:a.languages.javascript},{pattern:/((?:\b|\s|^)(?!(?:as|async|await|break|case|catch|class|const|continue|debugger|default|delete|do|else|enum|export|extends|finally|for|from|function|get|if|implements|import|in|instanceof|interface|let|new|null|of|package|private|protected|public|return|set|static|super|switch|this|throw|try|typeof|undefined|var|void|while|with|yield)(?![$\w\xA0-\uFFFF]))(?:(?!\s)[_$a-zA-Z\xA0-\uFFFF](?:(?!\s)[$\w\xA0-\uFFFF])*\s*)\(\s*|\]\s*\(\s*)(?!\s)(?:[^()\s]|\s+(?![\s)])|\([^()]*\))+(?=\s*\)\s*\{)/,lookbehind:!0,inside:a.languages.javascript}],constant:/\b[A-Z](?:[A-Z_]|\dx?)*\b/}),a.languages.insertBefore("javascript","string",{hashbang:{pattern:/^#!.*/,greedy:!0,alias:"comment"},"template-string":{pattern:/`(?:\\[\s\S]|\$\{(?:[^{}]|\{(?:[^{}]|\{[^}]*\})*\})+\}|(?!\$\{)[^\\`])*`/,greedy:!0,inside:{"template-punctuation":{pattern:/^`|`$/,alias:"string"},interpolation:{pattern:/((?:^|[^\\])(?:\\{2})*)\$\{(?:[^{}]|\{(?:[^{}]|\{[^}]*\})*\})+\}/,lookbehind:!0,inside:{"interpolation-punctuation":{pattern:/^\$\{|\}$/,alias:"punctuation"},rest:a.languages.javascript}},string:/[\s\S]+/}},"string-property":{pattern:/((?:^|[,{])[ \t]*)(["'])(?:\\(?:\r\n|[\s\S])|(?!\2)[^\\\r\n])*\2(?=\s*:)/m,lookbehind:!0,greedy:!0,alias:"property"}}),a.languages.insertBefore("javascript","operator",{"literal-property":{pattern:/((?:^|[,{])[ \t]*)(?!\s)[_$a-zA-Z\xA0-\uFFFF](?:(?!\s)[$\w\xA0-\uFFFF])*(?=\s*:)/m,lookbehind:!0,alias:"property"}}),a.languages.markup&&(a.languages.markup.tag.addInlined("script","javascript"),a.languages.markup.tag.addAttribute(/on(?:abort|blur|change|click|composition(?:end|start|update)|dblclick|error|focus(?:in|out)?|key(?:down|up)|load|mouse(?:down|enter|leave|move|out|over|up)|reset|resize|scroll|select|slotchange|submit|unload|wheel)/.source,"javascript")),a.languages.js=a.languages.javascript,function(){if("undefined"!==typeof a&&"undefined"!==typeof document){Element.prototype.matches||(Element.prototype.matches=Element.prototype.msMatchesSelector||Element.prototype.webkitMatchesSelector);var e="Loading…",t=function(e,t){return"✖ Error "+e+" while fetching file: "+t},n="✖ Error: File does not exist or is empty",i={js:"javascript",py:"python",rb:"ruby",ps1:"powershell",psm1:"powershell",sh:"bash",bat:"batch",h:"c",tex:"latex"},r="data-src-status",s="loading",o="loaded",l="failed",u="pre[data-src]:not(["+r+'="'+o+'"]):not(['+r+'="'+s+'"])';a.hooks.add("before-highlightall",(function(e){e.selector+=", "+u})),a.hooks.add("before-sanity-check",(function(t){var n=t.element;if(n.matches(u)){t.code="",n.setAttribute(r,s);var c=n.appendChild(document.createElement("CODE"));c.textContent=e;var g=n.getAttribute("data-src"),h=t.language;if("none"===h){var f=(/\.(\w+)$/.exec(g)||[,"none"])[1];h=i[f]||f}a.util.setLanguage(c,h),a.util.setLanguage(n,h);var m=a.plugins.autoloader;m&&m.loadLanguages(h),d(g,(function(e){n.setAttribute(r,o);var t=p(n.getAttribute("data-range"));if(t){var i=e.split(/\r\n?|\n/g),s=t[0],l=null==t[1]?i.length:t[1];s<0&&(s+=i.length),s=Math.max(0,Math.min(s-1,i.length)),l<0&&(l+=i.length),l=Math.max(0,Math.min(l,i.length)),e=i.slice(s,l).join("\n"),n.hasAttribute("data-start")||n.setAttribute("data-start",String(s+1))}c.textContent=e,a.highlightElement(c)}),(function(e){n.setAttribute(r,l),c.textContent=e}))}})),a.plugins.fileHighlight={highlight:function(e){for(var t,n=(e||document).querySelectorAll(u),i=0;t=n[i++];)a.highlightElement(t)}};var c=!1;a.fileHighlight=function(){c||(console.warn("Prism.fileHighlight is deprecated. Use `Prism.plugins.fileHighlight.highlight` instead."),c=!0),a.plugins.fileHighlight.highlight.apply(this,arguments)}}function d(e,a,i){var r=new XMLHttpRequest;r.open("GET",e,!0),r.onreadystatechange=function(){4==r.readyState&&(r.status<400&&r.responseText?a(r.responseText):r.status>=400?i(t(r.status,r.statusText)):i(n))},r.send(null)}function p(e){var t=/^\s*(\d+)\s*(?:(,)\s*(?:(\d+)\s*)?)?$/.exec(e||"");if(t){var n=Number(t[1]),a=t[2],i=t[3];return a?i?[n,Number(i)]:[n,void 0]:[n,n]}}}()}).call(this,n("c8ba"))},cabf:function(e,t,n){var a=n("ac16");a.__esModule&&(a=a.default),"string"===typeof a&&(a=[[e.i,a,""]]),a.locals&&(e.exports=a.locals);var i=n("499e").default;i("3072be6e",a,!0,{sourceMap:!1,shadowMode:!1})}}]);
|
16 |
+
//# sourceMappingURL=chunk-0c4e36c8.95d70738.js.map
|
dist/js/chunk-0cbfe13e.73856287.js
ADDED
@@ -0,0 +1,2 @@
|
|
|
|
|
|
|
1 |
+
(window["webpackJsonp"]=window["webpackJsonp"]||[]).push([["chunk-0cbfe13e"],{"00ce":function(e,t,r){"use strict";var o,n=r("a645"),i=r("417f"),a=r("dc99"),s=r("1409"),l=r("67ee"),c=r("0d25"),u=r("67d9"),p=Function,f=function(e){try{return p('"use strict"; return ('+e+").constructor;")()}catch(t){}},h=Object.getOwnPropertyDescriptor;if(h)try{h({},"")}catch(L){h=null}var y=function(){throw new c},d=h?function(){try{return y}catch(e){try{return h(arguments,"callee").get}catch(t){return y}}}():y,m=r("5156")(),g=r("0a36")(),b=Object.getPrototypeOf||(g?function(e){return e.__proto__}:null),v={},w="undefined"!==typeof Uint8Array&&b?b(Uint8Array):o,S={__proto__:null,"%AggregateError%":"undefined"===typeof AggregateError?o:AggregateError,"%Array%":Array,"%ArrayBuffer%":"undefined"===typeof ArrayBuffer?o:ArrayBuffer,"%ArrayIteratorPrototype%":m&&b?b([][Symbol.iterator]()):o,"%AsyncFromSyncIteratorPrototype%":o,"%AsyncFunction%":v,"%AsyncGenerator%":v,"%AsyncGeneratorFunction%":v,"%AsyncIteratorPrototype%":v,"%Atomics%":"undefined"===typeof Atomics?o:Atomics,"%BigInt%":"undefined"===typeof BigInt?o:BigInt,"%BigInt64Array%":"undefined"===typeof BigInt64Array?o:BigInt64Array,"%BigUint64Array%":"undefined"===typeof BigUint64Array?o:BigUint64Array,"%Boolean%":Boolean,"%DataView%":"undefined"===typeof DataView?o:DataView,"%Date%":Date,"%decodeURI%":decodeURI,"%decodeURIComponent%":decodeURIComponent,"%encodeURI%":encodeURI,"%encodeURIComponent%":encodeURIComponent,"%Error%":n,"%eval%":eval,"%EvalError%":i,"%Float32Array%":"undefined"===typeof Float32Array?o:Float32Array,"%Float64Array%":"undefined"===typeof Float64Array?o:Float64Array,"%FinalizationRegistry%":"undefined"===typeof FinalizationRegistry?o:FinalizationRegistry,"%Function%":p,"%GeneratorFunction%":v,"%Int8Array%":"undefined"===typeof Int8Array?o:Int8Array,"%Int16Array%":"undefined"===typeof Int16Array?o:Int16Array,"%Int32Array%":"undefined"===typeof Int32Array?o:Int32Array,"%isFinite%":isFinite,"%isNaN%":isNaN,"%IteratorPrototype%":m&&b?b(b([][Symbol.iterator]())):o,"%JSON%":"object"===typeof JSON?JSON:o,"%Map%":"undefined"===typeof Map?o:Map,"%MapIteratorPrototype%":"undefined"!==typeof Map&&m&&b?b((new Map)[Symbol.iterator]()):o,"%Math%":Math,"%Number%":Number,"%Object%":Object,"%parseFloat%":parseFloat,"%parseInt%":parseInt,"%Promise%":"undefined"===typeof Promise?o:Promise,"%Proxy%":"undefined"===typeof Proxy?o:Proxy,"%RangeError%":a,"%ReferenceError%":s,"%Reflect%":"undefined"===typeof Reflect?o:Reflect,"%RegExp%":RegExp,"%Set%":"undefined"===typeof Set?o:Set,"%SetIteratorPrototype%":"undefined"!==typeof Set&&m&&b?b((new Set)[Symbol.iterator]()):o,"%SharedArrayBuffer%":"undefined"===typeof SharedArrayBuffer?o:SharedArrayBuffer,"%String%":String,"%StringIteratorPrototype%":m&&b?b(""[Symbol.iterator]()):o,"%Symbol%":m?Symbol:o,"%SyntaxError%":l,"%ThrowTypeError%":d,"%TypedArray%":w,"%TypeError%":c,"%Uint8Array%":"undefined"===typeof Uint8Array?o:Uint8Array,"%Uint8ClampedArray%":"undefined"===typeof Uint8ClampedArray?o:Uint8ClampedArray,"%Uint16Array%":"undefined"===typeof Uint16Array?o:Uint16Array,"%Uint32Array%":"undefined"===typeof Uint32Array?o:Uint32Array,"%URIError%":u,"%WeakMap%":"undefined"===typeof WeakMap?o:WeakMap,"%WeakRef%":"undefined"===typeof WeakRef?o:WeakRef,"%WeakSet%":"undefined"===typeof WeakSet?o:WeakSet};if(b)try{null.error}catch(L){var x=b(b(L));S["%Error.prototype%"]=x}var O=function e(t){var r;if("%AsyncFunction%"===t)r=f("async function () {}");else if("%GeneratorFunction%"===t)r=f("function* () {}");else if("%AsyncGeneratorFunction%"===t)r=f("async function* () {}");else if("%AsyncGenerator%"===t){var o=e("%AsyncGeneratorFunction%");o&&(r=o.prototype)}else if("%AsyncIteratorPrototype%"===t){var n=e("%AsyncGenerator%");n&&b&&(r=b(n.prototype))}return S[t]=r,r},j={__proto__:null,"%ArrayBufferPrototype%":["ArrayBuffer","prototype"],"%ArrayPrototype%":["Array","prototype"],"%ArrayProto_entries%":["Array","prototype","entries"],"%ArrayProto_forEach%":["Array","prototype","forEach"],"%ArrayProto_keys%":["Array","prototype","keys"],"%ArrayProto_values%":["Array","prototype","values"],"%AsyncFunctionPrototype%":["AsyncFunction","prototype"],"%AsyncGenerator%":["AsyncGeneratorFunction","prototype"],"%AsyncGeneratorPrototype%":["AsyncGeneratorFunction","prototype","prototype"],"%BooleanPrototype%":["Boolean","prototype"],"%DataViewPrototype%":["DataView","prototype"],"%DatePrototype%":["Date","prototype"],"%ErrorPrototype%":["Error","prototype"],"%EvalErrorPrototype%":["EvalError","prototype"],"%Float32ArrayPrototype%":["Float32Array","prototype"],"%Float64ArrayPrototype%":["Float64Array","prototype"],"%FunctionPrototype%":["Function","prototype"],"%Generator%":["GeneratorFunction","prototype"],"%GeneratorPrototype%":["GeneratorFunction","prototype","prototype"],"%Int8ArrayPrototype%":["Int8Array","prototype"],"%Int16ArrayPrototype%":["Int16Array","prototype"],"%Int32ArrayPrototype%":["Int32Array","prototype"],"%JSONParse%":["JSON","parse"],"%JSONStringify%":["JSON","stringify"],"%MapPrototype%":["Map","prototype"],"%NumberPrototype%":["Number","prototype"],"%ObjectPrototype%":["Object","prototype"],"%ObjProto_toString%":["Object","prototype","toString"],"%ObjProto_valueOf%":["Object","prototype","valueOf"],"%PromisePrototype%":["Promise","prototype"],"%PromiseProto_then%":["Promise","prototype","then"],"%Promise_all%":["Promise","all"],"%Promise_reject%":["Promise","reject"],"%Promise_resolve%":["Promise","resolve"],"%RangeErrorPrototype%":["RangeError","prototype"],"%ReferenceErrorPrototype%":["ReferenceError","prototype"],"%RegExpPrototype%":["RegExp","prototype"],"%SetPrototype%":["Set","prototype"],"%SharedArrayBufferPrototype%":["SharedArrayBuffer","prototype"],"%StringPrototype%":["String","prototype"],"%SymbolPrototype%":["Symbol","prototype"],"%SyntaxErrorPrototype%":["SyntaxError","prototype"],"%TypedArrayPrototype%":["TypedArray","prototype"],"%TypeErrorPrototype%":["TypeError","prototype"],"%Uint8ArrayPrototype%":["Uint8Array","prototype"],"%Uint8ClampedArrayPrototype%":["Uint8ClampedArray","prototype"],"%Uint16ArrayPrototype%":["Uint16Array","prototype"],"%Uint32ArrayPrototype%":["Uint32Array","prototype"],"%URIErrorPrototype%":["URIError","prototype"],"%WeakMapPrototype%":["WeakMap","prototype"],"%WeakSetPrototype%":["WeakSet","prototype"]},A=r("0f7c"),P=r("9671"),k=A.call(Function.call,Array.prototype.concat),E=A.call(Function.apply,Array.prototype.splice),I=A.call(Function.call,String.prototype.replace),_=A.call(Function.call,String.prototype.slice),T=A.call(Function.call,RegExp.prototype.exec),N=/[^%.[\]]+|\[(?:(-?\d+(?:\.\d+)?)|(["'])((?:(?!\2)[^\\]|\\.)*?)\2)\]|(?=(?:\.|\[\])(?:\.|\[\]|%$))/g,C=/\\(\\)?/g,F=function(e){var t=_(e,0,1),r=_(e,-1);if("%"===t&&"%"!==r)throw new l("invalid intrinsic syntax, expected closing `%`");if("%"===r&&"%"!==t)throw new l("invalid intrinsic syntax, expected opening `%`");var o=[];return I(e,N,(function(e,t,r,n){o[o.length]=r?I(n,C,"$1"):t||e})),o},R=function(e,t){var r,o=e;if(P(j,o)&&(r=j[o],o="%"+r[0]+"%"),P(S,o)){var n=S[o];if(n===v&&(n=O(o)),"undefined"===typeof n&&!t)throw new c("intrinsic "+e+" exists, but is not available. Please file an issue!");return{alias:r,name:o,value:n}}throw new l("intrinsic "+e+" does not exist!")};e.exports=function(e,t){if("string"!==typeof e||0===e.length)throw new c("intrinsic name must be a non-empty string");if(arguments.length>1&&"boolean"!==typeof t)throw new c('"allowMissing" argument must be a boolean');if(null===T(/^%?[^%]*%?$/,e))throw new l("`%` may not be present anywhere but at the beginning and end of the intrinsic name");var r=F(e),o=r.length>0?r[0]:"",n=R("%"+o+"%",t),i=n.name,a=n.value,s=!1,u=n.alias;u&&(o=u[0],E(r,k([0,1],u)));for(var p=1,f=!0;p<r.length;p+=1){var y=r[p],d=_(y,0,1),m=_(y,-1);if(('"'===d||"'"===d||"`"===d||'"'===m||"'"===m||"`"===m)&&d!==m)throw new l("property names with quotes must have matching quotes");if("constructor"!==y&&f||(s=!0),o+="."+y,i="%"+o+"%",P(S,i))a=S[i];else if(null!=a){if(!(y in a)){if(!t)throw new c("base intrinsic for "+e+" exists, but the property is not available.");return}if(h&&p+1>=r.length){var g=h(a,y);f=!!g,a=f&&"get"in g&&!("originalValue"in g.get)?g.get:a[y]}else f=P(a,y),a=a[y];f&&!s&&(S[i]=a)}}return a}},"0a36":function(e,t,r){"use strict";var o={__proto__:null,foo:{}},n=Object;e.exports=function(){return{__proto__:o}.foo===o.foo&&!(o instanceof n)}},"0b16":function(e,t,r){"use strict";var o=r("9d88");function n(){this.protocol=null,this.slashes=null,this.auth=null,this.host=null,this.port=null,this.hostname=null,this.hash=null,this.search=null,this.query=null,this.pathname=null,this.path=null,this.href=null}var i=/^([a-z0-9.+-]+:)/i,a=/:[0-9]*$/,s=/^(\/\/?(?!\/)[^?\s]*)(\?[^\s]*)?$/,l=["<",">",'"',"`"," ","\r","\n","\t"],c=["{","}","|","\\","^","`"].concat(l),u=["'"].concat(c),p=["%","/","?",";","#"].concat(u),f=["/","?","#"],h=255,y=/^[+a-z0-9A-Z_-]{0,63}$/,d=/^([+a-z0-9A-Z_-]{0,63})(.*)$/,m={javascript:!0,"javascript:":!0},g={javascript:!0,"javascript:":!0},b={http:!0,https:!0,ftp:!0,gopher:!0,file:!0,"http:":!0,"https:":!0,"ftp:":!0,"gopher:":!0,"file:":!0},v=r("184d");function w(e,t,r){if(e&&"object"===typeof e&&e instanceof n)return e;var o=new n;return o.parse(e,t,r),o}function S(e){return"string"===typeof e&&(e=w(e)),e instanceof n?e.format():n.prototype.format.call(e)}function x(e,t){return w(e,!1,!0).resolve(t)}function O(e,t){return e?w(e,!1,!0).resolveObject(t):t}n.prototype.parse=function(e,t,r){if("string"!==typeof e)throw new TypeError("Parameter 'url' must be a string, not "+typeof e);var n=e.indexOf("?"),a=-1!==n&&n<e.indexOf("#")?"?":"#",l=e.split(a),c=/\\/g;l[0]=l[0].replace(c,"/"),e=l.join(a);var w=e;if(w=w.trim(),!r&&1===e.split("#").length){var S=s.exec(w);if(S)return this.path=w,this.href=w,this.pathname=S[1],S[2]?(this.search=S[2],this.query=t?v.parse(this.search.substr(1)):this.search.substr(1)):t&&(this.search="",this.query={}),this}var x=i.exec(w);if(x){x=x[0];var O=x.toLowerCase();this.protocol=O,w=w.substr(x.length)}if(r||x||w.match(/^\/\/[^@/]+@[^@/]+/)){var j="//"===w.substr(0,2);!j||x&&g[x]||(w=w.substr(2),this.slashes=!0)}if(!g[x]&&(j||x&&!b[x])){for(var A,P,k=-1,E=0;E<f.length;E++){var I=w.indexOf(f[E]);-1!==I&&(-1===k||I<k)&&(k=I)}P=-1===k?w.lastIndexOf("@"):w.lastIndexOf("@",k),-1!==P&&(A=w.slice(0,P),w=w.slice(P+1),this.auth=decodeURIComponent(A)),k=-1;for(E=0;E<p.length;E++){I=w.indexOf(p[E]);-1!==I&&(-1===k||I<k)&&(k=I)}-1===k&&(k=w.length),this.host=w.slice(0,k),w=w.slice(k),this.parseHost(),this.hostname=this.hostname||"";var _="["===this.hostname[0]&&"]"===this.hostname[this.hostname.length-1];if(!_)for(var T=this.hostname.split(/\./),N=(E=0,T.length);E<N;E++){var C=T[E];if(C&&!C.match(y)){for(var F="",R=0,L=C.length;R<L;R++)C.charCodeAt(R)>127?F+="x":F+=C[R];if(!F.match(y)){var D=T.slice(0,E),M=T.slice(E+1),U=C.match(d);U&&(D.push(U[1]),M.unshift(U[2])),M.length&&(w="/"+M.join(".")+w),this.hostname=D.join(".");break}}}this.hostname.length>h?this.hostname="":this.hostname=this.hostname.toLowerCase(),_||(this.hostname=o.toASCII(this.hostname));var B=this.port?":"+this.port:"",W=this.hostname||"";this.host=W+B,this.href+=this.host,_&&(this.hostname=this.hostname.substr(1,this.hostname.length-2),"/"!==w[0]&&(w="/"+w))}if(!m[O])for(E=0,N=u.length;E<N;E++){var q=u[E];if(-1!==w.indexOf(q)){var $=encodeURIComponent(q);$===q&&($=escape(q)),w=w.split(q).join($)}}var H=w.indexOf("#");-1!==H&&(this.hash=w.substr(H),w=w.slice(0,H));var G=w.indexOf("?");if(-1!==G?(this.search=w.substr(G),this.query=w.substr(G+1),t&&(this.query=v.parse(this.query)),w=w.slice(0,G)):t&&(this.search="",this.query={}),w&&(this.pathname=w),b[O]&&this.hostname&&!this.pathname&&(this.pathname="/"),this.pathname||this.search){B=this.pathname||"";var J=this.search||"";this.path=B+J}return this.href=this.format(),this},n.prototype.format=function(){var e=this.auth||"";e&&(e=encodeURIComponent(e),e=e.replace(/%3A/i,":"),e+="@");var t=this.protocol||"",r=this.pathname||"",o=this.hash||"",n=!1,i="";this.host?n=e+this.host:this.hostname&&(n=e+(-1===this.hostname.indexOf(":")?this.hostname:"["+this.hostname+"]"),this.port&&(n+=":"+this.port)),this.query&&"object"===typeof this.query&&Object.keys(this.query).length&&(i=v.stringify(this.query,{arrayFormat:"repeat",addQueryPrefix:!1}));var a=this.search||i&&"?"+i||"";return t&&":"!==t.substr(-1)&&(t+=":"),this.slashes||(!t||b[t])&&!1!==n?(n="//"+(n||""),r&&"/"!==r.charAt(0)&&(r="/"+r)):n||(n=""),o&&"#"!==o.charAt(0)&&(o="#"+o),a&&"?"!==a.charAt(0)&&(a="?"+a),r=r.replace(/[?#]/g,(function(e){return encodeURIComponent(e)})),a=a.replace("#","%23"),t+n+r+a+o},n.prototype.resolve=function(e){return this.resolveObject(w(e,!1,!0)).format()},n.prototype.resolveObject=function(e){if("string"===typeof e){var t=new n;t.parse(e,!1,!0),e=t}for(var r=new n,o=Object.keys(this),i=0;i<o.length;i++){var a=o[i];r[a]=this[a]}if(r.hash=e.hash,""===e.href)return r.href=r.format(),r;if(e.slashes&&!e.protocol){for(var s=Object.keys(e),l=0;l<s.length;l++){var c=s[l];"protocol"!==c&&(r[c]=e[c])}return b[r.protocol]&&r.hostname&&!r.pathname&&(r.pathname="/",r.path=r.pathname),r.href=r.format(),r}if(e.protocol&&e.protocol!==r.protocol){if(!b[e.protocol]){for(var u=Object.keys(e),p=0;p<u.length;p++){var f=u[p];r[f]=e[f]}return r.href=r.format(),r}if(r.protocol=e.protocol,e.host||g[e.protocol])r.pathname=e.pathname;else{var h=(e.pathname||"").split("/");while(h.length&&!(e.host=h.shift()));e.host||(e.host=""),e.hostname||(e.hostname=""),""!==h[0]&&h.unshift(""),h.length<2&&h.unshift(""),r.pathname=h.join("/")}if(r.search=e.search,r.query=e.query,r.host=e.host||"",r.auth=e.auth,r.hostname=e.hostname||e.host,r.port=e.port,r.pathname||r.search){var y=r.pathname||"",d=r.search||"";r.path=y+d}return r.slashes=r.slashes||e.slashes,r.href=r.format(),r}var m=r.pathname&&"/"===r.pathname.charAt(0),v=e.host||e.pathname&&"/"===e.pathname.charAt(0),w=v||m||r.host&&e.pathname,S=w,x=r.pathname&&r.pathname.split("/")||[],O=(h=e.pathname&&e.pathname.split("/")||[],r.protocol&&!b[r.protocol]);if(O&&(r.hostname="",r.port=null,r.host&&(""===x[0]?x[0]=r.host:x.unshift(r.host)),r.host="",e.protocol&&(e.hostname=null,e.port=null,e.host&&(""===h[0]?h[0]=e.host:h.unshift(e.host)),e.host=null),w=w&&(""===h[0]||""===x[0])),v)r.host=e.host||""===e.host?e.host:r.host,r.hostname=e.hostname||""===e.hostname?e.hostname:r.hostname,r.search=e.search,r.query=e.query,x=h;else if(h.length)x||(x=[]),x.pop(),x=x.concat(h),r.search=e.search,r.query=e.query;else if(null!=e.search){if(O){r.host=x.shift(),r.hostname=r.host;var j=!!(r.host&&r.host.indexOf("@")>0)&&r.host.split("@");j&&(r.auth=j.shift(),r.hostname=j.shift(),r.host=r.hostname)}return r.search=e.search,r.query=e.query,null===r.pathname&&null===r.search||(r.path=(r.pathname?r.pathname:"")+(r.search?r.search:"")),r.href=r.format(),r}if(!x.length)return r.pathname=null,r.search?r.path="/"+r.search:r.path=null,r.href=r.format(),r;for(var A=x.slice(-1)[0],P=(r.host||e.host||x.length>1)&&("."===A||".."===A)||""===A,k=0,E=x.length;E>=0;E--)A=x[E],"."===A?x.splice(E,1):".."===A?(x.splice(E,1),k++):k&&(x.splice(E,1),k--);if(!w&&!S)for(;k--;k)x.unshift("..");!w||""===x[0]||x[0]&&"/"===x[0].charAt(0)||x.unshift(""),P&&"/"!==x.join("/").substr(-1)&&x.push("");var I=""===x[0]||x[0]&&"/"===x[0].charAt(0);if(O){r.hostname=I?"":x.length?x.shift():"",r.host=r.hostname;j=!!(r.host&&r.host.indexOf("@")>0)&&r.host.split("@");j&&(r.auth=j.shift(),r.hostname=j.shift(),r.host=r.hostname)}return w=w||r.host&&x.length,w&&!I&&x.unshift(""),x.length>0?r.pathname=x.join("/"):(r.pathname=null,r.path=null),null===r.pathname&&null===r.search||(r.path=(r.pathname?r.pathname:"")+(r.search?r.search:"")),r.auth=e.auth||r.auth,r.slashes=r.slashes||e.slashes,r.href=r.format(),r},n.prototype.parseHost=function(){var e=this.host,t=a.exec(e);t&&(t=t[0],":"!==t&&(this.port=t.substr(1)),e=e.substr(0,e.length-t.length)),e&&(this.hostname=e)},t.parse=w,t.resolve=x,t.resolveObject=O,t.format=S,t.Url=n},"0d25":function(e,t,r){"use strict";e.exports=TypeError},"0f7c":function(e,t,r){"use strict";var o=r("688e");e.exports=Function.prototype.bind||o},1:function(e,t){},1409:function(e,t,r){"use strict";e.exports=ReferenceError},1696:function(e,t,r){"use strict";e.exports=function(){if("function"!==typeof Symbol||"function"!==typeof Object.getOwnPropertySymbols)return!1;if("symbol"===typeof Symbol.iterator)return!0;var e={},t=Symbol("test"),r=Object(t);if("string"===typeof t)return!1;if("[object Symbol]"!==Object.prototype.toString.call(t))return!1;if("[object Symbol]"!==Object.prototype.toString.call(r))return!1;var o=42;for(t in e[t]=o,e)return!1;if("function"===typeof Object.keys&&0!==Object.keys(e).length)return!1;if("function"===typeof Object.getOwnPropertyNames&&0!==Object.getOwnPropertyNames(e).length)return!1;var n=Object.getOwnPropertySymbols(e);if(1!==n.length||n[0]!==t)return!1;if(!Object.prototype.propertyIsEnumerable.call(e,t))return!1;if("function"===typeof Object.getOwnPropertyDescriptor){var i=Object.getOwnPropertyDescriptor(e,t);if(i.value!==o||!0!==i.enumerable)return!1}return!0}},"184d":function(e,t,r){"use strict";var o=r("f177"),n=r("2500"),i=r("bbc7");e.exports={formats:i,parse:n,stringify:o}},2500:function(e,t,r){"use strict";var o=r("a29f"),n=Object.prototype.hasOwnProperty,i=Array.isArray,a={allowDots:!1,allowEmptyArrays:!1,allowPrototypes:!1,allowSparse:!1,arrayLimit:20,charset:"utf-8",charsetSentinel:!1,comma:!1,decodeDotInKeys:!1,decoder:o.decode,delimiter:"&",depth:5,duplicates:"combine",ignoreQueryPrefix:!1,interpretNumericEntities:!1,parameterLimit:1e3,parseArrays:!0,plainObjects:!1,strictDepth:!1,strictNullHandling:!1},s=function(e){return e.replace(/&#(\d+);/g,(function(e,t){return String.fromCharCode(parseInt(t,10))}))},l=function(e,t){return e&&"string"===typeof e&&t.comma&&e.indexOf(",")>-1?e.split(","):e},c="utf8=%26%2310003%3B",u="utf8=%E2%9C%93",p=function(e,t){var r={__proto__:null},p=t.ignoreQueryPrefix?e.replace(/^\?/,""):e;p=p.replace(/%5B/gi,"[").replace(/%5D/gi,"]");var f,h=t.parameterLimit===1/0?void 0:t.parameterLimit,y=p.split(t.delimiter,h),d=-1,m=t.charset;if(t.charsetSentinel)for(f=0;f<y.length;++f)0===y[f].indexOf("utf8=")&&(y[f]===u?m="utf-8":y[f]===c&&(m="iso-8859-1"),d=f,f=y.length);for(f=0;f<y.length;++f)if(f!==d){var g,b,v=y[f],w=v.indexOf("]="),S=-1===w?v.indexOf("="):w+1;-1===S?(g=t.decoder(v,a.decoder,m,"key"),b=t.strictNullHandling?null:""):(g=t.decoder(v.slice(0,S),a.decoder,m,"key"),b=o.maybeMap(l(v.slice(S+1),t),(function(e){return t.decoder(e,a.decoder,m,"value")}))),b&&t.interpretNumericEntities&&"iso-8859-1"===m&&(b=s(String(b))),v.indexOf("[]=")>-1&&(b=i(b)?[b]:b);var x=n.call(r,g);x&&"combine"===t.duplicates?r[g]=o.combine(r[g],b):x&&"last"!==t.duplicates||(r[g]=b)}return r},f=function(e,t,r,o){for(var n=o?t:l(t,r),i=e.length-1;i>=0;--i){var a,s=e[i];if("[]"===s&&r.parseArrays)a=r.allowEmptyArrays&&(""===n||r.strictNullHandling&&null===n)?[]:[].concat(n);else{a=r.plainObjects?{__proto__:null}:{};var c="["===s.charAt(0)&&"]"===s.charAt(s.length-1)?s.slice(1,-1):s,u=r.decodeDotInKeys?c.replace(/%2E/g,"."):c,p=parseInt(u,10);r.parseArrays||""!==u?!isNaN(p)&&s!==u&&String(p)===u&&p>=0&&r.parseArrays&&p<=r.arrayLimit?(a=[],a[p]=n):"__proto__"!==u&&(a[u]=n):a={0:n}}n=a}return n},h=function(e,t,r,o){if(e){var i=r.allowDots?e.replace(/\.([^.[]+)/g,"[$1]"):e,a=/(\[[^[\]]*])/,s=/(\[[^[\]]*])/g,l=r.depth>0&&a.exec(i),c=l?i.slice(0,l.index):i,u=[];if(c){if(!r.plainObjects&&n.call(Object.prototype,c)&&!r.allowPrototypes)return;u.push(c)}var p=0;while(r.depth>0&&null!==(l=s.exec(i))&&p<r.depth){if(p+=1,!r.plainObjects&&n.call(Object.prototype,l[1].slice(1,-1))&&!r.allowPrototypes)return;u.push(l[1])}if(l){if(!0===r.strictDepth)throw new RangeError("Input depth exceeded depth option of "+r.depth+" and strictDepth is true");u.push("["+i.slice(l.index)+"]")}return f(u,t,r,o)}},y=function(e){if(!e)return a;if("undefined"!==typeof e.allowEmptyArrays&&"boolean"!==typeof e.allowEmptyArrays)throw new TypeError("`allowEmptyArrays` option can only be `true` or `false`, when provided");if("undefined"!==typeof e.decodeDotInKeys&&"boolean"!==typeof e.decodeDotInKeys)throw new TypeError("`decodeDotInKeys` option can only be `true` or `false`, when provided");if(null!==e.decoder&&"undefined"!==typeof e.decoder&&"function"!==typeof e.decoder)throw new TypeError("Decoder has to be a function.");if("undefined"!==typeof e.charset&&"utf-8"!==e.charset&&"iso-8859-1"!==e.charset)throw new TypeError("The charset option must be either utf-8, iso-8859-1, or undefined");var t="undefined"===typeof e.charset?a.charset:e.charset,r="undefined"===typeof e.duplicates?a.duplicates:e.duplicates;if("combine"!==r&&"first"!==r&&"last"!==r)throw new TypeError("The duplicates option must be either combine, first, or last");var n="undefined"===typeof e.allowDots?!0===e.decodeDotInKeys||a.allowDots:!!e.allowDots;return{allowDots:n,allowEmptyArrays:"boolean"===typeof e.allowEmptyArrays?!!e.allowEmptyArrays:a.allowEmptyArrays,allowPrototypes:"boolean"===typeof e.allowPrototypes?e.allowPrototypes:a.allowPrototypes,allowSparse:"boolean"===typeof e.allowSparse?e.allowSparse:a.allowSparse,arrayLimit:"number"===typeof e.arrayLimit?e.arrayLimit:a.arrayLimit,charset:t,charsetSentinel:"boolean"===typeof e.charsetSentinel?e.charsetSentinel:a.charsetSentinel,comma:"boolean"===typeof e.comma?e.comma:a.comma,decodeDotInKeys:"boolean"===typeof e.decodeDotInKeys?e.decodeDotInKeys:a.decodeDotInKeys,decoder:"function"===typeof e.decoder?e.decoder:a.decoder,delimiter:"string"===typeof e.delimiter||o.isRegExp(e.delimiter)?e.delimiter:a.delimiter,depth:"number"===typeof e.depth||!1===e.depth?+e.depth:a.depth,duplicates:r,ignoreQueryPrefix:!0===e.ignoreQueryPrefix,interpretNumericEntities:"boolean"===typeof e.interpretNumericEntities?e.interpretNumericEntities:a.interpretNumericEntities,parameterLimit:"number"===typeof e.parameterLimit?e.parameterLimit:a.parameterLimit,parseArrays:!1!==e.parseArrays,plainObjects:"boolean"===typeof e.plainObjects?e.plainObjects:a.plainObjects,strictDepth:"boolean"===typeof e.strictDepth?!!e.strictDepth:a.strictDepth,strictNullHandling:"boolean"===typeof e.strictNullHandling?e.strictNullHandling:a.strictNullHandling}};e.exports=function(e,t){var r=y(t);if(""===e||null===e||"undefined"===typeof e)return r.plainObjects?{__proto__:null}:{};for(var n="string"===typeof e?p(e,r):e,i=r.plainObjects?{__proto__:null}:{},a=Object.keys(n),s=0;s<a.length;++s){var l=a[s],c=h(l,n[l],r,"string"===typeof e);i=o.merge(i,c,r)}return!0===r.allowSparse?i:o.compact(i)}},2714:function(e,t,r){(function(t){var o="function"===typeof Map&&Map.prototype,n=Object.getOwnPropertyDescriptor&&o?Object.getOwnPropertyDescriptor(Map.prototype,"size"):null,i=o&&n&&"function"===typeof n.get?n.get:null,a=o&&Map.prototype.forEach,s="function"===typeof Set&&Set.prototype,l=Object.getOwnPropertyDescriptor&&s?Object.getOwnPropertyDescriptor(Set.prototype,"size"):null,c=s&&l&&"function"===typeof l.get?l.get:null,u=s&&Set.prototype.forEach,p="function"===typeof WeakMap&&WeakMap.prototype,f=p?WeakMap.prototype.has:null,h="function"===typeof WeakSet&&WeakSet.prototype,y=h?WeakSet.prototype.has:null,d="function"===typeof WeakRef&&WeakRef.prototype,m=d?WeakRef.prototype.deref:null,g=Boolean.prototype.valueOf,b=Object.prototype.toString,v=Function.prototype.toString,w=String.prototype.match,S=String.prototype.slice,x=String.prototype.replace,O=String.prototype.toUpperCase,j=String.prototype.toLowerCase,A=RegExp.prototype.test,P=Array.prototype.concat,k=Array.prototype.join,E=Array.prototype.slice,I=Math.floor,_="function"===typeof BigInt?BigInt.prototype.valueOf:null,T=Object.getOwnPropertySymbols,N="function"===typeof Symbol&&"symbol"===typeof Symbol.iterator?Symbol.prototype.toString:null,C="function"===typeof Symbol&&"object"===typeof Symbol.iterator,F="function"===typeof Symbol&&Symbol.toStringTag&&(typeof Symbol.toStringTag===C||"symbol")?Symbol.toStringTag:null,R=Object.prototype.propertyIsEnumerable,L=("function"===typeof Reflect?Reflect.getPrototypeOf:Object.getPrototypeOf)||([].__proto__===Array.prototype?function(e){return e.__proto__}:null);function D(e,t){if(e===1/0||e===-1/0||e!==e||e&&e>-1e3&&e<1e3||A.call(/e/,t))return t;var r=/[0-9](?=(?:[0-9]{3})+(?![0-9]))/g;if("number"===typeof e){var o=e<0?-I(-e):I(e);if(o!==e){var n=String(o),i=S.call(t,n.length+1);return x.call(n,r,"$&_")+"."+x.call(x.call(i,/([0-9]{3})/g,"$&_"),/_$/,"")}}return x.call(t,r,"$&_")}var M=r(1),U=M.custom,B=X(U)?U:null,W={__proto__:null,double:'"',single:"'"},q={__proto__:null,double:/(["\\])/g,single:/(['\\])/g};function $(e,t,r){var o=r.quoteStyle||t,n=W[o];return n+e+n}function H(e){return x.call(String(e),/"/g,""")}function G(e){return"[object Array]"===re(e)&&(!F||!("object"===typeof e&&F in e))}function J(e){return"[object Date]"===re(e)&&(!F||!("object"===typeof e&&F in e))}function K(e){return"[object RegExp]"===re(e)&&(!F||!("object"===typeof e&&F in e))}function z(e){return"[object Error]"===re(e)&&(!F||!("object"===typeof e&&F in e))}function V(e){return"[object String]"===re(e)&&(!F||!("object"===typeof e&&F in e))}function Q(e){return"[object Number]"===re(e)&&(!F||!("object"===typeof e&&F in e))}function Z(e){return"[object Boolean]"===re(e)&&(!F||!("object"===typeof e&&F in e))}function X(e){if(C)return e&&"object"===typeof e&&e instanceof Symbol;if("symbol"===typeof e)return!0;if(!e||"object"!==typeof e||!N)return!1;try{return N.call(e),!0}catch(t){}return!1}function Y(e){if(!e||"object"!==typeof e||!_)return!1;try{return _.call(e),!0}catch(t){}return!1}e.exports=function e(r,o,n,s){var l=o||{};if(te(l,"quoteStyle")&&!te(W,l.quoteStyle))throw new TypeError('option "quoteStyle" must be "single" or "double"');if(te(l,"maxStringLength")&&("number"===typeof l.maxStringLength?l.maxStringLength<0&&l.maxStringLength!==1/0:null!==l.maxStringLength))throw new TypeError('option "maxStringLength", if provided, must be a positive integer, Infinity, or `null`');var p=!te(l,"customInspect")||l.customInspect;if("boolean"!==typeof p&&"symbol"!==p)throw new TypeError("option \"customInspect\", if provided, must be `true`, `false`, or `'symbol'`");if(te(l,"indent")&&null!==l.indent&&"\t"!==l.indent&&!(parseInt(l.indent,10)===l.indent&&l.indent>0))throw new TypeError('option "indent" must be "\\t", an integer > 0, or `null`');if(te(l,"numericSeparator")&&"boolean"!==typeof l.numericSeparator)throw new TypeError('option "numericSeparator", if provided, must be `true` or `false`');var f=l.numericSeparator;if("undefined"===typeof r)return"undefined";if(null===r)return"null";if("boolean"===typeof r)return r?"true":"false";if("string"===typeof r)return pe(r,l);if("number"===typeof r){if(0===r)return 1/0/r>0?"0":"-0";var h=String(r);return f?D(r,h):h}if("bigint"===typeof r){var y=String(r)+"n";return f?D(r,y):y}var d="undefined"===typeof l.depth?5:l.depth;if("undefined"===typeof n&&(n=0),n>=d&&d>0&&"object"===typeof r)return G(r)?"[Array]":"[Object]";var m=ge(l,n);if("undefined"===typeof s)s=[];else if(ne(s,r)>=0)return"[Circular]";function b(t,r,o){if(r&&(s=E.call(s),s.push(r)),o){var i={depth:l.depth};return te(l,"quoteStyle")&&(i.quoteStyle=l.quoteStyle),e(t,i,n+1,s)}return e(t,l,n+1,s)}if("function"===typeof r&&!K(r)){var v=oe(r),w=ve(r,b);return"[Function"+(v?": "+v:" (anonymous)")+"]"+(w.length>0?" { "+k.call(w,", ")+" }":"")}if(X(r)){var O=C?x.call(String(r),/^(Symbol\(.*\))_[^)]*$/,"$1"):N.call(r);return"object"!==typeof r||C?O:he(O)}if(ue(r)){for(var A="<"+j.call(String(r.nodeName)),I=r.attributes||[],T=0;T<I.length;T++)A+=" "+I[T].name+"="+$(H(I[T].value),"double",l);return A+=">",r.childNodes&&r.childNodes.length&&(A+="..."),A+="</"+j.call(String(r.nodeName))+">",A}if(G(r)){if(0===r.length)return"[]";var U=ve(r,b);return m&&!me(U)?"["+be(U,m)+"]":"[ "+k.call(U,", ")+" ]"}if(z(r)){var q=ve(r,b);return"cause"in Error.prototype||!("cause"in r)||R.call(r,"cause")?0===q.length?"["+String(r)+"]":"{ ["+String(r)+"] "+k.call(q,", ")+" }":"{ ["+String(r)+"] "+k.call(P.call("[cause]: "+b(r.cause),q),", ")+" }"}if("object"===typeof r&&p){if(B&&"function"===typeof r[B]&&M)return M(r,{depth:d-n});if("symbol"!==p&&"function"===typeof r.inspect)return r.inspect()}if(ie(r)){var ee=[];return a&&a.call(r,(function(e,t){ee.push(b(t,r,!0)+" => "+b(e,r))})),de("Map",i.call(r),ee,m)}if(le(r)){var fe=[];return u&&u.call(r,(function(e){fe.push(b(e,r))})),de("Set",c.call(r),fe,m)}if(ae(r))return ye("WeakMap");if(ce(r))return ye("WeakSet");if(se(r))return ye("WeakRef");if(Q(r))return he(b(Number(r)));if(Y(r))return he(b(_.call(r)));if(Z(r))return he(g.call(r));if(V(r))return he(b(String(r)));if("undefined"!==typeof window&&r===window)return"{ [object Window] }";if("undefined"!==typeof globalThis&&r===globalThis||"undefined"!==typeof t&&r===t)return"{ [object globalThis] }";if(!J(r)&&!K(r)){var we=ve(r,b),Se=L?L(r)===Object.prototype:r instanceof Object||r.constructor===Object,xe=r instanceof Object?"":"null prototype",Oe=!Se&&F&&Object(r)===r&&F in r?S.call(re(r),8,-1):xe?"Object":"",je=Se||"function"!==typeof r.constructor?"":r.constructor.name?r.constructor.name+" ":"",Ae=je+(Oe||xe?"["+k.call(P.call([],Oe||[],xe||[]),": ")+"] ":"");return 0===we.length?Ae+"{}":m?Ae+"{"+be(we,m)+"}":Ae+"{ "+k.call(we,", ")+" }"}return String(r)};var ee=Object.prototype.hasOwnProperty||function(e){return e in this};function te(e,t){return ee.call(e,t)}function re(e){return b.call(e)}function oe(e){if(e.name)return e.name;var t=w.call(v.call(e),/^function\s*([\w$]+)/);return t?t[1]:null}function ne(e,t){if(e.indexOf)return e.indexOf(t);for(var r=0,o=e.length;r<o;r++)if(e[r]===t)return r;return-1}function ie(e){if(!i||!e||"object"!==typeof e)return!1;try{i.call(e);try{c.call(e)}catch(t){return!0}return e instanceof Map}catch(r){}return!1}function ae(e){if(!f||!e||"object"!==typeof e)return!1;try{f.call(e,f);try{y.call(e,y)}catch(t){return!0}return e instanceof WeakMap}catch(r){}return!1}function se(e){if(!m||!e||"object"!==typeof e)return!1;try{return m.call(e),!0}catch(t){}return!1}function le(e){if(!c||!e||"object"!==typeof e)return!1;try{c.call(e);try{i.call(e)}catch(t){return!0}return e instanceof Set}catch(r){}return!1}function ce(e){if(!y||!e||"object"!==typeof e)return!1;try{y.call(e,y);try{f.call(e,f)}catch(t){return!0}return e instanceof WeakSet}catch(r){}return!1}function ue(e){return!(!e||"object"!==typeof e)&&("undefined"!==typeof HTMLElement&&e instanceof HTMLElement||"string"===typeof e.nodeName&&"function"===typeof e.getAttribute)}function pe(e,t){if(e.length>t.maxStringLength){var r=e.length-t.maxStringLength,o="... "+r+" more character"+(r>1?"s":"");return pe(S.call(e,0,t.maxStringLength),t)+o}var n=q[t.quoteStyle||"single"];n.lastIndex=0;var i=x.call(x.call(e,n,"\\$1"),/[\x00-\x1f]/g,fe);return $(i,"single",t)}function fe(e){var t=e.charCodeAt(0),r={8:"b",9:"t",10:"n",12:"f",13:"r"}[t];return r?"\\"+r:"\\x"+(t<16?"0":"")+O.call(t.toString(16))}function he(e){return"Object("+e+")"}function ye(e){return e+" { ? }"}function de(e,t,r,o){var n=o?be(r,o):k.call(r,", ");return e+" ("+t+") {"+n+"}"}function me(e){for(var t=0;t<e.length;t++)if(ne(e[t],"\n")>=0)return!1;return!0}function ge(e,t){var r;if("\t"===e.indent)r="\t";else{if(!("number"===typeof e.indent&&e.indent>0))return null;r=k.call(Array(e.indent+1)," ")}return{base:r,prev:k.call(Array(t+1),r)}}function be(e,t){if(0===e.length)return"";var r="\n"+t.prev+t.base;return r+k.call(e,","+r)+"\n"+t.prev}function ve(e,t){var r=G(e),o=[];if(r){o.length=e.length;for(var n=0;n<e.length;n++)o[n]=te(e,n)?t(e[n],e):""}var i,a="function"===typeof T?T(e):[];if(C){i={};for(var s=0;s<a.length;s++)i["$"+a[s]]=a[s]}for(var l in e)te(e,l)&&(r&&String(Number(l))===l&&l<e.length||C&&i["$"+l]instanceof Symbol||(A.call(/[^\w$]/,l)?o.push(t(l,e)+": "+t(e[l],e)):o.push(l+": "+t(e[l],e))));if("function"===typeof T)for(var c=0;c<a.length;c++)R.call(e,a[c])&&o.push("["+t(a[c])+"]: "+t(e[a[c]],e));return o}}).call(this,r("c8ba"))},"2aa9":function(e,t,r){"use strict";var o=r("00ce"),n=o("%Object.getOwnPropertyDescriptor%",!0);if(n)try{n([],"length")}catch(i){n=null}e.exports=n},"3eb1":function(e,t,r){"use strict";var o=r("0f7c"),n=r("00ce"),i=r("d009"),a=r("0d25"),s=n("%Function.prototype.apply%"),l=n("%Function.prototype.call%"),c=n("%Reflect.apply%",!0)||o.call(l,s),u=r("71c9"),p=n("%Math.max%");e.exports=function(e){if("function"!==typeof e)throw new a("a function is required");var t=c(o,l,arguments);return i(t,1+p(0,e.length-(arguments.length-1)),!0)};var f=function(){return c(o,s,arguments)};u?u(e.exports,"apply",{value:f}):e.exports.apply=f},"417f":function(e,t,r){"use strict";e.exports=EvalError},4332:function(e,t,r){var o=r("b11f");o.__esModule&&(o=o.default),"string"===typeof o&&(o=[[e.i,o,""]]),o.locals&&(e.exports=o.locals);var n=r("499e").default;n("0883be1e",o,!0,{sourceMap:!1,shadowMode:!1})},5156:function(e,t,r){"use strict";var o="undefined"!==typeof Symbol&&Symbol,n=r("1696");e.exports=function(){return"function"===typeof o&&("function"===typeof Symbol&&("symbol"===typeof o("foo")&&("symbol"===typeof Symbol("bar")&&n())))}},5402:function(e,t,r){"use strict";var o=r("00ce"),n=r("545e"),i=r("2714"),a=r("0d25"),s=o("%WeakMap%",!0),l=o("%Map%",!0),c=n("WeakMap.prototype.get",!0),u=n("WeakMap.prototype.set",!0),p=n("WeakMap.prototype.has",!0),f=n("Map.prototype.get",!0),h=n("Map.prototype.set",!0),y=n("Map.prototype.has",!0),d=function(e,t){for(var r,o=e;null!==(r=o.next);o=r)if(r.key===t)return o.next=r.next,r.next=e.next,e.next=r,r},m=function(e,t){var r=d(e,t);return r&&r.value},g=function(e,t,r){var o=d(e,t);o?o.value=r:e.next={key:t,next:e.next,value:r}},b=function(e,t){return!!d(e,t)};e.exports=function(){var e,t,r,o={assert:function(e){if(!o.has(e))throw new a("Side channel does not contain "+i(e))},get:function(o){if(s&&o&&("object"===typeof o||"function"===typeof o)){if(e)return c(e,o)}else if(l){if(t)return f(t,o)}else if(r)return m(r,o)},has:function(o){if(s&&o&&("object"===typeof o||"function"===typeof o)){if(e)return p(e,o)}else if(l){if(t)return y(t,o)}else if(r)return b(r,o);return!1},set:function(o,n){s&&o&&("object"===typeof o||"function"===typeof o)?(e||(e=new s),u(e,o,n)):l?(t||(t=new l),h(t,o,n)):(r||(r={key:{},next:null}),g(r,o,n))}};return o}},"545e":function(e,t,r){"use strict";var o=r("00ce"),n=r("3eb1"),i=n(o("String.prototype.indexOf"));e.exports=function(e,t){var r=o(e,!!t);return"function"===typeof r&&i(e,".prototype.")>-1?n(r):r}},5748:function(e,t,r){"use strict";r.r(t);var o=function(){var e=this,t=e._self._c;return t("div",[t("meta",{attrs:{name:"viewport",content:"width=device-width, initial-scale=0.5, maximum-scale=0.5"}}),t("header",{staticClass:"controls"},[e.disableStore?e._e():t("StoreInput",{directives:[{name:"show",rawName:"v-show",value:!1,expression:"false"}],attrs:{sessionKey:"lotus-profilerSourceText"},model:{value:e.sourceText,callback:function(t){e.sourceText=t},expression:"sourceText"}}),e.disableStore?e._e():t("BoolStoreInput",{directives:[{name:"show",rawName:"v-show",value:!1,expression:"false"}],attrs:{sessionKey:"lotus-profilerBakingSheet"},model:{value:e.bakingSheet,callback:function(t){e.bakingSheet=t},expression:"bakingSheet"}}),t("input",{attrs:{type:"file"},on:{change:e.onScoreChange}}),t("button",{on:{click:e.homePlayer}},[e._v("⏮")]),t("button",{attrs:{disabled:!e.midiPlayer},on:{click:e.togglePlayer}},[e._v(e._s(e.midiPlayer&&e.midiPlayer.isPlaying?"⏸":"▶"))]),t("CheckButton",{attrs:{content:"ꕯ"},model:{value:e.showCursor,callback:function(t){e.showCursor=t},expression:"showCursor"}}),t("CheckButton",{attrs:{content:"♩"},model:{value:e.noteHighlight,callback:function(t){e.noteHighlight=t},expression:"noteHighlight"}}),t("CheckButton",{attrs:{content:"🍞"},model:{value:e.bakingSheet,callback:function(t){e.bakingSheet=t},expression:"bakingSheet"}}),e.fps?t("span",{staticClass:"fps"},[t("em",[e._v(e._s(e.fps.toFixed(1)))]),e._v("fps")]):e._e()],1),t("main",{on:{scroll:e.onScroll}},[!e.bakingSheet&&e.svgHashTable?t("SheetSigns",{directives:[{name:"show",rawName:"v-show",value:!1,expression:"false"}],ref:"signs",attrs:{hashTable:e.svgHashTable}}):e._e(),e.sheetDocument?t("SheetLive",{ref:"sheet",attrs:{doc:e.sheetDocument,midiNotation:e.midiNotation,pitchContextGroup:e.pitchContextGroup,midiPlayer:e.midiPlayer,scheduler:e.scheduler,showCursor:e.showCursor,noteHighlight:e.noteHighlight,bakingMode:e.bakingSheet,backgroundImages:e.bakingSheet?e.bakingImages:null,showPagesProgressively:e.showPagesProgressively},on:{"update:midiPlayer":function(t){e.midiPlayer=t},"update:midi-player":function(t){e.midiPlayer=t},midi:e.onMidi}}):e._e()],1),t("canvas",{directives:[{name:"show",rawName:"v-show",value:!1,expression:"false"}],ref:"canvas"})])},n=[],i=r("0b16"),a=r.n(i),s=r("b047"),l=r.n(s),c=(r("8701"),r("5027")),u=r("9224"),p=r("f634"),f=r("32c1"),h=r("1151"),y=r("a31e"),d=r("1c3a"),m=r("c0d2");const g=e=>{if(e){const t=e.match(/\d+/g);if(t){const e=t.map(Number).reverse();return e.reduce((e,t,r)=>e+t*1e3**r,0)}}return-1};class b{constructor({onStatus:e=((...e)=>e)}={}){this.onStatus=e}static fromJSON(e,{measureLayout:t=y["b"].Full,onStatus:r=((...e)=>e),jsonHandle:o=(e=>e)}={}){const n=new b({onStatus:r});return n.scoreJSON=o(Object(p["b"])(e,b.classDict)),n.checkVersion(),n.onStatus("json loaded"),n.scoreJSON.lilyNotation&&(n.matchedIds=n.scoreJSON.lilyNotation.idSet,n.scoreJSON.doc.updateMatchedTokens(n.matchedIds)),t&&n.loadNotation(t),n}loadNotation(e,t){const r=this.scoreJSON.lilyNotation;if(r){const o=r.getMeasureIndices(e);this.midiNotation=r.toPerformingNotationWithEvents(o,{trackList:t}),this.pitchContextGroup=r.getContextGroup(o);const n=r.toPerformingNotation(o,{withRestTied:!0});this.scheduler=y["e"].createFromNotation(n,this.scoreJSON.doc.getTokenMap()),this.onStatus("notation loaded")}}bakeSheet(e){return console.assert(!!this.scoreJSON.doc,"sheetDocument is null."),console.assert(!!this.scoreJSON.hashTable,"hashTable is null."),console.assert(!!this.matchedIds,"matchedIds is null."),this.onStatus("baking sheet"),d["a"]({sheetDocument:this.scoreJSON.doc,hashTable:this.scoreJSON.hashTable,matchedIds:this.matchedIds,canvas:e})}checkVersion(){const e=g(u.version),t=g(this.scoreJSON.version);t<g("0.6.1")&&console.warn(`This score bundle version[${this.scoreJSON.version}] is too low! The current Lotus API version is: ${u.version}.`),t>e&&console.warn(`The current Lotus API version[${u.version}] is behind this score bundle[${this.scoreJSON.version}]. If any score problem encountered, try to upgrade Lotus API.`),t>=0&&t<g("0.8.0")&&this.scoreJSON.doc.pages.forEach(e=>e.systems=e.rows)}}b.classDict={StaffToken:f["StaffToken"],SheetDocument:f["SheetDocument"],LilyNotation:y["d"],...y["c"],DictArray:m["a"],PitchContextTable:h["b"],PitchContext:h["a"]};var v=r("faa1");class w extends v["EventEmitter"]{constructor(e,{separator:t="\n\n\n\n"}={}){super(),this.reader=e,this.separator=t}async read(){let e="";while(1){const{done:t,value:r}=await this.reader.read();if(r){const t=new TextDecoder("utf-8").decode(r);e+=t;while(1){const t=e.indexOf(this.separator);if(!(t>=0))break;{const r=e.substr(0,t);this.emit("data",r),e=e.substr(t+this.separator.length)}}}if(t)break}e&&this.emit("data",e)}}var S=r("a139"),x=r("f7ce"),O=r("cdaf"),j=r("4e72"),A=r("ddcc"),P={name:"profiler",components:{SheetLive:S["a"],SheetSigns:x["a"],StoreInput:O["a"],BoolStoreInput:j["a"],CheckButton:A["a"]},data(){return{sourceText:null,sheetDocument:null,svgHashTable:null,midiNotation:null,scheduler:null,pitchContextGroup:null,midiPlayer:null,showCursor:!0,noteHighlight:!0,bakingSheet:!0,bakingImages:null,sourceBakingImages:null,fps:null,disableStore:!1,showPagesProgressively:!0}},async created(){this.logTime("created"),window.$main=this,this.watchFps();const e=a.a.parse(location.hash.substr(1),!0);e.query.score&&(this.disableStore=!0,this.loadScoreFromURL(e.query.score)),e.query.nobake&&(this.bakingSheet=!1)},methods:{logTime(e){console.log("[PROFILER]",e,performance.now())},onScoreChange(e){const t=e.target.files[0];if(t)return this.loadScoreFile(t)},onScroll:l()((function(){this.$refs.sheet.updatePageVisibility()}),60,{leading:!0}),async loadScoreFile(e){switch(this.logTime("file loading begin"),e.type){case"application/json":this.sourceText=await e.readAs("Text");break;case"application/zip":case"application/x-zip-compressed":{this.sourceText=null,await this.$nextTick();const{default:t}=await r.e("chunk-48b5b2a0").then(r.t.bind(null,"c4e3",7)),o=await t.loadAsync(e);this.sourceBakingImages=null;for(let e=0;1;++e){const t=o.file(`baking${e}.png`);if(!t)break;const r=await t.async("blob"),n=URL.createObjectURL(r);this.sourceBakingImages=this.sourceBakingImages||[],this.sourceBakingImages.push(n)}this.sourceBakingImages&&this.logTime(`baking images loaded [${this.sourceBakingImages.length}]`),this.sourceText=await o.file("score.json").async("text"),this.logTime("sourceText loaded.")}break;case"text/x-lilypond":case"text/lilypond-source":{const t=await e.readAs("Text"),r=new FormData;r.append("source",t),r.append("withLilyNotation",1);const o=await fetch("/advanced-engrave",{method:"POST",body:r});if(!o.ok)return this.error=await o.text(),void console.warn("advanced-engrave failed:",this.error);const n=new w(o.body.getReader());this.constructSheetFromStream(n)}break;default:console.log("unsupported type:",e.type)}},async loadScoreFromURL(e){this.logTime("URL fetching begin");const t=await fetch(e);if(!t.ok)return void console.warn("URL load failed:",await t.text());this.logTime("network responsed");const r=await t.blob();return this.loadScoreFile(r)},async loadSheet(){if(this.sheetDocument=null,this.midiNotation=null,this.scheduler=null,this.pitchContextGroup=null,this.bakingImages=null,this.sourceText){this.showPagesProgressively=!0;const e=b.fromJSON(this.sourceText,{onStatus:e=>this.logTime(e)});if(this.sheetDocument=e.scoreJSON.doc,this.pitchContextGroup=e.pitchContextGroup,this.midiNotation=e.midiNotation,this.scheduler=e.scheduler,this.svgHashTable=e.scoreJSON.hashTable,this.logTime("bundle parsed"),await this.$nextTick(),this.logTime("rendering initialized"),this.sourceBakingImages)this.bakingImages=this.sourceBakingImages,this.sourceBakingImages=null;else if(this.bakingSheet){this.bakingImages=[];const t=e.bakeSheet(this.$refs.canvas);this.logTime("baker loaded");for await(const e of t)this.bakingImages.push(e);this.logTime("baking finished")}await this.$nextTick(),this.logTime("rendering finished"),this.$refs.sheet.updatePageVisibility()}},async constructSheetFromStream(e){this.sheetDocument=null,this.midiNotation=null,this.scheduler=null,this.pitchContextGroup=null,this.bakingImages=null,this.svgHashTable={},this.showPagesProgressively=!1,this.bakingSheet=!1;const t=[];e.on("data",e=>{const r=Object(p["b"])(e,{...f,LilyNotation:y["d"],...y["c"]});if(console.log("data:",r),void 0!==r.page&&(t[r.page]=r.structure,this.svgHashTable={...this.svgHashTable,...r.hashTable},this.sheetDocument=new f["SheetDocument"]({pages:t})),r.lilyNotation){const e=r.lilyNotation,t=e.getMeasureIndices(y["b"].Full);this.midiNotation=e.toPerformingNotationWithEvents(t),this.pitchContextGroup=e.getContextGroup(t);const o=e.toPerformingNotation(t,{withRestTied:!0});this.scheduler=y["e"].createFromNotation(o,this.sheetDocument.getTokenMap())}}),await e.read()},onMidi(){},homePlayer(){this.midiPlayer&&this.midiPlayer.turnCursor(0)},togglePlayer(){this.midiPlayer&&(this.midiPlayer.isPlaying?this.midiPlayer.pause():this.midiPlayer.play())},async watchFps(){let e=performance.now(),t=0;while(1){await Object(c["a"])(),++t;const r=performance.now();r-e>1e3&&(this.fps=1e3*t/(r-e),t=0,e=r)}}},watch:{sourceText:"loadSheet"}},k=P,E=(r("ae16"),r("2877")),I=Object(E["a"])(k,o,n,!1,null,null,null);t["default"]=I.exports},"62e4":function(e,t){e.exports=function(e){return e.webpackPolyfill||(e.deprecate=function(){},e.paths=[],e.children||(e.children=[]),Object.defineProperty(e,"loaded",{enumerable:!0,get:function(){return e.l}}),Object.defineProperty(e,"id",{enumerable:!0,get:function(){return e.i}}),e.webpackPolyfill=1),e}},"64b0":function(e,t,r){"use strict";var o=r("71c9"),n=function(){return!!o};n.hasArrayLengthDefineBug=function(){if(!o)return null;try{return 1!==o([],"length",{value:1}).length}catch(e){return!0}},e.exports=n},"67d9":function(e,t,r){"use strict";e.exports=URIError},"67ee":function(e,t,r){"use strict";e.exports=SyntaxError},"688e":function(e,t,r){"use strict";var o="Function.prototype.bind called on incompatible ",n=Object.prototype.toString,i=Math.max,a="[object Function]",s=function(e,t){for(var r=[],o=0;o<e.length;o+=1)r[o]=e[o];for(var n=0;n<t.length;n+=1)r[n+e.length]=t[n];return r},l=function(e,t){for(var r=[],o=t||0,n=0;o<e.length;o+=1,n+=1)r[n]=e[o];return r},c=function(e,t){for(var r="",o=0;o<e.length;o+=1)r+=e[o],o+1<e.length&&(r+=t);return r};e.exports=function(e){var t=this;if("function"!==typeof t||n.apply(t)!==a)throw new TypeError(o+t);for(var r,u=l(arguments,1),p=function(){if(this instanceof r){var o=t.apply(this,s(u,arguments));return Object(o)===o?o:this}return t.apply(e,s(u,arguments))},f=i(0,t.length-u.length),h=[],y=0;y<f;y++)h[y]="$"+y;if(r=Function("binder","return function ("+c(h,",")+"){ return binder.apply(this,arguments); }")(p),t.prototype){var d=function(){};d.prototype=t.prototype,r.prototype=new d,d.prototype=null}return r}},"71c9":function(e,t,r){"use strict";var o=r("00ce"),n=o("%Object.defineProperty%",!0)||!1;if(n)try{n({},"a",{value:1})}catch(i){n=!1}e.exports=n},7992:function(e,t,r){"use strict";var o=r("71c9"),n=r("67ee"),i=r("0d25"),a=r("2aa9");e.exports=function(e,t,r){if(!e||"object"!==typeof e&&"function"!==typeof e)throw new i("`obj` must be an object or a function`");if("string"!==typeof t&&"symbol"!==typeof t)throw new i("`property` must be a string or a symbol`");if(arguments.length>3&&"boolean"!==typeof arguments[3]&&null!==arguments[3])throw new i("`nonEnumerable`, if provided, must be a boolean or null");if(arguments.length>4&&"boolean"!==typeof arguments[4]&&null!==arguments[4])throw new i("`nonWritable`, if provided, must be a boolean or null");if(arguments.length>5&&"boolean"!==typeof arguments[5]&&null!==arguments[5])throw new i("`nonConfigurable`, if provided, must be a boolean or null");if(arguments.length>6&&"boolean"!==typeof arguments[6])throw new i("`loose`, if provided, must be a boolean");var s=arguments.length>3?arguments[3]:null,l=arguments.length>4?arguments[4]:null,c=arguments.length>5?arguments[5]:null,u=arguments.length>6&&arguments[6],p=!!a&&a(e,t);if(o)o(e,t,{configurable:null===c&&p?p.configurable:!c,enumerable:null===s&&p?p.enumerable:!s,value:r,writable:null===l&&p?p.writable:!l});else{if(!u&&(s||l||c))throw new n("This environment does not support defining a property as non-configurable, non-writable, or non-enumerable.");e[t]=r}}},9671:function(e,t,r){"use strict";var o=Function.prototype.call,n=Object.prototype.hasOwnProperty,i=r("0f7c");e.exports=i.call(o,n)},"9d88":function(e,t,r){(function(e,o){var n;/*! https://mths.be/punycode v1.4.1 by @mathias */(function(i){t&&t.nodeType,e&&e.nodeType;var a="object"==typeof o&&o;a.global!==a&&a.window!==a&&a.self;var s,l=2147483647,c=36,u=1,p=26,f=38,h=700,y=72,d=128,m="-",g=/^xn--/,b=/[^\x20-\x7E]/,v=/[\x2E\u3002\uFF0E\uFF61]/g,w={overflow:"Overflow: input needs wider integers to process","not-basic":"Illegal input >= 0x80 (not a basic code point)","invalid-input":"Invalid input"},S=c-u,x=Math.floor,O=String.fromCharCode;function j(e){throw new RangeError(w[e])}function A(e,t){var r=e.length,o=[];while(r--)o[r]=t(e[r]);return o}function P(e,t){var r=e.split("@"),o="";r.length>1&&(o=r[0]+"@",e=r[1]),e=e.replace(v,".");var n=e.split("."),i=A(n,t).join(".");return o+i}function k(e){var t,r,o=[],n=0,i=e.length;while(n<i)t=e.charCodeAt(n++),t>=55296&&t<=56319&&n<i?(r=e.charCodeAt(n++),56320==(64512&r)?o.push(((1023&t)<<10)+(1023&r)+65536):(o.push(t),n--)):o.push(t);return o}function E(e){return A(e,(function(e){var t="";return e>65535&&(e-=65536,t+=O(e>>>10&1023|55296),e=56320|1023&e),t+=O(e),t})).join("")}function I(e){return e-48<10?e-22:e-65<26?e-65:e-97<26?e-97:c}function _(e,t){return e+22+75*(e<26)-((0!=t)<<5)}function T(e,t,r){var o=0;for(e=r?x(e/h):e>>1,e+=x(e/t);e>S*p>>1;o+=c)e=x(e/S);return x(o+(S+1)*e/(e+f))}function N(e){var t,r,o,n,i,a,s,f,h,g,b=[],v=e.length,w=0,S=d,O=y;for(r=e.lastIndexOf(m),r<0&&(r=0),o=0;o<r;++o)e.charCodeAt(o)>=128&&j("not-basic"),b.push(e.charCodeAt(o));for(n=r>0?r+1:0;n<v;){for(i=w,a=1,s=c;;s+=c){if(n>=v&&j("invalid-input"),f=I(e.charCodeAt(n++)),(f>=c||f>x((l-w)/a))&&j("overflow"),w+=f*a,h=s<=O?u:s>=O+p?p:s-O,f<h)break;g=c-h,a>x(l/g)&&j("overflow"),a*=g}t=b.length+1,O=T(w-i,t,0==i),x(w/t)>l-S&&j("overflow"),S+=x(w/t),w%=t,b.splice(w++,0,S)}return E(b)}function C(e){var t,r,o,n,i,a,s,f,h,g,b,v,w,S,A,P=[];for(e=k(e),v=e.length,t=d,r=0,i=y,a=0;a<v;++a)b=e[a],b<128&&P.push(O(b));o=n=P.length,n&&P.push(m);while(o<v){for(s=l,a=0;a<v;++a)b=e[a],b>=t&&b<s&&(s=b);for(w=o+1,s-t>x((l-r)/w)&&j("overflow"),r+=(s-t)*w,t=s,a=0;a<v;++a)if(b=e[a],b<t&&++r>l&&j("overflow"),b==t){for(f=r,h=c;;h+=c){if(g=h<=i?u:h>=i+p?p:h-i,f<g)break;A=f-g,S=c-g,P.push(O(_(g+A%S,0))),f=x(A/S)}P.push(O(_(f,0))),i=T(r,w,o==n),r=0,++o}++r,++t}return P.join("")}function F(e){return P(e,(function(e){return g.test(e)?N(e.slice(4).toLowerCase()):e}))}function R(e){return P(e,(function(e){return b.test(e)?"xn--"+C(e):e}))}s={version:"1.4.1",ucs2:{decode:k,encode:E},decode:N,encode:C,toASCII:R,toUnicode:F},n=function(){return s}.call(t,r,t,e),void 0===n||(e.exports=n)})()}).call(this,r("62e4")(e),r("c8ba"))},a29f:function(e,t,r){"use strict";var o=r("bbc7"),n=Object.prototype.hasOwnProperty,i=Array.isArray,a=function(){for(var e=[],t=0;t<256;++t)e.push("%"+((t<16?"0":"")+t.toString(16)).toUpperCase());return e}(),s=function(e){while(e.length>1){var t=e.pop(),r=t.obj[t.prop];if(i(r)){for(var o=[],n=0;n<r.length;++n)"undefined"!==typeof r[n]&&o.push(r[n]);t.obj[t.prop]=o}}},l=function(e,t){for(var r=t&&t.plainObjects?{__proto__:null}:{},o=0;o<e.length;++o)"undefined"!==typeof e[o]&&(r[o]=e[o]);return r},c=function e(t,r,o){if(!r)return t;if("object"!==typeof r&&"function"!==typeof r){if(i(t))t.push(r);else{if(!t||"object"!==typeof t)return[t,r];(o&&(o.plainObjects||o.allowPrototypes)||!n.call(Object.prototype,r))&&(t[r]=!0)}return t}if(!t||"object"!==typeof t)return[t].concat(r);var a=t;return i(t)&&!i(r)&&(a=l(t,o)),i(t)&&i(r)?(r.forEach((function(r,i){if(n.call(t,i)){var a=t[i];a&&"object"===typeof a&&r&&"object"===typeof r?t[i]=e(a,r,o):t.push(r)}else t[i]=r})),t):Object.keys(r).reduce((function(t,i){var a=r[i];return n.call(t,i)?t[i]=e(t[i],a,o):t[i]=a,t}),a)},u=function(e,t){return Object.keys(t).reduce((function(e,r){return e[r]=t[r],e}),e)},p=function(e,t,r){var o=e.replace(/\+/g," ");if("iso-8859-1"===r)return o.replace(/%[0-9a-f]{2}/gi,unescape);try{return decodeURIComponent(o)}catch(n){return o}},f=1024,h=function(e,t,r,n,i){if(0===e.length)return e;var s=e;if("symbol"===typeof e?s=Symbol.prototype.toString.call(e):"string"!==typeof e&&(s=String(e)),"iso-8859-1"===r)return escape(s).replace(/%u[0-9a-f]{4}/gi,(function(e){return"%26%23"+parseInt(e.slice(2),16)+"%3B"}));for(var l="",c=0;c<s.length;c+=f){for(var u=s.length>=f?s.slice(c,c+f):s,p=[],h=0;h<u.length;++h){var y=u.charCodeAt(h);45===y||46===y||95===y||126===y||y>=48&&y<=57||y>=65&&y<=90||y>=97&&y<=122||i===o.RFC1738&&(40===y||41===y)?p[p.length]=u.charAt(h):y<128?p[p.length]=a[y]:y<2048?p[p.length]=a[192|y>>6]+a[128|63&y]:y<55296||y>=57344?p[p.length]=a[224|y>>12]+a[128|y>>6&63]+a[128|63&y]:(h+=1,y=65536+((1023&y)<<10|1023&u.charCodeAt(h)),p[p.length]=a[240|y>>18]+a[128|y>>12&63]+a[128|y>>6&63]+a[128|63&y])}l+=p.join("")}return l},y=function(e){for(var t=[{obj:{o:e},prop:"o"}],r=[],o=0;o<t.length;++o)for(var n=t[o],i=n.obj[n.prop],a=Object.keys(i),l=0;l<a.length;++l){var c=a[l],u=i[c];"object"===typeof u&&null!==u&&-1===r.indexOf(u)&&(t.push({obj:i,prop:c}),r.push(u))}return s(t),e},d=function(e){return"[object RegExp]"===Object.prototype.toString.call(e)},m=function(e){return!(!e||"object"!==typeof e)&&!!(e.constructor&&e.constructor.isBuffer&&e.constructor.isBuffer(e))},g=function(e,t){return[].concat(e,t)},b=function(e,t){if(i(e)){for(var r=[],o=0;o<e.length;o+=1)r.push(t(e[o]));return r}return t(e)};e.exports={arrayToObject:l,assign:u,combine:g,compact:y,decode:p,encode:h,isBuffer:m,isRegExp:d,maybeMap:b,merge:c}},a645:function(e,t,r){"use strict";e.exports=Error},ae16:function(e,t,r){"use strict";r("4332")},b11f:function(e,t,r){var o=r("24fb");t=o(!1),t.push([e.i,'button.working{background-color:#dfd;font-weight:700}button.on{background-color:#efe;font-weight:700;box-shadow:0 0 4px #000}button.off{opacity:.7;filter:grayscale(100%)}button[disabled]{box-shadow:none;cursor:not-allowed!important}.controls fieldset{border:0;padding:0 1em}.controls fieldset,.controls fieldset>span{display:inline-block}.controls fieldset>*{margin:0 .2em}.controls>*,.controls fieldset>*{vertical-align:middle}header.controls{padding:1em;text-align:center;background-color:hsla(0,0%,100%,.6666666666666666)}header.controls button{font-size:24px}header.controls .dirty-badge{width:1em}header.controls .dirty-badge.dirty:before{content:"*"}header .fps{display:inline-block;margin:0 1em;color:#aaa}header .fps em{color:#000;display:inline-block;margin:0 .2em}main{width:100%;overflow:auto}main .sheet.live{white-space:nowrap;display:inline-block}main .sheet.live .page{display:inline-block;margin:1em;background-size:100%}main .sheet.live .cursor{fill:#add8e6}',""]),e.exports=t},bbc7:function(e,t,r){"use strict";var o=String.prototype.replace,n=/%20/g,i={RFC1738:"RFC1738",RFC3986:"RFC3986"};e.exports={default:i.RFC3986,formatters:{RFC1738:function(e){return o.call(e,n,"+")},RFC3986:function(e){return String(e)}},RFC1738:i.RFC1738,RFC3986:i.RFC3986}},d009:function(e,t,r){"use strict";var o=r("00ce"),n=r("7992"),i=r("64b0")(),a=r("2aa9"),s=r("0d25"),l=o("%Math.floor%");e.exports=function(e,t){if("function"!==typeof e)throw new s("`fn` is not a function");if("number"!==typeof t||t<0||t>4294967295||l(t)!==t)throw new s("`length` must be a positive 32-bit integer");var r=arguments.length>2&&!!arguments[2],o=!0,c=!0;if("length"in e&&a){var u=a(e,"length");u&&!u.configurable&&(o=!1),u&&!u.writable&&(c=!1)}return(o||c||!r)&&(i?n(e,"length",t,!0,!0):n(e,"length",t)),e}},dc99:function(e,t,r){"use strict";e.exports=RangeError},f177:function(e,t,r){"use strict";var o=r("5402"),n=r("a29f"),i=r("bbc7"),a=Object.prototype.hasOwnProperty,s={brackets:function(e){return e+"[]"},comma:"comma",indices:function(e,t){return e+"["+t+"]"},repeat:function(e){return e}},l=Array.isArray,c=Array.prototype.push,u=function(e,t){c.apply(e,l(t)?t:[t])},p=Date.prototype.toISOString,f=i["default"],h={addQueryPrefix:!1,allowDots:!1,allowEmptyArrays:!1,arrayFormat:"indices",charset:"utf-8",charsetSentinel:!1,commaRoundTrip:!1,delimiter:"&",encode:!0,encodeDotInKeys:!1,encoder:n.encode,encodeValuesOnly:!1,filter:void 0,format:f,formatter:i.formatters[f],indices:!1,serializeDate:function(e){return p.call(e)},skipNulls:!1,strictNullHandling:!1},y=function(e){return"string"===typeof e||"number"===typeof e||"boolean"===typeof e||"symbol"===typeof e||"bigint"===typeof e},d={},m=function e(t,r,i,a,s,c,p,f,m,g,b,v,w,S,x,O,j,A){var P=t,k=A,E=0,I=!1;while(void 0!==(k=k.get(d))&&!I){var _=k.get(t);if(E+=1,"undefined"!==typeof _){if(_===E)throw new RangeError("Cyclic object value");I=!0}"undefined"===typeof k.get(d)&&(E=0)}if("function"===typeof g?P=g(r,P):P instanceof Date?P=w(P):"comma"===i&&l(P)&&(P=n.maybeMap(P,(function(e){return e instanceof Date?w(e):e}))),null===P){if(c)return m&&!O?m(r,h.encoder,j,"key",S):r;P=""}if(y(P)||n.isBuffer(P)){if(m){var T=O?r:m(r,h.encoder,j,"key",S);return[x(T)+"="+x(m(P,h.encoder,j,"value",S))]}return[x(r)+"="+x(String(P))]}var N,C=[];if("undefined"===typeof P)return C;if("comma"===i&&l(P))O&&m&&(P=n.maybeMap(P,m)),N=[{value:P.length>0?P.join(",")||null:void 0}];else if(l(g))N=g;else{var F=Object.keys(P);N=b?F.sort(b):F}var R=f?String(r).replace(/\./g,"%2E"):String(r),L=a&&l(P)&&1===P.length?R+"[]":R;if(s&&l(P)&&0===P.length)return L+"[]";for(var D=0;D<N.length;++D){var M=N[D],U="object"===typeof M&&M&&"undefined"!==typeof M.value?M.value:P[M];if(!p||null!==U){var B=v&&f?String(M).replace(/\./g,"%2E"):String(M),W=l(P)?"function"===typeof i?i(L,B):L:L+(v?"."+B:"["+B+"]");A.set(t,E);var q=o();q.set(d,A),u(C,e(U,W,i,a,s,c,p,f,"comma"===i&&O&&l(P)?null:m,g,b,v,w,S,x,O,j,q))}}return C},g=function(e){if(!e)return h;if("undefined"!==typeof e.allowEmptyArrays&&"boolean"!==typeof e.allowEmptyArrays)throw new TypeError("`allowEmptyArrays` option can only be `true` or `false`, when provided");if("undefined"!==typeof e.encodeDotInKeys&&"boolean"!==typeof e.encodeDotInKeys)throw new TypeError("`encodeDotInKeys` option can only be `true` or `false`, when provided");if(null!==e.encoder&&"undefined"!==typeof e.encoder&&"function"!==typeof e.encoder)throw new TypeError("Encoder has to be a function.");var t=e.charset||h.charset;if("undefined"!==typeof e.charset&&"utf-8"!==e.charset&&"iso-8859-1"!==e.charset)throw new TypeError("The charset option must be either utf-8, iso-8859-1, or undefined");var r=i["default"];if("undefined"!==typeof e.format){if(!a.call(i.formatters,e.format))throw new TypeError("Unknown format option provided.");r=e.format}var o,n=i.formatters[r],c=h.filter;if(("function"===typeof e.filter||l(e.filter))&&(c=e.filter),o=e.arrayFormat in s?e.arrayFormat:"indices"in e?e.indices?"indices":"repeat":h.arrayFormat,"commaRoundTrip"in e&&"boolean"!==typeof e.commaRoundTrip)throw new TypeError("`commaRoundTrip` must be a boolean, or absent");var u="undefined"===typeof e.allowDots?!0===e.encodeDotInKeys||h.allowDots:!!e.allowDots;return{addQueryPrefix:"boolean"===typeof e.addQueryPrefix?e.addQueryPrefix:h.addQueryPrefix,allowDots:u,allowEmptyArrays:"boolean"===typeof e.allowEmptyArrays?!!e.allowEmptyArrays:h.allowEmptyArrays,arrayFormat:o,charset:t,charsetSentinel:"boolean"===typeof e.charsetSentinel?e.charsetSentinel:h.charsetSentinel,commaRoundTrip:!!e.commaRoundTrip,delimiter:"undefined"===typeof e.delimiter?h.delimiter:e.delimiter,encode:"boolean"===typeof e.encode?e.encode:h.encode,encodeDotInKeys:"boolean"===typeof e.encodeDotInKeys?e.encodeDotInKeys:h.encodeDotInKeys,encoder:"function"===typeof e.encoder?e.encoder:h.encoder,encodeValuesOnly:"boolean"===typeof e.encodeValuesOnly?e.encodeValuesOnly:h.encodeValuesOnly,filter:c,format:r,formatter:n,serializeDate:"function"===typeof e.serializeDate?e.serializeDate:h.serializeDate,skipNulls:"boolean"===typeof e.skipNulls?e.skipNulls:h.skipNulls,sort:"function"===typeof e.sort?e.sort:null,strictNullHandling:"boolean"===typeof e.strictNullHandling?e.strictNullHandling:h.strictNullHandling}};e.exports=function(e,t){var r,n,i=e,a=g(t);"function"===typeof a.filter?(n=a.filter,i=n("",i)):l(a.filter)&&(n=a.filter,r=n);var c=[];if("object"!==typeof i||null===i)return"";var p=s[a.arrayFormat],f="comma"===p&&a.commaRoundTrip;r||(r=Object.keys(i)),a.sort&&r.sort(a.sort);for(var h=o(),y=0;y<r.length;++y){var d=r[y],b=i[d];a.skipNulls&&null===b||u(c,m(b,d,p,f,a.allowEmptyArrays,a.strictNullHandling,a.skipNulls,a.encodeDotInKeys,a.encode?a.encoder:null,a.filter,a.sort,a.allowDots,a.serializeDate,a.format,a.formatter,a.encodeValuesOnly,a.charset,h))}var v=c.join(a.delimiter),w=!0===a.addQueryPrefix?"?":"";return a.charsetSentinel&&("iso-8859-1"===a.charset?w+="utf8=%26%2310003%3B&":w+="utf8=%E2%9C%93&"),v.length>0?w+v:""}},faa1:function(e,t,r){"use strict";var o,n="object"===typeof Reflect?Reflect:null,i=n&&"function"===typeof n.apply?n.apply:function(e,t,r){return Function.prototype.apply.call(e,t,r)};function a(e){console&&console.warn&&console.warn(e)}o=n&&"function"===typeof n.ownKeys?n.ownKeys:Object.getOwnPropertySymbols?function(e){return Object.getOwnPropertyNames(e).concat(Object.getOwnPropertySymbols(e))}:function(e){return Object.getOwnPropertyNames(e)};var s=Number.isNaN||function(e){return e!==e};function l(){l.init.call(this)}e.exports=l,e.exports.once=w,l.EventEmitter=l,l.prototype._events=void 0,l.prototype._eventsCount=0,l.prototype._maxListeners=void 0;var c=10;function u(e){if("function"!==typeof e)throw new TypeError('The "listener" argument must be of type Function. Received type '+typeof e)}function p(e){return void 0===e._maxListeners?l.defaultMaxListeners:e._maxListeners}function f(e,t,r,o){var n,i,s;if(u(r),i=e._events,void 0===i?(i=e._events=Object.create(null),e._eventsCount=0):(void 0!==i.newListener&&(e.emit("newListener",t,r.listener?r.listener:r),i=e._events),s=i[t]),void 0===s)s=i[t]=r,++e._eventsCount;else if("function"===typeof s?s=i[t]=o?[r,s]:[s,r]:o?s.unshift(r):s.push(r),n=p(e),n>0&&s.length>n&&!s.warned){s.warned=!0;var l=new Error("Possible EventEmitter memory leak detected. "+s.length+" "+String(t)+" listeners added. Use emitter.setMaxListeners() to increase limit");l.name="MaxListenersExceededWarning",l.emitter=e,l.type=t,l.count=s.length,a(l)}return e}function h(){if(!this.fired)return this.target.removeListener(this.type,this.wrapFn),this.fired=!0,0===arguments.length?this.listener.call(this.target):this.listener.apply(this.target,arguments)}function y(e,t,r){var o={fired:!1,wrapFn:void 0,target:e,type:t,listener:r},n=h.bind(o);return n.listener=r,o.wrapFn=n,n}function d(e,t,r){var o=e._events;if(void 0===o)return[];var n=o[t];return void 0===n?[]:"function"===typeof n?r?[n.listener||n]:[n]:r?v(n):g(n,n.length)}function m(e){var t=this._events;if(void 0!==t){var r=t[e];if("function"===typeof r)return 1;if(void 0!==r)return r.length}return 0}function g(e,t){for(var r=new Array(t),o=0;o<t;++o)r[o]=e[o];return r}function b(e,t){for(;t+1<e.length;t++)e[t]=e[t+1];e.pop()}function v(e){for(var t=new Array(e.length),r=0;r<t.length;++r)t[r]=e[r].listener||e[r];return t}function w(e,t){return new Promise((function(r,o){function n(r){e.removeListener(t,i),o(r)}function i(){"function"===typeof e.removeListener&&e.removeListener("error",n),r([].slice.call(arguments))}x(e,t,i,{once:!0}),"error"!==t&&S(e,n,{once:!0})}))}function S(e,t,r){"function"===typeof e.on&&x(e,"error",t,r)}function x(e,t,r,o){if("function"===typeof e.on)o.once?e.once(t,r):e.on(t,r);else{if("function"!==typeof e.addEventListener)throw new TypeError('The "emitter" argument must be of type EventEmitter. Received type '+typeof e);e.addEventListener(t,(function n(i){o.once&&e.removeEventListener(t,n),r(i)}))}}Object.defineProperty(l,"defaultMaxListeners",{enumerable:!0,get:function(){return c},set:function(e){if("number"!==typeof e||e<0||s(e))throw new RangeError('The value of "defaultMaxListeners" is out of range. It must be a non-negative number. Received '+e+".");c=e}}),l.init=function(){void 0!==this._events&&this._events!==Object.getPrototypeOf(this)._events||(this._events=Object.create(null),this._eventsCount=0),this._maxListeners=this._maxListeners||void 0},l.prototype.setMaxListeners=function(e){if("number"!==typeof e||e<0||s(e))throw new RangeError('The value of "n" is out of range. It must be a non-negative number. Received '+e+".");return this._maxListeners=e,this},l.prototype.getMaxListeners=function(){return p(this)},l.prototype.emit=function(e){for(var t=[],r=1;r<arguments.length;r++)t.push(arguments[r]);var o="error"===e,n=this._events;if(void 0!==n)o=o&&void 0===n.error;else if(!o)return!1;if(o){var a;if(t.length>0&&(a=t[0]),a instanceof Error)throw a;var s=new Error("Unhandled error."+(a?" ("+a.message+")":""));throw s.context=a,s}var l=n[e];if(void 0===l)return!1;if("function"===typeof l)i(l,this,t);else{var c=l.length,u=g(l,c);for(r=0;r<c;++r)i(u[r],this,t)}return!0},l.prototype.addListener=function(e,t){return f(this,e,t,!1)},l.prototype.on=l.prototype.addListener,l.prototype.prependListener=function(e,t){return f(this,e,t,!0)},l.prototype.once=function(e,t){return u(t),this.on(e,y(this,e,t)),this},l.prototype.prependOnceListener=function(e,t){return u(t),this.prependListener(e,y(this,e,t)),this},l.prototype.removeListener=function(e,t){var r,o,n,i,a;if(u(t),o=this._events,void 0===o)return this;if(r=o[e],void 0===r)return this;if(r===t||r.listener===t)0===--this._eventsCount?this._events=Object.create(null):(delete o[e],o.removeListener&&this.emit("removeListener",e,r.listener||t));else if("function"!==typeof r){for(n=-1,i=r.length-1;i>=0;i--)if(r[i]===t||r[i].listener===t){a=r[i].listener,n=i;break}if(n<0)return this;0===n?r.shift():b(r,n),1===r.length&&(o[e]=r[0]),void 0!==o.removeListener&&this.emit("removeListener",e,a||t)}return this},l.prototype.off=l.prototype.removeListener,l.prototype.removeAllListeners=function(e){var t,r,o;if(r=this._events,void 0===r)return this;if(void 0===r.removeListener)return 0===arguments.length?(this._events=Object.create(null),this._eventsCount=0):void 0!==r[e]&&(0===--this._eventsCount?this._events=Object.create(null):delete r[e]),this;if(0===arguments.length){var n,i=Object.keys(r);for(o=0;o<i.length;++o)n=i[o],"removeListener"!==n&&this.removeAllListeners(n);return this.removeAllListeners("removeListener"),this._events=Object.create(null),this._eventsCount=0,this}if(t=r[e],"function"===typeof t)this.removeListener(e,t);else if(void 0!==t)for(o=t.length-1;o>=0;o--)this.removeListener(e,t[o]);return this},l.prototype.listeners=function(e){return d(this,e,!0)},l.prototype.rawListeners=function(e){return d(this,e,!1)},l.listenerCount=function(e,t){return"function"===typeof e.listenerCount?e.listenerCount(t):m.call(e,t)},l.prototype.listenerCount=m,l.prototype.eventNames=function(){return this._eventsCount>0?o(this._events):[]}}}]);
|
2 |
+
//# sourceMappingURL=chunk-0cbfe13e.73856287.js.map
|
dist/js/chunk-117382e0.d47336d3.js
ADDED
@@ -0,0 +1,2 @@
|
|
|
|
|
|
|
1 |
+
(window["webpackJsonp"]=window["webpackJsonp"]||[]).push([["chunk-117382e0"],{"1e1a":function(e,t,a){"use strict";a("6b05")},"326e":function(e,t,a){var i=a("24fb");t=i(!1),t.push([e.i,".flex-engraver{width:100%;height:100vh}.flex-engraver header{position:absolute;width:100%;height:200px;background:#fafafa;display:flex;flex-direction:row;align-items:center;font-size:36px;overflow:hidden}.flex-engraver header>*{display:inline-block;font-size:inherit;margin:0 .5em}.flex-engraver header .source-list{min-width:8em}.flex-engraver header .gauge-view{height:100%}.flex-engraver header .dirty{font-weight:700;color:orange;cursor:pointer}.flex-engraver main{padding-top:200px;height:100%;background:#eee;white-space:nowrap}.flex-engraver main em{font-weight:700}.flex-engraver main .source-editor{height:calc(100% - 200px);vertical-align:top}.flex-engraver main .viewer{display:inline-block;position:relative;font-size:36px}.flex-engraver main .viewer .sheet-container{display:inline-block;resize:both;margin:2em;outline:1px solid #ccc;overflow:scroll;background:#fff}.flex-engraver main .viewer .sheet-container .sheet .page{margin:0}.flex-engraver main .viewer .sheet-container .loading-dots{background-color:transparent}.flex-engraver main .viewer .sheet-container .loading-dots .ellipsis{zoom:200%}.flex-engraver main .viewer .sheet-container .loading-dots .ellipsis>div{background-color:#4682b4}.flex-engraver main .viewer .container-size{display:inline-block;position:absolute;bottom:0;right:2em}.flex-engraver main .viewer .staff-size{display:inline-block;position:absolute;bottom:0;left:2em}.flex-engraver main .viewer .staff-size input{font-size:inherit}.flex-engraver main .viewer .staff-size .fit-staff-size{border:0}.flex-engraver main .viewer .staff-size .adjuster{display:inline-block;position:relative;margin:0 .6em;zoom:1.5}.flex-engraver main .viewer .staff-size .adjuster .slider{width:200px}.flex-engraver main .viewer .staff-size .adjuster .max,.flex-engraver main .viewer .staff-size .adjuster .min{display:inline-block;position:absolute;font-size:16px;top:3em;text-align:center}.flex-engraver main .viewer .staff-size .adjuster .max input,.flex-engraver main .viewer .staff-size .adjuster .min input{width:2em}.flex-engraver main .viewer .staff-size .adjuster .max .sheet,.flex-engraver main .viewer .staff-size .adjuster .min .sheet{position:absolute;top:120%;left:50%;transform:translate(-50%)}.flex-engraver main .viewer .staff-size .adjuster .min{left:0}.flex-engraver main .viewer .staff-size .adjuster .max{right:-2em}.flex-engraver main .staff-size-viewer{position:absolute;right:0;bottom:0}.flex-engraver main .staff-size-viewer input{width:4em}.flex-engraver.drag-hover header{background-color:#cfc;outline:4px dashed #4f4}",""]),e.exports=t},"6b05":function(e,t,a){var i=a("326e");i.__esModule&&(i=i.default),"string"===typeof i&&(i=[[e.i,i,""]]),i.locals&&(e.exports=i.locals);var n=a("499e").default;n("44bd81af",i,!0,{sourceMap:!1,shadowMode:!1})},e168:function(e,t,a){"use strict";a.r(t);var i=function(){var e=this,t=e._self._c;return t("div",{staticClass:"flex-engraver",class:{"drag-hover":e.dragHover},on:{dragover:function(t){t.preventDefault(),e.dragHover=!0},dragleave:function(t){e.dragHover=!1},drop:function(t){return t.preventDefault(),e.onDropFile.apply(null,arguments)}}},[t("header",[t("StoreInput",{directives:[{name:"show",rawName:"v-show",value:!1,expression:"false"}],attrs:{localKey:"lotus-flexEngraverContainerWidth"},model:{value:e.containerSize.offsetWidth,callback:function(t){e.$set(e.containerSize,"offsetWidth",t)},expression:"containerSize.offsetWidth"}}),t("StoreInput",{directives:[{name:"show",rawName:"v-show",value:!1,expression:"false"}],attrs:{localKey:"lotus-flexEngraverContainerHeight"},model:{value:e.containerSize.offsetHeight,callback:function(t){e.$set(e.containerSize,"offsetHeight",t)},expression:"containerSize.offsetHeight"}}),t("StoreInput",{directives:[{name:"show",rawName:"v-show",value:!1,expression:"false"}],attrs:{localKey:"lotus-flexEngraverChosenSourceIndex"},model:{value:e.chosenSourceIndex,callback:function(t){e.chosenSourceIndex=t},expression:"chosenSourceIndex"}}),t("StoreInput",{directives:[{name:"show",rawName:"v-show",value:!1,expression:"false"}],attrs:{localKey:"lotus-flexEngraverStaffSizeRangeMin"},model:{value:e.staffSizeRange.min,callback:function(t){e.$set(e.staffSizeRange,"min",t)},expression:"staffSizeRange.min"}}),t("StoreInput",{directives:[{name:"show",rawName:"v-show",value:!1,expression:"false"}],attrs:{localKey:"lotus-flexEngraverStaffSizeRangeMaX"},model:{value:e.staffSizeRange.max,callback:function(t){e.$set(e.staffSizeRange,"max",t)},expression:"staffSizeRange.max"}}),t("StoreInput",{directives:[{name:"show",rawName:"v-show",value:!1,expression:"false"}],attrs:{localKey:"lotus-flexEngraverFitStaffSize"},model:{value:e.fitStaffSize,callback:function(t){e.fitStaffSize=t},expression:"fitStaffSize"}}),t("BoolStoreInput",{directives:[{name:"show",rawName:"v-show",value:!1,expression:"false"}],attrs:{localKey:"lotus-flexEngraverFixStaffSize"},model:{value:e.fixStaffSize,callback:function(t){e.fixStaffSize=t},expression:"fixStaffSize"}}),t("select",{directives:[{name:"model",rawName:"v-model",value:e.chosenSourceIndex,expression:"chosenSourceIndex"}],staticClass:"source-list",on:{change:function(t){var a=Array.prototype.filter.call(t.target.options,(function(e){return e.selected})).map((function(e){var t="_value"in e?e._value:e.value;return t}));e.chosenSourceIndex=t.target.multiple?a:a[0]}}},e._l(e.sourceList,(function(a,i){return t("option",{key:i,domProps:{value:i}},[e._v(e._s(a.name))])})),0),t("span",{staticClass:"dirty",on:{click:e.saveSource}},[e._v(e._s(e.sourceDirty?"*":" "))]),t("button",{on:{click:e.removeCurrentSource}},[e._v("🗑")]),t("button",{on:{click:e.gauge}},[e._v("📏")]),t("button",{on:{click:e.renderSheet}},[e._v("🎼")]),t("button",{attrs:{title:"copy lilypond source"},on:{click:e.copySource}},[e._v("⎘")]),t("button",{attrs:{title:"export scores json"},on:{click:e.exportSourceList}},[e._v("⤓")]),e.gaugeSvgDoc?t("div",{staticClass:"gauge-view"},[e.gaugeSvgDoc?t("SheetSimple",{attrs:{documents:[e.gaugeSvgDoc]}}):e._e()],1):e._e()],1),t("main",[e.currentSource?t("SourceEditor",{attrs:{source:e.currentSource.content},on:{"update:source":function(t){return e.$set(e.currentSource,"content",t)}}}):e._e(),t("div",{staticClass:"viewer"},[t("div",{ref:"sheetContainer",staticClass:"sheet-container",style:{width:e.containerSize.offsetWidth+"px",height:e.containerSize.offsetHeight+"px"},on:{mousemove:e.updateContainerSize}},[e.containerSvgs?t("SheetSimple",{attrs:{documents:e.containerSvgs}}):e._e(),t("Loading",{directives:[{name:"show",rawName:"v-show",value:e.containerEngraving,expression:"containerEngraving"}]})],1),t("div",{staticClass:"container-size"},[t("span",[e._v(e._s(e.containerSize.width))]),e._v(" × "),t("span",[e._v(e._s(e.containerSize.height))])]),e.fitStaffSize?t("div",{staticClass:"staff-size"},[t("em",[e._v(e._s(e.fitStaffSize.toFixed(2)))]),e._v(" pt "),t("span",{staticClass:"adjuster"},[t("input",{directives:[{name:"model",rawName:"v-model",value:e.fixStaffSize,expression:"fixStaffSize"}],attrs:{type:"checkbox",title:"fix staff size"},domProps:{checked:Array.isArray(e.fixStaffSize)?e._i(e.fixStaffSize,null)>-1:e.fixStaffSize},on:{change:function(t){var a=e.fixStaffSize,i=t.target,n=!!i.checked;if(Array.isArray(a)){var r=null,s=e._i(a,r);i.checked?s<0&&(e.fixStaffSize=a.concat([r])):s>-1&&(e.fixStaffSize=a.slice(0,s).concat(a.slice(s+1)))}else e.fixStaffSize=n}}}),t("input",{directives:[{name:"model",rawName:"v-model.number",value:e.fitStaffSize,expression:"fitStaffSize",modifiers:{number:!0}}],staticClass:"slider",attrs:{type:"range",disabled:!e.fixStaffSize,min:e.staffSizeRange.min,max:e.staffSizeRange.max,step:"any"},domProps:{value:e.fitStaffSize},on:{change:e.delayRenderSheet,__r:function(t){e.fitStaffSize=e._n(t.target.value)},blur:function(t){return e.$forceUpdate()}}}),t("span",{staticClass:"min"},[t("input",{directives:[{name:"model",rawName:"v-model",value:e.staffSizeRange.min,expression:"staffSizeRange.min"}],attrs:{type:"number"},domProps:{value:e.staffSizeRange.min},on:{change:e.updateStaffSampleMin,input:function(t){t.target.composing||e.$set(e.staffSizeRange,"min",t.target.value)}}}),e.staffSampleSvgMin?t("SheetSimple",{attrs:{documents:[e.staffSampleSvgMin]}}):e._e()],1),t("span",{staticClass:"max"},[t("input",{directives:[{name:"model",rawName:"v-model",value:e.staffSizeRange.max,expression:"staffSizeRange.max"}],attrs:{type:"number"},domProps:{value:e.staffSizeRange.max},on:{change:e.updateStaffSampleMax,input:function(t){t.target.composing||e.$set(e.staffSizeRange,"max",t.target.value)}}}),e.staffSampleSvgMax?t("SheetSimple",{attrs:{documents:[e.staffSampleSvgMax]}}):e._e()],1)])]):e._e()])],1)])},n=[],r=a("8701"),s=a("5027"),o=a("010e"),c=a("bc9f"),l=a("f634"),f=a("32c1"),u=a("1f25"),h=a("3122"),g=a("cdaf"),S=a("4e72"),d=a("94c1"),v=a("488d");const m=e=>`\n\t\t#(set-global-staff-size ${e})\n\t\t\\paper {\n\t\t\tpaper-width = ${.16*e}\\cm\n\t\t\tpaper-height = ${.08*e}\\cm\n\t\t\ttop-margin = 0\n\t\t\tbottom-margin = 0\n\t\t\tleft-margin = 0.2\\cm\n\t\t\tright-margin = 0\n\t\t}\n\t\t\\layout\n\t\t{\n\t\t\tindent = 0\n\t\t}\n\t\t{c'1}\n\t`,p=e=>e.replace(/(?:>)[^<>]+lilypond.org(?=<)/g,"");var x={name:"flex-engraver",components:{SourceEditor:h["a"],StoreInput:g["a"],BoolStoreInput:S["a"],SheetSimple:d["a"],Loading:v["a"]},data(){return{containerSize:{width:100,height:100,offsetWidth:1215,offsetHeight:495},dragHover:!1,sourceList:[],chosenSourceIndex:0,sourceDirty:!1,gaugeSvgDoc:null,staffSizeRange:{min:10,max:40},containerSvgs:null,containerEngraving:!1,staffSampleSvgMin:null,staffSampleSvgMax:null,fitStaffSize:24,fixStaffSize:!1}},computed:{currentSource(){return this.sourceList[this.chosenSourceIndex]},currentSourceContent(){return this.currentSource&&this.currentSource.content},containerSizeHash(){return`${this.containerSize.width},${this.containerSize.height}`}},async created(){window.$main=this,this.loadSource(),this.lilyParser=await Object(o["a"])(a.e("chunk-2d0db258").then(a.t.bind(null,"6f2d",7))),console.log("Lilypond parser loaded."),this.updateStaffSamples()},async mounted(){await this.$nextTick(),this.updateContainerSize()},beforeDestroy(){this.checkAndSaveSource()},methods:{updateContainerSize({widthOffset:e=!0}={}){this.containerSize.width=this.$refs.sheetContainer.clientWidth,this.containerSize.height=this.$refs.sheetContainer.clientHeight,e&&(this.containerSize.offsetWidth=this.$refs.sheetContainer.offsetWidth,this.containerSize.offsetHeight=this.$refs.sheetContainer.offsetHeight)},async onDropFile(e){this.dragHover=!1;const t=e.dataTransfer.files[0];if(t)switch(t.type){case"text/x-lilypond":case"text/lilypond-source":const e=await t.readAs("Text"),a=t.name.replace(/\.ly$/,"");this.sourceList.push({name:a,content:e}),this.sourceDirty=!0;break}},removeCurrentSource(){this.sourceList.splice(this.chosenSourceIndex,1),this.chosenSourceIndex=Math.min(this.chosenSourceIndex,this.sourceList.length-1)},loadSource(){localStorage.lotusFlexEngraverSources&&(this.sourceList=JSON.parse(localStorage.lotusFlexEngraverSources),console.log("Source list loaded.")),this.sourceDirty=!1},saveSource(){localStorage.lotusFlexEngraverSources=JSON.stringify(this.sourceList),console.log("Source list saved."),this.sourceDirty=!1},checkAndSaveSource(){this.sourceDirty&&this.saveSource()},async gauge(){const e=20,t=1e4,a=2,i=new c["a"](await this.lilyParser.parse(this.currentSourceContent)),n=i.globalAttributes();n.staffSize.value=e,n.paperWidth.value.number=t,n.paperHeight.value.number=1e3,n.raggedLast.value=!0,n.topMargin.value=0,n.leftMargin.value=0;const r=i.toString();try{const s=await this.engrave(r,{tokenize:!0});this.gaugeSvgDoc=s.svgs[0],console.assert(1===s.svgs.length,"invalid page count:",s);const o=Object(l["b"])(s.doc,{StaffToken:f["StaffToken"],SheetDocument:f["SheetDocument"]}),u=o.pages[0].systems[0],h=t/o.pages[0].viewBox.width/e,g=u.width*h,S=(u.bottom-u.top)*h;n.paperWidth.value.number=a;const d=await this.engrave(i.toString(),{tokenize:!0});console.assert(1===d.svgs.length,"invalid page count:",d);const v=Object(l["b"])(d.doc,{StaffToken:f["StaffToken"],SheetDocument:f["SheetDocument"]}),m=v.pages[0].systems,p=Array(m.length-1).fill(null).map((e,t)=>m[t+1].y-m[t].y),x=Math.max(u.bottom-u.top,...p)*h-S,w=new c["a"](await this.lilyParser.parse(this.currentSourceContent));w.root.appendAssignment("naturalWidth",g),w.root.appendAssignment("naturalHeight",S),w.root.appendAssignment("systemSpacing",x),this.currentSource.content=w.toString(),this.checkAndSaveSource()}catch(s){console.warn("Engraving failed:",s)}},async engrave(e,{tokenize:t}={}){const a=new FormData;a.append("source",e),t&&a.append("tokenize",t);const i=await fetch("/engrave",{method:"POST",body:a});if(!i.ok)throw new Error(await i.text());return i.json()},async fitContainer(){if(!this.lilyParser||!this.currentSourceContent)return null;const e=new c["a"](await this.lilyParser.parse(this.currentSourceContent)),t=e.root.getField("naturalWidth"),a=e.root.getField("naturalHeight"),i=e.root.getField("systemSpacing");if(!t||!a||!i)return console.log("natural size is not set.",t,a),null;const n=t.value,r=a.value,s=i.value+.04,o=e.globalAttributes(),l=this.containerSize.width/u["a"],f=(this.containerSize.height-9)/u["a"],h=e=>o[e].value?o[e].value.number:null,g=h("leftMargin")||u["d"],S=h("rightMargin")||u["d"],d=h("topMargin")||u["f"],v=h("bottomMargin")||u["c"],m=f-d-v,p=l-g-S;let x=1,w=null;if(this.fixStaffSize){w=this.fitStaffSize;const e=(n-u["g"])*w/(p-u["g"]*w);x=Math.ceil(e-.2)}else{for(;x<1e3;++x){const e=m/(r*x+s*(x-1));if(e<this.staffSizeRange.min){Number.isFinite(w)||(w=e),--x,console.log("too samll vertical prefered staff size:",e,x);break}w=Math.min(e,this.staffSizeRange.max);const t=(n-u["g"])*w/(p-u["g"]*w);if(t<0)return void console.warn("Horizontal space too little:",t,p-u["g"]*w);if(t<x+.2){x=Math.max(Math.round(t),1),console.log("proper xsc:",t,x);break}}if(console.log("systemCount:",x,w),w<=this.staffSizeRange.min)return void console.warn("Vertical space too little:",w)}const z=(n-u["g"])*w/(p-u["g"]*w);console.log("horizontalNaturalCount:",z,x);const b=w*(r*x+s*(x-1)),y=Math.max(.9*(f-b)/2,0);if(o.topMargin.value={proto:"NumberUnit",number:y,unit:"\\cm"},1===x){const e=w*n,t=.9*(l-e)/2;t>1&&(o.leftMargin.value={proto:"NumberUnit",number:t,unit:"\\cm"},o.rightMargin.value={proto:"NumberUnit",number:t,unit:"\\cm"})}return o.staffSize.value=w,o.paperWidth.value={proto:"NumberUnit",number:l,unit:"\\cm"},o.paperHeight.value={proto:"NumberUnit",number:f,unit:"\\cm"},o.raggedLast.value=x<=1&&z<.8,this.fixStaffSize||(this.fitStaffSize=w),e.toString()},async renderSheet(){const e=await this.fitContainer();if(!e)return;this.containerEngraving=!0;const t=await this.engrave(e,{tokenize:!1});this.containerSvgs=t.svgs,this.containerSvgs=this.containerSvgs.map(p),this.containerEngraving=!1},async exportScore(){const e=await this.fitContainer();if(!e)return void console.warn("no source.");const t=await this.engrave(e,{tokenize:!0}),a={doc:Object(l["b"])(t.doc,{StaffToken:f["StaffToken"],SheetDocument:f["SheetDocument"]}),midi:t.midi,hashTable:t.hashTable},i=new Blob([JSON.stringify(a)]);Object(r["a"])(URL.createObjectURL(i),"score.json")},async copySource(){const e=await this.fitContainer();navigator.clipboard.writeText(e),console.log("Source copyed.")},async exportSourceList(){const e=JSON.stringify(this.sourceList),t=new Blob([e],{type:"text/plain"});Object(r["a"])(URL.createObjectURL(t),"FlexEngraverSources.json")},async delayRenderSheet(){await Object(s["b"])("renderSheet",500)&&this.renderSheet()},async engraveSample(e){const t=m(e),a=await this.engrave(t);return a.svgs[0]},async updateStaffSampleMin(){this.staffSampleSvgMin=p(await this.engraveSample(this.staffSizeRange.min))},async updateStaffSampleMax(){this.staffSampleSvgMax=p(await this.engraveSample(this.staffSizeRange.max))},updateStaffSamples(){this.updateStaffSampleMin(),this.updateStaffSampleMax()}},watch:{currentSourceContent(e,t){e&&void 0!==t&&(this.sourceDirty=!0)},async chosenSourceIndex(){this.gaugeSvgDoc=null,this.checkAndSaveSource(),await this.$nextTick(),this.sourceDirty=!1,this.renderSheet()},containerSizeHash:"delayRenderSheet",fixStaffSize(e){e||this.delayRenderSheet()}}},w=x,z=(a("1e1a"),a("2877")),b=Object(z["a"])(w,i,n,!1,null,null,null);t["default"]=b.exports}}]);
|
2 |
+
//# sourceMappingURL=chunk-117382e0.d47336d3.js.map
|
dist/js/chunk-2d0c53c7.d24941b8.js
ADDED
@@ -0,0 +1,2 @@
|
|
|
|
|
|
|
1 |
+
(window["webpackJsonp"]=window["webpackJsonp"]||[]).push([["chunk-2d0c53c7"],{"3ddd":function(C,J){C.exports="data:;base64,CiV7Cgljb25zdCByb290ID0gKHR5cGUsIGRhdGEpID0+ICh7X19wcm90b3R5cGU6ICJNZXNhdXJlTGF5b3V0IiwgdHlwZSwgZGF0YX0pOwoKCWNvbnN0IHNpbmdsZUxheW91dCA9IG4gPT4gKHtfX3Byb3RvdHlwZTogIlNpbmdsZU1MYXlvdXQiLCBtZWFzdXJlOiBOdW1iZXIobil9KTsKCWNvbnN0IGJsb2NrTGF5b3V0ID0gc2VxID0+ICh7X19wcm90b3R5cGU6ICJCbG9ja01MYXlvdXQiLCBzZXF9KTsKCWNvbnN0IHZvbHRhQmxvY2sgPSAodGltZXMsIGJvZHksIGFsdGVybmF0ZXMpID0+ICh7X19wcm90b3R5cGU6ICJWb2x0YU1MYXlvdXQiLCB0aW1lczogTnVtYmVyKHRpbWVzKSwgYm9keSwgYWx0ZXJuYXRlc30pOwoJY29uc3QgYWJhQmxvY2sgPSAobWFpbiwgcmVzdCkgPT4gKHtfX3Byb3RvdHlwZTogIkFCQU1MYXlvdXQiLCBtYWluLCByZXN0fSk7CgoJY29uc3Qgc2VnbWVudCA9IG4gPT4gKHtzZWdtZW50OiB0cnVlLCBsZW5ndGg6IE51bWJlcihuKX0pOwoKCWNvbnN0IGFsdGVybmF0ZXMgPSBpdGVtcyA9PiBpdGVtcy5tYXAoaXRlbSA9PiB7CgkJaWYgKGl0ZW0uX19wcm90b3R5cGUgPT09ICJCbG9ja01MYXlvdXQiKQoJCQlyZXR1cm4gaXRlbS5zZXE7CgoJCXJldHVybiBbaXRlbV07Cgl9KTsKCgljb25zdCByYW5nZSA9IChzdGFydCwgZW5kKSA9PiB7CgkJc3RhcnQgPSBOdW1iZXIoc3RhcnQpOwoJCWVuZCA9IE51bWJlcihlbmQpOwoKCQlpZiAoIShlbmQgPj0gc3RhcnQpKQoJCQl0aHJvdyBuZXcgRXJyb3IoYGludmFsaWQgbWVhc3VyZSByYW5nZTogJHtzdGFydH0uLiR7ZW5kfWApOwoKCQlyZXR1cm4gQXJyYXkoZW5kICsgMSAtIHN0YXJ0KS5maWxsKDApLm1hcCgoXywgaSkgPT4gc2luZ2xlTGF5b3V0KHN0YXJ0ICsgaSkpOwoJfTsKCgoJY29uc3Qgc2VyaWFsaXplU2VxID0gKGl0ZW0sIG9wdGlvbnMpID0+IHsKCQlpZiAoaXRlbS5zZWdtZW50KSB7CgkJCWNvbnN0IGluZGV4ID0gb3B0aW9ucy5pbmRleDsKCQkJb3B0aW9ucy5pbmRleCArPSBpdGVtLmxlbmd0aDsKCgkJCXJldHVybiBBcnJheShpdGVtLmxlbmd0aCkuZmlsbCgwKS5tYXAoKF8sIGkpID0+IHNpbmdsZUxheW91dChpbmRleCArIGkpKTsKCQl9CgoJCXJldHVybiBbc2VyaWFsaXplKGl0ZW0sIG9wdGlvbnMpXTsKCX07CgoJY29uc3Qgc2VyaWFsaXplID0gKGl0ZW0sIG9wdGlvbnMgPSB7aW5kZXg6IDF9KSA9PiB7CgkJY29uc3Qgc3BlYXJkID0gc2VxID0+IFtdLmNvbmNhdCguLi5zZXEubWFwKGl0ID0+IHNlcmlhbGl6ZVNlcShpdCwgb3B0aW9ucykpKTsKCgkJc3dpdGNoIChpdGVtLl9fcHJvdG90eXBlKSB7CgkJY2FzZSAiQmxvY2tNTGF5b3V0IjoKCQkJaXRlbS5zZXEgPSBzcGVhcmQoaXRlbS5zZXEpOwoKCQkJYnJlYWs7CgkJY2FzZSAiVm9sdGFNTGF5b3V0IjoKCQkJaXRlbS5ib2R5ID0gc3BlYXJkKGl0ZW0uYm9keSk7CgkJCWl0ZW0uYWx0ZXJuYXRlcyA9IGl0ZW0uYWx0ZXJuYXRlcyAmJiBpdGVtLmFsdGVybmF0ZXMubWFwKHNwZWFyZCk7CgoJCQlicmVhazsKCQljYXNlICJBQkFNTGF5b3V0IjoKCQkJaXRlbS5tYWluID0gc2VyaWFsaXplKGl0ZW0ubWFpbiwgb3B0aW9ucyk7CgkJCWl0ZW0ucmVzdCA9IHNwZWFyZChpdGVtLnJlc3QpOwoKCQkJYnJlYWs7CgkJfQoKCQlyZXR1cm4gaXRlbTsKCX07CiV9CgoKJWxleAoKJW9wdGlvbiBmbGV4IHVuaWNvZGUKCkEJCQkJCVthLXpdCk4JCQkJCVsxLTldCk4wCQkJCQlbMC05XQpVTlNJR05FRAkJCXtOfXtOMH0qCldPUkQJCQkJe0F9KwoKU1BFQ0lBTAkJCQlbKixcW1xdPD57fV0KCgolJQoKXHMrCQkJCQkJCQkJe30JLy8gc3BhY2VzCgp7U1BFQ0lBTH0JCQkJCQkJcmV0dXJuIHl5dGV4dDsKCntVTlNJR05FRH0JCQkJCQkJcmV0dXJuICdVTlNJR05FRCcKe1dPUkR9IjoiCQkJCQkJCXJldHVybiB5eXRleHQKIi4uIgkJCQkJCQkJcmV0dXJuIHl5dGV4dAoKPDxFT0Y+PgkJCQkJCQkJcmV0dXJuICdFT0YnOwoKCi9sZXgKCiVzdGFydCBzdGFydF9zeW1ib2wKCiUlCgpzdGFydF9zeW1ib2wKCTogbWVhc3VyZV9sYXlvdXQgRU9GCgkJewoJCQlyZXR1cm4gJDE7CgkJfQoJOwoKbWVhc3VyZV9sYXlvdXQKCTogaW5kZXhfd2lzZV9tZWFzdXJlX2xheW91dAoJCXskJCA9IHJvb3QobnVsbCwgJDEpO30KCXwgJ2k6JyBpbmRleF93aXNlX21lYXN1cmVfbGF5b3V0CgkJeyQkID0gcm9vdCgiaW5kZXgtd2lzZSIsICQyKTt9Cgl8ICdzOicgc2VnbWVudF93aXNlX21lYXN1cmVfbGF5b3V0CgkJeyQkID0gcm9vdCgic2VnbWVudC13aXNlIiwgc2VyaWFsaXplKCQyKSk7fQoJOwoKCmluZGV4X3dpc2VfbWVhc3VyZV9sYXlvdXQKCTogaXdfc2VxdWVuY2UKCQl7CgkJCWlmICgkMS5sZW5ndGggPT09IDEgJiYgJDFbMF0uX19wcm90b3R5cGUgPT09ICJCbG9ja01MYXlvdXQiKQoJCQkJJCQgPSAkMVswXTsKCQkJZWxzZQoJCQkJJCQgPSBibG9ja0xheW91dCgkMSk7CgkJfQoJOwoKaXdfc2VxdWVuY2UKCTogaXdfaXRlbQoJCXskJCA9IFskMV07fQoJfCByYW5nZQoJCXskJCA9ICQxO30KCXwgaXdfc2VxdWVuY2UgJywnIGl3X2l0ZW0KCQl7JCQgPSBbLi4uJDEsICQzXTt9Cgl8IGl3X3NlcXVlbmNlICcsJyByYW5nZQoJCXskJCA9IFsuLi4kMSwgLi4uJDNdO30KCTsKCnJhbmdlCgk6IFVOU0lHTkVEICcuLicgVU5TSUdORUQKCQl7JCQgPSByYW5nZSgkMSwgJDMpO30KCTsKCml3X2l0ZW0KCTogc2luZ2xlCgkJeyQkID0gJDE7fQoJfCBpd19ibG9ja19pdGVtCgkJeyQkID0gJDE7fQoJfCBpd192b2x0YQoJCXskJCA9ICQxO30KCXwgaXdfYWJhCgkJeyQkID0gJDE7fQoJOwoKc2luZ2xlCgk6IFVOU0lHTkVECgkJeyQkID0gc2luZ2xlTGF5b3V0KCQxKTt9Cgk7Cgppd19ibG9ja19pdGVtCgk6IGl3X2Jsb2NrCgkJeyQkID0gYmxvY2tMYXlvdXQoJDEpO30KCTsKCml3X2Jsb2NrCgk6ICdbJyBpd19zZXF1ZW5jZSAnXScKCQl7JCQgPSAkMjt9Cgk7Cgppd192b2x0YQoJOiBVTlNJR05FRCAnKicgaXdfYmxvY2sgaXdfb3B0aW9uYWxfYWx0ZXJuYXRlcwoJCXskJCA9IHZvbHRhQmxvY2soJDEsICQzLCAkNCk7fQoJOwoKaXdfb3B0aW9uYWxfYWx0ZXJuYXRlcwoJOiAlZW1wdHkKCQl7JCQgPSBudWxsO30KCXwgaXdfYWx0ZXJuYXRlcwoJCXskJCA9ICQxO30KCTsKCml3X2FsdGVybmF0ZXMKCTogJ3snIGl3X3NlcXVlbmNlICd9JwoJCXskJCA9IGFsdGVybmF0ZXMoJDIpO30KCTsKCml3X2FiYQoJOiAnPCcgaXdfaXRlbSAnLCcgaXdfc2VxdWVuY2UgJz4nCgkJeyQkID0gYWJhQmxvY2soJDIsICQ0KTt9Cgk7CgoKc2VnbWVudF93aXNlX21lYXN1cmVfbGF5b3V0Cgk6IHN3X3NlcXVlbmNlCgkJewoJCQlpZiAoJDEubGVuZ3RoID09PSAxICYmICQxWzBdLl9fcHJvdG90eXBlID09PSAiQmxvY2tNTGF5b3V0IikKCQkJCSQkID0gJDFbMF07CgkJCWVsc2UKCQkJCSQkID0gYmxvY2tMYXlvdXQoJDEpOwoJCX0KCTsKCnN3X3NlcXVlbmNlCgk6IHN3X2l0ZW0KCQl7JCQgPSBbJDFdO30KCXwgc3dfc2VxdWVuY2Ugc3dfaXRlbQoJCXskJCA9IFsuLi4kMSwgJDJdO30KCTsKCnN3X2l0ZW0KCTogc2VnbWVudAoJCXskJCA9IGJsb2NrTGF5b3V0KFskMV0pO30KCXwgc3dfYmxvY2tfaXRlbQoJCXskJCA9ICQxO30KCXwgc3dfdm9sdGEKCQl7JCQgPSAkMTt9Cgl8IHN3X2FiYQoJCXskJCA9ICQxO30KCTsKCnNlZ21lbnQKCTogVU5TSUdORUQKCQl7JCQgPSBzZWdtZW50KCQxKTt9Cgk7Cgpzd19ibG9ja19pdGVtCgk6IHN3X2Jsb2NrCgkJeyQkID0gYmxvY2tMYXlvdXQoJDEpO30KCTsKCnN3X2Jsb2NrCgk6ICdbJyBzd19zZXF1ZW5jZSAnXScKCQl7JCQgPSAkMjt9Cgk7Cgpzd192b2x0YQoJOiBVTlNJR05FRCAnKicgc3dfYmxvY2sgc3dfb3B0aW9uYWxfYWx0ZXJuYXRlcwoJCXskJCA9IHZvbHRhQmxvY2soJDEsICQzLCAkNCk7fQoJOwoKc3dfb3B0aW9uYWxfYWx0ZXJuYXRlcwoJOiAlZW1wdHkKCQl7JCQgPSBudWxsO30KCXwgc3dfYWx0ZXJuYXRlcwoJCXskJCA9ICQxO30KCTsKCnN3X2FsdGVybmF0ZXMKCTogJ3snIHN3X3NlcXVlbmNlICd9JwoJCXskJCA9IGFsdGVybmF0ZXMoJDIpO30KCTsKCnN3X2FiYQoJOiAnPCcgc3dfaXRlbSBzd19zZXF1ZW5jZSAnPicKCQl7JCQgPSBhYmFCbG9jaygkMiwgJDMpO30KCTsK"}}]);
|
2 |
+
//# sourceMappingURL=chunk-2d0c53c7.d24941b8.js.map
|
dist/js/chunk-2d0db258.a4804a7a.js
ADDED
@@ -0,0 +1,2 @@
|
|
|
|
|
|
|
1 |
+
(window["webpackJsonp"]=window["webpackJsonp"]||[]).push([["chunk-2d0db258"],{"6f2d":function(C,J){C.exports="data:;base64,
%{
	const preferNumber = x => Number.isFinite(Number(x)) ? Number(x) : x;

	const location = (begin, end) => ({proto: "_PLAIN", lines: [begin.first_line, end.last_line], columns: [begin.first_column, end.last_column]});

	const commandSubProtos = {
		markup: "MarkupCommand",
		repeat: "Repeat",
		relative: "Relative",
		parallelMusic: "ParallelMusic",
		time: "TimeSignature",
		partial: "Partial",
		times: "Times",
		tuplet: "Tuplet",
		clef: "Clef",
		key: "KeySignature",
		ottava: "OctaveShift",
		include: "Include",
		version: "Version",
		lyricmode: "LyricMode",
		chordmode: "ChordMode",
		grace: "Grace",
		acciaccatura: "Grace",
		appoggiatura: "Grace",
		slashedGrace: "Grace",
		afterGrace: "AfterGrace",
		language: "Language",
		transposition: "Transposition",
		stemUp: "StemDirection",
		stemDown: "StemDirection",
		stemNeutral: "StemDirection",
		change: "Change",
	};


	const root = (sections = []) => ({proto: "Root", sections});

	const appendSection = (list, item) => {
		list.sections.push(item);

		return list;
	};

	const string = exp => ({proto: "LiteralString", exp});

	const command = (cmd, ...args) => ({proto: commandSubProtos[cmd.substr(1)] || "Command", cmd: cmd.substr(1), args});

	const variable = name => ({proto: "Variable", name: name.substr(1)});

	const chord = (pitches, duration, {locations, post_events, ...options} = {}) => ({proto: "Chord", pitches, duration, post_events, _location: location(...locations), options: {...options, proto: "_PLAIN"}});

	const rest = ({name, duration, post_events = null, locations}) => ({proto: "Rest", name, duration, post_events, _location: location(...locations)});

	const chordElem = (pitch, {locations, ...options}) => ({proto: "ChordElement", pitch, _location: location(...locations), options: {...options, proto: "_PLAIN"}});

	const briefChord = (body, {locations, post_events = null} = {}) => ({proto: "BriefChord", body: {...body, proto: "_PLAIN"}, post_events, _location: location(...locations)});

	const block = (block, head, body = []) => ({proto: "Block", block, head, body});

	const inlineBlock = body => ({proto: "InlineBlock", body});

	const scheme = exp => ({proto: "Scheme", exp});

	const schemeFunction = (func, args) => ({proto: "SchemeFunction", func, args});

	const schemePair = (left, right) => ({proto: "SchemePair", left, right});

	const schemePointer = value => ({proto: "SchemePointer", value});

	const schemeEmbed = value => ({proto: "SchemeEmbed", value});

	const assignment = (key, value) => ({proto: "Assignment", key, value});

	const numberUnit = (number, unit) => ({proto: "NumberUnit", number: preferNumber(number), unit});

	const musicBlock = body => ({proto: "MusicBlock", body});

	const simultaneousList = list => ({proto: "SimultaneousList", list});

	const contextedMusic = (head, body, lyrics) => ({proto: "ContextedMusic", head, body, lyrics});

	const tempo = (beatsPerMinute, unit, text) => ({proto: "Tempo", beatsPerMinute: preferNumber(beatsPerMinute), unit: preferNumber(unit), text});

	const postEvent = (direction, arg) => ({proto: "PostEvent", direction, arg});

	const fingering = value => ({proto: "Fingering", value: preferNumber(value)});

	const markup = (head, body) => ({proto: "Markup", head, body});

	const lyric = (content, {locations, ...options}) => ({proto: "Lyric", content, _location: location(...locations), ...options});

	const duration = ({number, dots, multipliers}) => ({proto: "Duration", number, dots, multipliers});

	const comment = ({loc, ...data}) => ({proto: "Comment", _location: location(loc, loc), ...data});


	let lineHeadTable = {};
	let lineTailTable = {};

	const lineHead = (loc, term) => {
		if (!term || typeof term !== "object")
			return;

		if (!lineHeadTable[loc.first_line] || lineHeadTable[loc.first_line].column > loc.first_column)
			lineHeadTable[loc.first_line] = {column: loc.first_column, term};
	};

	const lineTail = (loc, term) => {
		if (!term || typeof term !== "object")
			return;

		if (!lineTailTable[loc.last_line] || lineTailTable[loc.last_line].column < loc.last_column)
			lineTailTable[loc.last_line] = {column: loc.last_column, term};
	};

	const lineRegister = (loc, term) => {
		lineHead(loc, term);
		lineTail(loc, term);
	};


	const attachComments = yy => {
		if (yy.$lotusComments && yy.$lotusComments.length > 0) {
			//console.log("attachComments:", yy.$lotusComments, lineTable);
			const headLineNumbers = Object.keys(lineHeadTable).map(Number);

			yy.$lotusComments.forEach(data => {
				const comm = comment(data);

				const tailLine = lineTailTable[data.loc.first_line];
				if (tailLine) {
					tailLine.term._tailComment = comm;
					return;
				}

				const line = headLineNumbers.find(line => line >= data.loc.last_line);
				if (Number.isFinite(line) && lineHeadTable[line]) {
					lineHeadTable[line].term._headComment = comm;
					return;
				}

				// TODO: attach on root's tail
			});
		}

		// reset parser states
		lineHeadTable = {};
		lineTailTable = {};
	};
%}


%lex

%option flex unicode

A					[a-zA-Z\200-\377]
AA					{A}|_
N					[0-9]
ANY_CHAR			(.|\n)
SYMBOL				{A}([-_]{A}|{A})*
COMMAND				\\{SYMBOL}
/* SPECIAL category is for every letter that needs to get passed to
 * the parser rather than being redefinable by the user */
SPECIAL				[-+*/=<>{}!?_^'',.:]
SHORTHAND			(.|\\.)
UNSIGNED			{N}+
E_UNSIGNED			\\{N}+
FRACTION			{N}+\/{N}+
//INT					[-]?{UNSIGNED}
//REAL				({INT}\.{N}*)|([-]?\.{N}+)
STRICTREAL			{UNSIGNED}\.{UNSIGNED}
WHITE				[ \n\t\f\r]
HORIZONTALWHITE		[ \t]
BLACK				[^ \n\t\f\r]
RESTNAME			[rRs](?=[\W\d_])
ESCAPED				[nt\\''""]
//EXTENDER			\_\_
//HYPHEN				\-\-
PRE_EXTENDER		\_(?=\_)
PRE_HYPHEN			\-(?=\-)
BOM_UTF8			\357\273\277

PHONET				[abcdefgqh]
PITCH				{PHONET}(([i][s])*|([e][s])*|[s][e][s]|[s]*|[f]*)(?=[\W\d_])
//PLACEHOLDER_PITCH	[s](?=[\W\d_^-])
//DURATION			"1"|"2"|"4"|"8"|"16"|"32"|"64"|"128"|"256"

//UNICODE_HAN			[\p{Script=Han}]

%%

// extra lex
// TODO: parse the dollar expression details
[$][(][^()]*[)]						return 'DOLLAR_SCHEME_EXPRESSION'

\s+									{}	// spaces
\%\{(.|\n)*?\%\}					yy.$lotusComments = yy.$lotusComments || []; yy.$lotusComments.push({text: yytext, loc: yylloc, scoped: true});	// scoped comments
\%.*								yy.$lotusComments = yy.$lotusComments || []; yy.$lotusComments.push({text: yytext, loc: yylloc});	// scoped comments
\"(\\\"|[^"])*\"					return 'STRING';

//{EXTENDER}							return 'EXTENDER';
//{HYPHEN}							return 'HYPHEN';
{PRE_EXTENDER}						return 'PRE_EXTENDER';
{PRE_HYPHEN}						return 'PRE_HYPHEN';

//"/+"								return CHORD_BASS;
//"^"								return CHORD_CARET;
//":"								return CHORD_COLON;
//"-"								return CHORD_MINUS;
//"/"								return CHORD_SLASH;

//"<"								return 'ANGLE_OPEN';
//">"								return 'ANGLE_CLOSE';
"<<"								return 'DOUBLE_ANGLE_OPEN';
">>"								return 'DOUBLE_ANGLE_CLOSE';

"\\\\"								return 'E_BACKSLASH';

{E_UNSIGNED}						return 'E_UNSIGNED';

"\\new"								return 'NEWCONTEXT';

"\\cm"								return 'CENTIMETER';
"\\mm"								return 'MILLIMETER';

"\\overrideProperty"				return 'OVERRIDEPROPERTY';

// binary commands
"\\relative"						return 'CMD_RELATIVE';
"\\tweak"							return 'CMD_TWEAK';
"\\key"								return 'CMD_KEY';
//"\\times"							return 'CMD_TIMES';
[\\][t][i][m][e][s]					return 'CMD_TIMES';
"\\afterGrace"						return 'CMD_AFTERGRACE';
"\\parallelMusic"					return 'CMD_PARALLELMUSIC';
"\\shape"							return 'CMD_SHAPE';
"\\tag"								return 'CMD_TAG';
"\\scaleDurations"					return 'CMD_SCALEDURATIONS';

// unitary commands
"\\clef"							return 'CMD_CLEF';
"\\time"							return 'CMD_TIME';
"\\stemUp"							return 'CMD_STEMUP';
"\\stemDown"						return 'CMD_STEMDOWN';
"\\stemNeutral"						return 'CMD_STEMNEUTRAL';
"\\bar"								return 'CMD_BAR';
"\\omit"							return 'CMD_OMIT';
"\\ottava"							return 'CMD_OTTAVA';
"\\barNumberCheck"					return 'CMD_BARNUMBERCHECK';
"\\partial"							return 'CMD_PARTIAL';
"\\mark"							return 'CMD_MARK';
"\\include"							return 'CMD_INCLUDE';
"\\tupletSpan"						return 'CMD_TUPLETSPAN';
"\\tuplet"							return 'CMD_TUPLET';
"\\tupletNeutral"					return 'CMD_TUPLETNEUTRAL';
"\\skip"							return 'CMD_SKIP';
"\\skip"(?=\d)						return 'CMD_SKIP';
"\\parenthesize"					return 'CMD_PARENTHESIZE';
"\\unfoldRepeats"					return 'CMD_UNFOLDREPEATS';
"\\grace"							return 'CMD_GRACE';
"\\acciaccatura"					return 'CMD_ACCIACCATURA';
"\\appoggiatura"					return 'CMD_APPOGGIATURA';
"\\slashedGrace"					return 'CMD_SLASHEDGRACE';
"\\language"						return 'CMD_LANGUAGE';
"\\once"							return 'CMD_ONCE';
"\\accidentalStyle"					return 'CMD_ACCIDENTALSTYLE';
"\\numericTimeSignature"			return 'CMD_NUMERICTIMESIGNATURE';
"\\defaultTimeSignature"			return 'CMD_DEFAULTTIMESIGNATURE';
"\\bendAfter"						return 'CMD_BENDAFTER';
"\\compoundMeter"					return 'CMD_COMPOUNDMETER';
"\\transposition"					return 'CMD_TRANSPOSITION';
"\\absolute"						return 'CMD_ABSOLUTE';
"\\hide"							return 'CMD_HIDE';
"\\crossStaff"						return 'CMD_CROSSSTAFF';
"\\keepWithTag"						return 'CMD_KEEPWITHTAG';
"\\articulate"						return 'CMD_ARTICULATE';

// zero commands
"\\tempoLegend"						return 'CMD_TEMPOLEGEND';
"\\fermata"							return 'CMD_FERMATA';
"\\mergeDifferentlyDottedOn"		return 'CMD_MERGEDIFFERENTLYDOTTEDON';
"\\mergeDifferentlyHeadedOn"		return 'CMD_MERGEDIFFERENTLYHEADEDON';
"\\voiceOne"						return 'CMD_VOICE_NUMBER';
"\\voiceTwo"						return 'CMD_VOICE_NUMBER';
"\\voiceThree"						return 'CMD_VOICE_NUMBER';
"\\voiceFour"						return 'CMD_VOICE_NUMBER';
//"\\voiceFive"						return 'CMD_VOICE_NUMBER';
"\\Score"							return 'CMD_SCORE';
"\\Voice"							return 'CMD_VOICE';
"\\Staff"							return 'CMD_STAFF';
"\\PianoStaff"						return 'CMD_PIANOSTAFF';
"\\arpeggio"						return 'CMD_ARPEGGIO';
"\\arpeggioArrowDown"				return 'CMD_ARPEGGIOARROWDOWN';
"\\arpeggioArrowUp"					return 'CMD_ARPEGGIOARROWUP';
"\\arpeggioNormal"					return 'CMD_ARPEGGIONORMAL';
"\\arpeggioBracket"					return 'CMD_ARPEGGIOBRACKET';
"\\arpeggioParenthesis"				return 'CMD_ARPEGGIOPARENTHESIS';
"\\arpeggioParenthesisDashed"		return 'CMD_ARPEGGIOPARENTHESISDASHED';
"\\glissando"						return 'CMD_GLISSANDO';
"\\mordent"							return 'CMD_MORDENT';
"\\musicglyph"						return 'CMD_MUSICGLYPH';
"\\powerChords"						return 'CMD_POWERCHORDS';
"\\prall"							return 'CMD_PRALL';
"\\sustainOff"						return 'CMD_SUSTAINOFF';
"\\sustainOn"						return 'CMD_SUSTAINON';
"\\trill"							return 'CMD_TRILL';
"\\turn"							return 'CMD_TURN';
"\\pointAndClickOff"				return 'CMD_POINTANDCLICKOFF';
"\\upbow"							return 'CMD_UPBOW';
"\\downbow"							return 'CMD_DOWNBOW';
"\\breathe"							return 'CMD_BREATHE';
"\\startTextSpan"					return 'CMD_STARTTEXTSPAN';
"\\stopTextSpan"					return 'CMD_STOPTEXTSPAN';
"\\flageolet"						return 'CMD_FLAGEOLET';
"\\slurDashed"						return 'CMD_SLURDASHED';
"\\slurSolid"						return 'CMD_SLURSOLID';
"\\break"							return 'CMD_BREAK';
"\\pageBreak"						return 'CMD_PAGEBREAK';
"\\startTrillSpan"					return 'CMD_STARTTRILLSPAN';
"\\stopTrillSpan"					return 'CMD_STOPTRILLSPAN';
"\\cadenzaOn"						return 'CMD_CADENZAON';
"\\cadenzaOff"						return 'CMD_CADENZAOFF';
"\\cresc"							return 'CMD_CRESC';
"\\crescTextCresc"					return 'CMD_CRESCTEXTCRESC';
"\\crescHairpin"					return 'CMD_CRESCHAIRPIN';
"\\dim"(?=[\W])						return 'CMD_DIM';
"\\dimTextDim"						return 'CMD_DIMTEXTDIM';
"\\dynamicUp"						return 'CMD_DYNAMICUP';
"\\hideNotes"						return 'CMD_HIDENOTES';
"\\unHideNotes"						return 'CMD_UNHIDENOTES';
"\\newSpacingSection"				return 'CMD_NEWSPACINGSECTION';
"\\noBeam"							return 'CMD_NOBEAM';
"\\oneVoice"						return 'CMD_ONEVOICE';
"\\phrasingSlurDown"				return 'CMD_PHRASINGSLURDOWN';
"\\phrasingSlurNeutral"				return 'CMD_PHRASINGSLURNEUTRAL';
"\\phrasingSlurUp"					return 'CMD_PHRASINGSLURUP';
"\\slurDown"						return 'CMD_SLURDOWN';
"\\slurNeutral"						return 'CMD_SLURNEUTRAL';
"\\slurUp"							return 'CMD_SLURUP';
"\\tieDown"							return 'CMD_TIEDOWN';
"\\tieNeutral"						return 'CMD_TIENEUTRAL';
"\\tieUp"							return 'CMD_TIEUP';
"\\tupletUp"						return 'CMD_TUPLETUP';
"\\tupletDown"						return 'CMD_TUPLETDOWN';
"\\shiftOn"							return 'CMD_SHIFTON';
"\\repeatTie"						return 'CMD_REPEATTIE';
"\\marcato"							return 'CMD_MARCATO';
"\\stopped"							return 'CMD_STOPPED';
"\\tenuto"							return 'CMD_TENUTO';
"\\staccatissimo"					return 'CMD_STACCATISSIMO';
"\\accent"							return 'CMD_ACCENT';
"\\staccato"						return 'CMD_STACCATO';
"\\portato"							return 'CMD_PORTATO';

"\\mp"(?=[\W])						return 'CMD_DYNAMIC_MARKINGS';
"\\mf"(?=[\W])						return 'CMD_DYNAMIC_MARKINGS';
"\\"[p]+(?=[\W])					return 'CMD_DYNAMIC_MARKINGS';
"\\"[f]+(?=[\W])					return 'CMD_DYNAMIC_MARKINGS';
"\\sf"(?=[\W])						return 'CMD_DYNAMIC_MARKINGS';
"\\sff"(?=[\W])						return 'CMD_DYNAMIC_MARKINGS';
"\\sfp"(?=[\W])						return 'CMD_DYNAMIC_MARKINGS';
"\\sfpp"(?=[\W])					return 'CMD_DYNAMIC_MARKINGS';
"\\fp"(?=[\W])						return 'CMD_DYNAMIC_MARKINGS';
"\\mfp"(?=[\W])						return 'CMD_DYNAMIC_MARKINGS';
"\\rf"(?=[\W])						return 'CMD_DYNAMIC_MARKINGS';
"\\rfz"(?=[\W])						return 'CMD_DYNAMIC_MARKINGS';
"\\sfz"(?=[\W])						return 'CMD_DYNAMIC_MARKINGS';
"\\sffz"(?=[\W])					return 'CMD_DYNAMIC_MARKINGS';
"\\fz"(?=[\W])						return 'CMD_DYNAMIC_MARKINGS';
"\\fzp"(?=[\W])						return 'CMD_DYNAMIC_MARKINGS';
"\\ffz"(?=[\W])						return 'CMD_DYNAMIC_MARKINGS';
"\\ppz"(?=[\W])						return 'CMD_DYNAMIC_MARKINGS';

"\\breve"							return 'CMD_BREVE';
"\\longa"							return 'CMD_LONGA';

// markup commands		
"\\version"							return 'CMD_VERSION';
"\\column"							return 'CMD_COLUMN';
"\\line"							return 'CMD_LINE';
"\\bold"							return 'CMD_BOLD';
"\\italic"							return 'CMD_ITALIC';
"\\box"								return 'CMD_BOX';
"\\whiteout"						return 'CMD_WHITEOUT';
"\\dynamic"							return 'CMD_DYNAMIC';
"\\abs-fontsize"					return 'CMD_ABS_FONTSIZE';
"\\with-color"						return 'CMD_WITH_COLOR';
"\\char"							return 'CMD_CHAR';
"\\center-column"					return 'CMD_CENTER_COLUMN';
"\\right-column"					return 'CMD_RIGHT_COLUMN';
"\\with-url"						return 'CMD_WITH_URL';
"\\sans"							return 'CMD_SANS';
"\\concat"							return 'CMD_CONCAT';
"\\maintainer"						return 'CMD_MAINTAINER';
"\\footnote"						return 'CMD_FOOTNOTE';
"\\natural"							return 'CMD_NATURAL';
"\\flat"							return 'CMD_FLAT';
"\\sharp"							return 'CMD_SHARP';
"\\hspace"							return 'CMD_HSPACE';
"\\footer"							return 'CMD_FOOTER';
"\\center-align"					return 'CMD_CENTER_ALIGN';
"\\right-align"						return 'CMD_RIGHT_ALIGN';
"\\general-align"					return 'CMD_GENERAL_ALIGN';
"\\lower"							return 'CMD_LOWER';
"\\finger"							return 'CMD_FINGER';
"\\fontsize"						return 'CMD_FONTSIZE';
"\\raise"							return 'CMD_RAISE';
"\\note"							return 'CMD_NOTE';
"\\circle"							return 'CMD_CIRCLE';
"\\pad-markup"						return 'CMD_PAD_MARKUP';
"\\smaller"							return 'CMD_SMALLER';
"\\normal-text"						return 'CMD_NORMAL_TEXT';

"\\huge"							return 'CMD_HUGE';
"\\large"							return 'CMD_LARGE';
"\\normalsize"						return 'CMD_NORMALSIZE';
"\\small"							return 'CMD_SMALL';
"\\tiny"							return 'CMD_TINY';
"\\teeny"							return 'CMD_TEENY';
"\\medium"							return 'CMD_MEDIUM';

// syntax commands		
"\\header"							return 'HEADER';
"\\markup"							return 'MARKUP';
"\\markuplist"						return 'MARKUPLIST';
"\\repeat"							return 'REPEAT';
"\\context"							return 'CONTEXT';
"\\accepts"							return 'ACCEPTS';
"\\addlyrics"						return 'ADDLYRICS';
"\\alias"							return 'ALIAS';
"\\alternative"						return 'ALTERNATIVE';
"\\book"							return 'BOOK';
"\\bookpart"						return 'BOOKPART';
"\\change"							return 'CHANGE';
"\\chordmode"						return 'CHORDMODE';
"\\chords"							return 'CHORDS';
"\\consists"						return 'CONSISTS';
"\\default"							return 'DEFAULT';
"\\defaultchild"					return 'DEFAULTCHILD';
"\\denies"							return 'DENIES';
"\\description"						return 'DESCRIPTION';
"\\drummode"						return 'DRUMMODE';
"\\drums"							return 'DRUMS';
"\\etc"								return 'ETC';
"\\figuremode"						return 'FIGUREMODE';
"\\figures"							return 'FIGURES';
"\\version-error"					return 'INVALID';
"\\layout"							return 'LAYOUT';
"\\lyricmode"						return 'LYRICMODE';
"\\lyrics"							return 'LYRICS';
"\\lyricsto"						return 'LYRICSTO';
"\\midi"							return 'MIDI';
"\\name"							return 'NAME';
"\\notemode"						return 'NOTEMODE';
"\\override"						return 'OVERRIDE';
"\\paper"							return 'PAPER';
"\\remove"							return 'REMOVE';
"\\rest"							return 'REST';
"\\revert"							return 'REVERT';
"\\score"							return 'SCORE';
"\\score-lines"						return 'SCORELINES';
"\\sequential"						return 'SEQUENTIAL';
"\\set"								return 'SET';
"\\simultaneous"					return 'SIMULTANEOUS';
"\\tempo"							return 'TEMPO';
"\\type"							return 'TYPE';
"\\unset"							return 'UNSET';
"\\with"							return 'WITH';

// simple commands
"\\<"								return 'CMD_CRESCENDO_BEGIN';
"\\>"								return 'CMD_DECRESCENDO_BEGIN';
"\\!"								return 'CMD_DYNAMICS_END';

{COMMAND}							return 'COMMAND';

{PITCH}								return 'PITCH';
//{PLACEHOLDER_PITCH}					return 'PLACEHOLDER_PITCH';
//{UNSIGNED}						return 'POST_UNSIGNED';
{RESTNAME}							return 'RESTNAME';

{FRACTION}							return 'FRACTION';
//{REAL}							return 'REAL';
{UNSIGNED}							return 'UNSIGNED';

//{INT}								return 'INT';

// CHORD_MODIFIER
[m][a][j](?=[\W\d])					return 'CHORD_MODIFIER_WORD';
m(?=[\W\d])							return 'CHORD_MODIFIER_WORD';
[a][u][g](?=[\W\d])					return 'CHORD_MODIFIER_WORD';
[d][i][m](?=[\W\d])					return 'CHORD_MODIFIER_WORD';
[s][u][s](?=[\W\d])					return 'CHORD_MODIFIER_WORD';

"inf.0"								return yytext;

{SYMBOL}							return 'SYMBOL';

"#f"								return 'SCM_FALSE';
"#t"								return 'SCM_TRUE';

"#x"[\da-fA-F]+						return 'SCM_HEX';

"#:"{SYMBOL}						return 'SCM_COLON';

"\\("								return yytext;
"\\)"								return yytext;

\.(?=\d)							return 'DOT_NUMBER_R';
//(?<=\d)\.							return 'DOT_NUMBER_L';

{SPECIAL}							return yytext;
\|									return 'DIVIDE';

[()]								return yytext;

"["									return yytext;
"]"									return yytext;

"#"									return yytext;
"~"									return yytext;
"`"									return yytext;

.									return 'UNKNOWN_CHAR';

<<EOF>>								return 'EOF';


/lex

%start start_symbol

%%

start_symbol
	: lilypond EOF
		{
			attachComments(yy);
			return $1;
		}
	//| embedded_lilypond
	;

lilypond
	: %empty
		{$$ = root();}
	| version
		{$$ = root([$1]); lineRegister(@1, $1);}
	| lilypond toplevel_expression
		{$$ = appendSection($1, $2); lineRegister(@2, $2);}
	| lilypond assignment
		{$$ = appendSection($1, $2); lineRegister(@2, $2);}
	;

version
	: CMD_VERSION literal_string
		{$$ = command($1, $2);}
	;

toplevel_expression
	: header_block
		{$$ = $1;}
	| composite_music
		{$$ = $1;}
	| full_markup
		{$$ = $1;}
	| output_def
		{$$ = $1;}
	| score_block
		{$$ = $1;}
	| book_block
		{$$ = $1;}
	| scm_identifier
		{$$ = $1;}
	//| full_markup_list
	//	{$$ = $1;}
	//| bookpart_block
	//| BOOK_IDENTIFIER
	//| SCM_TOKEN
	//| embedded_scm_active
	;

score_block
	: SCORE '{' score_body '}'
		{$$ = block("score", $1, $3);}
	;

book_block
	: BOOK '{' book_body '}'
		{$$ = block("book", $1, $3);}
	;

book_body
	: %empty
		{$$ = [];}
	//| BOOK_IDENTIFIER
	| book_body paper_block
		{$$.push($2);}
	//| book_body bookpart_block
	//	{$$.push($2);}
	| book_body score_block
		{$$.push($2);}
	| book_body composite_music
		{$$.push($2);}
	| book_body full_markup
		{$$.push($2);}
	| book_body full_markup_list
		{$$.push($2);}
	//| book_body SCM_TOKEN
	| book_body embedded_scm_active
		{$$.push($2);}
	| book_body lilypond_header
		{$$.push($2);}
	//| book_body error
	;

paper_block
	: output_def
		{$$ = $1;}
	;

header_block
	: lilypond_header
		{$$ = $1;}
	;

lilypond_header
	: HEADER '{' lilypond_header_body '}'
		{$$ = block("header", $1, $3);}
	;

lilypond_header_body
	: %empty
		{$$ = [];}
	| lilypond_header_body assignment
		{$$.push($2); lineRegister(@2, $2);}
	//| lilypond_header_body SCM_TOKEN
	//| lilypond_header_body embedded_scm_active
	;

assignment
	: assignment_id '=' identifier_init
		{$$ = assignment($1, $3);}
	| assignment_id '.' property_path '=' identifier_init
		{$$ = assignment($1 + "." + $3, $5);}
	//| markup_mode_word '=' identifier_init
	;

assignment_id
	: literal_string
		{$$ = $1;}
	| SYMBOL
		{$$ = $1;}
	// extra formula
	| PITCH
		{$$ = $1;}
	// extra formula
	| CHORD_MODIFIER_WORD
		{$$ = $1;}
	;

property_path
	: symbol_list_rev
		{$$ = $1;}
	;

symbol_list_rev
	: symbol_list_part
		{$$ = $1;}
	| symbol_list_rev '.' symbol_list_part
		{$$ = $1 + "." + $3;}
	| symbol_list_rev ',' symbol_list_part
		{$$ = $1 + "," + $3;}
	;

symbol_list_part
	: symbol_list_part_bare
		{$$ = $1;}
	| embedded_scm_bare
		{$$ = $1;}
	;

symbol_list_part_bare
	: SYMBOL
		{$$ = $1;}
	| symbol_list_element
		{$$ = $1;}
	;

symbol_list_element
	: literal_string
		{$$ = $1;}
	| UNSIGNED
		{$$ = $1;}
	;

identifier_init
	: identifier_init_nonumber
		{$$ = $1;}
	| number_expression
		{$$ = $1;}
	//| symbol_list_part_bare '.' property_path
	//	{$$ = $1 + "." + $3;}
	//| symbol_list_part_bare ',' property_path
	//	{$$ = $1 + "," + $3;}
	| post_event_nofinger post_events
		{$$ = [$1, $2];}
	;

number_expression
	: number_expression '+' number_term
	| number_expression '-' number_term
	| number_term
	;

number_term
	: number_factor
	| number_factor '*' number_factor
	| number_factor '/' number_factor
	;

number_factor
	: '-'  number_factor
	| bare_number
	;

identifier_init_nonumber
	: header_block
		{$$ = $1;}
	| music_assign
		{$$ = $1;}
	//| full_markup_list
	//	{$$ = $1;}
	| string
		{$$ = $1;}
	| pitch_or_music
		{$$ = $1;}
	| FRACTION
		{$$ = $1;}
	| embedded_scm
		{$$ = $1;}
	| score_block
		{$$ = $1;}
	| output_def
		{$$ = $1;}
	| context_modification
		{$$ = $1;}
	| book_block
		{$$ = $1;}
	//| bookpart_block
	//| context_def_spec_block
	//| partial_markup
	//| partial_function ETC
	;

string
	: literal_string
		{$$ = $1;}
	| SYMBOL
		{$$ = $1;}
	| full_markup
		{$$ = $1;}
	;

text
	: literal_string
		{$$ = $1;}
	| SYMBOL
		{$$ = $1;}
	| full_markup
		{$$ = $1;}
	| embedded_scm_bare
		{$$ = $1;}
	;

full_markup_list
	: MARKUPLIST
		{$$ = $1;}
	| markup_list
		{$$ = $1;}
	;

markup_list
	: markup_composed_list
		{$$ = $1;}
	| markup_uncomposed_list
		{$$ = [$1];}
	;

markup_composed_list
	: markup_head_1_list markup_uncomposed_list
		//{$$ = block("markup", $1, $2);}
		{$$ = [...$1, $2];}
	;

markup_head_1_list
	: markup_head_1_item
		{$$ = [$1];}
	| markup_head_1_list markup_head_1_item
		{$$ = $1.concat([$2]);}
	;

markup_head_1_item
	//: markup_function EXPECT_MARKUP markup_command_list_arguments
	//: markup_function markup_command_list_arguments
	//	{$$ = {func: $1, args: $2};}
	: markup_function
		//{$$ = {func: $1};}
		{$$ = $1;}
	;

// equivalent for MARKUP_FUNCTION in lilypond's parser.yy
markup_function
	//: CMD_COLUMN
	//	{$$ = $1;}
	: CMD_LINE
		{$$ = $1;}
	//| CMD_BOLD
	//	{$$ = $1;}
	//| CMD_ITALIC
	//	{$$ = $1;}
	//| markup_font_size
	//	{$$ = $1;}
	| CMD_BOX
		{$$ = $1;}
	| CMD_WHITEOUT
		{$$ = $1;}
	| CMD_DYNAMIC
		{$$ = $1;}
	//| CMD_CENTER_COLUMN
	//	{$$ = $1;}
	//| CMD_WITH_URL
	//	{$$ = $1;}
	//| CMD_SANS
	//	{$$ = $1;}
	//| CMD_CONCAT
	//	{$$ = $1;}
	| CMD_MAINTAINER
		{$$ = $1;}
	;

// extra syntax
markup_font_size
	: CMD_HUGE
		{$$ = $1;}
	| CMD_LARGE
		{$$ = $1;}
	| CMD_NORMALSIZE
		{$$ = $1;}
	| CMD_SMALL
		{$$ = $1;}
	| CMD_TINY
		{$$ = $1;}
	| CMD_TEENY
		{$$ = $1;}
	| CMD_MEDIUM
		{$$ = $1;}
	;

markup_uncomposed_list
	: markup_braced_list
		{$$ = $1;}
	//| markup_command_list
	//| markup_scm MARKUPLIST_IDENTIFIER
	//| SCORELINES '{' score_body '}'
	;

markup_braced_list
	: '{' markup_braced_list_body '}'
		{$$ = inlineBlock($2);}
	;

markup_braced_list_body
	: %empty
		{$$ = [];}
	| markup_braced_list_body markup
		{$$ = $1.concat([$2]);}
	| markup_braced_list_body markup_list
		{$$ = $1.concat($2);}
	;

markup
	: markup_head_1_list simple_markup
		//{$$ = $1.concat([$2]);}
		{$$ = markup($1, $2);}
	| simple_markup
		{$$ = $1;}
	;

simple_markup
	: markup_word
		{$$ = $1;}
	| simple_markup_noword
		{$$ = $1;}
	;

markup_word
	: literal_string
		{$$ = $1;}
	| SYMBOL
		{$$ = $1;}
	| unsigned_number
		{$$ = $1;}
	| CMD_MUSICGLYPH
		{$$ = command($1);}
	| CMD_NATURAL
		{$$ = $1;}
	| CMD_FLAT
		{$$ = $1;}
	| CMD_SHARP
		{$$ = $1;}
	| CMD_FOOTER
		{$$ = $1;}
	| scm_identifier
		{$$ = $1;}
	// extra formula
	| music_property_def
		{$$ = $1;}
	| context_change
		{$$ = $1;}
	| pitch_mode_music
		{$$ = $1;}
	// extra formula
	| PITCH
		{$$ = $1;}
	// extra formula
	| REAL
		{$$ = $1;}
	// extra formula
	| INT
		{$$ = $1;}
	// extra formula
	| general_text
		{$$ = $1;}
	;

// extra syntax
long_extender
	: PRE_EXTENDER "_"
		{$$ = $1 + $2}
	| PRE_EXTENDER long_extender
		{$$ = $1 + $2}
	;

// extra syntax
general_text
	: CHORD_MODIFIER_WORD
		{$$ = $1;}
	| "."
		{$$ = $1;}
	| "-"
		{$$ = $1;}
	| "_"
		{$$ = $1;}
	| PRE_HYPHEN
		{$$ = $1;}
	| long_extender
		{$$ = $1;}
	| "="
		{$$ = $1;}
	| "'"
		{$$ = $1;}
	| ","
		{$$ = $1;}
	| ":"
		{$$ = $1;}
	| "/"
		{$$ = $1;}
	| "("
		{$$ = $1;}
	| ")"
		{$$ = $1;}
	| "*"
		{$$ = $1;}
	| "~"
		{$$ = $1;}
	| "!"
		{$$ = $1;}
	| "?"
		{$$ = $1;}
	| UNKNOWN_CHAR
		{$$ = $1;}
	//| PLACEHOLDER_PITCH
	//	{$$ = $1;}
	| RESTNAME
		{$$ = $1;}
	;

simple_markup_noword
	: SCORE '{' score_body '}'
		//{$$ = {score: $3};}
		{$$ = block("score", $1, $3);}
	| markup_function markup_command_basic_arguments
		//{$$ = {func: $1, args: $2};}
		{$$ = command($1, ...$2);}
	//| markup_scm MARKUP_IDENTIFIER
	// extra formula
	| OVERRIDE scm_identifier
		{$$ = command($1, $2);}
	// extra formula
	| CMD_ABS_FONTSIZE scm_identifier markup
		{$$ = command($1, $2, $3);}
	// extra formula
	| CMD_ABS_FONTSIZE scm_identifier markup_list
		{$$ = command($1, $2, ...$3);}
	// extra formula
	| CMD_WITH_COLOR scm_identifier markup
		{$$ = command($1, $2, $3);}
	// extra formula
	| CMD_CHAR scm_identifier
		{$$ = command($1, $2);}
	// extra formula
	| CMD_HSPACE scm_identifier
		{$$ = command($1, $2);}
	// extra formula
	| CMD_SANS markup
		{$$ = command($1, $2);}
	// extra formula
	| CMD_SANS markup_list
		{$$ = command($1, ...$2);}
	// extra formula
	| CMD_CONCAT markup_list
		{$$ = command($1, ...$2);}
	// extra formula
	| CMD_COLUMN markup_list
		{$$ = command($1, ...$2);}
	// extra formula
	| CMD_CENTER_COLUMN markup_list
		{$$ = command($1, ...$2);}
	// extra formula
	| CMD_RIGHT_COLUMN markup_list
		{$$ = command($1, ...$2);}
	// extra formula
	| CMD_FOOTNOTE string string
		{$$ = command($1, $2, $3);}
	// extra formula
	| CMD_WITH_URL scalar string
		{$$ = command($1, $2, $3);}
	// extra formula
	| CMD_WITH_URL scalar markup_list
		{$$ = command($1, $2, ...$3);}
	// extra formula
	| CMD_BOLD markup_list
		{$$ = command($1, ...$2);}
	// extra formula
	| CMD_BOLD markup
		{$$ = command($1, $2);}
	// extra formula
	| markup_font_size markup
		{$$ = command($1, $2);}
	// extra formula
	| markup_font_size markup_list
		{$$ = command($1, ...$2);}
	// extra formula
	| CMD_ITALIC markup
		{$$ = command($1, $2);}
	// extra formula
	| CMD_ITALIC markup_list
		{$$ = command($1, ...$2);}
	// extra formula
	| CMD_CENTER_ALIGN markup_list
		{$$ = command($1, ...$2);}
	// extra formula
	| CMD_CENTER_ALIGN markup
		{$$ = command($1, $2);}
	// extra formula
	| CMD_RIGHT_ALIGN markup_list
		{$$ = command($1, ...$2);}
	// extra formula
	| CMD_RIGHT_ALIGN markup
		{$$ = command($1, $2);}
	// extra formula
	| CMD_GENERAL_ALIGN scm_identifier scm_identifier markup
		{$$ = command($1, $2, $3, $4);}
	// extra formula
	| CMD_LOWER scm_identifier
		{$$ = command($1, $2);}
	// extra formula
	| CMD_FINGER string
		{$$ = command($1, $2);}
	// extra formula
	| CMD_FONTSIZE scm_identifier markup
		{$$ = command($1, $2, $3);}
	// extra formula
	| CMD_FONTSIZE scm_identifier markup_list
		{$$ = command($1, $2, $3);}
	// extra formula
	| CMD_RAISE scm_identifier
		{$$ = command($1, $2);}
	// extra formula
	| CMD_NOTE scm_identifier scm_identifier markup
		{$$ = command($1, $2, $3, $4);}
	// extra formula
	| CMD_CIRCLE markup_list
		{$$ = command($1, ...$2);}
	// extra formula
	| CMD_CIRCLE markup
		{$$ = command($1, $2);}
	// extra formula
	| CMD_PAD_MARKUP scm_identifier
		{$$ = command($1, $2);}
	// extra formula
	| CMD_SMALLER markup
		{$$ = command($1, $2);}
	// extra formula
	| CMD_NORMAL_TEXT markup
		{$$ = command($1, $2);}
	;

markup_command_basic_arguments
	: %emtpy
		{$$ = [];}
	| /*EXPECT_MARKUP_LIST*/ markup_command_list_arguments markup_list
		{$$ = $1.concat($2);}
	| /*EXPECT_SCM*/ markup_command_list_arguments markup_command_embedded_lilypond
		{$$ = $1.concat($2);}
	//| EXPECT_SCM markup_command_list_arguments embedded_scm
	//| EXPECT_SCM markup_command_list_arguments mode_changed_music
	//| EXPECT_SCM markup_command_list_arguments MUSIC_IDENTIFIER
	//| EXPECT_SCM markup_command_list_arguments literal_string
	//| EXPECT_NO_MORE_ARGS
	;

markup_command_list_arguments
	: markup_command_basic_arguments
		{$$ = [$1];}
	| /*EXPECT_MARKUP*/ markup_command_list_arguments markup
		{$$ = $1.concat($2);}
	;

markup_command_embedded_lilypond
	: '{' embedded_lilypond '}'
		{$$ = $2;}
	;

embedded_lilypond
	: %empty
		{$$ = $1;}
	| identifier_init_nonumber
		{$$ = $1;}
	| embedded_lilypond_number
		{$$ = $1;}
	| post_event
		{$$ = $1;}
	//| duration post_events %prec ':'
	| music_embedded music_embedded music_list
		{$$ = [$1, $2, $3];}
	//| error
	//| INVALID embedded_lilypond
	;

embedded_lilypond_number
	: '-' embedded_lilypond_number
		{$$ = -$1;}
	| bare_number_common
		{$$ = $1;}
	//| UNSIGNED NUMBER_IDENTIFIER
	;

bare_number_common
	: REAL
		{$$ = Number($1);}
	//| NUMBER_IDENTIFIER
	//| REAL NUMBER_IDENTIFIER
	| number_identifier
		{$$ = $1;}
	| FRACTION
		{$$ = $1;}
	;

// extra syntax
dot
	: "."
		{$$ = $1;}
	| DOT_NUMBER_R
		{$$ = $1;}
	;

INT
	: UNSIGNED
		{$$ = Number($1);}
	| "-" UNSIGNED
		{$$ = -Number($2);}
	;

// extra syntax
positive_real
	: UNSIGNED DOT_NUMBER_R UNSIGNED
		{$$ = Number($1 + $2 + $3);}
	//| UNSIGNED DOT_NUMBER_L
	//	{$$ = Number($1 + $2);}
	| DOT_NUMBER_R UNSIGNED
		{$$ = Number($1 + $2);}
	;

REAL
	: positive_real
		{$$ = $1;}
	| "-" positive_real
		{$$ = -$2;}
	;

// equivalent for NUMBER_IDENTIFIER in lilypond's parser.yy
number_identifier
	: REAL number_unit
		{$$ = numberUnit($1, $2);}
	//| INT number_unit
	//	{$$ = numberUnit($1, $2);}
	| UNSIGNED number_unit
		{$$ = numberUnit($1, $2);}
	;

// addon term to construct number_identifier
number_unit
	: CENTIMETER
		{$$ = $1;}
	| MILLIMETER
		{$$ = $1;}
	;

score_body
	: score_items
		{$$ = $1;}
	//| score_body error
	;

score_items
	: %empty
		{$$ = [];}
	| score_items score_item
		{$$ = $1.concat([$2]); lineHead(@2, $2);}
	| score_items lilypond_header
		{$$ = $1.concat([$2]); lineHead(@2, $2);}
	;

score_item
	: music
		{$$ = $1;}
	| output_def
		{$$ = $1;}
	//: embedded_scm
	;

//markup_command_list
//	: MARKUP_LIST_FUNCTION markup_command_list_arguments
//	;

markup_scm
	: embedded_scm
		{$$ = $1;}
	;

embedded_scm
	: embedded_scm_bare
		{$$ = $1;}
	//| scm_function_call
	//| lookup
	;

scm_function_call
	: SCM_FUNCTION function_arglist
	;

function_arglist
	: function_arglist_nonbackup
		{$$ = $1;}
	//| EXPECT_OPTIONAL EXPECT_SCM function_arglist_skip_nonbackup DEFAULT
	;

function_arglist_nonbackup
	: function_arglist_common
	//| EXPECT_OPTIONAL EXPECT_SCM function_arglist_nonbackup post_event_nofinger
	//| EXPECT_OPTIONAL EXPECT_SCM function_arglist_nonbackup '-' UNSIGNED
	//| EXPECT_OPTIONAL EXPECT_SCM function_arglist_nonbackup '-' REAL
	//| EXPECT_OPTIONAL EXPECT_SCM function_arglist_nonbackup '-' NUMBER_IDENTIFIER
	//| EXPECT_OPTIONAL EXPECT_SCM function_arglist_nonbackup embedded_scm_arg
	//| EXPECT_OPTIONAL EXPECT_SCM function_arglist_nonbackup bare_number_common
	| function_arglist_nonbackup_reparse REPARSE pitch_or_music
	| function_arglist_nonbackup_reparse REPARSE duration
	| function_arglist_nonbackup_reparse REPARSE reparsed_rhythm
	| function_arglist_nonbackup_reparse REPARSE bare_number_common
	| function_arglist_nonbackup_reparse REPARSE SCM_ARG
	| function_arglist_nonbackup_reparse REPARSE lyric_element_music
	| function_arglist_nonbackup_reparse REPARSE symbol_list_arg
	;

function_arglist_common
	//: EXPECT_NO_MORE_ARGS
	: %empty
	//| EXPECT_SCM function_arglist_optional embedded_scm_arg
	//| EXPECT_SCM function_arglist_optional bare_number_common
	//| EXPECT_SCM function_arglist_optional post_event_nofinger
	//| EXPECT_SCM function_arglist_optional '-' NUMBER_IDENTIFIER
	| function_arglist_common_reparse REPARSE SCM_ARG
	| function_arglist_common_reparse REPARSE lyric_element_music
	| function_arglist_common_reparse REPARSE pitch_or_music
	| function_arglist_common_reparse REPARSE bare_number_common
	| function_arglist_common_reparse REPARSE duration
	| function_arglist_common_reparse REPARSE reparsed_rhythm
	| function_arglist_common_reparse REPARSE symbol_list_arg
	;

lookup
	: LOOKUP_IDENTIFIER
		{$$ = $1;}
	| LOOKUP_IDENTIFIER '.' symbol_list_rev
		{$$ = $1 + "." + $2;}
	;

symbol_list_part
	: symbol_list_part_bare
		{$$ = $1;}
	| embedded_scm_bare
		{$$ = $1;}
	;

embedded_scm_bare
	//: SCM_TOKEN
	//| SCM_IDENTIFIER
	: scm_identifier
		{$$ = $1;}
	;

// equivalent for SCM_IDENTIFIER in lilypond parser.yy
scm_identifier
	//: SCM_FALSE
	//	{$$ = scheme(false);}
	//| SCM_TRUE
	//	{$$ = scheme(true);}
	//| SCM_INT
	//	{$$ = scheme($1.substr(1));}
	//| "#" "'" SYMBOL
	//	{$$ = scheme("'" + $3);}
	: "#" scheme_expression
		{$$ = scheme($2);}
	| DOLLAR_SCHEME_EXPRESSION
		{$$ = $1;}
	;

composite_music
	: basic_music
		{$$ = $1;}
	| contexted_basic_music
		{$$ = $1;}
	//| basic_music new_lyrics
	;

contexted_basic_music
	: context_prefix contextable_music new_lyrics
		{$$ = contextedMusic($1, $2, $3);}
	| context_prefix contextable_music
		{$$ = contextedMusic($1, $2);}
	| context_prefix contexted_basic_music
		{$$ = contextedMusic($1, $2);}
	;

contextable_music
	: basic_music
		{$$ = $1;}
	| pitch_as_music
		{$$ = $1;}
	| event_chord
		{$$ = $1;}
	// extra formula
	| variable_command
		{$$ = $1;}
	;

new_lyrics
	: ADDLYRICS optional_context_mods lyric_mode_music
		{$$ = [{addLyrics: $3, mods: $2}];}
	| new_lyrics ADDLYRICS optional_context_mods lyric_mode_music
		{$$ = $1.concat([{addLyrics: $4, mods: $3}]);}
	;

lyric_mode_music
	: grouped_music_list
		{$$ = $1;}
	//| MUSIC_IDENTIFIER
	| music_identifier
		{$$ = $1;}
	;

context_prefix
	: CONTEXT symbol optional_id optional_context_mods
		//{$$ = {context: $2, assign: $3, mods: $4};}
		{$$ = command($1, $2, $3, ...$4);}
	| NEWCONTEXT symbol optional_id optional_context_mods
		//{$$ = {context: $2, new: true, assign: $3, mods: $4};}
		{$$ = command($1, $2, $3, ...$4);}
	;

optional_id
	: %empty
		{$$ = null;}
	| '=' simple_string
		{$$ = assignment(null, $2);}
	;

optional_context_mods
	: context_modification_mods_list
		{$$ = $1;}
	;

context_modification_mods_list
	: %empty
		{$$ = [];}
	| context_modification_mods_list context_modification
		{$$ = $1.concat($2);}
	;

basic_music
	: repeated_music
		{$$ = $1;}
	| music_bare
		{$$ = $1;}
	| LYRICSTO simple_string lyric_mode_music
		{$$ = command($1, $2, $3);}
	| LYRICSTO symbol '=' simple_string lyric_mode_music
		{$$ = command($1, assignment($2, $4), $5);}
	;

music_bare
	: grouped_music_list
		{$$ = $1;}
	| music_identifier
		{$$ = $1;}
	| mode_changed_music
		{$$ = $1;}
	;

mode_changed_music
	: mode_changing_head grouped_music_list
		{$$ = command($1, $2);}
	| mode_changing_head_with_context optional_context_mods grouped_music_list
		{$$ = command($1, ...$2, $3);}
	// extra formula
	| CHORDMODE chordmode_braced_music_list
		{$$ = command($1, $2);}
	// extra formula
	| LYRICMODE lyricmode_braced_music_list
		{$$ = command($1, $2);}
	;

// extra syntax
lyricmode_braced_music_list
	: '{' lyricmode_music_list '}'
		{$$ = musicBlock($2);}
	;

// extra syntax
lyricmode_music_list
	: %empty
		{$$ = [];}
	| lyricmode_music_list lyricmode_music
		{$$ = $1.concat([$2]);}
	;

// extra syntax
lyricmode_music
	: lyric_element_music
		{$$ = $1;}
	//| music_assign
	//	{$$ = $1;}
	| music_property_def
		{$$ = $1;}
	| lyricmode_music_identifier
		{$$ = $1;}
	| lyricmode_repeated_music
		{$$ = $1;}
	| lyricmode_braced_music_list
		{$$ = $1;}
	;

// extra syntax
lyricmode_music_identifier
	: zero_command
		{$$ = $1;}
	| CMD_TIME FRACTION
		{$$ = command($1, $2);}
	| CMD_BAR string
		{$$ = command($1, $2);}
	| CMD_OMIT property_path
		{$$ = command($1, $2);}
	| CMD_OTTAVA property_path
		{$$ = command($1, $2);}
	| CMD_BARNUMBERCHECK scm_identifier
		{$$ = command($1, $2);}
	| CMD_BARNUMBERCHECK unsigned_number
		{$$ = command($1, $2);}
	| CMD_MARK full_markup
		{$$ = command($1, $2);}
	| CMD_SKIP duration
		{$$ = command($1, $2);}
	| CMD_UNFOLDREPEATS lyricmode_music
		{$$ = command($1, $2);}
	| CMD_ONCE music_assign
		{$$ = command($1, $2);}
	| CMD_PARTIAL duration
		{$$ = command($1, $2);}
	| CMD_TUPLETSPAN duration
		{$$ = command($1, $2);}
	| CMD_TUPLETSPAN DEFAULT
		{$$ = command($1, $2);}
	| CMD_TUPLET FRACTION lyricmode_music
		{$$ = command($1, $2, $3);}
	| CMD_TUPLET FRACTION duration lyricmode_music
		{$$ = command($1, $2, $3, $4);}
	| CMD_TWEAK property_path scalar
		{$$ = command($1, $2, $3);}
	| CMD_TIMES FRACTION lyricmode_music
		{$$ = command($1, $2, $3);}
	| CMD_SHAPE scm_identifier symbol
		{$$ = command($1, $2, $3);}
	| CMD_ACCIDENTALSTYLE grob_prop_spec
		{$$ = command($1, $2);}
	| CMD_NUMERICTIMESIGNATURE lyricmode_music_identifier
		{$$ = command($1, $2);}
	| CMD_DEFAULTTIMESIGNATURE
		{$$ = command($1);}
	| CMD_BENDAFTER scm_identifier
		{$$ = command($1, $2);}
	| CMD_COMPOUNDMETER scm_identifier
		{$$ = command($1, $2);}
	;

// extra syntax
lyricmode_repeated_music
	: REPEAT simple_string unsigned_number lyricmode_braced_music_list
		{$$ = command($1, $2, $3, $4);}
	| REPEAT simple_string unsigned_number lyricmode_braced_music_list ALTERNATIVE lyricmode_braced_music_list
		{$$ = command($1, $2, $3, $4, command($5, $6));}
	;

// extra syntax
chordmode_braced_music_list
	: '{' chordmode_music_list '}'
		{$$ = musicBlock($2);}
	;

// extra syntax
chordmode_music_list
	: %empty
		{$$ = [];}
	| chordmode_music_list chordmode_music
		{$$ = $1.concat([$2]);}
	;

// extra syntax
chordmode_music
	: new_chord post_events
		{$$ = briefChord($1, {post_events: $2, locations: [@1, @2]});}
	| music_assign
		{$$ = $1;}
	| chordmode_repeated_music
		{$$ = $1;}
	| chordmode_braced_music_list
		{$$ = $1;}
	;

// extra syntax
chordmode_repeated_music
	: REPEAT simple_string unsigned_number chordmode_braced_music_list
		{$$ = command($1, $2, $3, $4);}
	| REPEAT simple_string unsigned_number chordmode_braced_music_list ALTERNATIVE chordmode_braced_music_list
		{$$ = command($1, $2, $3, $4, command($5, $6));}
	;

mode_changing_head_with_context
	: DRUMS
		{$$ = $1;}
	| FIGURES
		{$$ = $1;}
	| CHORDS
		{$$ = $1;}
	| LYRICS
		{$$ = $1;}
	;

mode_changing_head
	: NOTEMODE
		{$$ = $1;}
	| DRUMMODE
		{$$ = $1;}
	| FIGUREMODE
		{$$ = $1;}
	//| CHORDMODE
	//	{$$ = $1;}
	//| LYRICMODE
	//	{$$ = $1;}
	;

grouped_music_list
	: sequential_music
		{$$ = $1;}
	| simultaneous_music
		{$$ = $1;}
	;

simultaneous_music
	: SIMULTANEOUS braced_music_list
		{$$ = command($1, $2);}
	//| DOUBLE_ANGLE_OPEN music_list DOUBLE_ANGLE_CLOSE
	//	{$$ = simultaneousList($2);}
	| DOUBLE_ANGLE_OPEN multiple_voices_music_list DOUBLE_ANGLE_CLOSE
		{$$ = simultaneousList($2);}
	;

// extra syntax
multiple_voices_music_list
	: music_list
		{$$ = $1;}
	| multiple_voices_music_list E_BACKSLASH music_list
		{$$ = [...$1, $2, ...$3];}
	;

sequential_music
	: SEQUENTIAL braced_music_list
		{$$ = command($2);}
	| braced_music_list
		{$$ = $1;}
	;

braced_music_list
	: '{' music_list '}'
		{$$ = musicBlock($2);}
	;

music_list
	: %empty
		{$$ = [];}
	| music_list music_embedded
		{$$ = $1.concat([$2]); lineRegister(@2, $2);}
	;

music_embedded
	: music
		{$$ = $1;}
	| post_event
		{$$ = $1;}
	| music_embedded_backup
		{$$ = $1;}
	//| music_embedded_backup BACKUP lyric_element_music
	//| duration post_events %prec ':'
	;

music_embedded_backup
	: embedded_scm
		{$$ = $1;}
	;

music
	: music_assign
		{$$ = $1;}
	| pitch_as_music
		{$$ = $1;}
	//| lyric_element_music
	;

// extra syntax
variable_command
	: COMMAND
		{$$ = variable($1);}
	// some test case use \lower as a variable name!?
	| CMD_LOWER
		{$$ = variable($1);}
	;

lyric_element_music
	: lyric_element optional_notemode_duration post_events
		{$$ = lyric($1, {duration: $2, post_events: $3, locations: [@1, @3]});}
	// extra formula
	| variable_command optional_notemode_duration post_events
		{$$ = lyric($1, {duration: $2, post_events: $3, locations: [@1, @3]});}
	;

lyric_element
	: full_markup
		{$$ = $1;}
	| SYMBOL
		{$$ = $1;}
	| literal_string
		{$$ = $1;}
	//| LYRIC_ELEMENT
	// extra formula
	| general_text
		{$$ = $1;}
	// extra formula
	| "'"
		{$$ = $1;}
	// extra formula
	| UNSIGNED
		{$$ = $1;}
	// extra formula
	| PITCH
		{$$ = $1;}
	| UNKNOWN_CHAR
		{$$ = $1;}
	;

pitch_as_music
	: pitch_or_music
		{$$ = $1;}
	;

music_assign
	: simple_music
		{$$ = $1;}
	| composite_music
		{$$ = $1;}
	;

simple_music
	: event_chord
		{$$ = $1;}
	| music_property_def
		{$$ = $1;}
	| context_change
		{$$ = $1;}
	// extra formula
	| variable_command
		{$$ = $1;}
	// extra formula
	| COMMAND full_markup
		{$$ = command($1, $2);}
	;

context_change
	: CHANGE symbol '=' simple_string
		{$$ = command($1, assignment($2, $4));}
	;

music_property_def
	: OVERRIDE grob_prop_path '=' scalar
		{$$ = command($1, assignment($2, $4));}
	// extra formula
	| OVERRIDEPROPERTY grob_prop_spec scm_identifier
		{$$ = command($1, $2, $3);}
	//| REVERT simple_revert_context revert_arg
	//	{$$ = command($1, $2, $3);}
	| REVERT revert_arg
		{$$ = command($1, $2);}
	| SET context_prop_spec '=' scalar
		{$$ = command($1, assignment($2, $4));}
	| UNSET context_prop_spec
		{$$ = command($1, $2);}
	;

revert_arg
	//: revert_arg_backup BACKUP symbol_list_arg
	: revert_arg_backup
		{$$ = $1;}
	// extra formula
	| revert_arg_backup symbol_list_arg
		{$$ = [$1, $2];}
	;

revert_arg_backup
	: revert_arg_part
		{$$ = $1;}
	;

revert_arg_part
	: symbol_list_part
		{$$ = $1;}
	| revert_arg_backup '.' symbol_list_part
		{$$ = $1 + "." + $3;}
	//| revert_arg_backup BACKUP SCM_ARG '.' symbol_list_part
	//| revert_arg_backup BACKUP SCM_ARG ',' symbol_list_part
	//| revert_arg_backup BACKUP SCM_ARG symbol_list_part
	;

symbol_list_arg
	: SYMBOL_LIST
		{$$ = $1;}
	| SYMBOL_LIST '.' symbol_list_rev
		{$$ = $1.toString() + $2 + $3.toString();}
	| SYMBOL_LIST ',' symbol_list_rev
		{$$ = $1.toString() + $2 + $3.toString();}
	;

// extra syntax
SYMBOL_LIST
	: symbol_list_part
		{$$ = $1;}
	;

simple_revert_context
	: symbol_list_part
		{$$ = $1;}
	;

grob_prop_path
	: grob_prop_spec
		{$$ = [$1];}
	| grob_prop_spec property_path
		{$$ = [$1, $2];}
	;

grob_prop_spec
	: symbol_list_rev
		{$$ = $1;}
	;

context_prop_spec
	: symbol_list_rev
		{$$ = $1;}
	;

event_chord
	: note_chord_element
		{$$ = $1;}
	| tempo_event
		{$$ = $1;}
	| simple_element post_events
		{$$ = rest({...$1, post_events: $2, locations: [@1, @2]});}
	//| CHORD_REPETITION optional_notemode_duration post_events
	//| MULTI_MEASURE_REST optional_notemode_duration post_events
	;

tempo_event
	: TEMPO steno_duration '=' tempo_range
		//{$$ = {tempo: $4, unit: $2};}
		{$$ = tempo($4, $2);}
	| TEMPO text steno_duration '=' tempo_range
		//{$$ = {tempo: $5, unit: $3, text: $2};}
		{$$ = tempo($5, $3, $2);}
	| TEMPO text
		{$$ = tempo(undefined, undefined, $2);}
	| TEMPO CMD_TEMPOLEGEND
		{$$ = tempo(undefined, undefined, $2);}
	;

tempo_range
	: unsigned_number
		{$$ = $1;}
	| unsigned_number '-' unsigned_number
		{$$ = {from: $1, to: $2};}
	;

simple_element
	//: DRUM_PITCH optional_notemode_duration
	: RESTNAME optional_notemode_duration
		{$$ = {name: $1, duration: $2};}
	;

optional_notemode_duration
	: %empty
		{$$ = null;}
	| duration
		{$$ = $1;}
	;

duration
	: steno_duration multipliers
		//{$$ = $1 + $2;}
		{$$ = duration({...$1, multipliers: $2});}
	;

steno_duration
	: unsigned_number dots
		//{$$ = $1 + $2;}
		{$$ = duration({number: $1, dots: $2.length});}
	| DURATION_IDENTIFIER dots
		{$$ = duration({number: $1, dots: $2.length});}
	;

DURATION_IDENTIFIER
	: CMD_BREVE
		{$$ = $1;}
	| CMD_LONGA
		{$$ = $1;}
	;

dots
	: %empty
		{$$ = "";}
	| dots dot
		{$$ = $1 + $2;}
	;

multipliers
	: %empty
		{$$ = [];}
	| multipliers '*' unsigned_number
		{$$ = [...$1, $3];}
	| multipliers '*' FRACTION
		{$$ = [...$1, $3];}
	//| multipliers '*' multiplier_scm
	;

repeated_music
	: REPEAT simple_string unsigned_number music
		{$$ = command($1, $2, $3, $4);}
	| REPEAT simple_string unsigned_number music ALTERNATIVE braced_music_list
		{$$ = command($1, $2, $3, $4, command($5, $6));}
	;

unsigned_number
	: UNSIGNED
		{$$ = $1;}
	//| POST_UNSIGNED
	//	{$$ = $1;}
	//| NUMBER_IDENTIFIER
	//| embedded_scm
	;

simple_string
	: literal_string
		{$$ = $1;}
	| SYMBOL
		{$$ = $1;}
	;

// all kinds commands in music list, seems named as MUSIC_IDENTIFIER in lilypond's parser.yy
music_identifier
	: zero_command
		{$$ = $1;}
	| CMD_CLEF string
		{$$ = command($1, $2);}
	| CMD_TIME FRACTION
		{$$ = command($1, $2);}
	| CMD_BAR string
		{$$ = command($1, $2);}
	| CMD_OMIT property_path
		{$$ = command($1, $2);}
	| CMD_OTTAVA property_path
		{$$ = command($1, $2);}
	| CMD_BARNUMBERCHECK scm_identifier
		{$$ = command($1, $2);}
	| CMD_BARNUMBERCHECK unsigned_number
		{$$ = command($1, $2);}
	| CMD_MARK full_markup
		{$$ = command($1, $2);}
	// TODO:
	| CMD_INCLUDE string
		{$$ = command($1, $2);}
	| CMD_SKIP duration
		{$$ = command($1, $2);}
	//| CMD_PARENTHESIZE property_path
	//	{$$ = command($1, $2);}
	| CMD_UNFOLDREPEATS music
		{$$ = command($1, $2);}
	| CMD_GRACE music
		{$$ = command($1, $2);}
	| CMD_ACCIACCATURA music
		{$$ = command($1, $2);}
	| CMD_APPOGGIATURA music
		{$$ = command($1, $2);}
	| CMD_SLASHEDGRACE music
		{$$ = command($1, $2);}
	| CMD_LANGUAGE string
		{$$ = command($1, $2);}
	| CMD_ONCE music_assign
		{$$ = command($1, $2);}
	| CMD_PARTIAL duration
		{$$ = command($1, $2);}
	| CMD_TUPLETSPAN duration
		{$$ = command($1, $2);}
	| CMD_TUPLETSPAN DEFAULT
		{$$ = command($1, $2);}
	| CMD_TUPLET FRACTION music
		{$$ = command($1, $2, $3);}
	| CMD_TUPLET FRACTION duration music
		{$$ = command($1, $2, $3, $4);}
	| CMD_TWEAK property_path scalar
		{$$ = command($1, $2, $3);}
	| CMD_KEY PITCH COMMAND
		{$$ = command($1, $2, $3);}
	| CMD_TIMES FRACTION music
		{$$ = command($1, $2, $3);}
	| CMD_AFTERGRACE music music
		{$$ = command($1, $2, $3);}
	| CMD_PARALLELMUSIC scm_identifier composite_music
		{$$ = command($1, $2, $3);}
	| CMD_SHAPE scm_identifier symbol
		{$$ = command($1, $2, $3);}
	| CMD_ACCIDENTALSTYLE grob_prop_spec
		{$$ = command($1, $2);}
	| CMD_NUMERICTIMESIGNATURE music_identifier
		{$$ = command($1, $2);}
	| CMD_DEFAULTTIMESIGNATURE
		{$$ = command($1);}
	| CMD_BENDAFTER scm_identifier
		{$$ = command($1, $2);}
	| CMD_COMPOUNDMETER scm_identifier
		{$$ = command($1, $2);}
	| CMD_TRANSPOSITION pitch
		{$$ = command($1, chordElem($2, {locations: [@2, @2]}));}
	| CMD_ABSOLUTE music
		{$$ = command($1, $2);}
	| CMD_HIDE symbol
		{$$ = command($1, $2);}
	| CMD_CROSSSTAFF composite_music
		{$$ = command($1, $2);}
	| CMD_KEEPWITHTAG symbol
		{$$ = command($1, $2);}
	| CMD_TAG symbol composite_music
		{$$ = command($1, $2, $3);}
	| CMD_SCALEDURATIONS FRACTION composite_music
		{$$ = command($1, $2, $3);}
	| CMD_TUPLETNEUTRAL music
		{$$ = command($1, $2);}
	| CMD_ARTICULATE music
		{$$ = command($1, $2);}
	| markup_font_size music
		{$$ = command($1, $2);}
	| pitch_mode_music
		{$$ = $1;}
	| "("
		{$$ = $1;}
	| ")"
		{$$ = $1;}
	| "["
		{$$ = $1;}
	| "]"
		{$$ = $1;}
	| DIVIDE
		{$$ = {proto: "Divide"};}
	| expressive_mark
		{$$ = $1;}
	;

// extra syntax
zero_command
	: CMD_FERMATA
		{$$ = command($1);}
	| CMD_STEMUP
		{$$ = command($1);}
	| CMD_STEMDOWN
		{$$ = command($1);}
	| CMD_STEMNEUTRAL
		{$$ = command($1);}
	| CMD_MERGEDIFFERENTLYDOTTEDON
		{$$ = command($1);}
	| CMD_MERGEDIFFERENTLYHEADEDON
		{$$ = command($1);}
	| CMD_VOICE_NUMBER
		{$$ = command($1);}
	| CMD_SCORE
		{$$ = command($1);}
	| CMD_VOICE
		{$$ = command($1);}
	| CMD_STAFF
		{$$ = command($1);}
	| CMD_PIANOSTAFF
		{$$ = command($1);}
	| CMD_ARPEGGIO
		{$$ = command($1);}
	| CMD_ARPEGGIOARROWDOWN
		{$$ = command($1);}
	| CMD_ARPEGGIOARROWUP
		{$$ = command($1);}
	| CMD_ARPEGGIONORMAL
		{$$ = command($1);}
	| CMD_ARPEGGIOBRACKET
		{$$ = command($1);}
	| CMD_ARPEGGIOPARENTHESIS
		{$$ = command($1);}
	| CMD_ARPEGGIOPARENTHESISDASHED
		{$$ = command($1);}
	| CMD_GLISSANDO
		{$$ = command($1);}
	| CMD_MORDENT
		{$$ = command($1);}
	| CMD_POWERCHORDS
		{$$ = command($1);}
	| CMD_PRALL
		{$$ = command($1);}
	//| CMD_SUSTAINOFF
	//	{$$ = command($1);}
	//| CMD_SUSTAINON
	//	{$$ = command($1);}
	| CMD_TRILL
		{$$ = command($1);}
	| CMD_TURN
		{$$ = command($1);}
	| CMD_POINTANDCLICKOFF
		{$$ = command($1);}
	| CMD_UPBOW
		{$$ = command($1);}
	| CMD_DOWNBOW
		{$$ = command($1);}
	| CMD_BREATHE
		{$$ = command($1);}
	| CMD_STARTTEXTSPAN
		{$$ = command($1);}
	| CMD_STOPTEXTSPAN
		{$$ = command($1);}
	| CMD_FLAGEOLET
		{$$ = command($1);}
	| CMD_SLURDASHED
		{$$ = command($1);}
	| CMD_SLURSOLID
		{$$ = command($1);}
	| CMD_BREAK
		{$$ = command($1);}
	| CMD_PAGEBREAK
		{$$ = command($1);}
	| CMD_STARTTRILLSPAN
		{$$ = command($1);}
	| CMD_STOPTRILLSPAN
		{$$ = command($1);}
	| CMD_CADENZAON
		{$$ = command($1);}
	| CMD_CADENZAOFF
		{$$ = command($1);}
	| CMD_CRESC
		{$$ = command($1);}
	| CMD_CRESCTEXTCRESC
		{$$ = command($1);}
	| CMD_CRESCHAIRPIN
		{$$ = command($1);}
	| CMD_DIM
		{$$ = command($1);}
	| CMD_DIMTEXTDIM
		{$$ = command($1);}
	| CMD_DYNAMICUP
		{$$ = command($1);}
	| CMD_HIDENOTES
		{$$ = command($1);}
	| CMD_UNHIDENOTES
		{$$ = command($1);}
	| CMD_NEWSPACINGSECTION
		{$$ = command($1);}
	| CMD_NOBEAM
		{$$ = command($1);}
	| CMD_ONEVOICE
		{$$ = command($1);}
	| CMD_PHRASINGSLURDOWN
		{$$ = command($1);}
	| CMD_PHRASINGSLURNEUTRAL
		{$$ = command($1);}
	| CMD_PHRASINGSLURUP
		{$$ = command($1);}
	| CMD_SLURDOWN
		{$$ = command($1);}
	| CMD_SLURNEUTRAL
		{$$ = command($1);}
	| CMD_SLURUP
		{$$ = command($1);}
	| CMD_TIEDOWN
		{$$ = command($1);}
	| CMD_TIENEUTRAL
		{$$ = command($1);}
	| CMD_TIEUP
		{$$ = command($1);}
	| CMD_PARENTHESIZE
		{$$ = command($1);}
	| CMD_TUPLETUP
		{$$ = command($1);}
	| CMD_TUPLETDOWN
		{$$ = command($1);}
	| CMD_SHIFTON
		{$$ = command($1);}
	| CMD_REPEATTIE
		{$$ = command($1);}
	;

// extra syntax
expressive_mark
	: CMD_CRESCENDO_BEGIN
		{$$ = $1;}
	| CMD_DECRESCENDO_BEGIN
		{$$ = $1;}
	| CMD_DYNAMICS_END
		{$$ = $1;}
	| CMD_FERMATA
		{$$ = $1;}
	| CMD_TWEAK property_path scalar
		{$$ = command($1, $2, $3);}
	| CMD_BENDAFTER scm_identifier
		{$$ = command($1, $2);}
	| CMD_DYNAMIC_MARKINGS
		{$$ = command($1);}
	| CMD_MARCATO
		{$$ = command($1);}
	| CMD_STOPPED
		{$$ = command($1);}
	| CMD_TENUTO
		{$$ = command($1);}
	| CMD_STACCATISSIMO
		{$$ = command($1);}
	| CMD_ACCENT
		{$$ = command($1);}
	| CMD_STACCATO
		{$$ = command($1);}
	| CMD_PORTATO
		{$$ = command($1);}
	| CMD_MORDENT
		{$$ = command($1);}
	| CMD_PRALL
		{$$ = command($1);}
	| CMD_TURN
		{$$ = command($1);}
	| CMD_ARPEGGIO
		{$$ = command($1);}
	| CMD_TRILL
		{$$ = command($1);}
	| CMD_STARTTRILLSPAN
		{$$ = command($1);}
	| CMD_STOPTRILLSPAN
		{$$ = command($1);}
	| "~"
		{$$ = $1;}
	| "("
		{$$ = $1;}
	| ")"
		{$$ = $1;}
	| "\("
		{$$ = $1;}
	| "\)"
		{$$ = $1;}
	;

// extra syntax
pitch_mode_music
	: CMD_RELATIVE pitch music
		{$$ = command($1, chordElem($2, {locations: [@2, @2]}), $3);}
	| CMD_RELATIVE music
		{$$ = command($1, $2);}
	;

pitch_or_music
	//: pitch exclamations questions octave_check maybe_notemode_duration erroneous_quotes optional_rest post_events
	: pitch exclamations questions optional_notemode_duration optional_rest post_events
		{$$ = chord([chordElem($1, {locations: [@1, @1]})], $4, {exclamations: $2, questions: $3, rest: $5, post_events: $6, locations: [@1, @6]});}
	//| new_chord post_events
	//	{$$ = briefChord($1, {post_events: $2});}
	;

new_chord
	//: steno_tonic_pitch maybe_notemode_duration
	: pitch optional_notemode_duration
		{$$ = {pitch: $1, duration: $2};}
	//| steno_tonic_pitch optional_notemode_duration chord_separator chord_items
	| pitch optional_notemode_duration chord_separator chord_items
		{$$ = {pitch: $1, duration: $2, separator: $3, items: $4};}
	;

chord_items
	: %empty
		{$$ = [];}
	| chord_items chord_item
		{$$ = $1.concat($2);}
	;

chord_item
	: chord_separator
		{$$ = $1;}
	| step_numbers
		{$$ = $1;}
	| CHORD_MODIFIER
		{$$ = $1;}
	;

// m, m7, dim, dim7, aug, maj, maj7
CHORD_MODIFIER
	: CHORD_MODIFIER_WORD
		{$$ = $1;}
	//| CHORD_MODIFIER_WORD UNSIGNED
	//	{$$ = $1 + $2;}
	;

step_numbers
	: step_number
		{$$ = $1;}
	| step_numbers dot step_number
		{$$ = $1 + $2 + $3;}
	;

step_number
	: UNSIGNED
		{$$ = $1;}
	| UNSIGNED '+'
		{$$ = $1 + $2;}
	| UNSIGNED CHORD_MINUS
		{$$ = $1 + $2;}
	;

maybe_notemode_duration
	: %empty
		{$$ = null;}
	| duration
		{$$ = $1;}
	;

steno_tonic_pitch
	: TONICNAME_PITCH quotes
		{$$ = $1 + $2;}
	;

CHORD_BASS
	: "/" "+"
		{$$ = $1 + $2;}
	;

CHORD_CARET
	: "^"
		{$$ = $1;}
	;

CHORD_COLON
	: ":"
		{$$ = $1;}
	;

CHORD_MINUS
	: "-"
		{$$ = $1;}
	;

CHORD_SLASH
	: "/"
		{$$ = $1;}
	;

chord_separator
	: CHORD_COLON
		{$$ = $1;}
	| CHORD_CARET
		{$$ = $1;}
	//| CHORD_SLASH steno_tonic_pitch
	| CHORD_SLASH pitch
		{$$ = $1 + $2;}
	//| CHORD_BASS steno_tonic_pitch
	| CHORD_BASS pitch
		{$$ = $1 + $2;}
	;

exclamations
	: %empty
		{$$ = [];}
	| exclamations '!'
		{$$ = $1.concat($2);}
	;

questions
	: %empty
		{$$ = [];}
	| questions '?'
		{$$ = $1.concat($2);}
	;

post_events
	: %empty
		{$$ = [];}
	| post_events post_event
		{$$ = $1.concat($2);}
	;

note_chord_element
	: chord_body optional_notemode_duration post_events
		{$$ = chord($1, $2, {withAngle: true, post_events: $3, locations: [@1, @2]});}
	;

chord_body
	: "<" chord_body_elements ">"
		{$$ = $2;}
	//| FIGURE_OPEN figure_list FIGURE_CLOSE
	;

chord_body_elements
	: %empty
		{$$ = [];}
	| chord_body_elements chord_body_element
		{$$ = $1.concat([$2]);}
	;

chord_body_element
	//: pitch_or_tonic_pitch exclamations questions octave_check post_events %prec ':'
	: pitch_or_tonic_pitch exclamations questions post_events
		//{$$ = $1 + $2 + $3 + $4;}
		{$$ = chordElem($1, {locations: [@1, @4], exclamations: $2, questions: $3, post_events: $4});}
	//| DRUM_PITCH post_events %prec ':' 
	| music_function_chord_body
		{$$ = $1;}
	//| post_event
	;

music_function_chord_body
	//: music_function_call
	//| MUSIC_IDENTIFIER
	//| embedded_scm
	: music_identifier
		{$$ = $1;}
	;

pitch_or_tonic_pitch
	: pitch
		{$$ = $1;}
	//| steno_tonic_pitch
	;

/*// extra syntax
pitches
	:	pitches pitch
		{$$ = $1.concat([$2]);}
	|	pitch
		{$$ = [$1];}
	;*/

pitch
	: PITCH quotes
		{$$ = $1 + $2;}
	//| steno_pitch
	// extra formula
	//| PLACEHOLDER_PITCH
	//	{$$ = $1;}
	;

quotes
	: %empty
		{$$ = "";}
	| sub_quotes
		{$$ = $1;}
	| sup_quotes
		{$$ = $1;}
	;

sup_quotes
	: "'"
		{$$ = $1;}
	| sup_quotes "'"
		{$$ = $1 + $2;}
	;

sub_quotes
	: ","
		{$$ = $1;}
	| sub_quotes ","
		{$$ = $1 + $2;}
	;

post_event
	: post_event_nofinger
		{$$ = $1;}
	| '-' fingering
		//{$$ = {type: "fingering", direction: "middle", value: $2};}
		{$$ = postEvent("middle", fingering($2));}
	;

HYPHEN
	: PRE_HYPHEN "-"
		{$$ = "--";}
	| PRE_HYPHEN PRE_HYPHEN
		{$$ = "--";}
	;

EXTENDER
	: PRE_EXTENDER "_"
		{$$ = "__";}
	| PRE_EXTENDER PRE_EXTENDER
		{$$ = "__";}
	;

post_event_nofinger
	: '^' fingering
		//{$$ = {direction: "up", fingering: $2};}
		{$$ = postEvent("up", fingering($2));}
	| '_' fingering
		//{$$ = {direction: "down", fingering: $2};}
		{$$ = postEvent("down", fingering($2));}
	| direction_less_event
		{$$ = $1;}
	| script_dir music_function_call
		{$$ = postEvent($1, $2);}
	| HYPHEN
		{$$ = $1;}
	| EXTENDER
		{$$ = $1;}
	| script_dir direction_reqd_event
		{$$ = postEvent($1, $2);}
	| script_dir direction_less_event
		{$$ = postEvent($1, $2);}
	// extra formula
	| script_dir zero_command
		{$$ = postEvent($1, $2);}
	// extra formula
	| script_dir expressive_mark
		{$$ = postEvent($1, $2);}
	// extra formula
	| expressive_mark
		{$$ = postEvent(null, $1);}
	// extra formula
	| CMD_SUSTAINOFF
		{$$ = postEvent(null, $1);}
	// extra formula
	| CMD_SUSTAINON
		{$$ = postEvent(null, $1);}
	// extra formula
	| "["
		{$$ = $1;}
	// extra formula
	| "]"
		{$$ = $1;}
	// extra formula
	| script_dir "["
		{$$ = postEvent($1, $2);}
	// extra formula
	| script_dir "]"
		{$$ = postEvent($1, $2);}
	// extra formula
	| "("
		{$$ = $1;}
	// extra formula
	| ")"
		{$$ = $1;}
	// extra formula
	| script_dir variable_command
		{$$ = postEvent($1, $2);}
	;

direction_reqd_event
	: gen_text_def
		{$$ = $1;}
	| script_abbreviation
		{$$ = $1;}
	;

gen_text_def
	: full_markup
		{$$ = $1;}
	| literal_string
		{$$ = $1;}
	| SYMBOL
		{$$ = $1;}
	| embedded_scm
		{$$ = $1;}
	;

script_abbreviation
	: '^'
		{$$ = $1;}
	| '+'
		{$$ = $1;}
	| '-' 
		{$$ = $1;}
 	| '!'
		{$$ = $1;}
	| '>'
		{$$ = $1;}
	| dot
		{$$ = $1;}
	| '_'
		{$$ = $1;}
	;

direction_less_event
	: string_number_event
		{$$ = $1;}
	//| EVENT_IDENTIFIER
	| tremolo_type
		{$$ = $1;}
	| event_function_event
		{$$ = $1;}
	;

string_number_event
	: E_UNSIGNED
		{$$ = $1;}
	;

tremolo_type
	: ':'
		{$$ = ":";}
	| ':' UNSIGNED
		{$$ = ":" + $2;}
	;

event_function_event
	: EVENT_FUNCTION function_arglist
	;

music_function_call
	: MUSIC_FUNCTION function_arglist
	;

script_dir
	: "_"
		{$$ = "down";}
	| "^"
		{$$ = "up";}
	| "-"
		{$$ = "middle";}
	;

fingering
	: UNSIGNED
		{$$ = $1;}
	;

full_markup
	: markup_mode markup_top
		{$$ = command($1, ...$2);}
	| markup_mode_word
		{$$ = $1;}
	;

markup_mode
	: MARKUP
		{$$ = $1;}
	;

markup_top
	: markup_list
		{$$ = $1;}
	| markup_head_1_list simple_markup
		//{$$ = {head: $1, body: $2};}
		{$$ = $1.concat([$2]);}
	| simple_markup_noword
		{$$ = [$1];}
	;

markup_mode_word
	: markup_mode markup_word
		{$$ = command($1, $2);}
	;

output_def
	: output_def_body '}'
		{$$ = $1;}
	;

output_def_body
	: output_def_head_with_mode_switch '{'
		{$$ = block("score", $1);}
	| output_def_body assignment
		{
			$1.body.push($2);
			$$ = $1;
		}
	| output_def_body music_or_context_def
		{
			$1.body.push($2);
			$$ = $1;
		}
	| output_def_body scm_identifier
		{
			$1.body.push($2);
			$$ = $1;
		}
	//| output_def_body embedded_scm_active
	//| output_def_body SCM_TOKEN
	//| output_def_body error
	;

output_def_head_with_mode_switch
	: output_def_head
		{$$ = $1;}
	;

output_def_head
	: PAPER
		{$$ = $1;}
	| MIDI
		{$$ = $1;}
	| LAYOUT
		{$$ = $1;}
	;

music_or_context_def
	: music_assign
		{$$ = $1;}
	| context_def_spec_block
		{$$ = $1;}
	;

context_def_spec_block
	: CONTEXT '{' context_def_spec_body '}'
		{$$ = block("context", $1, $3);}
	;

context_def_spec_body
	: %empty
		{$$ = [];}
	| context_def_spec_body context_mod
		{$$ = $1.concat([$2]);}
	| context_def_spec_body context_modification
		{$$ = $1.concat([$2]);}
	| context_def_spec_body context_mod_arg
		{$$ = $1.concat([$2]);}
	;

context_mod_arg
	: embedded_scm
		{$$ = $1;}
	| composite_music
		{$$ = $1;}
	;

context_modification
	: WITH '{' context_mod_list '}'
		{$$ = command($1, inlineBlock($3));}
	| WITH context_modification_arg
		{$$ = command($1, $2);}
	;

context_modification_arg
	: embedded_scm
		{$$ = $1;}
	//| MUSIC_IDENTIFIER
	| music_identifier
		{$$ = $1;}
	;

context_mod_list
	: %empty
		{$$ = [];}
	| context_mod_list context_mod
		{$$ = $1.concat($2);}
	| context_mod_list context_mod_arg
		{$$ = $1.concat($2);}
	;

context_mod
	: property_operation
		{$$ = $1;}
	//| context_def_mod SYMBOL
	| context_def_mod symbol
		{$$ = command($1, $2);}
	//| context_def_mod embedded_scm
	;

property_operation
	: symbol '=' scalar
		{$$ = assignment($1, $3);}
	| UNSET symbol
		{$$ = command($1, $2);}
	| OVERRIDE revert_arg '=' scalar
		{$$ = command($1, assignment($2, $4));}
	| REVERT revert_arg
		{$$ = command($1, $2);}
	;

symbol
	: literal_string
		{$$ = $1;}
	| SYMBOL
		{$$ = $1;}
	| embedded_scm_bare
		{$$ = $1;}
	;

scalar
	//: embedded_scm_arg
	: pitch_or_music
		{$$ = $1;}
	| scm_identifier
		{$$ = $1;}
	| bare_number
		{$$ = $1;}
	| '-' bare_number
		{$$ = -$1;}
	| string
		{$$ = $1;}
	| symbol_list_part_bare '.' property_path
		{$$ = $1 + "." + $3;}
	| symbol_list_part_bare ',' property_path
		{$$ = $1 + "," + $3;}
	// extra formula
	| UNSIGNED ',' property_path
		{$$ = $1 + "," + $3;}
	// extra formula
	| COMMAND
		{$$ = $1;}
	;

bare_number
	: bare_number_common
		{$$ = $1;}
	| UNSIGNED
		{$$ = Number($1);}
	//| UNSIGNED NUMBER_IDENTIFIER
	;

context_def_mod
	: CONSISTS
		{$$ = $1;}
	| REMOVE
		{$$ = $1;}
	| ACCEPTS
		{$$ = $1;}
	| DEFAULTCHILD
		{$$ = $1;}
	| DENIES
		{$$ = $1;}
	| ALIAS
		{$$ = $1;}
	| TYPE
		{$$ = $1;}
	| DESCRIPTION
		{$$ = $1;}
	| NAME
		{$$ = $1;}
	;

embedded_scm_active
	//: SCM_IDENTIFIER
	: scm_identifier
		{$$ = $1;}
	| scm_function_call
		{$$ = $1;}
	| lookup
		{$$ = $1;}
	;


/*// extra syntax, maybe the substitution for embedded_scm_active in lilypond's parser
embedded_scheme_expression
	: "#" scheme_expression
		{$$ = scheme($2);}
	;*/

// extra syntax
scheme_expression
	: SCM_TRUE
		{$$ = true;}
	| SCM_FALSE
		{$$ = false;}
	| SCM_HEX
		{$$ = $1}
	| SCM_COLON
		{$$ = $1}
	| bare_number
		{$$ = $1;}
	| INT
		{$$ = $1;}
	| "(" ")"
		{$$ = null;}
	| "+" "inf.0"
		{$$ = $1 + $2;}
	| "-" "inf.0"
		{$$ = $1 + $2;}
	| "(" scheme_expression "." scheme_expression ")"
		{$$ = schemePair($2, $4);}
	| "(" scheme_expression scheme_args ")"
		{$$ = schemeFunction($2, $3);}
	| scheme_token
		{$$ = $1;}
	| scheme_token "?"
		{$$ = $1 + $2;}
	| "'" scheme_expression
		{$$ = schemePointer($2);}
	| "`" scheme_expression
		{$$ = schemePointer($2);}
	| "#" "{" lilypond "#" "}"
		{$$ = schemeEmbed($3);}
	| "*"
		{$$ = $1;}
	;

scheme_args
	: %empty
		{$$ = [];}
	| scheme_args scheme_expression
		{$$ = $1.concat($2);}
	;

scheme_token
	: bare_number
		{$$ = $1;}
	| symbol
		{$$ = $1;}
	| symbol ":" scheme_token
		{$$ = $1 + $2 + $3;}
	| symbol ":" ":" scheme_token
		{$$ = $1 + $2 + $3 + $4;}
	;

optional_rest
	: %empty
		{$$ = null;}
	| REST
		{$$ = $1;}
	;

// extra syntax, the substitution of STRING
literal_string
	: STRING
		{$$ = string($1);}
	;
"}}]);
|
2 |
+
//# sourceMappingURL=chunk-2d0db258.a4804a7a.js.map
|
dist/js/chunk-40965e1a.74707226.js
ADDED
@@ -0,0 +1,2 @@
|
|
|
|
|
|
|
1 |
+
(window["webpackJsonp"]=window["webpackJsonp"]||[]).push([["chunk-40965e1a"],{"00fd":function(t,e,s){var o=s("9e69"),a=Object.prototype,i=a.hasOwnProperty,n=a.toString,r=o?o.toStringTag:void 0;function c(t){var e=i.call(t,r),s=t[r];try{t[r]=void 0;var o=!0}catch(c){}var a=n.call(t);return o&&(e?t[r]=s:delete t[r]),a}t.exports=c},"0146":function(t,e,s){var o=s("24fb");e=o(!1),e.push([t.i,'@font-face{font-family:lotus-music;src:url(data:application/font-woff;charset=utf-8;base64,d09GRgABAAAAAD7MAA0AAAAAe2wAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAABGRlRNAAA+sAAAABoAAAAchdwGK0dERUYAAD6QAAAAHQAAAB4AJwDmT1MvMgAAAaQAAABMAAAAYFZHYnRjbWFwAAADrAAAAHgAAAF6QRmARmdhc3AAAD6IAAAACAAAAAj//wADZ2x5ZgAABegAADXeAABuyE84zEhoZWFkAAABMAAAADMAAAA2FTAd12hoZWEAAAFkAAAAIAAAACQLrwCRaG10eAAAAfAAAAG5AAADgMqNB5Bsb2NhAAAEJAAAAcIAAAHCZHxJJG1heHAAAAGEAAAAHwAAACABKgDPbmFtZQAAO8gAAADwAAABlR1fbG5wb3N0AAA8uAAAAc8AAAJOAWBqCnjaY2BkYGAA4u+nNqjH89t8ZeBm/AIUYbgh/X4ijP6/45ccmyLrMiCXg4EJJAoAilANywB42mNgZGBgXfZLjkGXTfH/jv+FbIoMQBEU8AAAicIGSHjaY2BkYGB4wHCGgYMBBJiAmJEBJOYA5jMAAC59AfoAeNpjYGYSZJzAwMrAwOjKGMfAwOAGpT8wSDA0MjAwMbAyM8AADwMSCEhzTWFwYBRg/M844f8cBl3WZayPgMKMYMmLYFKBgREA5q0LanjajZNBSFRRFIbPPU+HEoxE1Kk200BKi1yEGzfSooUVRotCF2KEITotDHEhLWanO9EJChcuNGiWIUzJQLRt584ocGGLQXRnWZCBXr/77h0cH4g++PjPO+eee847974oI7fFPUWYEDUXRUwr9iN7YC6hP+ErrOBvgHERbZdI76A9vG+KROTIS3gKAzANZWKj6H/0MdyFe6zdRh/CE+gN9oLHxaroCL5F6PJoFjoTZE2zdp5E1lm/Bw+ovW7LtXsmMZfhWqLuJPqO3LfEugMGUnAf5uEb666if1wONa8cq4tr0+k1z4POiNGS3VXXxwf2TAdf7McuhV7ee+QGvlfk9onU0ZP5gs3M6shNXcfOSEojtBFfm0g9s4w4D+2HWdjg/QJ5v/lUNz/OXYeAGhFnpzlPsk8p2n1jWf+Z+OvAC3hO/A3+DKyxbhW1vs94ruqRPLocCPeuilRgKNSZwncz2BV/n6oapel5rGZ23DkdDHu2hJy/2B2isf/HGRTsoSmgOWtlCc2LuvtsPsFHu2fm2G8n/BP7wLzlH2zBmo+5uyLf0VtSL7+wn8UYGWbP4fg84n/OzfkImZ+IaQAAAHjaY2BgYGaAYBkGRgYQKAHyGMF8FoYIIC3EIAAUYWJgYBRitGSsY5zLuIbxJeP///9BihkFGOUZrRkbGBcwbmB8CxL79/nf83/P/j399/jfg393oWaiAUY2BrgEI9BkkOmoChiGBCiHsyws4cw4PBokEEwApVkbCgAAABQAFAAUACIAMgBOAJIAkgDCAOoBAgE0AX4B3AIsAmwCyAM8A2wDkgOqA9wEIARIBGQErgUIBTIFeAXABfwGRgaKBqgGxAbqBx4HTgeeB9QICAgiCFQIogkICYQKGAo8Cn4K1gtGC84L6Av8DAoMQgx2DJQMzg0oDZYOHA64Dt4PJA+CD/gQhBCUEKwQuBDGENQRDBEyEXIRzBJAEtAS7BL6EwgTFBMgEy4TPhNME14TbBN6E5ATnhOyE74T0BPwFAQUFhREFFgUfhU6Fa4WNhZKFmAXZBhkGSAZ2Bp6GsIbEhs6G2oboBvoHCYcnB1KHi4esB8AH44gXiFuIgYimiMaI9wkYiTUJSolVCWYJdwmPCaQJuQm8icIJyYnSidwJ+AoQChyKKYpWCoEKtorKiuEK7AryCvUK+Ar9CwILCwsPixgLHIslCyoLMos1izwLP4tIi08LVAtYi10LYYtmC5sLyIvNC9QL5Qv2C/yMAowJjA6MHowjDDKMRIxZDGcMfQySDKCMuQzFDNgM3QziDO0M9Q0ADQ8NFI0gjSeNOQ1OjVsNao17DYmNmw2qDbgNvI3BjcaNzQ3UDdkN2Q3ZAAAeNq9fQmcG1d9/7wZaWZ0jmY0l66RRiNpVsfqXEl737ter+94ba/t2I5z2I5xEidxEpML4jihJeRoQkwoBRIgIQQCoQmU/MPRFtq0tLQfKLSUlsK/Lf2npVACKSUhlv/vvRlpd71rx05M19Zqzvf7ve/veL/fu5YgiSpBkD+lEgRFMARR43XehJ8q+cXqySepxOs/hEcnJwkCECRBEEeIF+FzRJMxa9f19l734osvwovwHvrFgDjhJAjdMBkjCJh948P67e95D7yI3jz1KyCDIXjf136CTtKiXG30APnS1Rsjv+3TJEmTfv/3wdBJCR3iIk+dhL/dxFNEN6QpKrIkM2W5IisyQzM0/F2uZ5qNZsMsm41KwywCfFquw9tOkgQscAIAv9GXlyYBxQLAwi8KkDQA1EOMn3P64EMs7ZFJv4ekgKK4aJ8LAJeX8gKv2+MDFMli/lt3ET8ENxERAlYnWGcko87Um/hTk/BHYvCHgmcPTmpl+KNZX5dOth5/jv/qC94D3vu9V3i/+gL/xy/A7/vh+Qvgd+/n77fwWU/8KfgxREchCImv8Y1aVZZEP2BoU2o26j1mhqG1SpdWCWSKyeSAYXKS8PvkB1tvz1STUjqfijz+8ZlKqkxTgmDx+xHiGbAbStSDy5OavFGHJdW1B7UjRz5HBv8qffInf/VH74i9w5bfqbcDN3EzkSAIp2FmzCQHjHqzCgmXQL2myLXqCKhJDC2JcQDcO1hHXKPd23rd4VTvNjetxWh2R18q1feugSSVGtzCyn54ZctgKj2w1U2HI7SbIByojuAaXMcIkSZKBKEsqSez5KzZqTXTOfpEB4B/7xy9y4LiqRUQmVwZHs8CUA6Cat1KZsDVBEfkiQFiBOqmCCk2msOg3pMxoaJxAIl6ZXEzGTNT74EaRzN+IIkyfEAiM+N9E1sTfv/AtomB0Y+1lcD6+qPxgbGxfny7/y8md2xdo44NxVZVdl0/rm6c+4X3hRXV5J9Cs1t2bt62LjI6HFtV3nXDu7HKIDxbPyFHMZ4hogtZ8VI8OYCYU6DIFNlpYgjNJqyRJNJUB9LvdIA8zH7ww/TVggpG6auD6gSbK05t+dUKqH7AffCZT1zmXvPZXX1rRtzrPrO3b33RLQ7uOLIJmvoiHUT8fYQcxjoYJJJEBvOH9fAsnAGI7otQR59rczO8wE3r9Y7ivv0sTIQspYZe6lQ/+AHxJ4Qf6tqQLVkkVxMLlnkjNe9IF8sW/KDUVR3uVqSRUr70tjMawI/LuVIpVx4qdY8PV8XSttGxXK4s9A1/9UyG8bDQGBobHO4TurePTMxBvmkkV8pl24lJNIhRYhZa5Vls5ex4LjMj8PBKdmQDTroWAX7qHI3r3HRikeERls9rEIeJZy2fqjBnsTN447DKJTkV/lhfZnIzvXEte8Veuoeeoav0+lnX3t10jV5Fu2fYmXb5HyL2ED+BOEqo9UBNBsRNQrgxPWYSItLYU3Q7/SIXEjlRD7oEUBL0FOv0SWDD14JhSaU4ZnPdLqtJzBOftfypArlEqoz4mudWw3/9f36dAMI3CdePcMO2P/0icRfUOh09jRRNYqCiDZNNS9HExYp2V7eD5ARnIUo6XaQQLTgFP+XojgpCdFLjgaDlHYBkWI5y5DQhGMs7KZ/X6cT+9EPEF4j/xHpiQP91eh2XnCk9RpKOgYaJvmHNv1C1ah7kQluqbRDUsugSSluCrECWgloXRmL9HwfDgYXjkBIJ0Jt75YifxdBgbExwgPge5EO1rCxpGU5VQThnjCSqZhUcyNU+XcvhXwd8Yj4n+R6ZHVu/fmTt7Dj8vX60smZNY+B5jF2JeI34JvTJCOlqszEEGLNZ40VZee2bCQ87qe48JOjfjPO+5DcPub/d+lIyQlmYm2AO8lGD1iIayQwSMS0ZZg/25kYSmQc0CQMzV6uiE4xQs4oMvw7m9ifmIh7eF4xTNf9H14xFotnQuFncMZVN5aXpETndqLi8qcEPjlKUO6CEyt2efbObI04hEFdT9GDNSGf0WiUZiIdFEKMb7XbVBJOQpxHEE0MbGBrLcjlAK9UGMkpYyQZmENqsxVUS8TgMkNwQy4wGFBlMNrsnh7qamUg4xfipqBT1h2/u71uTpwDNCILQk+yVilG9HGr0lEUjkNwxnoyI4UbYNMI+jyI7XD41syeeKW4UKZKh/XGR51KhRtTt9fgTA4keF+B8YZvnElgN8Y+gWAC6PkWypWDUG80aQqxWb0o1sLoSSPwVlMfTO9SHEqtmy7/9tsqeIJRFHEkmqG99vm/LDnPr/i+1yyzAMstLcEBiwEKA9gBPepo9SFuMpgZr3kQPMQgqUOiqbO82coxLkMcUBuixDAzkemIlPeKMjcZ7x4XrMyWqNjCyZayRrSjhXEDxqmNKgHIHM3Eq7Un357/rZH0Xe3br0219XQ9lkiEqi7nBOqEBpCEI/hHQgzSXkbAYkCJjPa5VwfpqcWhbKV+pb+9OSXImEkpxZDgtu6lYPVSZPuDl+gPevt65/i5jrH/nruFxxZ+QQ6ahuFOsdyypDXOBwYF4OJFQ9d023lmwhfg76OuJNEIXtjcGJN9ACJQARqRegxqKFQJqCIQKbPG63Zlo7KH3dQXoSlwIRDVZlxIJKsCEYSCdTmh+IUL9xVo1PtR9zfTh600xlvGFo73HVNHFGBrQJdw+mmAG4iBB74GRaAPRtl4oGtmEYVDGsh1lGDDQfUEQZMxOA8xcvm4gX9kw3dN9XbYxNLuaC0bTk40qL4W4ypre/oTTpaixi+oXRRPNETM70tAjwXCtUuu6hw+5/d2JZIz0Z74X17rMXKIHcMIi2egoPkxj2WTMjsFA9Ot+gMVjWbCJzVpWLHTA+lJ9R2GWTocjqQBQZ+M9anW6hoSVu6HZ56C5hMbRDsf88KpKLmSmQv444/3DADdUHTCNsT5mbi4Y96VSvjiWyRXgEdAFYwcCWDKBbRMPdbQJHokMpU22ZJ7464nhKkg0h9KKMyT+PTDULmMIy7P1XfAI8QGU6QCmCEzogRVs2eARuXzHN0+Uk95MZrD1Q3ZgTRgYf6/yLiU9iDKnUz+lXqJewnmYB3q/IEHwurTsH/XS6+ryj9D+Z/ud11VQoH5OxOAJMrC6FdBSMEUKAMmKVyFLGgCFqaGn5ncK4mpRBIFdaVVUpVJXrjX8FzcfuoEb+ehlXUfnQP3yR1eDe8Bd3NMHp+/bvPrG/U8+9NdL6YTRCd/TDOIAq47dFoXcLxKOAgqtn3WtFoVnnxyZuqgohYKh9C744lFw3/1fBvXNN+a/+BB3yzU3/+U1rb996vKjMxfdu+ptHyWsWOjUDnIf8QTGRIZUNKypBl+j4IdearvAvuyEFS1CLbHqWGvUhgG5zzDGDcOY7V8127d+tn96ZqD1LL4E9vJBf0rxS4IRS9CUrj4B1raeA2svu1sPh5ITY0Y41Ppv69IXtaLE+72hYm9WB07DbvuOgd8jbod6Alv61FJ+mOV8gN9bxEHxNMoTCyRvWUYK07oNYn2Q4FGWll6ghd0CrHqP3dQiBwFJFcaHx8tyajw+kjRzoXiIrW0fU8OjmXxe7K12++SxgYoSUinn63/HOSarZsAZjmhNyx/dien4EBXLGTYwBVC4Yyo3eGx4UAmZw/sqmYEDMkW127k7ySR8J471zcChJlL9Je9DVYgDmUzOBQpqb2xrvlOay+E1jkyD47fce2By/KLpdNV58s87BFZ5613H5m0aRUijhmtvNGwiJbCUithuRU072Bct6yOL1UCA87FsZeJAh3JE3WpOC16SHC4e1sKpKX30P3ZfpJbLqbyqDeTGWrMdNrpG5HXl+YsbgwGvnh+ebY7VGqXEiLlpof5F6tfE1CLeMufE3IoXNGBz66DYyvjBFdldfJyNbG3EJJt1MemNDRVWt0bbvL93Ge+pvtMuaFuVNSPGxEJdHN8jtpwvzm984Rwlcabjs0nIcf2yWr7xheUydL6N2H9hZHjhpfympO90vAlczqYfIEjcim09uMjWM3lwmn3PXOE+3a4vBl7bmO1y+rE9ZxfrGYw2SsiRv5HuMGFl6kb3We1Xi+XGwL1nlbnNw6+J/qUyJ03jzdnq3XX3eRjpdoHUzhf7fmyba84Ns/O2SFVed9j9ZmwxHol2j5En36INWvWDNrjrNBuErBq/MctbU3RfCJur+qj7LrCtQZ/0MPFt4krU6xCHWTDMg82m0lRu3yptnZPm4P8tEtjSPp4T7TiBBr+AGTTuz680FYZSmuZrr776x9/95Hdfe+01wMNvq+xusMbqz5fqQAJr/mYjmN2A5ECd+sKp9SBMPAPpSlZks5CSKijOwF33prLAEQgfePBAPTyYWX3RzrGRnRfffvHdi3icefDAgdUz9XD4s6Po1sUjy3h+/QTMD+/uxDYoyjd5g7fj/IV+BJwyw9gmNpYsJUMwxKlIqbjkYKs7xtTQWGZrZKhHCUknGWrDYL3sV2S/E8zUc7wzGh2z4lVE5zYisJwKzDNHE8NJ6H/Hh6cL/ZnBWjgsv36SmhivlSqL300tehdmQbh3AWpo08rpOXB6eelyXovE3KEENzd9JB7wLS6/5YqFMsk1+fj49Bh97y0Xzx8zG5vbJBfRHFhEE414NG1tbrY7MWiYp+LLQ6Bn4erprBwYiyguyun1erzdo/FVRkTLDvKNuMfnYANr5UtN7xLerh3LxiMJb9afqBY87pk9OzaaI1qlr6x0+bsnQrVwyPB6B+J37YnOG2OLef71LzHPaxbzDLmzUpAm7tDAnZioo70xRHZqAQPlZRdOr8HBsYoL0KrLmxe10FQxLAnUNaVhCniF6fS8Gll8vLQuqwuDmjMv6Ds97tmd5fGUNmVK+Zw+0jc7HMpyTGCgefF8cYPUlz7tijyUo/5lUd2Gcd12XYi6XcjK/kYqvvyK4wM2FAiL1+7GWFzzG8HiNwXO/xJQy684N2PoLJvmTh0nRBQ/2R4D91/YbgRwlptwXzZru43WrZZjAN6LO76C6PiHyKmPEQU7zz+zI6gt8RogsrL5uz/Z1XEQrZNnM3hwZFt1kXPAtnEf5GWSGEa8nJeJ1xqL9QZE3ti03dO1jvhbHz4/YZFixlgkaluX/xbzvvE8eb8QlVlascrMm67YClX9S7WytKq4rhqu66Vvta4XuvJLgcivv4BArODUj8u1xdCgeIgmb8BxE5EGBiBvaHneCf77tZYXvNKOq6iPtOOqNNCBYoAa9ZGWt3UReOW1p1qHn8LPVluH23MiAHGCeBnPlzDqtRMnToCfWdePgD7relqv66Cv9SLoO3Jb+x147/al94B9r/VeAMC7CAP3E9s9eEw7LmoPV8JwCQ8B1gCoqn4uwdW4fEwPcjX/iFoOh7PJ/O/uHcj9Sb+pdoulHmVYVGJKqDcr1sKNqqlIyl/P76yts+ndBGhiipAhPcoIdlJgqytWQX2h9FFj93Qm1deVCGWdlJsrFLtaG8Bmo5hMFPMRgfUwFOVfZZe1BtYsBwteVpasVEfA6cXLqPw1Xzb2oPKziUjGCXUjXGwsIud2FoogenKnTbCkuV1+GuS/ukCeBaSjQ/8Bsg/5Jhc4B/qol/0MPD3QalmVtpii3IlsY/FpoOg5nUky+nqbyULU5+ZoUPpqm2W3jwaJ4Jl4/g5VIy56Czyf4Rqqx3daz79RPepLzvNcz+LzoGFwlPzrvWev2B+tVE/7VI2mbD0jw45u4sBblc25X2yS4daR85OiFl96HtdOl7Ij8Nr+85KyK/1GF6AeIL/zP+AV8GM8JkA4jbqO/Kyk12sUPASvaK0ZTQNf0OC36/Bh8Mr+/a2v33EHzqVDoEBdj32JgXpvQ6EQtaMdN3yYCJ36N3QvCO/BO69/mFrVzsEJGNdhP6hDv4Xee83KF2kiZM2sUOB1+M5r9vVfgAcIH7ruRD7rgdZh8EB7LBE6Th5U0D0XqAOSP/lf8NzV+p8u4LLefQ+k9QFUS6ceBIXWt993BHDt/rgH4b1D2Nvao8qgUEg+kCwcurOaTlfvxM8chM+cwLzCVBQUTuAf+/1xeO9qXDb01rBsUPgyvGLdOw6Pbkd5dVpcGJZFPdyFWOzY8KHM8OVhtav3QOfZWzv1wwXd+iu7nHfBe0cIFpYD6eNkFTExYhbNbn/sxIk+TXPh524BeeIJq548yLe+cxc8JxfV34kRUEwGg7Bj+y6Iw0+efBI/08YhsICE0xAVaxCuUEq+O1kCzD+n1Ho1nD50rJZO99yRnDdUNWm3W22MGAslSETBSF0Of05Y3/i5Nl5OjBhTN6y6Jv8AwvbdP7DKuhnj5ieii8cg7CR9aY4+MTIKk/M7xlCGHmIrO8fUyGB2qzxQLfmVicFL/IrIOaaruYAzFBtZxidESafqhmRz8BJMg1u/fOkl67k25sFFqCtWDwH8pywIgM5rMT3GOETvlbYsyoYkOB28+0oCz6ECJy8j73bcDXPVncQ+4hiuUxxFRahiaD6h1emxaLqAYYqKhq7gSUP2Yz0w6LEGtJsN6zmlWZUp1Gtht9QoLgrisdQgOmQofLUqifj1RtPEg2Wo5Cp595PdYNc22tdbT6ZAKGCqtEOPGQ5G0XRP2hVzMY6HXKlG4e59Pf1dOx9OBIK5VdqOSvcGI3f5bU/zXLQvxXHSdHdIPhiLlGdnygm/955VhdHNNDe4ZstQ/Orauu1d+wcT9Mc3fLGy6QoXSCm9acZhZmQVRBJ50R8OqkDhHDTgHjDNe/37GrfNzrz7jg/cdlVpR+2GiPfHOw9edfmGZ8mHqa6YJHl3dk1lmuv6N+bnXj+aGgyVWh8L1eT6R8EV3Fxxpujyk9Lh0bEb5A9dOnXobW+f1XXcv/Szk98iPwpjIA7q0cbOSG+7hwmljgtTIEwkDsbE4Nr9nUVgIPxqI8CaIcHYo8IN630NkB/lAooa51jBH/JzWhkfqdr7Ds0e/hm9odgjbOkfVYV0MBveMh4WRUnjYpGAoZRn+krqbvnQ6lLuakkI6LIosCzPvSwaCQUd6rvmvzG080apR08FVu279trtpZTkYZkZSXU4XV41pcf8/qFidWeUCw1l1uN5mBT5EXASatceHG3b5oKaIjyZwOo6R3qbQR1RNb7RlLESdHI1ZFHtPkycY9tRHk6vkxloyrDKGkA5JPmRm/ZOV0bN/q7cwLUesded11U6vT2d3mK6lEAmpGfdDjqgrlZ9MkUyLkn0SHxUCo72TFxy0+WbtVy8WIzUQhsPnPybG25ovXzweDg5rHUnY5EcrX1te2q8lql/srfloWCUW/tMPa1GEml+qDKf7mMcLO+gXG4pEuNZVpa1ET10x1Vb3+5ivLOr3S7m9taXqdfsMczWSeLTOPZlrJYHDdh+ur//SH9/6yR4ZSHWbq0GKnjceg4gXwDRAWrrJaB2f7m1mrr49Rb55Ek0kMBYNux8glCIy4hriXcQ9xO/S3zeyocXJjRY8xnQTJrgBbVw1BeKAvE4qDaUnkwJWKdQgG/W+klTz6YSsb5S/2wkElOk7uNvwR18MqWFu0WOT7pyDl00AiVJimtl+tydhOMHhUQkEsk0BK6udYlCQjVoh/Fr/1v1HY6fqlpS9PvcNJO6xRVhvXlfTJXKre+/SZeC9OD1QduX74V6cCvxCPE48dwF9ugdP9WxYqhRb1LSQVt5TtMdbP1vqQ14D9Sg9GBfaWCmifRHP3dpjyN1SQVsbeHKkjscgOryVoX9eg0qUTONdChumnGoQs4k/Y9vRtItD1Yb1cFaWpPzxxS3t2zNFYFtyrPktN2mHCBuJI6/YcvCnG/TYqvAcrdinsUVkM+u2Bqd6K+dY3O04xOnOwVzRcNeudFatenynYPn2GqtAfJKBr9vRWu15+icDJDPdtryfcT1dhz1m8D9dNMzz2ZDbxn1E0sNyVzROi4A5if9y+1j3xkUHc0Pz4JZ8s+JODEC44rdxJUwFiaAAaHIo5nqNWvSbJ1H4TAHjKYCsagFYcY2AqB/M9BFMY6WNaD0oRmsBWF0UaNEOg/wFdRvnMHHtoOUKZMJ4msSuq9Yh2jKJugRyFiXzkXJuzWGSWS9XifJM9d1+7dw2qVOnnSJ94sk6ZCdmx1+mvXe4LmZY9yO8kjhuJPzO51e3nn9hBQWOXNuZHz4HkkoSKYUCnrjqSzb1OLk7Y6oeCrEe71uR5Q45aJ1xxqH46THS3o8QqXicPvdFPB6Xc4oiLR+lGRA67Fc0efyBb8n+Rm/X4hFJUmQwPEPfKD1g7DgOx6WHsmGokdyKZyPkTC6OGXPe7SmGS6aF9yeAorWFkHFzOC505Yto66Tx+OR5rbC3mjPxoktU83hYkavjHEuCrj9McnjGJjitYlEXU74e3KrY/31zPT+A+tG07mJZH+X3lApkna5fV6w69qLKukpNZqdxfHOPHgcfIIoE402R1BAGWuWbL0mLeEKCsHEszJphscsMTxiD7M1r21vlA7OabN9ExZbHkDFpRDrYARfkCtt8n1jGHP11Ke5Dav2bHlX8JmnjOy0zRkb4FgnoBytjcze1Y+AE/qVd+9tz8P7HyK9ZG6iguK3eo1f4ZNe/EO9e3frx0v+4x4EGOTdDH5NuGEmGW7PM8N9NEFgAIc9BRpJg/QZspLUVSV5Vcv5DvDr1j/xXDmZVJXAUVVPFpNJcKjlBL/eW+Z5eFHnrbwOJRnr7PIFNAvyNAqGTcVyMVXyX9pEDmIicxap/8tzISWZLC6hhKkVITVF1S1sIMFXIC0K00KzFNO4JwtQCzQpm+ZpdB2vYEJMm/o6fLplJR4swq3bTmflNHZwfjwHvgTcsNbZdu8GQw/YBh9soJTCQMjCD9NjZKxZzTBh/9KeISNRpfyVl3okx00UQ8cG1VVrI/RabeucbOqwfD3y+PBN+n1cIiGCdYyXvOHhbHY04dzYX8ysn7JwPzVH1THtAWIVsfmN6DvPfncRd+lFx1T9zJy2fuvM90CqXY3Wb525QlecuYqZRXVFNjHneNqu63biChhz3vYW6/um0Fh87Hj6zSFztnvk7y1H7fzwOxdEF0OLetNOzdF6B9ubiDthVvcB4uO/UXzfMvqLj2n9LGi/58JLyRHoyOY9F15e5yXBBVFSBA39QQLLcSNx8RtKD7tF2OrCpg0FLg0YzcFvayWGgVeLKPZHVlBPfrW5xC0kzgLdp0r5RIYaJr2MqIytWk8yXkken14PvEZdNaRcNeqopGN0jyGGADnqURMgeGbEntsaCfo/SLOlL/w97XCUvvAdmrthKlLPwp+g3/1UENSPLGgzijXmSAp8ELXsQBGZpJ1W4oVYNFOTDGvcHt9Aa5EkK0WsjgCIEa41SYJev+jzSkAWErwkbLsk6Ymlk8WRZKki+byk5nQ1iqPjGyI9/RNlcA3X55YCHp71OfxKWO/Tr8v2rzfrNGziGa9P5iIU6yTv+YE+UazafdhzlA/yt/8s/LVXXSxl74w3bL5h3r2sOHyD8i6rUKyoz3Qtqc9HVrhm11FY+voTZ6r4ZatqjdOqvdI1BEXxtJcROhgbhwKxef+Fw+ZNg3bGGw75TaJ5vghfYNjPQxQrScfWXacO5fMX/wvyufCCO+MNZ2KZRCPlNyvRCyLlcxT9jXtX1XrepOjfmjqspCE4FqbGoX7UiUPnrSG2WIHd33Ne7VKjSY2tZJSrsiuJofVMSTd0QSDPoX1SSdg+BTSPcm52hwBsfWNNye+VfE+v3GRxrv8TBKao2bkDuQ3iVSOubOOFJvScDS2Gri1CC+WuSRNC0+4Nww/hV1AyC1FCPTsMTlt7SsDOucmtECwJUDlyRQ8mcCTpseCqDUC4PsT5Yg5ABj3esFZwhah4ICkpmpDQgx4atpVkF6eRkUo0GivMY5QULhh1Lsco4PczLgjRWKn2Ta8AQm63i1V8nBR1h0xd9nhoj6jnJI/f4aJ4keclPaRlQko7xyLvgTiliJ1oDBLbstzA9Wk0q5I10IO1A0KDlrTWzDfZ9N/zIq+5/RHWRXppzqeVklrUL3JcBHBfmhYkMfSZ3z6P2OCfn/AxHsrvdtNulgMZox7weDnfo/rGG84rZqBw/b8P628Q24h7EQJQ+ZU2ALWzAPC/ok3f/xbJx+usm/TRvC9eScU1TuICMZJ7cVZQpPDz7/pNads/fZrkgsDnhqrD8l2ZJu/18b4njW03XHglxDKgGOzbNuPRxcUKhnd8OBf9cp5z/8BCEKom9DeKQZVz6EtY7MA8FNVzVq371dk6HigCnPoZ+Dk4RAwSU8QGlGcYSeSwTasLdARYcwNkxu7rq+NZQVCFUD2bNcbqZJZEBqch1nrnalPpMfGa42YD/NyIUCRg1WjQY05PAAfJhPh61unzKnqim1S5cFQK0ONJo2tVvRIP6pLsJFOFYuthTS9XwsEr4llSDgU8f29qAs2Agq4f+2fG6WEKPdVAwh+NhrhtjMQnRkO3PJHmWFblOLK3pOa7HZ51eubtbj4hWT7nF+Db4FqiZK1WonEVe2AV/QC1SnQe4BrVm9DYRkCNqbbX1+PdIBomRIJGhgK+LfCwNi41wvqZ/LhnJuSJCjJoHvDuVyaDXVxCLksyk8sJ0S86PJGZhlPydSd/IKlxnqbJbl9CPJYs+bkoTX3UmC7fSDpcilfqDQe9gPF8JCIpdlz2K6iZn8JzB2m8hs6am1FvKBqIA6iA4IM31mJbJVkV3W4/e2tVm5NVRcjs3sc6ZyjSx7A+JnGQdU45SU+iPR+oRe4mXkY7XaSRTuO18bBOnaM61Hzkh5EHWTgidx9zs076Sg9DOzd1jqZWvPjh97Ekvdvl3G19bew/7dzm4wlyFjTQnlRppr1G39onaOGsbsV3aAsTq3FAnMxezbJOFpNioCthnYxF9k4nzXiOaFPo66AzwJbecxzSdDnZLzxcwwcb+x+h3fuf//Am2r3LyfZ29tyi5oj/QntLQD46cFgs0J2jBWbwXDoLGOvIWrAHD6m5O5mA86CPoR0bGRdDH/S4GHKT0w2P0LXJY5Az70Enw7o2Ugzjs45IBiqRA/E7/egjrHOX27kH/naR5B58sqn//bRnl5N076FIt/1NAqcDVmCTjeMt5EHiLqTNdh98CRSBYa0aR2MWaE5RezgWc20NFqG2oN07rAHyoCDlGmsuPRzrFpNBLqy6u4K61xFV4r1GgLvzot1X96WSEq+G2e6MWo00jECQjHSL5sycLFBhKarqMS0q+ApSObGfn9naTVOqHFFTMUVQ1WywvJRPe64Ynn9hT9lHPfEcOH2sC54a7ZEoGUJdk8mD471xJerwJIUuj+qj3ByfMgYOX7KusSnF52JRKRrmxZw7FOLdUd7ov3rX5jt9g1t8AU2L8Q6JCysKTeXnZtb0Vgui6PGGBCWWVqKSl6O7t85w7T3YCsS/4Pl+znramjl14sS6E3jnNngPXGLPp4B3pc4T6AOC8LGvo0etvnl45ft2X3/n6aVv4LcesN+y3rT73UGB3N4ZJ1jy9vIS0If0wrf/fElJ4NSrsKCnoQdBO/cg94aAVAwUPeBBrJr8uOUlBDdyEqso6CQS+16x/YnkubUa2yYid3KjXZ9T5McJNzFKrLVn4Vi98iYlt6VlZiyZ4jkAlBHMdKa1KXI7lkHjk3EQBAurgWVwKiZK0agU1NQ/2pzu3zyQX01TRqjoT4mJies/W//ijtTI3EDeSNNUMlR00UZQYNmZo59rPYvei0iiBj4ZjEW7YrF/bv1DOTO4auBAIauae3LRWnG49W+gu2oMzAzuk9SuUKYnL3kkvtY9BITWTiGuBcVolMD7RrwIXgBfI9JEAcbt/cQYmu9pjbBaY0joXx227jg0gB7JWgEp1Qy0rBuNMyX9pKQYUnWYZOoGghq80O9zc9lJfazg9TtJZ3+/6s+P6+N52NB7+t/tohOiLsXdzo9PiAm38/IJlzPxecVISH3+vj7vQCDN08FAXJkKWud8ms74QuNq5v3vT2e7tfj7013dN78/NB5KW7E0/PUg1M0g2j1AN6wNRyBHKLNKQGagc+hsm0M+mKyWdxxq7tlVS3q37AbCnmJP6yfxohCIhyW9viMznVyzo+f+y6861DgalDQuIApaDI+vPUo+CAl58Q53i5Zzy2jLlQRyMTVrH5Fhsg6OxcNyQoP6EwABSKX1U0glWSvPX4Xo1mCJuOi4RWa6drFF1R4LIg+CZwgVRiLbiMuJa4nbUQtoJEcA9nAW0RGYU8ojpInbcLwDEAfQNEY8qG3thmMWSRTfMnbdRwB2gva8Lw4gxu3ZlngUN91ZIBkHDcXapcWaH0pudLlY4KdCcTU5ZAYkypMEpMMR/dhGNU7BtEJxKH7O72L9pF+eopweOiA+sfW6XDG8KlXJZ0rVrT7PRMpIgEYwEImqgUgiAlofD2vFgqb0z0y5hzesiSYq5UTkCy5fvy+czHZt5/zgPhjlkNJl2cuigak05+AcgocMMimlKHC7dNoJ472fVIe1ZK3fn6aH093xyECJz9J6JvlooGAaaibu7k3G9GQsXlAj4j2VUFzanImnMon2eBt5OXiBCBA9xEUw8j1EHIVWbeUcmRFgVmGKwTQYDG4Pnl2wCN0OtiOgynSmcsAAcfG8OgtfHDtiDOs9zhXwteAdMkcrgRDVG2IFt79bX7MhlHBEWLfskP0BfwKCOk3RHqeLlp7ozvb35YqR6bSi5jLlCgR2HAGrqbGgW/Dw2Xi49f5wDAHbh4Gd7a9WEmGwuTu7kfN93+EgPR+sR7suj7jYkRwHLFCF4MUid3GSdnpgqAx+UtE1iGofl6YTWrqQ6MD6dFQQVnlLmVTYgXCNJroVC1dlU3oSw4r9yKvUw+BZIgwj6BuIe4j32fHNYtWNg7bu+nG/SB13guBOEdjeoW9kvp38AX7kmqxgzwnRSjMGY+9oZTeY1fbePWjBUMZsooANbcnXwPMQ7OCFR9M7gN2g4r6VhyOLldrr80KtJkmHM/okgA2rVyJzAdlfk3oowc27a2rD6Yp1MfkunlFlnhYiTkqicgA4vUrrR4UPR7o3XKxLxWF9OOBTQwGlO+r2xpNdlzZPyCrb+vkXezwuh9vLeoxrfL4a+bVgIFblookw+Wqoo/IMC3WepQElMQ7GJ3HsNSH1jnfGfNwddzIcSU2sCYXDIUDpb3MwgM48NT7tiRUmk9NcIhryACYULAR0MVoQpMgJ3t1ax/gpHyu1ZDflYSdajwuKww92B7YlJDITdxFEu41OQD8TJkxoBSguOZObMM/oGUBiBTsvrmDah1e21U+cyURxrvIq+AbUpTRauQzeiti/cZ4CAs+cP7rt/CJEfJ6YRGscFGvKMeQPKXPGbKOqyJ9X1K4hSi+QdFAJ8S5Gi+i5qFxIDsRKAu8e1MUw5RQ9vE+XAoZGtNs3cAqctGIvlz0JAJxq/UsVxG7Hv8HJFmV97NzJnmtBKAaPZlOw1M/w9V8SJxaunzhxwk393F4fQtpzMzp7IwEynT504oSLevl1jnrZlsmKz5048f10eslzIE/cTj4D4yUYfQUXoiV+4fD2TEg10+FQpv1NHlbNTNXMtH5ofVvxqAM8BgDyLDAe7QZ18FhrD2i0vn7sdn3Xbr2DzWPgIvgM33lqcXhlv0GcCgeFCPr8DX639Q/WWXhh7+HHrL2HPQDtPTy1ZzZ0BPhav4CXO3Rk8KhFBz8VWUJHHr1iPnRMb1NBE4rAo617O1TadKwyCOABtQgA8rH3wvfgg5fAz49AZCktD6IVXESlid/6RpvIauvt620i4YVSMC3iX4kXUWlNxvzXa6/HOy0jHfgWsZsoo33JmktXX+z2e6Wg3x8Nij4/gB94Kkb8XvzOT2BeEFy6fiWYsXXiq7D1HIRxkYxKtHuKrC0ysb+FzeVRaEdGQAiruqsqhUpK2BFNUoYaCgpxNR+Ry9H23lTEcStj0NFsdpRd4nUmm6amN25Ytcp7dfPqng27M411uwf31vYuWrvjwfsMGihChcFpDRTuMo4fN+7i70LfNhY/suQbxAj+6Noj2patWDcea30ezNqymQN78AoXa4WUDua01ksamPuzP4PXLexOwXKw/BSsJz+Kzm3TjgAwCwt5rLNmE5ZzCV5zpFurrMBc61MaUGFQvOfP/gyvGTl1klLB1wkfcQdxL/Ew9r4MFISJun4Va7kCyiiweHDnhImOcCiC+iitWY8YXPgdt1cFQL/XnnwGC2tiDw3LxHP1rGVv0H83MQ3UjYe3AYRlWju9oq3GoLtHexA3TNR5BUlRapDnPC4gBji3a13Ar5W0HLOTZbXInMvBCMaMEQCXZN01l4N0ucJs45qYV2EEsiumib67tg6W6qrcFxO8UZoVKNdunfRUPaKux0qcALoLwTB9kaAJepDc0fMQdXTz3gEhoHtZTUn+Y8bZTHFB0u0OcCLwsOvkglgwtXF67WqPXBQv83f1qXmvOO31s84gX0pPNDIkWwYSvByMeEZ3p7O9UpzazKYdLmVAXu/luBlfQIxXqUCYFJXynEPjS40aFZRCuwo7kkF+ZOZ4akgVzF5PrOF0f7Zf9qfj4Y4dXgl2E7cS77Lmo1qrKVAvcs2a2YiU3d7bUIaxTKOOOqdLINm5CYPEBt6UkjZxLAmfQxm/0VnEASVYtyJ1vBUvlFUVnWCBJa1oE0sH9X6hdVHgylRACHhcHhfDsknN6+F5VY36fRRdMgEZo3h3BF4LhBI66yMFryvJp9Oc4dZY2qUnQgk14uaB5shlHRH4aDjLhbQkwwGNTQVS7+PDLE3GfPwzaRoEeBgNCW4WuDQ/5yGVQDnh8nsC3UnG4Q95Iw5Q8ftdZIjXXCDgYd0GmUvSpE5Skot1kXSCiyg8TIMjrpDIGklWVD0qTZa7rOfdQop0plxBloUF8Qu+sQA+vMgHFwoHDxZOgHTrH+DlRc98CPUKpPFsYL7GI4DqaG3i/K3zmcOFycyJ4fn5YcC2fnXdeBmw7ViHPEBuhe9J9oy2pe9iemnGmhKJYnbTbqvJA9tRqd2TmdbXLG6+BeLJ3Mj06oGsoYGBtQeuXDdwYmj7IoKtSy2OG1QoKoRCfCxEaYVmM9fmYx/kIwgjhCxqc5ZTDGJeqNM5/OFysndjhj7Wrjb4zmkEWx6Lkdb/nAZIu81cDbF2ddpM5NftttLyi1+32sk/yCxuk/7Tfsdq/9A7Vrv33tZ/AHmt3eiJmU5f0t9aKw5hbYZAzeqUgweWsMaLbWmNjhYnx8ujbRkTr4AgolLBjq4GTemqRk5zlRuu1HjzqluGZxA/5KnPgSj4HJEiumC7Y234rMhxsikxyDQN3LPGSExnh6Gg1My8u1730bTQ7zYifI4VAsH5oSH/0JAP/V+zMbnu0hSffiUXksY4VQgmoyNT5oCka4M3D8OnBvF/oF/RyFttwcvkPPEsjk0YtIaQnOdbwRlhG5/43Xs8T9pjXI+B74K3EwKhoXXrQQXFzWgvXBSjQrnXm0p79jnqVl/U0LeAl5aOyILPJa39ytGtczejj/buSy95D/qA0GsBmqJpj/gV0eU077NuH91q3X33EtohGEFXCGIEWP2ejGm3ERxeIqYwRvoMLNy/saqF8zIrTCVUubRutyiwtPI4yC/jxjk3lA2nmuScJOtdvRvdAtV37+kcLfCTJIqoH7SJmh/oOxkTb1yM2cLbxUqilSrWqmdi7GsB6N3WesCqSjJMC6MR02ykk6Wi3wvYZbxFXGQqPAGc4Xx3AUzFw7GY0lPQQiq3bzlmjg6PJN4RP40ip1oTBhSSFVyciaHXr7jCJ+bz+k2gvoyBl+9//vno9HQ0fzYZIQkNdTCBuUIemCNoB2OFofMA7938xrC8yHsjiT4x7KUpUJx3eagAQ28QPGHvbDJZqgteQC9jTvOCQrQWZp0kdYWbylAp6PLV1cV4LMJdejZ85LYU053O9UZNwXuAYCm29/tFqxObZ2AXZFUzH/N4+6rjYDrlpRNKKlv0GxUflyh9aDmnWo7PqrO1AuMdUpRgNWNMGr1GxJgK68sYXcDVS6hory0gUUYTD99BVJ1n4mdqR2TzdtHj5JiAC/DLbS7CtP7zdZdDdnTJyw3O2aGJcp5uog/v42ktB7M62OpWNKw0mIVuora24zUpZ2RM8MoFvabS3jIfqYvbubCSiBb6xlQ1qVeaEeBYxmrIK6lyQsoNPccpJUUwEpxDVOM8n4gaejmfEAVDz5hS6PRKwITy1B7wGeJxvDK7PR5krbJBwxJxqysLLV/4TNaYzgwaiZjugdx4BFPO5oZ0RYqzrNPPXD65/nfWjypTrm1CUoeHY7Lki1s+8zLwKeITyB+1x70WxptQ7zhj9jTaI01ot2h8TMpKAyb3n+JgTBHwq0U1wIXUgE+ICpzTCUIsI6QMiaMdrNujuSFFife4XC6P171rdnLrQw9tH5uee2gmUnGBaC06u2pv3V/m+OfjYzOXjBcFX+8aWO9Tp64G98M8AmW1JmM0lWYNJJ+bTV/z7Or0g7/c98gvL2vn99cRjxL3oedQswLdd/PRxoYI/B+79qm+vus+2Wc9hzLgfyU+h/Y/bya59qQCezTRljms8oFIKkX1juZKMN+o4uORfMloGEfSaQfl7R3VerVU2kH6ekfgUXs/mLuIy1BW2EziWLSOh+CUalOi5/2CW3JziszxbokRVCC5PX4z7AQy/M5E2m34c+A5qKMwvg22la0Hql6js1E8Urzvbbz62nUbDl/b+u47dwy9LSqP9xarO94J3n/t+g3XHP69a7+y453jzVJ16GBUOYbXYv/i1M/AV8AfQlw47DXboyB2hxuAfswaJEFNcXvb9l09qzddVRTqqzdeVQT6OD/WeMfOTWNTgcnmeH+htm22dHjDbJ0vXb1pfQUkJwIT4xfteEd9KjjZPz9yKCJ1YpL7wWeh9bEoWgjWglJap57sf7r1GEj8Tg+4rXVnTw+YtPPUl4l/B1nIYczKU20L66SqjDU9BGr4vwtyaZWeE+TKtGFWQ0UjwrKe7mTkb7qm0hubya7J1IY6md63ueH3usj9mzqxKfGnsHwfRJai4wCqERQ2rO+fXkQJYfp9dMFVn3RfCbKtV10OIIrKJxruO6w4gngVvldCXKGxaDwmiXw/HgKkGaMHD1qiOS8MVCGcKFoK9Wpmsp4RBd0XphUpUMrx9SDvkUe1fB/I5FiKBF6fh6U1LbXjHzOKFshyNJNit5mUNF4URbXszed8fncA5m6uRCxixzQw9sri9hki1NYH0+5ww0u5sC/Di+2a1lHG/rMD8K610for6Vw6KCSzSQUG8iLpdcOsBhqlQ2QCAboma5TTBfMzkgaSTwb18I7/F80LUV0wuKiHi+YdvI/1sl5fWktJLEnKPqESdPoFj5N18Szw+yLZBbx/BXnVUCttMLRcsxA3sIeqWbNWFDSL4Vd0H0xsZfphOs96b3eqxUzwYqU/wCW794tr6fyVYpVhRTFsiuJQbdBscj4tQ07O3mfj8W+QBordOn9MwDJfO1WXaHt+BwyuYLZmINX/t65iLGxkColsV6o+S/MwPXKSLiaVCMlyIsWLA1oqDrJN7/5IXi+nE6OeMOVivR4PJxbSSZ8S4D7tsfX1G7h+7b/LkcR/qaqJPGV7oMfuU2fwGDL+IxnNxjc8rgDHA9LpYchAn2p2NXyCXpmbH+jpczl9khjskfyvBbxeKuD1MIzb9S6drUVycm7/2ruec3tg7pbpSmWKqTbGyGZSqP4cQPtzNdCov0kbeL2ygXDG+zjhP4DR/Hc61B0QLqFrI3GJNBNhH2fwYkqkSbfCqb4gF1AZkM2VNseGk7Gp6WSFNDjW0XvE7WbpoJcKckncdr8MaEhTIhJoT7o0VsPFf1YAD+o38No11CHNZDod0g1AO4b1G7v0zJZIKmZmElqaMlQuTqldyapTzfeqckhNfSsfKYS3iKrZo6cShiKooaSc54OhiCm+IwRF0q2IAbndB/1T4u8gL+iv6Nh/G8WaNCYxNHQUTKeRRr0nVoIuN/4uWxhR1WA45KS4Wiol8FpqpLGj6fPFZJ6KRbJfmRiecFNCIBwNhTnwzqArnSivG/BFQBqmeA5qA8Td2ToMusEDyJcKGbya0qxbHduNmgTpxNBqdDQCG8PD2jXpe7NGeP0GTZvdXNDCfVdfFeP8EX9KaR3uXXt8VVR0cUxYnTm+rulVpdHD8wOiEOd6S4dHJI/PZ+2fdJh4GfwOatdCsDS0OdDLn/lM6zClR08q0XbbN0NcQjyPPG3T2i2sp3nJ0BA7P/9M8+1Hm66xn47j5wB8bi98zoVyDrxtLrQNAB9jh55xjR29cdzVPHoTanNbB4m94H0wRiOCGT7QVOy9/ue3b3dlOJeUbR28HzDAdZ+45rqZ2295YS3m823EJeAE6rusBfDiX2szQg1cMs9mOJb1m2yh0Hr1vuCa69YdvXv0ujXBGdQDek5/n+7/A+0tffQAAHjadY4xTsNAEEWfEycIgSgR5RZpba1XFMiiSRNRpaBIn2JlWbK80joRN+AIXIIjUHIMDsARaPk2mwIkvLLmzd//Zwe45IWM8cu44CbxjDNuE89Z8Zw4l/6WeCH/R+Kl9C85s/xcyvWUGnnGldI/POeB+8S59NfEC/nfEy+lf7KmI9Ay4OkxbFWfYN2FdvC92Xo1j9IajnLuiWp9c+z2go2SPYepRjm88o4Sq1rr/zu7nu4dBZVOIbKqdxoU+sMmxMYbV1pTm9PrtXGuqKrC2Uqu3+NOK8JOfZTaTusYjRxXYOfj0IbeVKX9N/sNm+w8JnjafcTXa1NhAEDx86VJbjrSWvdCrXvGJE1ta1y11r231nV7m8Zrk+/Gmxv3rhMHiqBPigtBxT0RVHBrHS0O9EV8UHDjfjb5Bzxwftj4f2+TC2zCJoRIE3bSsOPAiYKLdDLIJAs32eTQgFwa0ojGNKEpzWhOC1rSita0oS3tyKM9HehIJzrTha50ozs96EkvPPTGiw8/+QQooA+FFFFMX4L0oz8DGMggShhMKUMoYyjDGM4IRjKK0YxhLOMYzwQmMonJTGEq05jODGZSzixmM4e5zEMVDuHkNke5wwnWcZNX3OUd1zkjFOES6SKDS1zmAhfZxDnO85ptvOcGZ0WmyOIw69nANfbwgY3sYCv7OMYRatjNT36xnb1s5hZv+MF+jvOH3/zlECd5wD1OUYHGTiqpJcR9HvKURzzmCR+p4hl11HOaMN/ZxUue84L5fOYrW1iATjVRIkgOYLCQGBaLWMwnlrCMpSxnJSu4wkFWs4o1rOUL37gq3CJb5LjjlqppqqXH43rUUMqiqmYa0hE2jURMSUjd6yvy2jWfvzhJwJfCnyI/RSBJQaEiK+IxVQu5PNKwKkNVQadHJiKRYK40ZFQ1q3UZNkNWwpTBf6Wgf18AAAAAAf//AAJ42mNgZGBg4AFiMSBmYmAEwvtAzALmMQAADTMBDgAAAHjaY2BgYGQAgqtL1DlA9A3p9xNhNABAIwaQAAA=) format("woff");font-weight:400;font-style:normal}',""]),t.exports=e},1310:function(t,e){function s(t){return null!=t&&"object"==typeof t}t.exports=s},"1a8c":function(t,e){function s(t){var e=typeof t;return null!=t&&("object"==e||"function"==e)}t.exports=s},"1c3a":function(t,e,s){"use strict";s.d(e,"b",(function(){return h})),s.d(e,"a",(function(){return u}));var o=s("2b0e"),a=s("a139"),i=s("f7ce"),n=s("1f25");const r=o["a"].extend(a["a"]),c=o["a"].extend(i["a"]),l=async(t,e)=>{const s="data:image/svg+xml,"+encodeURIComponent(n["h"]+t),o=await new Promise((t,e)=>{const o=new Image;o.onload=()=>t(o),o.onerror=t=>{console.warn("Error when loading svg image:",s),e(t)},o.src=s});e.width=o.width,e.height=o.height;const a=e.getContext("2d");a.drawImage(o,0,0);const i=await new Promise(t=>e.toBlob(e=>t(e),"image/png"));return URL.createObjectURL(i)},d=(t,e,s)=>{const o=(new DOMParser).parseFromString(t,"text/xml"),a=o.childNodes[0];for(const n of a.childNodes)switch(n.tagName){case"text":/www\.lilypond\.org/.test(n.textContent)&&a.removeChild(n);break;case"a":const t=n.getAttribute("xlink:href"),s=t.match(/:(\d+:\d+:\d+)$/);if(s){const t=s[1];e.has(t)&&a.removeChild(n)}break}const i=a.outerHTML;return l(i,s)},h=async(t,e,s)=>{const o=[];for(const a of t)o.push(await d(a,e,s));return o},f=(t,e)=>{for(const s of t.children)if("use"===s.tagName){const o=s.getAttribute("xlink:href");if(o){const a=o.substr(1),i=e[a];if(i){const e=i.cloneNode(!0);e.classList.add(...s.classList),(e.classList.contains("staff-line")||e.classList.contains("line")||e.classList.contains("slur"))&&e.children[0].setAttribute("stroke","black"),t.insertBefore(e,s),t.removeChild(s)}}}else f(s,e)},u=async function*({sheetDocument:t,signs:e,hashTable:s,matchedIds:o,canvas:a}){console.assert(!!t,"sheetDocument is null."),console.assert(!!o,"matchedIds is null."),console.assert(!!a,"canvas is null."),console.assert(e||s,"signs & hashTable is both null.");const i=new r({propsData:{doc:t,partialVisible:!1}}).$mount(document.createElement("div"));await i.$nextTick(),e||(e=new c({propsData:{hashTable:s}}).$mount(document.createElement("div")));const n=e.$el.children[0],d=[...n.children].reduce((t,e)=>(t[e.id]=e,t),{}),h=[...i.$el.children];for(const r of h)o.forEach(t=>{r.querySelectorAll(`g[data-href='${t}']`).forEach(t=>t.parentElement.removeChild(t))}),f(r,d),yield await l(r.outerHTML,a)}},2462:function(t,e,s){var o=s("6df9");o.__esModule&&(o=o.default),"string"===typeof o&&(o=[[t.i,o,""]]),o.locals&&(t.exports=o.locals);var a=s("499e").default;a("6677ab5e",o,!0,{sourceMap:!1,shadowMode:!1})},"29f3":function(t,e){var s=Object.prototype,o=s.toString;function a(t){return o.call(t)}t.exports=a},"2b3e":function(t,e,s){var o=s("585a"),a="object"==typeof self&&self&&self.Object===Object&&self,i=o||a||Function("return this")();t.exports=i},"2be7":function(t,e,s){var o=s("24fb"),a=s("0146");e=o(!1),e.i(a),e.push([t.i,"",""]),t.exports=e},"2bff":function(t,e,s){var o=s("2be7");o.__esModule&&(o=o.default),"string"===typeof o&&(o=[[t.i,o,""]]),o.locals&&(t.exports=o.locals);var a=s("499e").default;a("1b71e8f0",o,!0,{sourceMap:!1,shadowMode:!1})},3729:function(t,e,s){var o=s("9e69"),a=s("00fd"),i=s("29f3"),n="[object Null]",r="[object Undefined]",c=o?o.toStringTag:void 0;function l(t){return null==t?void 0===t?r:n:c&&c in Object(t)?a(t):i(t)}t.exports=l},"3ed0":function(t,e,s){var o=s("24fb");e=o(!1),e.push([t.i,":root{--lotus-token-default-color:#000;--lotus-token-on-color:#0af}",""]),t.exports=e},"408c":function(t,e,s){var o=s("2b3e"),a=function(){return o.Date.now()};t.exports=a},"4a93":function(t,e,s){"use strict";s("f7f3")},"4cef":function(t,e){var s=/\s/;function o(t){var e=t.length;while(e--&&s.test(t.charAt(e)));return e}t.exports=o},"50f4":function(t,e,s){"use strict";s("8b2a")},"585a":function(t,e,s){(function(e){var s="object"==typeof e&&e&&e.Object===Object&&e;t.exports=s}).call(this,s("c8ba"))},"5f3d":function(t,e,s){var o=s("b1a2");o.__esModule&&(o=o.default),"string"===typeof o&&(o=[[t.i,o,""]]),o.locals&&(t.exports=o.locals);var a=s("499e").default;a("0a968266",o,!0,{sourceMap:!1,shadowMode:!1})},"6c07":function(t,e,s){var o=s("81d1");o.__esModule&&(o=o.default),"string"===typeof o&&(o=[[t.i,o,""]]),o.locals&&(t.exports=o.locals);var a=s("499e").default;a("2c493a70",o,!0,{sourceMap:!1,shadowMode:!1})},"6c11":function(t,e,s){"use strict";s("2bff")},"6df9":function(t,e,s){var o=s("24fb"),a=s("3ed0");e=o(!1),e.i(a),e.push([t.i,".token .line[data-v-5fc078a1],.token .slur[data-v-5fc078a1],.token .staff-line[data-v-5fc078a1]{stroke:var(--lotus-token-default-color)}.token use[data-v-5fc078a1]{fill:var(--lotus-token-default-color)}.token.matched use.on[data-v-5fc078a1]{fill:var(--lotus-token-on-color);stroke-width:.1;stroke:var(--lotus-token-on-color)}",""]),t.exports=e},"81d1":function(t,e,s){var o=s("24fb");e=o(!1),e.push([t.i,".check-button[data-v-58f51ec2]{cursor:pointer}",""]),t.exports=e},"8a81":function(t,e,s){"use strict";s("6c07")},"8b2a":function(t,e,s){var o=s("b1f3");o.__esModule&&(o=o.default),"string"===typeof o&&(o=[[t.i,o,""]]),o.locals&&(t.exports=o.locals);var a=s("499e").default;a("715033e0",o,!0,{sourceMap:!1,shadowMode:!1})},"8d74":function(t,e,s){var o=s("4cef"),a=/^\s+/;function i(t){return t?t.slice(0,o(t)+1).replace(a,""):t}t.exports=i},"9e69":function(t,e,s){var o=s("2b3e"),a=o.Symbol;t.exports=a},a139:function(t,e,s){"use strict";var o=function(){var t=this,e=t._self._c;return e("div",{staticClass:"sheet live"},t._l(t.shownPages,(function(s,o){return e("svg",{key:o,ref:"pages",refInFor:!0,staticClass:"page",style:{["background-image"]:t.backgroundImages&&t.backgroundImages[o]&&`url(${t.backgroundImages[o]})`},attrs:{xmlns:"http://www.w3.org/2000/svg",width:s.width,height:s.height,viewBox:`${s.viewBox.x} ${s.viewBox.y} ${s.viewBox.width} ${s.viewBox.height}`},on:{DOMNodeInserted:t.onPageChanged}},[t.partialVisible&&s.hidden?t._e():e("g",[t.showMark?e("g",{staticClass:"mark"},[t._l(s.systems,(function(o,a){return e("g",{key:a,staticClass:"system",attrs:{transform:`translate(${o.x}, ${o.y})`},on:{mousemove:function(e){t.enablePointer&&t.onMousemovePad(o,e)},mouseleave:function(e){t.enablePointer&&t.onMouseleavePad(o,e)},click:function(e){return t.onClickPad(o,e)}}},[e("rect",{attrs:{x:0,y:o.top,width:o.width,height:o.bottom-o.top}}),t._t("system",null,{system:o,page:s})],2)})),t._t("page",null,{page:s})],2):t._e(),t.bakingMode?t._e():e("g",[t.watermark?e("g",{staticClass:"wm"},[e("image",{attrs:{href:t.watermark,x:(t.doc.pageSize.width-t.watermarkSize.width)/2/t.svgScale,y:(t.doc.pageSize.height-t.watermarkSize.height)/2/t.svgScale,width:t.watermarkSize.width/2/t.svgScale,height:t.watermarkSize.height/2/t.svgScale}})]):t._e(),e("g",{staticClass:"page-tokens"},t._l(s.tokens,(function(t,s){return e("SheetToken",{key:s,attrs:{token:t}})})),1),t._l(s.systems,(function(o,a){return e("g",{key:a,staticClass:"system",attrs:{transform:`translate(${o.x}, ${o.y})`}},[t.showCursor&&t.cursorPosition&&t.cursorPosition.system===o.index?e("rect",{staticClass:"cursor",attrs:{x:t.cursorPosition.x,y:o.top-.5,width:"1",height:o.bottom-o.top+1}}):t._e(),e("g",t._l(o.tokens,(function(t,s){return e("SheetToken",{key:s,attrs:{token:t}})})),1),t._l(o.staves,(function(a,i){return e("g",{key:i,staticClass:"staff",attrs:{transform:`translate(${a.x}, ${a.y})`}},[e("g",t._l(a.tokens,(function(t,s){return e("SheetToken",{key:s,attrs:{token:t}})})),1),t._l(a.measures,(function(s,o){return e("g",{key:o,staticClass:"measure"},t._l(s.tokens,(function(s,o){return e("SheetToken",{key:o,attrs:{token:s,classes:{matched:t.statusMap.has(s.href),mismatched:s.is("NOTEHEAD")&&!t.statusMap.has(s.href),tied:s.tied,attached:Number.isFinite(s.stemX),highlight:t.highlightSymbol&&s.is(t.highlightSymbol)},showTitle:t.showMark,scale:t.enabledFont?s.scale2:null},on:{click:function(e){return t.$emit("click-token",s,e)}}})})),1)})),t.showMark?e("g",{staticClass:"mark"},[t._t("staff",null,{staff:a,system:o,page:s})],2):t._e(),e("g",{staticClass:"markings"},t._l(a.markings,(function(s){return e("g",{key:s.index,class:s.cls,attrs:{transform:`translate(${s.x}, ${s.y+a.yRoundOffset})`}},[e("text",[t._v(t._s(s.text))]),s.alterText?e("text",{staticClass:"alter",attrs:{x:"-0.2",y:"0"}},[t._v(t._s(s.alterText))]):t._e()])})),0)],2)}))],2)}))],2),t.bakingMode?e("g",{staticClass:"bake"},t._l(s.systems,(function(s,o){return e("g",{key:o,staticClass:"system",attrs:{transform:`translate(${s.x}, ${s.y})`}},[t.showCursor&&t.cursorPosition&&t.cursorPosition.system===s.index?e("rect",{staticClass:"cursor",attrs:{x:t.cursorPosition.x,y:s.top-.5,width:"1",height:s.bottom-s.top+1}}):t._e(),t._l(s.staves,(function(s,o){return e("g",{key:o,staticClass:"staff",attrs:{transform:`translate(${s.x}, ${s.y})`}},[t._l(s.measures,(function(s,o){return e("g",{key:o,staticClass:"measure"},t._l(s.matchedTokens,(function(s,o){return e("g",{key:o,staticClass:"token matched",class:{tied:s.tied},attrs:{transform:`translate(${s.x}, ${s.y})`+(s.scale&&1!==s.scale?` scale(${s.scale})`:""),"data-track":s.track,"data-index":s.index}},[e("text",{attrs:{"data-href":s.href}},[t._v(t._s(s.fontUnicode))])])})),0)})),e("g",{staticClass:"markings"},t._l(s.markings,(function(o){return e("g",{key:o.index,class:o.cls,attrs:{transform:`translate(${o.x}, ${o.y+s.yRoundOffset})`}},[e("text",[t._v(t._s(o.text))]),o.alterText?e("text",{staticClass:"alter",attrs:{x:"-0.2",y:"0"}},[t._v(t._s(o.alterText))]):t._e()])})),0)],2)}))],2)})),0):t._e()])])})),0)},a=[],i=s("2b0e"),n=s("4b63"),r=s("6a98"),c=s("5027");class l{constructor(t=!1){t&&this.lock()}get locked(){return!!this.resolve}lock(){return console.assert(!this.locked,"[SingleLock] duplicated locking, last locking has't been released yet."),this.promise=new Promise(t=>this.resolve=t),this.promise}release(t){this.resolve&&(this.resolve(t),this.resolve=null)}wait(){return this.promise}}class d{constructor(t=Date){this.tasks={},this.handlers={},this.timer=t}clear(){Object.values(this.handlers).forEach(t=>clearTimeout(t)),this.tasks={},this.handlers={}}getTask(t){const e=Math.max(t-this.timer.now(),0);return this.tasks[t]||(this.tasks[t]=new Promise(s=>{this.handlers[t]=setTimeout(s,e)}).then(()=>{delete this.tasks[t],delete this.handlers[t]})),this.tasks[t]}appendTask(t,e){this.tasks[t]=this.getTask(t).then(e)}}var h=function(){var t=this,e=t._self._c;return e("g",{staticClass:"token",class:t.classes,attrs:{transform:`translate(${t.token.x}, ${t.token.y})`+(t.scale?` scale(${t.scale.x}, ${t.scale.y})`:""),"data-index":t.token.index,"data-href":t.token.href,"data-track":t.token.track},on:{click:function(e){return t.$emit("click",e)}}},[e("use",{class:t.token.classes,attrs:{"data-href":t.token.href,"xlink:href":"#sign-"+t.token.hash}}),t.showTitle&&t.token.href?e("title",[t._v(t._s(t.token.href))]):t._e()])},f=[],u={name:"sheet-token",props:{token:Object,classes:Object,showTitle:Boolean,scale:Object}},g=u,A=(s("b30c"),s("2877")),m=Object(A["a"])(g,h,f,!1,null,"5fc078a1",null),k=m.exports;class p{add(){}remove(){}}class y{constructor(t){this.elems=t}add(t){this.elems.forEach(e=>e.classList.add(t))}remove(t){this.elems.forEach(e=>e.classList.remove(t))}get value(){return this.elems[0]&&this.elems[0].value}set value(t){this.elems.forEach(e=>e.value=t)}}const v=(t,e=document)=>{const s=e.querySelectorAll(`.token *[data-href='${t}']`);return 0===s.length?new p:1===s.length?s[0].classList:new y(s)},w=Object({NODE_ENV:"production",BASE_URL:""}).VUE_APP_DEFAULT_WATERMARK;var x={name:"sheet-live",components:{SheetToken:k},props:{doc:Object,midiNotation:Object,pitchContextGroup:Array,showMark:Boolean,showCursor:{type:Boolean,default:!0},noteHighlight:{type:Boolean,default:!0},bakingMode:{type:Boolean,default:!1},backgroundImages:Array,enablePointer:{type:Boolean,default:!1},showPagesProgressively:{type:Boolean,default:!1},partialVisible:{type:Boolean,default:!0},scheduler:Object,watermark:{type:String,default:w},enabledFont:Boolean,highlightSymbol:String},data(){return{midiPlayer:null,statusMap:new Map,shownPages:[],watermarkSize:{width:256,height:256}}},computed:{progressTicks(){return this.midiPlayer&&this.midiPlayer.progressTicks},cursorPosition(){return this.midiPlayer&&this.scheduler?this.scheduler.lookupPosition(this.progressTicks):null},cursorPageIndex(){if(!this.cursorPosition||!this.doc)return null;const t=this.doc.systems[this.cursorPosition.system];return console.assert(t,"invalid cursor system index:",this.cursorPosition),t?t.pageIndex:null},cursorSystemIndex(){return this.cursorPosition&&this.doc?this.cursorPosition.system:null},cursorRowIndex(){return this.cursorSystemIndex},svgScale(){const t=this.doc&&this.doc.pages[0];return t?this.doc.pageSize.width/t.viewBox.width:1},isPlaying(){return this.midiPlayer&&this.midiPlayer.isPlaying}},created(){if(this.pageLoadingLock=new l,this.schedulePool=new d(performance),this.preparePlayer(),this.showPages(),this.watermark){const t=new Image;t.src=this.watermark,t.onload=()=>{this.watermarkSize.width=t.naturalWidth,this.watermarkSize.height=t.naturalHeight}}},methods:{onPlayerMidi(t,e){if(this.$emit("midi",t,e),this.noteHighlight&&t.ids){let s=null;const o=t.ids;switch(t.subtype){case"noteOn":s=()=>o.forEach(t=>{const e=this.statusMap.get(t);e&&e.add("on")});break;case"noteOff":s=()=>o.forEach(t=>{const e=this.statusMap.get(t);e&&e.remove("on")});break}s&&this.schedulePool.appendTask(e,s)}},setNoteStatus(t,e,s){if(this.midiNotation){const o=this.midiNotation.notes[t];o?o.ids&&o.ids.forEach(t=>{const o=this.statusMap.get(t);o&&(s?o.add(e):o.remove(e))}):console.warn("invalid note index:",t,this.midiNotation.notes.length)}},clearNoteStatus(){for(const t of this.statusMap.values())t.value=""},updateTokenStatus(){if(this.midiNotation&&this.noteHighlight)for(const t of this.midiNotation.notes){const e=this.midiPlayer.isPlaying&&this.midiPlayer.progressTime>=t.start&&this.midiPlayer.progressTime<t.start+t.duration;t.ids&&t.ids.forEach(t=>{const s=this.statusMap.get(t);s&&(e?s.add("on"):s.remove("on"))})}},async preparePlayer(){if(this.statusMap.clear(),this.midiPlayer&&(this.midiPlayer.dispose(),this.midiPlayer=null),this.midiNotation&&(this.updateMidiPlayer(),await this.$nextTick(),await this.pageLoadingLock.wait(),this.updateStatusMap(),!this.scheduler)){const t=this.doc&&this.doc.getTokenMap();if(t){for(const s of t.values())i["a"].set(s,"on",s.on||!1);const e=r["a"].createFromNotation(this.midiNotation,t);this.$emit("update:scheduler",e)}}},updateMidiPlayer(){this.midiPlayer&&this.midiPlayer.dispose(),this.midiPlayer=new n["MidiPlayer"](this.midiNotation,{cacheSpan:400,onMidi:(t,e)=>this.onPlayerMidi(t,e),onTurnCursor:()=>this.updateTokenStatus()})},updateStatusMap(){this.midiNotation&&this.midiNotation.notes.forEach(t=>t.ids&&t.ids.forEach(t=>{this.statusMap.get(t)||this.statusMap.set(t,v(t,this.$el))}))},updateStatusMapInPage(t){const e=t.querySelectorAll(".token *[data-href]");e.forEach(e=>{const s=e.dataset.href;this.statusMap.set(s,v(s,t))})},addMarkingByTick(t,e,s,{id:o,cls:a,text:i="",xoffset:n=0}={}){if(!this.pitchContextGroup)return void console.warn("[addMarkingByTick]\tpitchContextGroup is required.");const r=this.pitchContextGroup[s];if(!r)return void console.warn("[addMarkingByTick]\tinvalid staffIndex:",s,this.pitchContextGroup.length);const c=this.scheduler.lookupPosition(t);if(!c)return void console.warn("[addMarkingByTick]\tinvalid tick:",t);const l=r.lookup(t);if(!l)return console.warn("no context at tick:",t,r),null;const{y:d,alter:h}=l.pitchToY(e);return this.doc.addMarking(c.system,s,{x:c.x+n,y:d,text:i,alter:h,id:o,cls:a})},addMarkingByNote(t,e,{id:s=null,cls:o,text:a=""}={}){console.assert(this.midiNotation,"[addMarkingByNote]\tmidiNotation is null.");const i=this.midiNotation.notes[t];if(i){if(!s){if(!i.ids)return null;s=i.ids[0]}return this.addMarkingByTick(i.startTick,e,i.staffTrack,{id:s,cls:o,text:a,xoffset:1.2})}console.warn("[addMarkingByNote]\tinvalid noteIndex:",t,this.midiNotation.notes.length)},removeMarking(t){this.doc.removeMarking(t)},clearMarkings(){this.doc.clearMarkings()},async showPages(){if(this.shownPages=[],this.doc)if(this.showPagesProgressively){await this.pageLoadingLock.wait(),this.pageLoadingLock.lock();for(let t=0;t<this.doc.pages.length;++t)this.shownPages.push(this.doc.pages[t]),await this.$nextTick(),await Object(c["a"])();this.pageLoadingLock.release()}else this.shownPages=this.doc.pages},onDocChanged(){this.clearNoteStatus(),this.clearMarkings(),this.showPages()},eventToSystemPosition(t,e){return{x:e.offsetX/this.svgScale-t.x,y:e.offsetY/this.svgScale-t.y}},eventToPointer(t,e){const s=this.eventToSystemPosition(t,e),o=t.index,a=this.doc.lookupMeasureIndex(o,s.x),i=this.scheduler&&this.scheduler.lookupTick({system:o,x:s.x});return{systemIndex:o,measureIndex:a,tick:i,...s}},onMousemovePad(t,e){this.$emit("pointerUpdate",this.eventToPointer(t,e))},onMouseleavePad(){this.$emit("pointerUpdate",null)},onClickPad(t,e){this.$emit("pointerClick",this.eventToPointer(t,e),e)},updatePageVisibility(){this.$refs.pages?this.$refs.pages.forEach((t,e)=>{const s=t.getBoundingClientRect(),o=this.shownPages[e],a=s.top>window.innerHeight||s.bottom<0||s.left>window.innerWidth||s.right<0;!!o.hidden!==a&&i["a"].set(o,"hidden",a)}):console.log("[updatePageVisibility] $refs.pages is null:",this.$refs.pages)},onPageChanged(t){t.target&&"g"===t.target.nodeName&&this.updateStatusMapInPage(t.target)}},watch:{midiNotation:"preparePlayer",midiPlayer(t){this.$emit("update:midiPlayer",t)},async bakingMode(){await this.$nextTick(),await this.pageLoadingLock.wait(),this.updateStatusMap(),this.updateTokenStatus()},doc:"onDocChanged",cursorPageIndex(t){this.$emit("cursorPageShift",t)},cursorSystemIndex(t){this.$emit("cursorSystemShift",t)},isPlaying(t){t||this.schedulePool.clear()}}},b=x,B=(s("4a93"),s("6c11"),s("50f4"),Object(A["a"])(b,o,a,!1,null,"dc74d054",null));e["a"]=B.exports},b047:function(t,e,s){var o=s("1a8c"),a=s("408c"),i=s("b4b0"),n="Expected a function",r=Math.max,c=Math.min;function l(t,e,s){var l,d,h,f,u,g,A=0,m=!1,k=!1,p=!0;if("function"!=typeof t)throw new TypeError(n);function y(e){var s=l,o=d;return l=d=void 0,A=e,f=t.apply(o,s),f}function v(t){return A=t,u=setTimeout(b,e),m?y(t):f}function w(t){var s=t-g,o=t-A,a=e-s;return k?c(a,h-o):a}function x(t){var s=t-g,o=t-A;return void 0===g||s>=e||s<0||k&&o>=h}function b(){var t=a();if(x(t))return B(t);u=setTimeout(b,w(t))}function B(t){return u=void 0,p&&l?y(t):(l=d=void 0,f)}function C(){void 0!==u&&clearTimeout(u),A=0,l=g=d=u=void 0}function M(){return void 0===u?f:B(a())}function P(){var t=a(),s=x(t);if(l=arguments,d=this,g=t,s){if(void 0===u)return v(g);if(k)return clearTimeout(u),u=setTimeout(b,e),y(g)}return void 0===u&&(u=setTimeout(b,e)),f}return e=i(e)||0,o(s)&&(m=!!s.leading,k="maxWait"in s,h=k?r(i(s.maxWait)||0,e):h,p="trailing"in s?!!s.trailing:p),P.cancel=C,P.flush=M,P}t.exports=l},b1a2:function(t,e,s){var o=s("24fb");e=o(!1),e.push([t.i,".sign line[data-v-7dcb4590],.sign polygon[data-v-7dcb4590]{stroke:inherit}.sign path[data-v-7dcb4590],.sign polygon[data-v-7dcb4590],.sign rect[data-v-7dcb4590]{fill:inherit}.sign path[data-v-7dcb4590]{stroke:inherit}.sign .font-char[data-v-7dcb4590]{font-family:var(--music-font-family);font-size:var(--music-font-size)}",""]),t.exports=e},b1f3:function(t,e,s){var o=s("24fb"),a=s("3ed0");e=o(!1),e.i(a),e.push([t.i,".sheet .bake .token text{fill:var(--lotus-token-default-color)}.sheet .bake .token text.on{fill:var(--lotus-token-on-color);stroke-width:.1;stroke:var(--lotus-token-on-color)}",""]),t.exports=e},b30c:function(t,e,s){"use strict";s("2462")},b4b0:function(t,e,s){var o=s("8d74"),a=s("1a8c"),i=s("ffd6"),n=NaN,r=/^[-+]0x[0-9a-f]+$/i,c=/^0b[01]+$/i,l=/^0o[0-7]+$/i,d=parseInt;function h(t){if("number"==typeof t)return t;if(i(t))return n;if(a(t)){var e="function"==typeof t.valueOf?t.valueOf():t;t=a(e)?e+"":e}if("string"!=typeof t)return 0===t?t:+t;t=o(t);var s=c.test(t);return s||l.test(t)?d(t.slice(2),s?2:8):r.test(t)?n:+t}t.exports=h},ddcc:function(t,e,s){"use strict";var o=function(){var t=this,e=t._self._c;return e("button",{class:`check-button ${t.classes} ${t.value?"on":"off"}`,domProps:{innerHTML:t._s(t.content)},on:{click:t.onClick}})},a=[],i={name:"check-button",props:{value:Boolean,classes:{type:String,default:""},content:String},methods:{onClick(){this.$emit("input",!this.value)}}},n=i,r=(s("8a81"),s("2877")),c=Object(r["a"])(n,o,a,!1,null,"58f51ec2",null);e["a"]=c.exports},e179:function(t,e,s){var o=s("24fb"),a=s("3ed0");e=o(!1),e.i(a),e.push([t.i,".sheet .mark[data-v-dc74d054]{opacity:0}.sheet .mark .locator text[data-v-dc74d054]{font-size:2px;text-anchor:start;pointer-events:none}.sheet .mark rect[data-v-dc74d054]{fill:transparent}.sheet .cursor[data-v-dc74d054],.sheet .wm[data-v-dc74d054]{pointer-events:none}.sheet .bake[data-v-dc74d054]{font-family:var(--music-font-family)}.sheet .bake .token text[data-v-dc74d054]{pointer-events:none}.sheet .bake .token text[data-v-dc74d054],.sheet .markings text[data-v-dc74d054]{-webkit-user-select:none;-moz-user-select:none;user-select:none;font-size:var(--music-font-size)}.sheet .markings text[data-v-dc74d054]{font-family:var(--music-font-family)}.sheet .markings .alter[data-v-dc74d054]{text-anchor:end}",""]),t.exports=e},eaac:function(t,e,s){"use strict";s("5f3d")},f7ce:function(t,e,s){"use strict";var o=function(){var t=this,e=t._self._c;return e("svg",{staticClass:"sheet-signs",attrs:{xmlns:"http://www.w3.org/2000/svg"}},[e("defs",t._l(t.signs,(function(s){return e("g",{key:s.id,staticClass:"sign",attrs:{id:"sign-"+s.id,transform:s.def.scale&&!s.glyph&&`scale(${s.def.scale.x}, ${s.def.scale.y})`}},[s.glyph?e("text",{staticClass:"font-char",attrs:{"text-anchor":"start"},domProps:{innerHTML:t._s(s.glyph)}}):t._e(),"path"!==s.def.type||s.glyph?t._e():e("path",{attrs:{d:s.def.d,"stroke-width":s.def["stroke-width"]}}),"rect"===s.def.type?e("rect",{attrs:{x:s.def.width>=0?0:s.def.width,y:s.def.height>=0?0:s.def.height,width:Math.abs(s.def.width),height:Math.abs(s.def.height)}}):t._e(),"line"===s.def.type?e("line",{attrs:{x1:"0",y1:"0",x2:s.def.width,y2:s.def.height,"stroke-width":s.def["stroke-width"],"stroke-dasharray":s.def["stroke-dasharray"]}}):t._e(),"polygon"===s.def.type?e("polygon",{attrs:{points:s.def.points,"stroke-width":s.def["stroke-width"]}}):t._e(),"text"===s.def.type?e("text",{attrs:{"font-size":s.def["font-size"],"font-weight":s.def["font-weight"],"font-style":s.def["font-style"],"text-anchor":s.def["text-anchor"],fill:s.def.color}},[e("tspan",[t._v(t._s(s.def.text))])]):t._e()])})),0)])},a=[],i=s("32c1"),n={name:"sheet-signs",props:{hashTable:Object,enabledFont:Boolean},computed:{signs(){return this.hashTable?Object.entries(this.hashTable).map(([t,e])=>({id:t,def:e,glyph:this.enabledFont?i["glyph"].glyphHash[t]&&i["glyph"].GlyphUnicode[i["glyph"].glyphHash[t]]:null})):[]}}},r=n,c=(s("eaac"),s("2877")),l=Object(c["a"])(r,o,a,!1,null,"7dcb4590",null);e["a"]=l.exports},f7f3:function(t,e,s){var o=s("e179");o.__esModule&&(o=o.default),"string"===typeof o&&(o=[[t.i,o,""]]),o.locals&&(t.exports=o.locals);var a=s("499e").default;a("7daddbf7",o,!0,{sourceMap:!1,shadowMode:!1})},ffd6:function(t,e,s){var o=s("3729"),a=s("1310"),i="[object Symbol]";function n(t){return"symbol"==typeof t||a(t)&&o(t)==i}t.exports=n}}]);
|
2 |
+
//# sourceMappingURL=chunk-40965e1a.74707226.js.map
|
dist/js/chunk-48b5b2a0.3db5a0aa.js
ADDED
The diff for this file is too large to render.
See raw diff
|
|
dist/js/chunk-a06ef50c.1caef24f.js
ADDED
@@ -0,0 +1,2 @@
|
|
|
|
|
|
|
1 |
+
(window["webpackJsonp"]=window["webpackJsonp"]||[]).push([["chunk-a06ef50c"],{1773:function(e,t,n){"use strict";n.d(t,"a",(function(){return s})),n.d(t,"e",(function(){return l})),n.d(t,"f",(function(){return c})),n.d(t,"c",(function(){return i})),n.d(t,"d",(function(){return u})),n.d(t,"b",(function(){return f}));var r=n("27e2"),o=n("43fb"),a=n("6977");function s(e,t){let n=t;for(;;){if(!Array.isArray(n))break;{if(0===n.length)break;n=Object(o["i"])(e,n);const t=Object(r["d"])(n[0]);if(!t)break;{const r=Object(o["h"])(e,t,n);if(!r)break;n=r.fn(r.actualArgs)}}Object(a["d"])(e)}return n}function l(e,t){if(!Array.isArray(t)||!Object(r["d"])(t[0],e.config.reservedNames.quote))throw new Error("[SX] stripQuote: token is not quoted.");return t[1]}function c(e,t){return Array.isArray(t)&&Object(r["d"])(t[0],e.config.reservedNames.quote)?t[1]:t}function i(e){return function(){return Object(o["e"])(e)}}function u(e,t){return Object(a["b"])("(compiler)resolveValueSymbol_dynamic",t),function(){return Object(o["j"])(e,{symbol:t})}}function f(e){return function(t){return Object(a["b"])(e,t)}}},"27e2":function(e,t,n){"use strict";function r(e,t){return[{symbol:e.config.reservedNames.quote},t]}function o(e,t){if(Array.isArray(t)&&0<t.length){const n=f(t);if(n&&n.symbol===e.config.reservedNames.quote)return!0}return!1}function a(e,t){return[{symbol:e.config.reservedNames.backquote},t]}function s(e,t){if(Array.isArray(t)&&0<t.length){const n=f(t);if(n&&n.symbol===e.config.reservedNames.backquote)return!0}return!1}function l(e,t){return[{symbol:e.config.reservedNames.unquote},t]}function c(e,t){if(Array.isArray(t)&&0<t.length){const n=f(t);if(n&&n.symbol===e.config.reservedNames.unquote)return!0}return!1}function i(e,t){return[{symbol:e.config.reservedNames.spread},t]}function u(e,t){return[{symbol:e.config.reservedNames.splice},t]}function f(e,t){return e&&"object"===typeof e&&Object.prototype.hasOwnProperty.call(e,"symbol")?void 0!==t?e.symbol===t?e:null:e:null}n.d(t,"f",(function(){return r})),n.d(t,"c",(function(){return o})),n.d(t,"a",(function(){return a})),n.d(t,"b",(function(){return s})),n.d(t,"i",(function(){return l})),n.d(t,"e",(function(){return c})),n.d(t,"h",(function(){return i})),n.d(t,"g",(function(){return u})),n.d(t,"d",(function(){return f}))},"43fb":function(e,t,n){"use strict";n.d(t,"l",(function(){return a})),n.d(t,"i",(function(){return l})),n.d(t,"h",(function(){return c})),n.d(t,"k",(function(){return u})),n.d(t,"j",(function(){return f})),n.d(t,"a",(function(){return m})),n.d(t,"c",(function(){return b})),n.d(t,"f",(function(){return $})),n.d(t,"m",(function(){return p})),n.d(t,"e",(function(){return _})),n.d(t,"d",(function(){return d})),n.d(t,"g",(function(){return g})),n.d(t,"b",(function(){return j}));var r=n("27e2"),o=n("6977");function a(e){switch(typeof e){case"object":case"symbol":case"function":return NaN;default:return Number(e)}}function s(e,t){for(let n=0;n<t.length;n++){const o=Array.isArray(t[n])&&Object(r["d"])(t[n][0],e.config.reservedNames.unquote);o&&(t=t.slice(0,n).concat([j(e,t[n][1])],t.slice(n+1))),Array.isArray(t[n])&&(t=t.slice(0),t[n]=s(e,t[n]))}return l(e,t)}function l(e,t){if(e.config.enableSplice)for(let n=t.length-1;n>=0;n--){const o=Array.isArray(t[n])&&Object(r["d"])(t[n][0],e.config.reservedNames.splice);o&&(t=t.slice(0,n).concat(t[n][1],t.slice(n+1)))}return t}function c(e,t,n){const r=e.macroMap.get(t.symbol);let o=null;if(r){let a=r;const s=n.slice(1);while(a){if(!a.formalArgs)return{fn:a.fn(e,t.symbol),actualArgs:n};{const r=y(e,t.symbol,a.formalArgs,Boolean(a.lastIsSpread),s);if(!r.error)return{fn:a.fn(e,t.symbol,r.formalArgs),actualArgs:n.slice(0,1).concat(r.actualArgs)};o=r.error,a=a.next}}if(o)throw new Error(o)}return!1}function i(e,t){if("function"===typeof t)return t;const n=e.funcMap.get(t.symbol);if(n)return n.fn(e,t.symbol);{const n=f(e,t);if("function"===typeof n)return n;if(e.config.funcSymbolResolverFallback)return e.config.funcSymbolResolverFallback(e,t.symbol);if(e.config.raiseOnUnresolvedSymbol)throw new Error(`[SX] resolveFunctionSymbol: Unresolved symbol: ${t.symbol}.`);return t.symbol}}function u(e,t,n){for(let o=e.scopes.length-1;o>0;o--){const n=e.scopes[o];if(n&&Object.prototype.hasOwnProperty.call(n.scope,t.symbol))return n.scope;if(n.capturedScopes&&Object.prototype.hasOwnProperty.call(n.capturedScopes,t.symbol))return n.capturedScopes[t.symbol];if(!n.isBlockLocal)break}const r=d(e);return Object.prototype.hasOwnProperty.call(r.scope,t.symbol)?r.scope:n?null:_(e).scope}function f(e,t){const n=u(e,t,!0);if(n)return n[t.symbol];const r=e.symbolMap.get(t.symbol);if(r)return r.fn(e,t.symbol);if(e.config.valueSymbolResolverFallback)return e.config.valueSymbolResolverFallback(e,t.symbol);if(e.config.raiseOnUnresolvedSymbol)throw new Error(`[SX] resolveValueSymbol: Unresolved symbol: ${t.symbol}.`);return t.symbol}function m(e,t){const n={};for(const r of t){const t=u(e,r,!0);if(null===t)throw new Error("[SX] collectCapturedVariables: Unresolved symbols "+r);Object(o["c"])("collectCapturedVariables",n,r.symbol),n[r.symbol]=t}return n}function b(e){const t=[];for(let n=e.scopes.length-1;n>0;n--){const r=e.scopes[n];if(r.capturedScopes&&t.unshift(r.capturedScopes),!r.isBlockLocal)break}return t.length>0?Object.assign({},...t):void 0}function $(e,t,n,r){e.scopes.push({isBlockLocal:n,scope:t,capturedScopes:r})}function p(e){if(e.scopes.length<2)throw new Error("[SX] uninstallScope: Unable to pop stack.");return e.scopes.pop()}function _(e){return e.scopes[e.scopes.length-1]}function d(e){return e.scopes[0]}function y(e,t,n,o,a){if(n=n.slice(0),a=a.slice(0),a.length+(o?1:0)<n.length)return{error:`[SX] macro call (${t}): Actual args too short: actual ${a.length} / formal ${n.length}.`};for(let s=n.length-(o?2:1);s>=0;s--){let e=n[s].symbol;if(e.startsWith("!")){if(n[s].symbol=n[s].symbol.slice(1),e=n[s].symbol,!Object(r["d"])(a[s]))return{error:`[SX] macro call (${t}): Actual arg(${s}: ${e}) is not symbol.`}}else if(e.startsWith("<")&&e.endsWith(">")){if(n[s].symbol=n[s].symbol.slice(1,-1),e=n[s].symbol,!Object(r["d"])(a[s],e))return{error:`[SX] macro call (${t}): Actual arg(${s}: ${e}) is not expected symbol.`};n=n.slice(0,s).concat(n.slice(s+1)),a=a.slice(0,s).concat(a.slice(s+1))}else{const o=e.lastIndexOf(":");if(0<o){const l=e.slice(o+1);switch(l){case"number":if("number"!==typeof a[s])return{error:`[SX] macro call (${t}): Actual arg(${s}: ${e}) is not number.`};break;case"string":if("string"!==typeof a[s])return{error:`[SX] macro call (${t}): Actual arg(${s}: ${e}) is not string.`};break;case"function":if(!Array.isArray(a[s])||!Object(r["d"])(a[s][0]))return{error:`[SX] macro call (${t}): Actual arg(${s}: ${e}) is not function.`};break;case"list":if(!Array.isArray(a[s]))return{error:`[SX] macro call (${t}): Actual arg(${s}: ${e}) is not list.`};break;case"symbol":if(!Object(r["d"])(a[s]))return{error:`[SX] macro call (${t}): Actual arg(${s}: ${e}) is not symbol.`};break;case"any":break;default:return{error:`[SX] macro call (${t}): Formal arg(${s}: ${e}) is unknown type ${l}.`}}n[s].symbol=n[s].symbol.slice(0,o)}}}return{formalArgs:n,actualArgs:a}}function g(e,t,n){if(Array.isArray(n[n.length-1])){const r=n.slice(0,n.length-1),o=n[n.length-1];if(o&&"object"===typeof o[0]&&o[0].symbol===e.config.reservedNames.if&&Array.isArray(o[3])&&"object"===typeof o[3][0]&&o[3][0].symbol===e.config.reservedNames.self){const n=`$__tempvar__$$ec${e.evalCount++}$$_`,a=t.map((e,t)=>({symbol:`${n}_$i${t}_${e.symbol}`}));return[[{symbol:e.config.reservedNames.until},o[1],[{symbol:e.config.reservedNames.let},[...a],...r,...o[3].slice(1).map((t,n)=>[{symbol:e.config.reservedNames.set},a[n],t]),...a.map((n,r)=>[{symbol:e.config.reservedNames.set},t[r],n])]],...r,o[2]]}}return n}function j(e,t){if(Object(o["d"])(e),null===t||void 0===t)return t;let n=t;for(;;){if(!Array.isArray(n))break;{if(0===n.length)return n;n=l(e,n);const t=Object(r["d"])(n[0]);if(!t)break;{const r=c(e,t,n);if(!r)break;n=r.fn(r.actualArgs)}}Object(o["d"])(e)}if(Array.isArray(n)){if(n=n.slice(0),0<n.length){const t=Object(r["d"])(n[0]);if(t){if(t.symbol===e.config.reservedNames.quote)return n.slice(1,2)[0];if(t.symbol===e.config.reservedNames.backquote)return n=n.slice(1,2)[0],Array.isArray(n)&&(n=s(e,n)),n;if(t.symbol===e.config.reservedNames.eval)return j(e,j(e,n.slice(1,2)[0]))}const o=[];for(let s=1;s<n.length;s++){const t=Array.isArray(n[s])&&Object(r["d"])(n[s][0],e.config.reservedNames.spread);if(t){o.push(s);const t=j(e,n[s][1]);n[s]=Array.isArray(t)?t:[t]}else n[s]=j(e,n[s])}for(const e of o.reverse())n=n.slice(0,e).concat(n[e],n.slice(e+1));let a;if(a="function"===typeof n[0]?n[0]:t?i(e,t):j(e,n[0]),"function"!==typeof a)throw new Error(`[SX] evaluate: First item of list is not a function: ${JSON.stringify(n)}.`);n=a(...n.slice(1))}}else if(e.config.wrapExternalValue&&Object.prototype.hasOwnProperty.call(n,"value"))n=n.value;else if(Object.prototype.hasOwnProperty.call(n,"symbol"))n=f(e,n);else if(Object.prototype.hasOwnProperty.call(n,"car")){const t=j(e,n.car),r=j(e,n.cdr);if(Array.isArray(r)){const e=r.slice(0);e.unshift(t),n=e}else n={car:t,cdr:r}}else Object.prototype.hasOwnProperty.call(n,"dotted")?n=[j(e,n.dotted)]:Object.prototype.hasOwnProperty.call(n,"comment")&&(n=[]);return n}},6921:function(e,t,n){"use strict";n.d(t,"a",(function(){return r}));const r=Function("return this")()},6977:function(e,t,n){"use strict";n.d(t,"d",(function(){return a})),n.d(t,"a",(function(){return s})),n.d(t,"b",(function(){return i})),n.d(t,"c",(function(){return u}));var r=n("c541"),o=n("6921");function a(e){if(e.evalCount++,e.config.maxEvalCount&&e.config.maxEvalCount<e.evalCount)throw new r["b"]}function s(e,t,n,r){if(t.length<n)throw new Error(`[SX] ${e}: Invalid argument length: expected: ${n} / args: ${t.length}.`);if(r&&r<t.length)throw new Error(`[SX] ${e}: Invalid argument length: expected: ${r} / args: ${t.length}.`);return t}const l={}.constructor,c=Function;function i(e,t){if("__proto__"===t||"__defineGetter__"===t||"__defineSetter__"===t||"__lookupGetter__"===t||"__lookupSetter__"===t)throw new Error(`[SX] ${e}: Invalid var name ${t}.`);if("prototype"===t||"constructor"===t)throw new Error(`[SX] ${e}: Invalid var name ${t}.`);if(l.hasOwnProperty(t))throw new Error(`[SX] ${e}: Invalid var name ${t}.`);if("call"===t||"arguments"===t||"caller"===t)throw new Error(`[SX] ${e}: Invalid var name ${t}.`);return t}function u(e,t,n){if(t===o["a"]||"__proto__"===n||"__defineGetter__"===n||"__defineSetter__"===n||"__lookupGetter__"===n||"__lookupSetter__"===n)throw new Error(`[SX] ${e}: Invalid var name ${n}.`);if(("prototype"===n||"constructor"===n)&&(null===t||void 0===t||"function"===typeof t))throw new Error(`[SX] ${e}: Invalid var name ${n}.`);if((null===t||void 0===t||t===l)&&l.hasOwnProperty(n))throw new Error(`[SX] ${e}: Invalid var name ${n}.`);if(null===t||void 0===t||t===c){let t=c;while(t){if(t.hasOwnProperty(n))throw new Error(`[SX] ${e}: Invalid var name ${n}.`);t=t.__proto__}}if("function"===typeof t&&!t.hasOwnProperty(n))throw new Error(`[SX] ${e}: Invalid var name ${n}.`);return n}},7941:function(module,__webpack_exports__,__webpack_require__){"use strict";__webpack_require__.d(__webpack_exports__,"a",(function(){return compileLambda}));var _ast__WEBPACK_IMPORTED_MODULE_0__=__webpack_require__("27e2"),_evaluate__WEBPACK_IMPORTED_MODULE_1__=__webpack_require__("43fb"),_errors__WEBPACK_IMPORTED_MODULE_2__=__webpack_require__("6977"),_compile_ops_helpers__WEBPACK_IMPORTED_MODULE_3__=__webpack_require__("1773"),_compile_ops__WEBPACK_IMPORTED_MODULE_4__=__webpack_require__("d8d6");function compileCore(e,t,n,r,o){function a(e){const t=new Map;for(const n of c.varNames.entries())t.set(n[0],n[1]);e(),c.varNames=t}function s(e){let t="";if(null===e)t+="(null)";else if(void 0===e)t+="(void 0)";else switch(typeof e){case"boolean":case"number":t+=`(${String(e)})`;break;case"object":if(Object(_ast__WEBPACK_IMPORTED_MODULE_0__["d"])(e)){const n=e;Object(_errors__WEBPACK_IMPORTED_MODULE_2__["c"])("(compiler)compileValue",c.varNames,n.symbol),c.varNames.has(n.symbol)?t+=`(${c.varNames.get(n.symbol)})`:(o[c.varsCount]=n.symbol,t+=`(_$_vars[${String(c.varsCount++)}])`);break}default:o[c.varsCount]=e,t+=`(_$_vars[${String(c.varsCount++)}])`;break}return t}function l(t,n){let r="";const a=t[n];if(Array.isArray(a))if(0<a.length){const t=Object(_compile_ops_helpers__WEBPACK_IMPORTED_MODULE_3__["a"])(e,a);if(Array.isArray(t))if(0<t.length)if(Object(_ast__WEBPACK_IMPORTED_MODULE_0__["d"])(t[0])){const n=t[0],a=t.slice(1);Object(_errors__WEBPACK_IMPORTED_MODULE_2__["c"])("(compiler)compileToken",i,n.symbol),i.has(n.symbol)?r+=i.get(n.symbol)(t,a):n.symbol===e.config.reservedNames.spread?r+=`...(${a.map(t=>l([Object(_compile_ops_helpers__WEBPACK_IMPORTED_MODULE_3__["f"])(e,t)],0)).join(",")})`:e.funcMap.has(n.symbol)?(o[c.varsCount]=e.funcMap.get(n.symbol).fn(e,""),r+=`((_$_vars[${String(c.varsCount++)}])(${a.map(t=>l([Object(_compile_ops_helpers__WEBPACK_IMPORTED_MODULE_3__["f"])(e,t)],0)).join(",")}))`):c.varNames.has(n.symbol)?r+=`(${String(c.varNames.get(n.symbol))})(${a.map(t=>l([Object(_compile_ops_helpers__WEBPACK_IMPORTED_MODULE_3__["f"])(e,t)],0)).join(",")})`:(o[c.varsCount]=Object(_compile_ops_helpers__WEBPACK_IMPORTED_MODULE_3__["d"])(e,n.symbol),r+=`((_$_vars[${String(c.varsCount++)}])()(${a.map(t=>l([Object(_compile_ops_helpers__WEBPACK_IMPORTED_MODULE_3__["f"])(e,t)],0)).join(",")}))`)}else switch(typeof t[0]){case"function":o[c.varsCount]=t[0],r+=`((_$_vars[${String(c.varsCount++)}])(${t.slice(1).map((t,n,r)=>l([Object(_compile_ops_helpers__WEBPACK_IMPORTED_MODULE_3__["f"])(e,t)],0)).join(",")}))`;break;default:throw new Error(`[SX] compileToken: First item of list is not a function: ${JSON.stringify(t[0])}.`)}else r+="([])";else r+=s(a)}else r+="([])";else r+=s(a);return r}const c={_$_vars:o,varsCount:1,varNames:new Map,varNamesCount:0,varDefs:"var x0;",ops:new Map,makeScope:a,compileToken:l};Object(_compile_ops__WEBPACK_IMPORTED_MODULE_4__["a"])(e,c);const i=c.ops;e.config.enableTailCallOptimization&&(r=Object(_evaluate__WEBPACK_IMPORTED_MODULE_1__["g"])(e,t,r)),c.varNames.set(e.config.reservedNames.thiz,"(this===(Function('return this')())?null:(this===void 0?null:this))");const u=Object(_evaluate__WEBPACK_IMPORTED_MODULE_1__["c"])(e);if(u)for(const b in u)Object.prototype.hasOwnProperty.call(u,b)&&(o[c.varsCount++]=b,o[c.varsCount]=u[b],c.varNames.set(b,`(_$_vars[${String(c.varsCount)}][_$_vars[${String(c.varsCount-1)}]])`),c.varsCount++);const f=""+t.map((e,r)=>(c.varNames.set(t[r].symbol,"a"+r),`${n&&r===t.length-1?"...":""}a${r}`)).join(","),m=`return(${r.map((e,t)=>l(r,t)).join(",")})`;return`(function(${f}){"strict";${c.varDefs}${m}})`}function evalCompiledLambda(_$_state,_$_vars,code){return _$_vars[0]=eval(code),_$_vars[0]}function compileLambda(e,t,n,r){const o=[];return evalCompiledLambda(e,o,compileCore(e,t,n,r,o))}},c541:function(e,t,n){"use strict";n.d(t,"a",(function(){return r})),n.d(t,"b",(function(){return o})),n.d(t,"c",(function(){return a}));class r extends Error{constructor(e){super(e)}}class o extends r{constructor(){super("[SX] evaluate: The maximum count of evaluations has been exceeded.")}}class a extends r{constructor(e){super(`[SX] ${e}: Unexpected termination of script.`)}}},d8d6:function(e,t,n){"use strict";n.d(t,"a",(function(){return s}));var r=n("27e2"),o=n("6977"),a=n("1773");function s(e,t){const{_$_vars:n,ops:s,makeScope:l,compileToken:c}=t;s.set(e.config.reservedNames.quote,(function(e,r){let o="";return n[t.varsCount]=e[1],o+=`(_$_vars[${String(t.varsCount++)}])`,o})),s.set(e.config.reservedNames.self,(function(t,n){let r="";return r+=`((_$_vars[0])(${n.map(t=>c([Object(a["f"])(e,t)],0)).join(",")}))`,r})),s.set("$__if",(function(t,n){let r="";return Object(o["a"])("compileToken:$__if",n,2,3),r+=`(${c(t,1)}?(${c([Object(a["e"])(e,t[2])],0)}):(${c([Object(a["e"])(e,t[3])],0)}))`,r})),s.set("$__if-null",(function(t,n){let r="";return Object(o["a"])("compileToken:$__if-null",n,2,2),r+=`((()=>{let _$_rv=${c(t,1)};return _$_rv?_$_rv:(${c([Object(a["e"])(e,t[2])],0)}});})())`,r})),s.set("$__cond",(function(t,n){let r="";Object(o["a"])("compileToken:$__cond",n,1),r+="(";for(let o=0;o<n.length;o+=2)r+=`${c([Object(a["e"])(e,n[o])],0)}?(${c([Object(a["e"])(e,n[o+1])],0)}):(`;r+="null";for(let e=0;e<n.length;e+=2)r+=")";return r+=")",r})),s.set("$__while",(function(t,n){let r="";return Object(o["a"])("compileToken:$__while",n,1),r+=`((()=>{let _$_rv=null;while(${c([Object(a["e"])(e,t[1])],0)}){_$_rv=${t.slice(2).map(t=>c([Object(a["e"])(e,t)],0)).join(",")}}return _$_rv})())`,r})),s.set("$__do-while",(function(t,n){let r="";return Object(o["a"])("compileToken:$__do-until",n,1),r+=`((()=>{let _$_rv=null;do{_$_rv=${t.slice(2).map(t=>c([Object(a["e"])(e,t)],0)).join(",")}}}while(${c([Object(a["e"])(e,t[1])],0)})return _$_rv)())`,r})),s.set("$__until",(function(t,n){let r="";return Object(o["a"])("compileToken:$__until",n,1),r+=`((()=>{let _$_rv=null;while(!${c([Object(a["e"])(e,t[1])],0)}){_$_rv=${t.slice(2).map(t=>c([Object(a["e"])(e,t)],0)).join(",")}}return _$_rv})())`,r})),s.set("$__do-until",(function(t,n){let r="";return Object(o["a"])("compileToken:$__do-until",n,1),r+=`((()=>{let _$_rv=null;do{_$_rv=${t.slice(2).map(t=>c([Object(a["e"])(e,t)],0)).join(",")}}}while(!${c([Object(a["e"])(e,t[1])],0)})return _$_rv)())`,r})),s.set("$__repeat",(function(n,s){let i="";if(Object(o["a"])("compileToken:$__repeat",s,2),!Object(r["d"])(s[0]))throw new Error("[SX] compileToken: $__repeat : args[0] is not symbol.");return l(()=>{const r="v"+t.varNamesCount++;t.varNames.set(s[0].symbol,r),i+=`(((_$_n)=>{let _$_rv=null;for(let ${r}=0;${r}<_$_n;${r}++){_$_rv=${n.slice(2).map(t=>c([Object(a["e"])(e,t)],0)).join(",")}}return _$_rv})(${c(s,1)}))`}),i})),s.set("$__for",(function(n,s){let i="";if(Object(o["a"])("compileToken:$__for",s,2),!Object(r["d"])(s[0]))throw new Error("[SX] compileToken: $__for : args[0] is not symbol.");return l(()=>{const r="v"+t.varNamesCount++;t.varNames.set(s[0].symbol,r),i+=`(((_$_l)=>{let _$_rv=null;for(const ${r} of _$_l){_$_rv=${n.slice(2).map(t=>c([Object(a["e"])(e,t)],0)).join(",")}}return _$_rv})(${c(s,1)}))`}),i})),s.set("$__scope",(function(n,s){let i="";if(Object(o["a"])("compileToken:$__scope",s,2),!Array.isArray(n[3]))throw new Error("[SX] compileToken: $__scope : args[2] is not array.");return l(()=>{for(const s of Object(a["e"])(e,n[3])){let e="";if(Array.isArray(s)){if(s.length<1)throw new Error("[SX] compileToken: $__scope : args[0][?] is too short.");if(!Object(r["d"])(s[0]))throw new Error("[SX] compileToken: $__scope : args[0][?][0] is not symbol.");e=s[0].symbol,i+=`(${"v"+t.varNamesCount}=${c(s,1)})`}else{if(!Object(r["d"])(s))throw new Error("[SX] compileToken: $__scope : args[0][?] is not symbol.");e=s.symbol}t.varDefs+=`var v${t.varNamesCount}=void 0;`,t.varNames.set(e,"v"+t.varNamesCount++)}const o=""+n.slice(4).map(t=>c([Object(a["e"])(e,t)],0)).join(",");i+=n[2]?`[${o}]`:`(${o})`}),i})),s.set("$__try",(function(r,s){let i="";return Object(o["a"])("compileToken:$__try",s,1,2),i+=`((()=>{try{${c([Object(a["e"])(e,r[1])],0)}}catch(e${t.varNamesCount}){let e${t.varNamesCount+1}=(_$_vars[${String(t.varsCount)}])();`,n[t.varsCount++]=Object(a["c"])(e),l(()=>{t.varNames.set("$error","e"+t.varNamesCount++),t.varNames.set("$parent","e"+t.varNamesCount++),i+=c([Object(a["e"])(e,r[2])],0)+"}})())"}),i})),s.set(e.config.reservedNames.raise,(function(t,n){let r="";return r+=`((()=>{throw ${c([Object(a["f"])(e,t[1])],0)}})())`,r})),s.set("$boolean",(function(e,t){let n="";return Object(o["a"])("compileToken:$boolean",t,1,1),n+=`((x0=${c(t,0)}),(Array.isArray(x0)&&x0.length===0?false:boolean(x0)))`,n})),s.set("$__get",(function(s,l){let i="";Object(o["a"])("compileToken:$__get",l,1);const u=Object(a["e"])(e,s[1]),f=Object(r["d"])(u)?u.symbol:"string"===typeof u?u:null;if("string"!==typeof f)throw new Error(`[SX] compileToken: $__get : operand is not symbol: ${JSON.stringify(s[1])}.`);let m="";t.varNames.has(f)?m=t.varNames.get(f):(n[t.varsCount]=Object(a["d"])(e,f),m=`_$_vars[${String(t.varsCount++)}]`);const b=e=>(n[t.varsCount]=e,`_$_vars[${String(t.varsCount++)}]`);return i+=`((${m})${s.slice(2).map((e,t,n)=>`[${Object(r["d"])(n[t])?b(n[t].symbol):c(n,t)}]`).join("")})`,i})),s.set("$__let",(function(n,s){let l="";Object(o["a"])("compileToken:$__let",s,2,2);const i=Object(a["e"])(e,n[1]),u=Object(r["d"])(i)?i.symbol:"string"===typeof i?i:null;if("string"!==typeof u)throw new Error(`[SX] compileToken: $__let : operand is not rvalue: ${JSON.stringify(n[1])}.`);return t.varNames.has(u)||(t.varDefs+=`var v${t.varNamesCount}=void 0;`,t.varNames.set(u,"v"+t.varNamesCount++)),l+=`(${t.varNames.get(u)}=${c(n,2)})`,l})),s.set("$__set",(function(s,l){let i="";Object(o["a"])("compileToken:$__set",l,2);const u=Object(a["e"])(e,s[1]),f=Object(r["d"])(u)?u.symbol:"string"===typeof u?u:Array.isArray(u)?Object(r["d"])(u[0])?u[0].symbol:"string"===typeof u[0]?u[0]:null:null;if("string"!==typeof f)throw new Error(`[SX] compileToken: $__set : operand is not rvalue: ${JSON.stringify(s[1])}.`);let m="";t.varNames.has(f)?m=t.varNames.get(f):(n[t.varsCount]=Object(a["d"])(e,f),m=`_$_vars[${String(t.varsCount++)}]`);const b=e=>(n[t.varsCount]=Object(o["b"])("compileToken:$__set",e),`_$_vars[${String(t.varsCount++)}]`);let $=!1,p="";const _=e=>($||(n[t.varsCount]=Object(a["b"])("compileToken:$__set"),p=String(t.varsCount++),$=!0),`(_$_vars[${p}](${e}))`);return i+=`((${m})${(Array.isArray(u)?u.slice(1):[]).map((e,t,n)=>`[${Object(r["d"])(n[t])?b(n[t].symbol):_(c(n,t))}]`).join("")}=${c(s,2)})`,i})),s.set(e.config.reservedNames.not,(function(e,t){let n="";return Object(o["a"])("compileToken:$not",t,1,1),n+=`(!${c(t,0)})`,n})),s.set("$__and",(function(t,n){let r="";return Object(o["a"])("compileToken:$__and",n,1),r+=`(${n.map(t=>c([Object(a["e"])(e,t)],0)).join("&&")})`,r})),s.set("$__or",(function(t,n){let r="";return Object(o["a"])("compileToken:$__or",n,1),r+=`(${n.map(t=>c([Object(a["e"])(e,t)],0)).join("||")})`,r})),s.set("===",(function(e,t){let n="";return Object(o["a"])("compileToken:===",t,2,2),n+=`(${t.map((e,t,n)=>c(n,t)).join("===")})`,n})),s.set("!==",(function(e,t){let n="";return Object(o["a"])("compileToken:!==",t,2,2),n+=`(${t.map((e,t,n)=>c(n,t)).join("!==")})`,n})),s.set("==",(function(e,t){let n="";return Object(o["a"])("compileToken:==",t,2,2),n+=`(${t.map((e,t,n)=>c(n,t)).join("==")})`,n})),s.set("!=",(function(e,t){let n="";return Object(o["a"])("compileToken:!=",t,2,2),n+=`(${t.map((e,t,n)=>c(n,t)).join("!=")})`,n})),s.set("<",(function(e,t){let n="";return Object(o["a"])("compileToken:<",t,2,2),n+=`(${t.map((e,t,n)=>c(n,t)).join("<")})`,n})),s.set("<=",(function(e,t){let n="";return Object(o["a"])("compileToken:<=",t,2,2),n+=`(${t.map((e,t,n)=>c(n,t)).join("<=")})`,n})),s.set(">",(function(e,t){let n="";return Object(o["a"])("compileToken:>",t,2,2),n+=`(${t.map((e,t,n)=>c(n,t)).join(">")})`,n})),s.set(">=",(function(e,t){let n="";return Object(o["a"])("compileToken:<=",t,2,2),n+=`(${t.map((e,t,n)=>c(n,t)).join(">=")})`,n})),s.set("$concat",(function(t,n){let a="";Object(o["a"])("compileToken:$concat",n,1);let s="";if(Array.isArray(n[0])&&Object(r["d"])(n[0][0],e.config.reservedNames.spread)){const e=c(n[0],1);s=`(${e}[0]).concat((${e}.length>1?${e}[1]:(typeof ${e}[0]==='string'?'':[])),`}else s=c(n,0)+".concat(";return a+=`(${s}${n.slice(1).map((e,t,n)=>c(n,t)).join(",")}))`,a})),s.set("+",(function(t,n){let a="";Object(o["a"])("compileToken:+",n,1);let s=!1;return n.map((t,n,o)=>{Array.isArray(t)&&Object(r["d"])(t[0],e.config.reservedNames.spread)&&(s=!0)}),a+=s?`((()=>{let _$_rv=[];${n.map((e,t,n)=>`_$_rv.push(${c(n,t)})`).join(";")};return _$_rv.reduce((x,y)=>x+y);})())`:`(${n.map((e,t,n)=>c(n,t)).join("+")})`,a})),s.set("-",(function(t,n){let a="";Object(o["a"])("compileToken:-",n,1);let s=!1;return n.map((t,n,o)=>{Array.isArray(t)&&Object(r["d"])(t[0],e.config.reservedNames.spread)&&(s=!0)}),a+=s?`((()=>{let _$_rv=[];${n.map((e,t,n)=>`_$_rv.push(${c(n,t)})`).join(";")};return (_$_rv.length>1?(_$_rv.reduce((x,y)=>x-y)):(_$_rv.length>0?-_$_rv[0]:NaN));})())`:`(${t.length>2?n.map((e,t,n)=>c(n,t)).join("-"):`-(${String(c(t,1))})`})`,a})),s.set("*",(function(t,n){let a="";Object(o["a"])("compileToken:*",n,2);let s=!1;return n.map((t,n,o)=>{Array.isArray(t)&&Object(r["d"])(t[0],e.config.reservedNames.spread)&&(s=!0)}),a+=s?`((()=>{let _$_rv=[];${n.map((e,t,n)=>`_$_rv.push(${c(n,t)})`).join(";")};return _$_rv.reduce((x,y)=>x*y);})())`:`(${n.map((e,t,n)=>c(n,t)).join("*")})`,a})),s.set("**",(function(t,n){let a="";Object(o["a"])("compileToken:**",n,2);let s=!1;return n.map((t,n,o)=>{Array.isArray(t)&&Object(r["d"])(t[0],e.config.reservedNames.spread)&&(s=!0)}),a+=s?`((()=>{let _$_rv=[];${n.map((e,t,n)=>`_$_rv.push(${c(n,t)})`).join(";")};return _$_rv.reduce((x,y)=>x**y);})())`:`(${n.map((e,t,n)=>c(n,t)).join("**")})`,a})),s.set("/",(function(t,n){let a="";Object(o["a"])("compileToken:/",n,2);let s=!1;return n.map((t,n,o)=>{Array.isArray(t)&&Object(r["d"])(t[0],e.config.reservedNames.spread)&&(s=!0)}),a+=s?`((()=>{let _$_rv=[];${n.map((e,t,n)=>`_$_rv.push(${c(n,t)})`).join(";")};return _$_rv.reduce((x,y)=>x/y);})())`:`(${n.map((e,t,n)=>c(n,t)).join("/")})`,a})),s.set("%",(function(t,n){let a="";Object(o["a"])("compileToken:%",n,2);let s=!1;return n.map((t,n,o)=>{Array.isArray(t)&&Object(r["d"])(t[0],e.config.reservedNames.spread)&&(s=!0)}),a+=s?`((()=>{let _$_rv=[];${n.map((e,t,n)=>`_$_rv.push(${c(n,t)})`).join(";")};return _$_rv.reduce((x,y)=>x%y);})())`:`(${n.map((e,t,n)=>c(n,t)).join("%")})`,a})),s.set("<<",(function(e,t){let n="";return Object(o["a"])("compileToken:<<",t,2,2),n+=`(${t.map((e,t,n)=>c(n,t)).join("<<")})`,n})),s.set(">>",(function(e,t){let n="";return Object(o["a"])("compileToken:>>",t,2,2),n+=`(${t.map((e,t,n)=>c(n,t)).join(">>")})`,n})),s.set(">>>",(function(e,t){let n="";return Object(o["a"])("compileToken:>>>",t,2,2),n+=`(${t.map((e,t,n)=>c(n,t)).join(">>>")})`,n})),s.set("$bit-not",(function(e,t){let n="";return Object(o["a"])("compileToken:$bit-not",t,1,1),n+=`(~(${c(e,1)}))`,n})),s.set("$bit-and",(function(e,t){let n="";return Object(o["a"])("compileToken:$bit-and",t,2,2),n+=`(${t.map((e,t,n)=>c(n,t)).join("&")})`,n})),s.set("$bit-or",(function(e,t){let n="";return Object(o["a"])("compileToken:$bit-or",t,2,2),n+=`(${t.map((e,t,n)=>c(n,t)).join("|")})`,n})),s.set("$bit-xor",(function(e,t){let n="";return Object(o["a"])("compileToken:$bit-xor",t,2,2),n+=`(${t.map((e,t,n)=>c(n,t)).join("^")})`,n}))}},f587:function(e,t,n){"use strict";n.r(t),n.d(t,"FatalError",(function(){return r["a"]})),n.d(t,"MaxEvaluationCountError",(function(){return r["b"]})),n.d(t,"ScriptTerminationError",(function(){return r["c"]})),n.d(t,"quote",(function(){return o["f"]})),n.d(t,"isQuoted",(function(){return o["c"]})),n.d(t,"backquote",(function(){return o["a"]})),n.d(t,"isBackquoted",(function(){return o["b"]})),n.d(t,"wrapByUnquote",(function(){return o["i"]})),n.d(t,"isUnquoted",(function(){return o["e"]})),n.d(t,"spread",(function(){return o["h"]})),n.d(t,"splice",(function(){return o["g"]})),n.d(t,"isSymbol",(function(){return o["d"]})),n.d(t,"defaultReservedNames",(function(){return a})),n.d(t,"defaultConfig",(function(){return s})),n.d(t,"SExpression",(function(){return T})),n.d(t,"SExpressionAsync",(function(){return X})),n.d(t,"S",(function(){return x})),n.d(t,"L",(function(){return Qn})),n.d(t,"LS",(function(){return Zn})),n.d(t,"lisp",(function(){return er})),n.d(t,"L_async",(function(){return tr})),n.d(t,"LS_async",(function(){return nr})),n.d(t,"lisp_async",(function(){return rr})),n.d(t,"LM",(function(){return or})),n.d(t,"LM_async",(function(){return ar})),n.d(t,"LSX",(function(){return gr})),n.d(t,"LSX_async",(function(){return jr})),n.d(t,"runScriptTags",(function(){return Or})),n.d(t,"installCore",(function(){return St})),n.d(t,"installArithmetic",(function(){return Zt})),n.d(t,"installSequence",(function(){return Dn})),n.d(t,"installJsx",(function(){return yr})),n.d(t,"installConcurrent",(function(){return Yn})),n.d(t,"builtinOperators",(function(){return hr})),n.d(t,"builtinMacros",(function(){return vr})),n.d(t,"builtinSymbols",(function(){return wr}));var r=n("c541"),o=n("27e2");const a={eval:"$eval",quote:"$quote",backquote:"$backquote",unquote:"$unquote",spread:"$spread",splice:"$splice",car:"$car",cdr:"$cdr",cons:"$cons",atom:"$atom",eq:"$eq",list:"$list",let:"$clisp-let",lambda:"$lambda",self:"$self",defun:"$defun",thiz:"$this",if:"$if",cond:"$cond",while:"$while",doWhile:"$do-while",until:"$until",doUntil:"$do-until",get:"$get",defvar:"$clisp-defvar",setq:"$clisp-setq",set:"$set",call:"$call",not:"$not",and:"$and",or:"$or",isSymbol:"$is-symbol",gensym:"$gensym",raise:"$raise",catch:"$catch",Template:"Template"},s={raiseOnUnresolvedSymbol:!1,enableEvaluate:!0,enableHereDoc:!0,enableSpread:!0,enableSplice:!0,enableShorthands:!0,enableVerbatimStringLiteral:!0,enableTailCallOptimization:!0,enableRegExpMatchOperators:!0,enableCompilationOperators:!0,stripComments:!1,wrapExternalValue:!0,returnMultipleRoot:!1,maxEvalCount:0,reservedNames:a,symbols:[],macros:[],funcs:[]};function l(e){return"object"===typeof e&&Object.prototype.hasOwnProperty.call(e,"eof")}function c(e){return"string"===typeof e&&0===e.trim().length}function i(e){return"string"===typeof e&&/^[0-9\+\-]$/.test(e)}function u(e){return"string"===typeof e&&/^[0-9]$/.test(e)}function f(e){return"string"===typeof e&&!c(e)&&!i(e)}function m(e){return`line: ${e.line} / strings: ${e.index} / pos: ${e.pos} :${e.strings.length>e.index?e.strings[e.index].slice(e.pos,e.pos+20):""}`}function b(e,t,n){if(e.strings.length<=e.index)return{eof:!0};if(e.strings[e.index].length<=e.pos){if(!e.values||e.values.length<=e.index)return e.pos=0,e.index++,b(e);{const t={value:e.values[e.index]};return e.pos=0,e.index++,t}}if(t)for(const r of t){const t=e.strings[e.index].slice(e.pos,e.pos+r.length);if(t===r)return e.pos+=r.length,e.line+=t.split("\n").length-1,{eof:!1,eofSeq:r}}{let t=e.strings[e.index].slice(e.pos,e.pos+1);if(e.pos++,"\n"===t&&e.line++,!n&&"\\"===t){if(e.strings[e.index].length<=e.pos)throw new Error(`[SX] getChar: Invalid syntax at: ${m(e)}.`);switch(t=e.strings[e.index].slice(e.pos,e.pos+1),e.pos++,t){case"b":t="\b";break;case"t":t="\t";break;case"n":t="\n";break;case"v":t="\v";break;case"f":t="\f";break;case"r":t="\r";break;case"U":case"u":if("{"===e.strings[e.index].slice(e.pos,e.pos+1)){let n="";for(let t=0;t<6;t++){const r=e.strings[e.index].slice(e.pos+t,e.pos+1+t);if("}"===r){if(0===t)throw new Error(`[SX] getChar: Invalid syntax at: ${m(e)}.`);e.pos+=t;break}if(!/^[0-9A-Fa-f]{1}$/.test(n))throw new Error(`[SX] getChar: Invalid syntax at: ${m(e)}.`);n+=r}if("}"!==e.strings[e.index].slice(e.pos,e.pos+1))throw new Error(`[SX] getChar: Invalid syntax at: ${m(e)}.`);e.pos++,t=String.fromCodePoint(Number.parseInt(n,16))}else{const n=e.strings[e.index].slice(e.pos,e.pos+4);if(!/^[0-9A-Fa-f]{4}$/.test(n))throw new Error(`[SX] getChar: Invalid syntax at: ${m(e)}.`);e.pos+=4,t=String.fromCodePoint(Number.parseInt(n,16))}break}}return t}}function $(e,t,n,r){const o=e.index,a=e.pos,s=e.line,l=[];try{for(let o=0;o<t;o++)l.push(b(e,n,r))}finally{e.index=o,e.pos=a,e.line=s}return l}function p(e,t,n){const r=e.index,o=e.pos,a=e.line;let s;try{s=b(e,t,n)}finally{e.index=r,e.pos=o,e.line=a}return s}function _(e){let t=p(e);while(!l(t)&&c(t))b(e),t=p(e)}function d(e,t){let n="",r=p(e,t);while(!l(r)){if("string"!==typeof r)break;if(/^0[XxOoBb][0-9]*$/.test(n+r))b(e,t),n+=r;else{if(!/^[0-9\+\-\.EeInfinityNaN]+$/.test(n+r))break;b(e,t),n+=r}r=p(e,t)}if(!/^([\+\-]?\d*\.?\d+(?:[Ee][\+\-]?\d+)?)|(0[XxOoBb][0-9]+)|([\+\-]Infinity)|(NaN)$/.test(n))throw new Error(`[SX] parseNumber: Invalid syntax at: ${m(e)}.`);return Number(n)}function y(e,t){let n="",r=p(e,t);while(!l(r)){if("string"===typeof r){if(c(r))break;if("#"===r&&"|"===$(e,2,t)[1])break;if(!/^[^.;()"]+$/.test(n+r))break;b(e,t),n+=r}else{if("object"!==typeof r||!Object.prototype.hasOwnProperty.call(r,"value"))throw new Error(`[SX] parseSymbol: Invalid syntax at: ${m(e)}.`);b(e,t);r.value;n+=String(r)}r=p(e,t)}if(e.config.enableShorthands){let t=null;if(t=n.match(/^:((?:\:[^=:]+?)+?)=$/)){const n=t[1].slice(1).split(":"),r=[{symbol:e.config.reservedNames.splice},[{symbol:e.config.reservedNames.set},n]];return r}if(t=n.match(/^:((?:\:[^@:]+?)+?)@([^@:]+?)$/)){const n=t[1].slice(1).split(":"),r=[{symbol:e.config.reservedNames.splice},[{symbol:e.config.reservedNames.call},[{symbol:e.config.reservedNames.get},...n],{symbol:t[2]}]];return r}if(t=n.match(/^:((?:\:[^:]+?)+?)$/)){const n=t[1].slice(1).split(":"),r=[{symbol:e.config.reservedNames.get},...n];return r}}return{symbol:n}}function g(e,t,n,o,a,s){const c=n?[...t,n]:t,i=[],u=[];for(;;){let t="",f=p(e,c,a);while(!l(f)){if("string"===typeof f)b(e,c,a),t+=f;else{if("object"!==typeof f||!Object.prototype.hasOwnProperty.call(f,"value"))throw new Error(`[SX] parseStringOrComment: Invalid syntax at: ${m(e)}.`);b(e,c,a);f.value;t+=String(f)}f=p(e,c,a)}if(b(e,c,a),!0===f.eof&&!s)throw new r["c"]("parseStringOrComment");if(i.push(t),f.eofSeq!==n)break;u.push(S(e,o,[]))}return{strings:i,values:u}}function j(e,t){return g(e,['"'],null,")",t,!1).strings[0]}function O(e,t,n){const r=[t];n&&r.push(n);const o=g(e,['"""'],"%%%(",")",!1,!1);for(let a=0;a<o.strings.length;a++)r.push(o.strings[a]),a<o.values.length&&r.push(o.values[a]);return r}function h(e){return{comment:g(e,["\r","\n"],null,")",!1,!0).strings[0]}}function v(e){return{comment:g(e,["|#"],null,")",!1,!1).strings[0]}}function w(e){_(e);let t=p(e);while(!l(t)){switch(t){case")":throw new Error(`[SX] parseOneToken: Invalid syntax at: ${m(e)}.`);case"(":return b(e),S(e,")",[]);case"'":case"`":case",":{b(e);const n=p(e);let r=!1;","===t&&"@"===n&&(b(e),r=!0),_(e);const a=("'"===t?o["f"]:"`"===t?o["a"]:o["i"])(e,w(e));return r?Object(o["g"])(e,a):a}case".":{b(e);const t=$(e,2);return e.config.enableSpread&&"."===t[0]&&"."===t[1]?(b(e),b(e),_(e),Object(o["h"])(e,w(e))):(_(e),{dotted:w(e)})}case";":return b(e),h(e);case"#":{const t=$(e,2);return"|"===t[1]?(b(e),b(e),v(e)):y(e)}case'"':{b(e);const t=$(e,4);if(e.config.enableHereDoc&&'"'===t[0]&&'"'===t[1]){let n=!0;l(t[2])||c(t[2])||(i(t[2])?"+"!==t[2]&&"-"!==t[2]||u(t[3])||(n=!1):f(t[2])&&(n=!1)),b(e),b(e);let r=null,o=null;if(n)r={symbol:e.config.reservedNames.Template};else{if(r=y(e,["@"]),null===r)throw new Error(`[SX] parseOneToken: Invalid syntax at: ${m(e)}.`);if("number"===typeof r)throw new Error(`[SX] parseOneToken: Invalid syntax at: ${m(e)}.`);const t=$(e,2);if("@"===t[0]){if("{"!==t[1])throw new Error(`[SX] parseOneToken: Invalid syntax at: ${m(e)}.`);b(e),b(e);const n=S(e,"}",[{symbol:"@"}]);Array.isArray(n)&&(o=n)}}return O(e,r,o)}return j(e,!1)}case"@":if(e.config.enableVerbatimStringLiteral){const t=$(e,2);if('"'===t[1])return b(e),b(e),j(e,!0)}default:if("string"!==typeof t){if("object"===typeof t&&Object.prototype.hasOwnProperty.call(t,"value"))return b(e),e.config.wrapExternalValue?t:t.value;throw new Error(`[SX] parseOneToken: Invalid syntax at: ${m(e)}.`)}if(c(t))break;if(i(t)){if("+"===t||"-"===t){const t=$(e,2);if(!u(t[1]))return y(e)}return d(e)}if(f(t))return y(e);throw new Error(`[SX] parseOneToken: Invalid syntax at: ${m(e)}.`)}_(e),t=p(e)}throw new r["c"]("parseOneToken")}function S(e,t,n){const o=n.slice(0);let a=!1;_(e);let s=p(e);while(!l(s)){switch(s){case t:return b(e),a?o[0]:o;default:{const t=w(e);if("object"===typeof t&&Object.prototype.hasOwnProperty.call(t,"dotted")){if(1!==o.length)throw new Error(`[SX] parseList: Invalid syntax at: ${m(e)}.`);a=!0,Array.isArray(t)?(t.unshift(o.pop()),o.push(t)):o.push({car:o.pop(),cdr:t.dotted})}else if("object"===typeof t&&Object.prototype.hasOwnProperty.call(t,"comment"))e.config.stripComments||o.push(t);else{if(a)throw new Error(`[SX] parseList: Invalid syntax at: ${m(e)}.`);o.push(t)}}break}_(e),s=p(e)}throw new r["c"]("parseList")}function A(e){const t=[];_(e);let n=p(e);while(!l(n)){switch(n){case"(":b(e),t.push(S(e,")",[]));break;case"'":case"`":for(b(e),_(e);;){const r=w(e);if("object"!==typeof r||!Object.prototype.hasOwnProperty.call(r,"comment")){t.push(("'"===n?o["f"]:o["a"])(e,r));break}e.config.stripComments||t.push(r)}break;case";":b(e),e.config.stripComments?h(e):t.push(h(e));break;case"#":{const n=$(e,2);"|"===n[1]?(b(e),b(e),e.config.stripComments?v(e):t.push(v(e))):(b(e),e.config.stripComments?h(e):t.push(h(e)))}break;case'"':{const n=$(e,3);if('"'===n[1]&&'"'===n[2]){t.push(w(e));break}}default:throw new Error(`[SX] parseInitialState: Invalid syntax at: ${m(e)}.`)}_(e),n=p(e)}return t}var E=n("43fb"),N=function(e,t,n,r){function o(e){return e instanceof n?e:new n((function(t){t(e)}))}return new(n||(n=Promise))((function(n,a){function s(e){try{c(r.next(e))}catch(t){a(t)}}function l(e){try{c(r["throw"](e))}catch(t){a(t)}}function c(e){e.done?n(e.value):o(e.value).then(s,l)}c((r=r.apply(e,t||[])).next())}))};function I(e,t,n,r){return{strings:"string"===typeof n?[n]:n,values:r||[],index:0,pos:0,line:0,evalCount:0,scopes:[{isBlockLocal:!1,scope:t}],macroMap:new Map(e.macros.map(e=>[e.name,e])),funcMap:new Map(e.funcs.map(e=>[e.name,e])),symbolMap:new Map(e.symbols.map(e=>[e.name,e])),config:e}}function k(e,t,n){return e.strings="string"===typeof t?[t]:t,e.values=n||[],e.index=0,e.pos=0,e.line=0,e.evalCount=0,e}function T(e){let t=e||Object.assign({},s),n={},r=[];const o=(e,n)=>{if(t.enableEvaluate)for(let t=0;t<n.length;t++)n[t]=Object(E["b"])(e,n[t]);return t.returnMultipleRoot?1===n.length?n[0]:n:n[n.length-1]},a=(e,...a)=>{const s=I(t,Object.assign({},n),e,a);return o(s,r.concat(A(s)))};return a.evaluateAST=e=>{const a=I(t,Object.assign({},n),"");return o(a,r.concat(e))},a.repl=()=>{const e=I(t,Object.assign({},n),"");o(e,r.slice(0));const a=(t,...n)=>(k(e,t,n),o(e,A(e)));return a.sync=a,a},a.setGlobals=e=>(n=Object.assign({},e||{}),a),a.appendGlobals=e=>(n=Object.assign({},n,e||{}),a),a.setStartup=(e,...o)=>{const s=I(t,Object.assign({},n),e,o);return r=A(s),a},a.setStartupAST=e=>(r=e,a),a.appendStartup=(e,...o)=>{const s=I(t,Object.assign({},n),e,o);return r=r.concat(A(s)),a},a.appendStartupAST=e=>(r=r.concat(e),a),a.install=e=>(t=e(t),a),a}function X(e){let t=e||Object.assign({},s),n={},r=[];const o=(e,n)=>N(this,void 0,void 0,(function*(){if(t.enableEvaluate)for(let t=0;t<n.length;t++)n[t]=Object(E["b"])(e,n[t]),"object"===typeof n[t]&&null!==n[t]&&"function"===typeof n[t].then&&(n[t]=yield n[t]);return t.returnMultipleRoot?1===n.length?n[0]:n:n[n.length-1]})),a=(e,...a)=>N(this,void 0,void 0,(function*(){const s=I(t,Object.assign({},n),e,a);return o(s,r.concat(A(s)))}));return a.evaluateAST=e=>{const a=I(t,Object.assign({},n),"");return o(a,r.concat(e))},a.repl=()=>{const e=(e,n)=>{if(t.enableEvaluate)for(let t=0;t<n.length;t++)n[t]=Object(E["b"])(e,n[t]);return t.returnMultipleRoot?1===n.length?n[0]:n:n[n.length-1]},a=I(t,Object.assign({},n),"");o(a,r.slice(0));const s=(e,...t)=>N(this,void 0,void 0,(function*(){return k(a,e,t),o(a,A(a))})),l=(t,...n)=>(k(a,t,n),e(a,A(a)));return s.sync=l,s},a.setGlobals=e=>(n=Object.assign({},e||{}),a),a.appendGlobals=e=>(n=Object.assign({},n,e||{}),a),a.setStartup=(e,...o)=>{const s=I(t,Object.assign({},n),e,o);return r=A(s),a},a.setStartupAST=e=>(r=e,a),a.appendStartup=(e,...o)=>{const s=I(t,Object.assign({},n),e,o);return r=r.concat(A(s)),a},a.appendStartupAST=e=>(r=r.concat(e),a),a.install=e=>(t=e(t),a),a}const x=(()=>{const e=Object.assign({},s);return e.enableEvaluate=!1,e.returnMultipleRoot=!0,T(e)})();var C=n("6921"),M=n("7941"),P=n("6977");const D=(e,t)=>(...e)=>{Object(P["a"])("$car",e,1,1);const t=R(...e);if(!Array.isArray(t))throw new Error("[SX] $car: Invalid argument(s): args[0] is not array.");if(0===t.length)throw new Error("[SX] $car: Invalid argument(s): args[0] is nil.");return t[0]},L=(D(null,null),(e,t)=>(...e)=>{Object(P["a"])("$cdr",e,1,1);const t=R(...e);if(!Array.isArray(t))throw new Error("[SX] $cdr: Invalid argument(s): args[0] is not array.");if(0===t.length)throw new Error("[SX] $cdr: Invalid argument(s): args[0] is nil.");return t.slice(1)}),q=(L(null,null),(e,t)=>(...e)=>{Object(P["a"])("$cons",e,2,2);let{car:t,cdr:n}=G(...e);return null===t&&(t=[]),null===n&&(n=[]),Array.isArray(n)?(n=n.slice(0),n.unshift(t),n):{car:t,cdr:n}}),U=(q(null,null),(e,t)=>(...e)=>{const t=e.slice(0,1);return 1===t.length?t[0]:null}),R=U(null,null),F=(e,t)=>(...e)=>{const t=e.slice(1,2);return 1===t.length?t[0]:null},B=F(null,null),W=(e,t)=>(...e)=>{const t=e.slice(e.length-1,e.length);return 1===t.length?t[0]:null},K=(W(null,null),(e,t)=>(...e)=>{const t=e.slice(1);return 0<t.length?t:null}),V=(K(null,null),(e,t)=>(...e)=>{let t=e.slice(0,1);t=1===t.length?t[0]:null;let n=e.slice(1,2);return n=1===n.length?n[0]:null,{car:t,cdr:n}}),G=V(null,null),J=(e,t)=>(...e)=>{Object(P["a"])("$atom",e,1,1);const t=R(...e);if(null===t||void 0===t)return!0;if(Array.isArray(t))return 0===t.length;switch(typeof t){case"number":case"string":case"function":case"boolean":return!0;case"object":return!!Object(o["d"])(t)}return!1},H=(J(null,null),(e,t)=>(...e)=>{Object(P["a"])("$eq",e,2,2);const{car:t,cdr:n}=G(...e);return t===n}),z=(H(null,null),(e,t)=>(...e)=>{Object(P["a"])("$notEq",e,2,2);const{car:t,cdr:n}=G(...e);return t!==n}),Y=(z(null,null),(e,t)=>(...e)=>e.slice(0)),Q=(Y(null,null),(e,t,n)=>(...t)=>{Object(P["a"])("$__scope",t,3);const r=R(...t),a=B(...t),{car:s,cdr:l}=G(...t.slice(2));let c=null;const i={};if(Array.isArray(s))for(const n of s)if(Array.isArray(n)){const t=G(...n),r=Object(o["d"])(t.car),a=r?r.symbol:String(t.car);Object(P["c"])("$__scope",i,a),i[a]=Object(E["b"])(e,t.cdr)}else{const e=Object(o["d"])(n),t=e?e.symbol:String(n);Object(P["c"])("$__scope",i,t),i[t]=null}Object(E["f"])(e,i,r,n);try{if(4<t.length)if(a){c=[];for(const n of t.slice(3))c.push(Object(E["b"])(e,n))}else for(const n of t.slice(3))c=Object(E["b"])(e,n);else c=Object(E["b"])(e,l)}finally{Object(E["m"])(e)}return c}),Z=(e,t)=>(...t)=>{Object(P["a"])("$__globalScope",t,1);const n=R(...t),r=B(...t);let o=null;Object(E["f"])(e,Object(E["d"])(e).scope,!0);try{if(2<t.length)if(n){o=[];for(const n of t.slice(1))o.push(Object(E["b"])(e,n))}else for(const n of t.slice(1))o=Object(E["b"])(e,n);else o=Object(E["b"])(e,r)}finally{Object(E["m"])(e)}return o},ee=(e,t)=>(...t)=>{Object(P["a"])("$__capture",t,1);const n=t[0];if(!Array.isArray(n))throw new Error("[SX] $__lambda: Invalid argument(s): args[0] is not array.");let r=null;const o=Object(E["a"])(e,n);Object(E["f"])(e,{},!0,o);try{for(const n of t.slice(1))r=Object(E["b"])(e,n)}finally{Object(E["m"])(e)}return r},te=(e,t)=>(...n)=>{Object(P["a"])("$__lambda",n,2);const r=n[0];if(!Array.isArray(r))throw new Error("[SX] $__lambda: Invalid argument(s): args[0] is not array.");let a=!1;for(let t=0;t<r.length;t++){const n=r[t];if(t===r.length-1&&e.config.enableSpread&&Array.isArray(n)&&Object(o["d"])(n[0],e.config.reservedNames.spread)){if(!Object(o["d"])(n[1]))throw new Error(`[SX] $__lambda: Invalid formal argument(s): item(s) of args[${t}] is not symbol.`);r[t]=n[1],a=!0}else if(!Object(o["d"])(n))throw new Error(`[SX] $__lambda: Invalid formal argument(s): item(s) of args[${t}] is not symbol.`)}let s=n.slice(1);e.config.enableTailCallOptimization&&(s=Object(E["g"])(e,r,s));const l=Object(E["c"])(e),c=function(...n){if(n.length+(a?1:0)<r.length)throw new Error(`[SX] func call: Actual args too short: actual ${n.length} / formal ${r.length}.`);return Q(e,t,l)(!1,!1,[[e.config.reservedNames.self,c],[e.config.reservedNames.thiz,this===C["a"]||void 0===this?null:Object(o["f"])(e,this)],...r.map((t,s)=>[t.symbol,Object(o["f"])(e,a&&s===r.length-1?n.slice(s):n[s])])],...s)};return c},ne=(e,t)=>(...t)=>{Object(P["a"])("$$__lambda",t,2);const n=t[0];if(!Array.isArray(n))throw new Error("[SX] $$__lambda: Invalid argument(s): args[0] is not array.");let r=!1;for(let s=0;s<n.length;s++){const t=n[s];if(s===n.length-1&&e.config.enableSpread&&Array.isArray(t)&&Object(o["d"])(t[0],e.config.reservedNames.spread)){if(!Object(o["d"])(t[1]))throw new Error(`[SX] $$__lambda: Invalid formal argument(s): item(s) of args[${s}] is not symbol.`);n[s]=t[1],r=!0}else if(!Object(o["d"])(t))throw new Error(`[SX] $$__lambda: Invalid formal argument(s): item(s) of args[${s}] is not symbol.`)}const a=t.slice(1);return Object(M["a"])(e,n,r,a)},re=(e,t)=>(...n)=>{Object(P["a"])("$__defun",n,3);const r=R(...n),o=te(e,t)(...n.slice(1));return Object(P["c"])("$__defun",e.funcMap,r.symbol),e.funcMap.set(r.symbol,{name:r.symbol,fn:(e,t)=>o}),o},oe=(e,t)=>(...n)=>{Object(P["a"])("$$__defun",n,3);const r=R(...n),o=ne(e,t)(...n.slice(1));return Object(P["c"])("$$__defun",e.funcMap,r.symbol),e.funcMap.set(r.symbol,{name:r.symbol,fn:(e,t)=>o}),o},ae=(e,t)=>(...t)=>{Object(P["a"])("$__refun",t,1,1);const n=R(...t);Object(P["c"])("$__refun",e.funcMap,n.symbol);const r=e.funcMap.get(n.symbol);if(!r)throw new Error(`[SX] $__refun: function ${n.symbol} is not defined.`);return r.fn(e,n.symbol)},se=(e,t)=>(...n)=>{Object(P["a"])("$__defmacro",n,3);const r=R(...n),a=n[1];if(!Array.isArray(a))throw new Error("[SX] $__defmacro: Invalid argument(s): args[1] is not array.");let s=!1;for(let t=0;t<a.length;t++){const n=a[t];if(t===a.length-1&&e.config.enableSpread&&Array.isArray(n)&&Object(o["d"])(n[0],e.config.reservedNames.spread)){if(!Object(o["d"])(n[1]))throw new Error(`[SX] $__defmacro: Invalid formal argument(s): item(s) of args[${t}] is not symbol.`);a[t]=n[1],s=!0}else if(!Object(o["d"])(n))throw new Error(`[SX] $__defmacro: Invalid formal argument(s): item(s) of args[${t}] is not symbol.`)}const l=n.slice(2),c=Object(E["c"])(e),i=n=>(...r)=>Q(e,t,c)(!1,!1,[[e.config.reservedNames.self,i],...n.map((t,a)=>[t.symbol,Object(o["f"])(e,s&&a===n.length-1?r.slice(a):r[a])])],...l),u={name:r.symbol,fn:(e,t,n)=>e=>i(n)(...e.slice(1)),formalArgs:a,lastIsSpread:s};if(Object(P["c"])("$__defmacro",e.macroMap,r.symbol),e.macroMap.has(r.symbol)){let t=e.macroMap.get(r.symbol);if(t.next=u,t&&t.formalArgs)if(t.formalArgs.length<a.length)e.macroMap.set(r.symbol,u),u.next=t;else{let e=t;t=t.next;while(t){if(t.formalArgs&&t.formalArgs.length<a.length){e.next=u,u.next=t;break}e=t,t=t.next}}}else e.macroMap.set(r.symbol,u);return i},le=(e,t)=>(...e)=>{Object(P["a"])("$apply",e,1);const t=R(...e);if("function"!==typeof t)throw new Error("[SX] $apply: Invalid argument(s): args[0] is not function.");return(...n)=>t.apply(null,e.slice(1).concat(n))},ce=(le(null,null),(e,t)=>(...t)=>{Object(P["a"])("$__call",t,2);const{car:n,cdr:r}=G(...t),a=Object(o["d"])(r),s=a?a.symbol:Object(E["b"])(e,r);return Object(P["c"])("$__call",n,s),Function.prototype.apply.call(n[s],n,t.slice(2))}),ie=(e,t)=>(...n)=>{Object(P["a"])("$__try",n,1,2);let a=[];try{a=Object(E["b"])(e,n[0])}catch(s){if(s instanceof r["a"])throw s;a=1<n.length?Q(e,t)(!0,!1,[["$error",Object(o["f"])(e,s)],["$parent",Object(o["f"])(e,Object(E["e"])(e))]],n[1]):null}return a},ue=(e,t)=>(...e)=>{const t=R(...e);throw t},fe=(ue(null,null),(e,t)=>(...t)=>{Object(P["a"])("$__if",t,2,3);const n=R(...t);let r=[];return r=Se(n)?Object(E["b"])(e,t[1]):2<t.length?Object(E["b"])(e,t[2]):null,r}),me=(e,t)=>(...t)=>{Object(P["a"])("$__ifNull",t,2,2);const{car:n,cdr:r}=G(...t);let o=[];return o=Te(n,null)?Object(E["b"])(e,r):n,o},be=(e,t)=>(...t)=>{Object(P["a"])("$__cond",t,1);for(let n=0;n<t.length-1;n+=2){const r=t[n],o=t[n+1];if(Se(Object(E["b"])(e,r)))return Object(E["b"])(e,o)}return null},$e=(e,t)=>(...t)=>{Object(P["a"])("$__while",t,1);const n=R(...t),r=t.slice(1);let o=null;while(Se(Object(E["b"])(e,n)))for(const t of r)o=Object(E["b"])(e,t);return o},pe=(e,t)=>(...t)=>{Object(P["a"])("$__doWhile",t,1);const n=R(...t),r=t.slice(1);let o=null;do{for(const t of r)o=Object(E["b"])(e,t)}while(Se(Object(E["b"])(e,n)));return o},_e=(e,t)=>(...t)=>{Object(P["a"])("$__until",t,1);const n=R(...t),r=t.slice(1);let o=null;while(Ee(Object(E["b"])(e,n)))for(const t of r)o=Object(E["b"])(e,t);return o},de=(e,t)=>(...t)=>{Object(P["a"])("$__doUntil",t,1);const n=R(...t),r=t.slice(1);let o=null;do{for(const t of r)o=Object(E["b"])(e,t)}while(Ee(Object(E["b"])(e,n)));return o},ye=(e,t)=>(...t)=>{Object(P["a"])("$__repeat",t,2);const n=Object(o["d"])(R(...t));if(!n)throw new Error("[SX] $__repeat: Invalid argument(s): item(s) of args[0] is not symbol.");Object(P["c"])("$__repeat",{},n.symbol);const r=Object(E["k"])(e,n,!1),a=Object(E["l"])(B(...t)),s=t.slice(2);let l=null;for(let o=0;o<a;o++){r[n.symbol]=o;for(const t of s)l=Object(E["b"])(e,t)}return l},ge=(e,t)=>(...t)=>{Object(P["a"])("$__for",t,2);const n=Object(o["d"])(R(...t));if(!n)throw new Error("[SX] $__for: Invalid argument(s): item(s) of args[0] is not symbol.");Object(P["c"])("$__for",{},n.symbol);const r=Object(E["k"])(e,n,!1),a=B(...t);if(!Array.isArray(a))throw new Error("[SX] $__for: Invalid argument(s): item(s) of args[1] is not array.");const s=t.slice(2);let l=null;for(const o of a){r[n.symbol]=o;for(const t of s)l=Object(E["b"])(e,t)}return l},je=(e,t)=>(...e)=>{Object(P["a"])("$pipe",e,1);let t=e[0];for(let n=1;n<e.length;n++)t=e[n](t);return t},Oe=(je(null,null),(e,t)=>(...t)=>{Object(P["a"])("$__get",t,1);let n=null,r=Object(o["d"])(t[0]);if(!r)switch(typeof t[0]){case"string":case"number":r={symbol:String(t[0])};break;default:n=Object(E["b"])(e,t[0]);break}if(r){const t=Object(E["k"])(e,r,!0);if(!t)throw new Error(`[SX] $__get: Invalid argument(s): args[0]: symbol "${r.symbol}" is not defined.`);n=t[r.symbol]}for(let a=1;a<t.length;a++){let s=t[a],l=!0;while(l)switch(typeof s){case"function":n=s(n),l=!1;break;case"object":if(Array.isArray(s))s=Object(E["b"])(e,s);else if(r=Object(o["d"])(s),r)s=r.symbol;else{if(!Object.prototype.hasOwnProperty.call(s,"value"))throw new Error("[SX] $__get: Invalid argument(s): invalid name path.");s=Object(E["b"])(e,s)}break;case"number":s<0&&(s=n.length+s);case"string":Object(P["c"])("$__get",n,s),n=n[s],l=!1;break;default:throw new Error("[SX] $__get: Invalid argument(s): invalid name path.")}}return n}),he=(e,t)=>(...t)=>{Object(P["a"])("$__let",t,2,2);let n=Object(o["d"])(R(...t));if(!n){if("string"!==typeof t[0])throw new Error("[SX] $__let: Invalid argument(s): invalid name.");n={symbol:t[0]}}Object(P["c"])("$__let",{},n.symbol);const r=Object(E["k"])(e,n,!1);return r[n.symbol]=t[1],t[1]},ve=(e,t)=>(...t)=>{Object(P["a"])("$__set",t,2,2);let n=[];Array.isArray(t[0])?n=t[0]:n.push(t[0]);let r=Object(o["d"])(n[0]);if(!r){if("string"!==typeof n[0])throw new Error("[SX] $__set: Invalid argument(s): invalid name.");Object(P["c"])("$__set",{},n[0]),r={symbol:n[0]}}let a=Object(E["k"])(e,r,!0);if(null===a)throw new Error(`[SX] $__set: Unresolved symbol: ${r.symbol}.`);let s=!1;for(let l=0;l<n.length;l++){let c=n[l],i=!0;const u=l===n.length-1;while(i)switch(typeof c){case"function":a=c(a),i=!1;break;case"object":if(Array.isArray(c))c=Object(E["b"])(e,c);else if(r=Object(o["d"])(c),r)c=r.symbol;else{if(!Object.prototype.hasOwnProperty.call(c,"value"))throw new Error("[SX] $__set: Invalid argument(s): invalid name.");c=Object(E["b"])(e,c)}break;case"number":c<0&&(c=a.length+c);case"string":Object(P["c"])("$__set",a,c),u?(a[c]=t[1],s=!0):a=a[c],i=!1;break;default:throw new Error("[SX] $__set: Invalid argument(s): invalid name.")}}if(!s)throw new Error("[SX] $__set: Invalid argument(s): last path is not lvalue.");return t[1]},we=(e,t)=>(...e)=>{Object(P["a"])("$boolean",e,1,1);const t=R(...e);return(!Array.isArray(t)||0!==t.length)&&Boolean(t)},Se=we(null,null),Ae=(e,t)=>(...e)=>(Object(P["a"])("$not",e,1,1),!Se(...e)),Ee=Ae(null,null),Ne=(e,t)=>(...t)=>{Object(P["a"])("$__and",t,1);let n=null;for(let r=0;r<t.length;r++){const o=Object(E["b"])(e,t[r]);if(!Se(o))return o;n=o}return n},Ie=(Ne(null,null),(e,t)=>(...t)=>{Object(P["a"])("$__or",t,1);let n=null;for(let r=0;r<t.length;r++){const o=Object(E["b"])(e,t[r]);if(Se(o))return o;n=o}return n}),ke=(Ie(null,null),(e,t)=>(...e)=>{Object(P["a"])("$ambiguousEq",e,2,2);let{car:t,cdr:n}=G(...e);return Array.isArray(t)&&0===t.length&&(t=null),Array.isArray(n)&&0===n.length&&(n=null),void 0===t&&(t=null),void 0===n&&(n=null),t==n}),Te=ke(null,null),Xe=(e,t)=>(...e)=>!Te(...e),xe=(Xe(null,null),(e,t)=>(...e)=>{Object(P["a"])("$lt",e,2,2);const{car:t,cdr:n}=G(...e);return Object(E["l"])(t)<Object(E["l"])(n)}),Ce=(xe(null,null),(e,t)=>(...e)=>{Object(P["a"])("$le",e,2,2);const{car:t,cdr:n}=G(...e);return Object(E["l"])(t)<=Object(E["l"])(n)}),Me=(Ce(null,null),(e,t)=>(...e)=>{Object(P["a"])("$gt",e,2,2);const{car:t,cdr:n}=G(...e);return Object(E["l"])(t)>Object(E["l"])(n)}),Pe=(Me(null,null),(e,t)=>(...e)=>{Object(P["a"])("$ge",e,2,2);const{car:t,cdr:n}=G(...e);return Object(E["l"])(t)>=Object(E["l"])(n)}),De=(Pe(null,null),(e,t)=>(...e)=>{Object(P["a"])("$typeof",e,1,1);const t=R(...e);if(null===t)return"null";const n=typeof t;switch(n){case"object":return Array.isArray(t)?"list":"object";case"symbol":return"js-symbol";default:return n}}),Le=(De(null,null),(e,t)=>(...e)=>{if(Object(P["a"])("$symbol",e,1,1),"string"===typeof e[0])return{symbol:e[0]};throw new Error("[SX] $symbol: Invalid argument(s): item(s) of args[0] is not string.")}),qe=(Le(null,null),(e,t)=>(...t)=>{Object(P["a"])("$__gensym",t,0,1);const n=`$__tempvar__$$ec${e.evalCount++}$$_`,r={symbol:n+"_$gensym"};if(1===t.length){const n=Object(o["d"])(t[0]);if(n)he(e,"")(n,r);else{if("string"!==typeof t[0])throw new Error("[SX] $__gensym: Invalid argument(s): item(s) of args[0] is not symbol.");he(e,"")({symbol:t[0]},r)}}return r}),Ue=(e,t)=>(...e)=>{if(Object(P["a"])("$isSymbol",e,1,2),1===e.length)return!!Object(o["d"])(e[0]);if("string"===typeof e[1])return!!Object(o["d"])(e[0],e[1]);throw new Error("[SX] $isSymbol: Invalid argument(s): item(s) of args[1] is not string.")},Re=(Ue(null,null),(e,t)=>(...e)=>(Object(P["a"])("$isNull",e,1,1),null===R(...e))),Fe=(Re(null,null),(e,t)=>(...e)=>{Object(P["a"])("$isNil",e,1,1);const t=R(...e);return Array.isArray(t)&&0===t.length}),Be=(Fe(null,null),(e,t)=>(...e)=>(Object(P["a"])("$isUndefined",e,1,1),void 0===R(...e))),We=(Be(null,null),(e,t)=>(...e)=>(Object(P["a"])("$isList",e,1,1),Array.isArray(R(...e)))),Ke=(We(null,null),(e,t)=>(...e)=>(Object(P["a"])("$isString",e,1,1),"string"===typeof R(...e))),Ve=(Ke(null,null),(e,t)=>(...e)=>(Object(P["a"])("$isNumber",e,1,1),"number"===typeof R(...e))),Ge=(Ve(null,null),(e,t)=>(...e)=>(Object(P["a"])("$isNaN",e,1,1),Number.isNaN(R(...e)))),Je=(Ge(null,null),(e,t)=>(...e)=>(Object(P["a"])("$isFinite",e,1,1),Number.isFinite(R(...e)))),He=(Je(null,null),(e,t)=>(...e)=>(Object(P["a"])("$isInteger",e,1,1),Number.isInteger(R(...e)))),ze=(He(null,null),(e,t)=>(...e)=>(Object(P["a"])("$toString",e,1,1),String(R(...e)))),Ye=(ze(null,null),(e,t)=>(...e)=>(Object(P["a"])("$toNumber",e,1,1),Object(E["l"])(R(...e)))),Qe=(Ye(null,null),(e,t)=>(...t)=>{const n={};for(const r of t){if(!(Array.isArray(r)&&0<r.length))throw new Error("[SX] $__toObject: Invalid argument(s): args[?] is not array.");{const t=Object(o["d"])(r[0]),a=t?t.symbol:String(Object(E["b"])(e,r[0]));Object(P["c"])("$__#",n,a),1===r.length?n[a]=!0:2===r.length?n[a]=Object(E["b"])(e,r[1]):n[a]=Object(E["b"])(e,[{symbol:e.config.reservedNames.list}].concat(r.slice(1)))}}return n}),Ze=[C["a"],Object.__proto__,{}.__proto__,Function.__proto__],et=(e,t)=>(...e)=>{if(Object(P["a"])("$objectAssign",e,1),Ze.includes(e[0]))throw new Error("[SX] $objectAssign: Invalid argument: args[0] is blacklisted object.");return Object.assign(e[0],...e.slice(1))},tt=(et(null,null),(e,t)=>(...e)=>(Object(P["a"])("$jsonStringify",e,1,1),JSON.stringify(R(...e)))),nt=(tt(null,null),(e,t)=>(...e)=>{Object(P["a"])("$jsonParse",e,1,1);const t=R(...e);if("string"!==typeof t)throw new Error("[SX] $jsonParse: Invalid argument(s): args[0] is not string.");return JSON.parse(t)}),rt=(nt(null,null),(e,t)=>(...e)=>Date.now()),ot=(rt(null,null),(e,t)=>(...e)=>{Object(P["a"])("$datetimeFromIso",e,1,1);const t=R(...e);if("string"!==typeof t)throw new Error("[SX] $datetimeFromIso: Invalid argument(s): args[0] is not string.");if(!/^(?:(?:-[0-9]{6,})|[0-9]{4,})-(?:[0-1][0-9])-(?:[0-3][0-9])(?:T(?:[0-2][0-9])(?:[:](?:[0-6][0-9])(?:[:](?:[0-6][0-9])(?:.[0-9]{1,})?)?)?(?:Z|[-+][0-9]{2}(?:[:]?[0-6][0-9])?)?)?$/.test(t))throw new Error(`[SX] $datetimeFromIso: Invalid datetime (pattern unmatched): ${t}.`);const n=new Date(t).getTime();if(Number.isNaN(n))throw new Error(`[SX] $datetimeFromIso: Invalid datetime: ${t}.`);return n}),at=(ot(null,null),(e,t)=>(...e)=>{Object(P["a"])("$datetime",e,3,7);let t="";const n=Number(e[0]);t+=n>=0?String(n).padStart(4,"0"):"-"+String(-n).padStart(6,"0"),t+="-"+String(Number(e[1])).padStart(2,"0"),t+="-"+String(Number(e[2])).padStart(2,"0"),e.length>=4&&(t+="T"+String(Number(e[3])).padStart(2,"0"),e.length>=5?t+=":"+String(Number(e[4])).padStart(2,"0"):t+=":00",e.length>=6&&(t+=":"+String(Number(e[5])).padStart(2,"0")),e.length>=7&&(t+="."+String(Number(e[6])).padStart(3,"0").slice(0,3)),t+="Z");const r=new Date(t).getTime();if(Number.isNaN(r))throw new Error(`[SX] $datetime: Invalid datetime: ${t}.`);return r}),st=(at(null,null),(e,t)=>(...e)=>{Object(P["a"])("$datetimeLc",e,3,7);let t="";const n=Number(e[0]);t+=n>=0?String(n).padStart(4,"0"):"-"+String(-n).padStart(6,"0"),t+="-"+String(Number(e[1])).padStart(2,"0"),t+="-"+String(Number(e[2])).padStart(2,"0"),e.length>=4?(t+="T"+String(Number(e[3])).padStart(2,"0"),e.length>=5?t+=":"+String(Number(e[4])).padStart(2,"0"):t+=":00",e.length>=6&&(t+=":"+String(Number(e[5])).padStart(2,"0")),e.length>=7&&(t+="."+String(Number(e[6])).padStart(3,"0").slice(0,3))):t+="T00:00:00.000";const r=new Date(t).getTime();if(Number.isNaN(r))throw new Error(`[SX] $datetimeLc: Invalid datetime: ${t}.`);return r}),lt=(st(null,null),(e,t)=>(...e)=>{Object(P["a"])("$datetimeToIsoString",e,1,1);const t=R(...e);if("number"!==typeof t)throw new Error("[SX] $datetimeToIsoString: Invalid argument(s): args[0] is not number.");const n=new Date(t);if(Number.isNaN(n.getTime()))throw new Error(`[SX] $datetimeToIsoString: Invalid datetime: ${t}.`);return n.toISOString()}),ct=(lt(null,null),(e,t)=>(...e)=>{Object(P["a"])("$datetimeToComponents",e,1,1);const t=R(...e);if("number"!==typeof t)throw new Error("[SX] $datetimeToComponents: Invalid argument(s): args[0] is not number.");const n=new Date(t);if(Number.isNaN(n.getTime()))throw new Error(`[SX] $datetimeToComponents: Invalid datetime: ${t}.`);return[n.getUTCFullYear(),n.getUTCMonth()+1,n.getUTCDate(),n.getUTCHours(),n.getUTCMinutes(),n.getUTCSeconds(),n.getUTCMilliseconds(),0,n.getUTCDay()]}),it=(ct(null,null),(e,t)=>(...e)=>{Object(P["a"])("$datetimeToComponentsLc",e,1,1);const t=R(...e);if("number"!==typeof t)throw new Error("[SX] $datetimeToComponentsLc: Invalid argument(s): args[0] is not number.");const n=new Date(t);if(Number.isNaN(n.getTime()))throw new Error(`[SX] $datetimeToComponentsLc: Invalid datetime: ${t}.`);return[n.getFullYear(),n.getMonth()+1,n.getDate(),n.getHours(),n.getMinutes(),n.getSeconds(),n.getMilliseconds(),-n.getTimezoneOffset(),n.getDay()]}),ut=(it(null,null),(e,t)=>(...t)=>{if(Object(P["a"])("$match",t,2,3),!e.config.enableRegExpMatchOperators)throw new Error("[SX] $match: Operator is disabled by configuration.");if(2===t.length){const e=new RegExp(t[0]);return e.exec(t[1])}{const e=new RegExp(t[0],t[1]);return e.exec(t[2])}}),ft=(ut(null,null),(e,t)=>(...e)=>(console.log(...e),null)),mt=(ft(null,null),(e,t)=>(...e)=>(console.error(...e),null)),bt=(mt(null,null),(e,t)=>(...e)=>(console.trace(...e),null)),$t=(bt(null,null),(e,t)=>(...e)=>(console.time(...e),null)),pt=($t(null,null),(e,t)=>(...e)=>(console.timeEnd(...e),null)),_t=(pt(null,null),(e,t)=>(...e)=>(console.timeLog(...e),null)),dt=(_t(null,null),[{name:"$car",fn:D},{name:"$cdr",fn:L},{name:"$cons",fn:q},{name:"$first",fn:U},{name:"$second",fn:F},{name:"$last",fn:W},{name:"$progn",fn:W},{name:"$rest",fn:K},{name:"$first-and-second",fn:V},{name:"$atom",fn:J},{name:"$eq",fn:H},{name:"===",fn:H},{name:"$not-eq",fn:z},{name:"!==",fn:z},{name:"$list",fn:Y},{name:"$__scope",fn:Q},{name:"$__global",fn:Z},{name:"$__capture",fn:ee},{name:"$__lambda",fn:te},{name:"$__defun",fn:re},{name:"$__refun",fn:ae},{name:"$__defmacro",fn:se},{name:"$apply",fn:le},{name:"$__call",fn:ce},{name:"$__try",fn:ie},{name:"$raise",fn:ue},{name:"$__if",fn:fe},{name:"$__if-null",fn:me},{name:"$__cond",fn:be},{name:"$__while",fn:$e},{name:"$__do-while",fn:pe},{name:"$__until",fn:_e},{name:"$__do-until",fn:de},{name:"$__repeat",fn:ye},{name:"$__for",fn:ge},{name:"$pipe",fn:je},{name:"$__get",fn:Oe},{name:"$__let",fn:he},{name:"$__set",fn:ve},{name:"$boolean",fn:we},{name:"$not",fn:Ae},{name:"$__and",fn:Ne},{name:"$__or",fn:Ie},{name:"==",fn:ke},{name:"!=",fn:Xe},{name:"<",fn:xe},{name:"<=",fn:Ce},{name:">",fn:Me},{name:">=",fn:Pe},{name:"$typeof",fn:De},{name:"$symbol",fn:Le},{name:"$__gensym",fn:qe},{name:"$is-symbol",fn:Ue},{name:"$is-null",fn:Re},{name:"$is-nil",fn:Fe},{name:"$is-undefined",fn:Be},{name:"$is-list",fn:We},{name:"$is-string",fn:Ke},{name:"$is-number",fn:Ve},{name:"$is-NaN",fn:Ge},{name:"$is-finite",fn:Je},{name:"$is-integer",fn:He},{name:"$to-string",fn:ze},{name:"$to-number",fn:Ye},{name:"$__#",fn:Qe},{name:"$object-assign",fn:et},{name:"$json-stringify",fn:tt},{name:"$now",fn:rt},{name:"$datetime-from-iso",fn:ot},{name:"$datetime",fn:at},{name:"$datetime-lc",fn:st},{name:"$datetime-to-iso-string",fn:lt},{name:"$datetime-to-components",fn:ct},{name:"$datetime-to-components-lc",fn:it},{name:"$json-parse",fn:nt},{name:"$match",fn:ut},{name:"$console-log",fn:ft},{name:"$console-error",fn:mt},{name:"$console-trace",fn:bt},{name:"$console-time",fn:$t},{name:"$console-time-end",fn:pt},{name:"$console-time-log",fn:_t}]),yt=[{name:"$$__lambda",fn:ne},{name:"$$__defun",fn:oe}];var gt=[].concat(dt,yt);const jt=[{name:"$scope",fn:(e,t)=>t=>[{symbol:"$__scope"},t[1],t[2],...t.slice(3).map(t=>Object(o["f"])(e,t))]},{name:"$local",fn:(e,t)=>t=>[{symbol:"$__scope"},!0,!1,...t.slice(1).map(t=>Object(o["f"])(e,t))]},{name:"$global",fn:(e,t)=>t=>[{symbol:"$__global"},!1,...t.slice(1).map(t=>Object(o["f"])(e,t))]},{name:"$clisp-let",fn:(e,t)=>t=>[{symbol:"$__scope"},!0,!1,...t.slice(1).map(t=>Object(o["f"])(e,t))]},{name:"$capture",fn:(e,t)=>t=>[{symbol:"$__capture"},...t.slice(1).map(t=>Object(o["f"])(e,t))]},{name:"$closure",fn:(e,t)=>t=>{const n=Object(o["d"])(t[2],"use");if(!n)throw new Error("[SX] $closure: Invalid syntax: missing 'use' keyword.");return[{symbol:"$__capture"},Object(o["f"])(e,t[3]),Object(o["f"])(e,[{symbol:"$__lambda"},Object(o["f"])(e,t[1]),...t.slice(4).map(t=>Object(o["f"])(e,t))])]}},{name:"|->",fn:(e,t)=>e=>[{symbol:"$closure"},...e.slice(1)]},{name:"$lambda",fn:(e,t)=>t=>[{symbol:"$__lambda"},...t.slice(1).map(t=>Object(o["f"])(e,t))]},{name:"->",fn:(e,t)=>t=>[{symbol:"$__lambda"},...t.slice(1).map(t=>Object(o["f"])(e,t))]},{name:"$defun",fn:(e,t)=>t=>[{symbol:"$__defun"},...t.slice(1).map(t=>Object(o["f"])(e,t))]},{name:"$refun",fn:(e,t)=>t=>[{symbol:"$__refun"},...t.slice(1).map(t=>Object(o["f"])(e,t))]},{name:"<-",fn:(e,t)=>t=>[{symbol:"$__refun"},...t.slice(1).map(t=>Object(o["f"])(e,t))]},{name:"$defmacro",fn:(e,t)=>t=>[{symbol:"$__defmacro"},...t.slice(1).map(t=>Object(o["f"])(e,t))]},{name:"$call",fn:(e,t)=>t=>(Object(P["a"])("$call",t,3),[{symbol:"$__call"},t[1],Object(o["f"])(e,t[2]),...t.slice(3)])},{name:"$try",fn:(e,t)=>t=>[{symbol:"$__try"},...t.slice(1).map(t=>Object(o["f"])(e,t))]},{name:"$if",fn:(e,t)=>t=>[{symbol:"$__if"},t[1],...t.slice(2).map(t=>Object(o["f"])(e,t))]},{name:"$if-null",fn:(e,t)=>t=>[{symbol:"$__if-null"},t[1],...t.slice(2).map(t=>Object(o["f"])(e,t))]},{name:"??",fn:(e,t)=>t=>[{symbol:"$__if-null"},t[1],...t.slice(2).map(t=>Object(o["f"])(e,t))]},{name:"$cond",fn:(e,t)=>t=>[{symbol:"$__cond"},...t.slice(1).map(t=>Object(o["f"])(e,t))]},{name:"$while",fn:(e,t)=>t=>[{symbol:"$__while"},...t.slice(1).map(t=>Object(o["f"])(e,t))]},{name:"$do-while",fn:(e,t)=>t=>[{symbol:"$__do-while"},...t.slice(1).map(t=>Object(o["f"])(e,t))]},{name:"$until",fn:(e,t)=>t=>[{symbol:"$__until"},...t.slice(1).map(t=>Object(o["f"])(e,t))]},{name:"$do-until",fn:(e,t)=>t=>[{symbol:"$__do-until"},...t.slice(1).map(t=>Object(o["f"])(e,t))]},{name:"$repeat",fn:(e,t)=>t=>{const n=Object(o["d"])(t[2],"of");if(!n)throw new Error("[SX] $repeat: Invalid syntax: missing 'of' keyword.");return[{symbol:"$__repeat"},Object(o["f"])(e,t[1]),t[3],...t.slice(4).map(t=>Object(o["f"])(e,t))]}},{name:"$for",fn:(e,t)=>t=>{const n=Object(o["d"])(t[2],"of");if(!n)throw new Error("[SX] $for: Invalid syntax: missing 'of' keyword.");return[{symbol:"$__for"},Object(o["f"])(e,t[1]),t[3],...t.slice(4).map(t=>Object(o["f"])(e,t))]}},{name:"$get",fn:(e,t)=>t=>[{symbol:"$__get"},...t.slice(1).map(t=>Object(o["f"])(e,t))]},{name:"$let",fn:(e,t)=>t=>(Object(P["a"])("$let",t,3,3),[{symbol:"$__let"},Object(o["f"])(e,t[1]),t[2]])},{name:"$clisp-defvar",fn:(e,t)=>t=>(Object(P["a"])("$clisp-defvar",t,3,3),[{symbol:"$global"},[{symbol:"$__let"},Object(o["f"])(e,t[1]),t[2]]])},{name:"$set",fn:(e,t)=>t=>(Object(P["a"])("$set",t,3,3),[{symbol:"$__set"},Object(o["f"])(e,t[1]),t[2]])},{name:"$clisp-setq",fn:(e,t)=>t=>(Object(P["a"])("$clisp-setq",t,3,3),[{symbol:"$__set"},Object(o["f"])(e,t[1]),t[2]])},{name:"$and",fn:(e,t)=>t=>[{symbol:"$__and"},...t.slice(1).map(t=>Object(o["f"])(e,t))]},{name:"$or",fn:(e,t)=>t=>[{symbol:"$__or"},...t.slice(1).map(t=>Object(o["f"])(e,t))]},{name:"$gensym",fn:(e,t)=>t=>[{symbol:"$__gensym"},...t.slice(1).map(t=>Object(o["f"])(e,t))]},{name:"#",fn:(e,t)=>t=>[{symbol:"$__#"},...t.slice(1).map(t=>Object(o["f"])(e,t))]}],Ot=[{name:"$$closure",fn:(e,t)=>t=>{const n=Object(o["d"])(t[2],"use");if(!n)throw new Error("[SX] $closure: Invalid syntax: missing 'use' keyword.");return[{symbol:"$__capture"},Object(o["f"])(e,t[3]),Object(o["f"])(e,[{symbol:"$$__lambda"},Object(o["f"])(e,t[1]),...t.slice(4).map(t=>Object(o["f"])(e,t))])]}},{name:"|=>",fn:(e,t)=>e=>[{symbol:"$$closure"},...e.slice(1)]},{name:"$$lambda",fn:(e,t)=>t=>[{symbol:"$$__lambda"},...t.slice(1).map(t=>Object(o["f"])(e,t))]},{name:"=>",fn:(e,t)=>t=>[{symbol:"$$__lambda"},...t.slice(1).map(t=>Object(o["f"])(e,t))]},{name:"$$defun",fn:(e,t)=>t=>[{symbol:"$$__defun"},...t.slice(1).map(t=>Object(o["f"])(e,t))]}];var ht=[].concat(jt,Ot);const vt=[{name:"nil",fn:(e,t)=>[]},{name:"null",fn:(e,t)=>null},{name:"undefined",fn:(e,t)=>{}},{name:"true",fn:(e,t)=>!0},{name:"#true",fn:(e,t)=>!0},{name:"#t",fn:(e,t)=>!0},{name:"false",fn:(e,t)=>!1},{name:"#false",fn:(e,t)=>!1},{name:"#f",fn:(e,t)=>!1},{name:"#Number:Infinity",fn:(e,t)=>Number.POSITIVE_INFINITY},{name:"+Infinity",fn:(e,t)=>Number.POSITIVE_INFINITY},{name:"-Infinity",fn:(e,t)=>Number.NEGATIVE_INFINITY},{name:"#Number:Epsilon",fn:(e,t)=>Number.EPSILON},{name:"#Number:MaxValue",fn:(e,t)=>Number.MAX_VALUE},{name:"#Number:MinValue",fn:(e,t)=>Number.MIN_VALUE},{name:"#Number:MinSafeInteger",fn:(e,t)=>Number.MAX_SAFE_INTEGER},{name:"#Number:MinSafeInteger",fn:(e,t)=>Number.MIN_SAFE_INTEGER},{name:"NaN",fn:(e,t)=>Number.NaN}];var wt=vt;function St(e){return e.funcs=(e.funcs||[]).concat(dt).concat(e.enableCompilationOperators?yt:[]),e.macros=(e.macros||[]).concat(jt).concat(e.enableCompilationOperators?Ot:[]),e.symbols=(e.symbols||[]).concat(wt),e}const At=(e,t)=>(...e)=>{Object(P["a"])("$bitLShift",e,2,2);let{car:t,cdr:n}=G(...e);return t=Object(E["l"])(t),n=Object(E["l"])(n),0<=n?n<32?t<<n:0:n>-32?t>>>-n:0},Et=(At(null,null),(e,t)=>(...e)=>{Object(P["a"])("$bitSRShift",e,2,2);let{car:t,cdr:n}=G(...e);return t=Object(E["l"])(t),n=Object(E["l"])(n),0<=n?n<32?t>>n:2147483648&t?-1:0:n>-32?t<<-n:0}),Nt=(Et(null,null),(e,t)=>(...e)=>{Object(P["a"])("$bitURShift",e,2,2);let{car:t,cdr:n}=G(...e);return t=Object(E["l"])(t),n=Object(E["l"])(n),0<=n?n<32?t>>>n:0:n>-32?t<<-n:0}),It=(Nt(null,null),(e,t)=>(...e)=>{Object(P["a"])("$bitNot",e,1,1);const t=R(...e);return~Object(E["l"])(t)}),kt=(It(null,null),(e,t)=>(...e)=>{Object(P["a"])("$bitAnd",e,2);const t=R(...e);return e.slice(1).reduce((e,t)=>Object(E["l"])(e)&Object(E["l"])(t),Object(E["l"])(t))}),Tt=(kt(null,null),(e,t)=>(...e)=>{Object(P["a"])("$bitOr",e,2);const t=R(...e);return e.slice(1).reduce((e,t)=>Object(E["l"])(e)|Object(E["l"])(t),Object(E["l"])(t))}),Xt=(Tt(null,null),(e,t)=>(...e)=>{Object(P["a"])("$bitXor",e,2);const t=R(...e);return e.slice(1).reduce((e,t)=>Object(E["l"])(e)^Object(E["l"])(t),Object(E["l"])(t))}),xt=(Xt(null,null),(e,t)=>(...e)=>(Object(P["a"])("$add",e,1),e.reduce((e,t)=>Object(E["l"])(e)+Object(E["l"])(t),0))),Ct=(xt(null,null),(e,t)=>(...e)=>{Object(P["a"])("$sub",e,1);const t=R(...e),n=e.slice(1);return 0===n.length?-Object(E["l"])(t):e.slice(1).reduce((e,t)=>Object(E["l"])(e)-Object(E["l"])(t),Object(E["l"])(t))}),Mt=(Ct(null,null),(e,t)=>(...e)=>{Object(P["a"])("$mul",e,2);const t=R(...e);return e.slice(1).reduce((e,t)=>Object(E["l"])(e)*Object(E["l"])(t),Object(E["l"])(t))}),Pt=(Mt(null,null),(e,t)=>(...e)=>{Object(P["a"])("$sup",e,2);const t=R(...e);return e.slice(1).reduce((e,t)=>Math.pow(Object(E["l"])(e),Object(E["l"])(t)),Object(E["l"])(t))}),Dt=(Pt(null,null),(e,t)=>(...e)=>{Object(P["a"])("$div",e,2);const t=R(...e);return e.slice(1).reduce((e,t)=>Object(E["l"])(e)/Object(E["l"])(t),Object(E["l"])(t))}),Lt=(Dt(null,null),(e,t)=>(...e)=>{Object(P["a"])("$mod",e,2);const t=R(...e);return e.slice(1).reduce((e,t)=>Object(E["l"])(e)%Object(E["l"])(t),Object(E["l"])(t))}),qt=(Lt(null,null),(e,t)=>(...e)=>Math.max(...e.map(e=>Object(E["l"])(e)))),Ut=(qt(null,null),(e,t)=>(...e)=>Math.min(...e.map(e=>Object(E["l"])(e)))),Rt=(Ut(null,null),(e,t)=>(...e)=>{const t=e.map(e=>Object(E["l"])(e));return t.length>0?t.reduce((e,t)=>e+t,0)/t.length:NaN}),Ft=(Rt(null,null),(e,t)=>(...e)=>(Object(P["a"])("$floor",e,1,1),Math.floor(Object(E["l"])(R(...e))))),Bt=(Ft(null,null),(e,t)=>(...e)=>(Object(P["a"])("$ceil",e,1,1),Math.ceil(Object(E["l"])(R(...e))))),Wt=(Bt(null,null),(e,t)=>(...e)=>(Object(P["a"])("$round",e,1,1),Math.round(Object(E["l"])(R(...e))))),Kt=(Wt(null,null),(e,t)=>(...e)=>(Object(P["a"])("$abs",e,1,1),Math.abs(Object(E["l"])(R(...e))))),Vt=(Kt(null,null),(e,t)=>(...e)=>(Object(P["a"])("$sign",e,1,1),Math.sign(Object(E["l"])(R(...e))))),Gt=(Vt(null,null),[{name:"<<",fn:At},{name:"$bit-l-shift",fn:At},{name:">>",fn:Et},{name:"$bit-sr-shift",fn:Et},{name:">>>",fn:Nt},{name:"$bit-ur-shift",fn:Nt},{name:"$bit-not",fn:It},{name:"$bit-and",fn:kt},{name:"$bit-or",fn:Tt},{name:"$bit-xor",fn:Xt},{name:"+",fn:xt},{name:"$add",fn:xt},{name:"$sum",fn:xt},{name:"-",fn:Ct},{name:"$sub",fn:Ct},{name:"$neg",fn:Ct},{name:"*",fn:Mt},{name:"$mul",fn:Mt},{name:"**",fn:Pt},{name:"$sup",fn:Pt},{name:"/",fn:Dt},{name:"$div",fn:Dt},{name:"%",fn:Lt},{name:"$mod",fn:Lt},{name:"$max",fn:qt},{name:"$min",fn:Ut},{name:"$avg",fn:Rt},{name:"$floor",fn:Ft},{name:"$ceil",fn:Bt},{name:"$round",fn:Wt},{name:"$abs",fn:Kt},{name:"$sign",fn:Vt}]);var Jt=Gt;const Ht=[{name:"$incl",fn:(e,t)=>e=>(Object(P["a"])("$incl",e,2,2),[{symbol:"$set"},e[1],[{symbol:"$add"},e[1],1]])},{name:"++",fn:(e,t)=>e=>(Object(P["a"])("++",e,2,2),[{symbol:"$set"},e[1],[{symbol:"$add"},e[1],1]])},{name:"$decl",fn:(e,t)=>e=>(Object(P["a"])("$decl",e,2,2),[{symbol:"$set"},e[1],[{symbol:"$add"},e[1],-1]])},{name:"--",fn:(e,t)=>e=>(Object(P["a"])("--",e,2,2),[{symbol:"$set"},e[1],[{symbol:"$add"},e[1],-1]])},{name:"$incln",fn:(e,t)=>e=>{if(Object(P["a"])("$incln",e,3,3),"number"!==typeof e[2])throw new Error("[SX] $incln: Invalid parameter: arg(1) is not number.");return[{symbol:"$set"},e[1],[{symbol:"$add"},e[1],e[2]]]}},{name:"+=",fn:(e,t)=>e=>{if(Object(P["a"])("+=",e,3,3),"number"!==typeof e[2])throw new Error("[SX] +=: Invalid parameter: arg(1) is not number.");return[{symbol:"$set"},e[1],[{symbol:"$add"},e[1],e[2]]]}},{name:"$decln",fn:(e,t)=>e=>{if(Object(P["a"])("$decln",e,3,3),"number"!==typeof e[2])throw new Error("[SX] $decln: Invalid parameter: arg(1) is not number.");return[{symbol:"$set"},e[1],[{symbol:"$add"},e[1],-e[2]]]}},{name:"-=",fn:(e,t)=>e=>{if(Object(P["a"])("-=",e,3,3),"number"!==typeof e[2])throw new Error("[SX] -=: Invalid parameter: arg(1) is not number.");return[{symbol:"$set"},e[1],[{symbol:"$add"},e[1],-e[2]]]}}];var zt=Ht;const Yt=[];var Qt=Yt;function Zt(e){return e.funcs=(e.funcs||[]).concat(Jt),e.macros=(e.macros||[]).concat(zt),e.symbols=(e.symbols||[]).concat(Qt),e}class en{constructor(e){this.data=e}orderBy(e){let t;return t=Array.isArray(e)?(t,n)=>{for(const r of e)if("string"===typeof r){if(t[r]>n[r])return 1;if(t[r]<n[r])return-1}else{const e="desc"===r[1]?-1:1;if(t[r[0]]>n[r[0]])return 1*e;if(t[r[0]]<n[r[0]])return-1*e}return 0}:e,new en(this.data.slice(0).sort(t))}groupBy(e){let t;t=Array.isArray(e)?(t,n)=>{for(const r of e)if(t[r]!==n[r])return!1;return!0}:e;const n=[];let r=0,o=1;for(;o<this.data.length;o++)t(this.data[r],this.data[o],o,this.data)||(n.push(this.data.slice(r,o)),r=o);return n.push(this.data.slice(r,o)),new en(n)}groupEvery(e){if("number"===typeof e)return this.groupBy((t,n,r,o)=>r%e!==0);{const t=Object.assign({first:e.intermediate,last:e.intermediate},e),n=this.groupBy((e,n,r,o)=>t.single>=o.length?r%t.single!==0:r<=t.first?r%t.first!==0:(r-t.first)%t.intermediate!==0);return 1===n.data.length?t.single<n.data[0].length&&n.data.push([]):t.first<n.data[0].length&&n.data.unshift([]),n.data.length>1&&n.data[n.data.length-1].length>t.last&&n.data.push([]),n}}where(e){return new en(this.data.filter(e))}select(e){return e?this.data.map(e):this.data}}function tn(e){return new en(e)}const nn=(e,t)=>(...t)=>{Object(P["a"])("$range",t,2,3);const{car:n,cdr:r}=G(...t),o=Object(E["l"])(n)||0,a=Object(E["l"])(r)||0,s=t.length>2&&Object(E["l"])(t[2])||(o<=a?1:-1),l=Math.sign(a-o)+Math.sign(s)!==0?Math.floor(Math.abs(a-o)/Math.abs(s))+1:0;return e.evalCount+=l,Object(E["b"])(e,0),Array.from({length:l},(e,t)=>o+t*s)},rn=(e,t)=>(...e)=>{Object(P["a"])("$length",e,1,1);const t=R(...e);switch(typeof t){case"object":if(!("length"in t))break;case"string":return t.length}throw new Error("[SX] $length: Invalid argument type: object has no property 'length'.")},on=(rn(null,null),(e,t)=>(...e)=>{Object(P["a"])("$trim",e,1,1);const t=R(...e);if("string"===typeof t)return t.trim();throw new Error("[SX] $trim: Invalid argument type: args[0] is not string.")}),an=(on(null,null),(e,t)=>(...e)=>{Object(P["a"])("$trimHead",e,1,1);const t=R(...e);if("string"===typeof t)return t.trimLeft();throw new Error("[SX] $trimHead: Invalid argument type: args[0] is not string.")}),sn=(an(null,null),(e,t)=>(...e)=>{Object(P["a"])("$trimTail",e,1,1);const t=R(...e);if("string"===typeof t)return t.trimRight();throw new Error("[SX] $trimTail: Invalid argument type: args[0] is not string.")}),ln=(sn(null,null),(e,t)=>(...e)=>{if(Object(P["a"])("$replaceAll",e,3,3),"string"===typeof e[0]&&"string"===typeof e[1]&&"string"===typeof e[2])return e[0].split(e[1]).join(e[2]);throw new Error("[SX] $replaceAll: Invalid argument type: args[0] or [1] or [2] is not string.")}),cn=(ln(null,null),(e,t)=>(...e)=>{if(Object(P["a"])("$split",e,2,2),"string"===typeof e[0]&&"string"===typeof e[1])return e[0].split(e[1]);throw new Error("[SX] $split: Invalid argument type: args[0] or [1] is not string.")}),un=(cn(null,null),(e,t)=>(...e)=>{if(Object(P["a"])("$join",e,1,2),Array.isArray(e[0]),e.length>1){if("string"===typeof e[1])return e[0].join(e[1]);throw new Error("[SX] $join: Invalid argument type: args[1] is not string.")}return e[0].join()}),fn=(un(null,null),(e,t)=>(...e)=>{Object(P["a"])("$concat",e,1);const t=R(...e);switch(typeof t){case"object":if(!("concat"in t))break;case"string":return t.concat(...e.slice(1))}throw new Error("[SX] $concat: Invalid argument type: object has no property 'concat'.")}),mn=(fn(null,null),(e,t)=>(...e)=>{if(Object(P["a"])("$slice",e,2,3),3===e.length&&("string"===typeof e[2]||Array.isArray(e[2])))return e[2].slice(Object(E["l"])(e[0]),Object(E["l"])(e[1]));if(2===e.length&&("string"===typeof e[1]||Array.isArray(e[1])))return e[1].slice(Object(E["l"])(e[0]));throw new Error(`[SX] $slice: Invalid argument type: args[${e.length-1}] is not string or array.`)}),bn=(mn(null,null),(e,t)=>(...e)=>{if(Object(P["a"])("$top",e,2,2),"string"===typeof e[1]||Array.isArray(e[1]))return e[1].slice(0,Object(E["l"])(e[0]));throw new Error("[SX] $top: Invalid argument type: args[1] is not string or array.")}),$n=(bn(null,null),(e,t)=>(...e)=>{if(Object(P["a"])("$tail",e,2,2),"string"===typeof e[1]||Array.isArray(e[1])){const t=-Object(E["l"])(e[0]);return e[1].slice(t>=0||Number.isNaN(t)?e[1].length:t)}throw new Error("[SX] $tail: Invalid argument type: args[1] is not string or array.")}),pn=($n(null,null),(e,t)=>(...e)=>(Object(P["a"])("$push",e,2,2),Array.isArray(e[0]),e[0].push(e[1]),e[0])),_n=(pn(null,null),(e,t)=>(...e)=>{Object(P["a"])("$pop",e,1,1),Array.isArray(e[0]);{const t=e[0].pop();return t}}),dn=(_n(null,null),(e,t)=>(...e)=>{Object(P["a"])("$__at",e,2,2);const{car:t,cdr:n}=G(...e);return n[t]}),yn=(dn(null,null),(e,t)=>(...e)=>{Object(P["a"])("$reverse",e,1,1);const t=R(...e);if(Array.isArray(t))return t.slice(0).reverse();throw new Error("[SX] $reverse: Invalid argument type: args[0] is not array.")}),gn=(yn(null,null),(e,t)=>(...e)=>{Object(P["a"])("$reverse!",e,1,1);const t=R(...e);if(Array.isArray(t))return t.reverse();throw new Error("[SX] $reverse!: Invalid argument type: args[0] is not array.")}),jn=(gn(null,null),(e,t)=>(...e)=>{Object(P["a"])("$find",e,2,2);const{car:t,cdr:n}=G(...e);if(Array.isArray(t))return t.find(n);throw new Error("[SX] $find: Invalid argument type: args[0] is not array.")}),On=(jn(null,null),(e,t)=>(...e)=>{Object(P["a"])("$filter",e,2,2);const{car:t,cdr:n}=G(...e);if(Array.isArray(t))return t.filter(n);throw new Error("[SX] $filter: Invalid argument type: args[0] is not array.")}),hn=(On(null,null),(e,t)=>(...e)=>{Object(P["a"])("$map",e,2,2);const{car:t,cdr:n}=G(...e);if(Array.isArray(t))return t.map(n);throw new Error("[SX] $map: Invalid argument type: args[0] is not array.")}),vn=(hn(null,null),(e,t)=>(...e)=>{Object(P["a"])("$reduce",e,2,3);const{car:t,cdr:n}=G(...e);if(Array.isArray(t))return e.length<3?t.reduce(n):t.reduce(n,e[2]);throw new Error("[SX] $reduce: Invalid argument type: args[0] is not array.")}),wn=(vn(null,null),(e,t)=>(...e)=>{Object(P["a"])("$reduceFromTail",e,2,3);const{car:t,cdr:n}=G(...e);if(Array.isArray(t))return e.length<3?t.reduceRight(n):t.reduceRight(n,e[2]);throw new Error("[SX] $reduceFromTail: Invalid argument type: args[0] is not array.")}),Sn=(wn(null,null),(e,t)=>(...e)=>{Object(P["a"])("$sort",e,2,2);const{car:t,cdr:n}=G(...e);if(Array.isArray(t))return t.slice(0).sort(n);throw new Error("[SX] $sort: Invalid argument type: args[0] is not array.")}),An=(Sn(null,null),(e,t)=>(...e)=>{Object(P["a"])("$sort!",e,2,2);const{car:t,cdr:n}=G(...e);if(Array.isArray(t))return t.sort(n);throw new Error("[SX] $sort!: Invalid argument type: args[0] is not array.")}),En=(An(null,null),(e,t)=>(...e)=>{Object(P["a"])("$group-every",e,2,2);const{car:t,cdr:n}=G(...e);if(!Array.isArray(n))throw new Error("[SX] $group-every: Invalid argument type: args[1] is not array.");return tn(n).groupEvery(t).select()}),Nn=(En(null,null),(e,t)=>(...e)=>{Object(P["a"])("$group-by",e,2,2);const{car:t,cdr:n}=G(...e);if(!Array.isArray(n))throw new Error("[SX] $group-by: Invalid argument type: args[1] is not array.");return tn(n).groupBy(t).select()}),In=(Nn(null,null),(e,t)=>(...e)=>{Object(P["a"])("$order-by",e,2,2);const{car:t,cdr:n}=G(...e);if(!Array.isArray(n))throw new Error("[SX] $order-by: Invalid argument type: args[1] is not array.");return tn(n).orderBy(t).select()}),kn=(In(null,null),(e,t)=>(...e)=>{Object(P["a"])("$where",e,2,2);const{car:t,cdr:n}=G(...e);if("function"!==typeof e[0])throw new Error("[SX] $where: Invalid argument type: args[0] is not function.");if(!Array.isArray(n))throw new Error("[SX] $where: Invalid argument type: args[1] is not array.");return tn(n).where(t).select()}),Tn=(kn(null,null),[{name:"$range",fn:nn},{name:"$length",fn:rn},{name:"$trim",fn:on},{name:"$trim-head",fn:an},{name:"$trim-tail",fn:sn},{name:"$replace-all",fn:ln},{name:"$split",fn:cn},{name:"$join",fn:un},{name:"$concat",fn:fn},{name:"$slice",fn:mn},{name:"$top",fn:bn},{name:"$tail",fn:$n},{name:"$push",fn:pn},{name:"$pop",fn:_n},{name:"$__at",fn:dn},{name:"$reverse",fn:yn},{name:"$reverse!",fn:gn},{name:"$find",fn:jn},{name:"$filter",fn:On},{name:"$map",fn:hn},{name:"$reduce",fn:vn},{name:"$reduce-from-head",fn:vn},{name:"$reduce-from-tail",fn:wn},{name:"$sort",fn:Sn},{name:"$sort!",fn:An},{name:"$group-every",fn:En},{name:"$group-by",fn:Nn},{name:"$order-by",fn:In},{name:"$where",fn:kn}]);var Xn=Tn;const xn=[{name:"$[",fn:(e,t)=>e=>{const t=Object(o["d"])(e[2],"]");if(!t)throw new Error("[SX] $repeat: Invalid syntax: missing ']' keyword.");return[{symbol:"$__at"},e[1],e[3]]}}];var Cn=xn;const Mn=[];var Pn=Mn;function Dn(e){return e.funcs=(e.funcs||[]).concat(Xn),e.macros=(e.macros||[]).concat(Cn),e.symbols=(e.symbols||[]).concat(Pn),e}const Ln=(e,t)=>(...t)=>{Object(P["a"])("$__letAsync",t,2,2);let n=t[1];return"object"===typeof n&&"function"===typeof n.then||(n=Promise.resolve(n)),n=n.then(n=>{try{return he(e,"")(t[0],n),n}catch(r){return Promise.reject(r)}}),n},qn=(Ln(null,null),(e,t)=>(...t)=>{Object(P["a"])("$__setAsync",t,2,2);let n=t[1];return"object"===typeof n&&"function"===typeof n.then||(n=Promise.resolve(n)),n=n.then(n=>{try{return ve(e,"")(t[0],n),n}catch(r){return Promise.reject(r)}}),n}),Un=(qn(null,null),(e,t)=>(...e)=>{Object(P["a"])("$then",e,2,3);let t=e[0];if("object"===typeof t&&"function"===typeof t.then||(t=Promise.resolve(t)),"function"===typeof e[2])t=t.then(e[1],e[2]);else{if("function"!==typeof e[1])throw new Error("[SX] $then: Invalid argument(s): args[1] is not function.");t=t.then(e[1])}return t}),Rn=(Un(null,null),(e,t)=>(...e)=>{const t=e.slice(0);for(let n=0;n<t.length;n++)"object"===typeof t[n]&&"function"===typeof t[n].then||(t[n]=Promise.resolve(t[n]));return Promise.all(t)}),Fn=(Rn(null,null),(e,t)=>(...e)=>{const t=e.slice(0);for(let o=0;o<t.length;o++)"object"===typeof t[o]&&"function"===typeof t[o].then||(t[o]=Promise.resolve(t[o]));const n=e=>new Promise((t,n)=>e.then(n,t)),r=e=>n(Promise.all(e.map(n)));return r(t)}),Bn=(Fn(null,null),(e,t)=>(...e)=>{Object(P["a"])("$resolvePipe",e,1);let t=e[0];"object"===typeof t&&"function"===typeof t.then||(t=Promise.resolve(t));const n=e.slice(1);for(let o=0;o<n.length;o++)if("function"!==typeof n[o]){const e=n[o];n[o]=()=>e}let r=t;for(const o of n)r=r.then(o);return r}),Wn=(Bn(null,null),(e,t)=>(...e)=>{Object(P["a"])("$resolveFork",e,1);let t=e[0];"object"===typeof t&&"function"===typeof t.then||(t=Promise.resolve(t));const n=e.slice(1);for(let s=0;s<n.length;s++)if("function"!==typeof n[s]){const e=n[s];n[s]=()=>e}const r=new Array(n.length),o=new Array(n.length),a=[];for(let s=0;s<n.length;s++)a.push(new Promise((e,t)=>{r[s]=r=>{let o=n[s](r);"object"===typeof o&&"function"===typeof o.then||(o=Promise.resolve(o)),o.then(t=>e(t)).catch(e=>t(e))},o[s]=t}));return t.then(e=>r.forEach(t=>t(e)),e=>o.forEach(t=>t(e))),a}),Kn=(Wn(null,null),[{name:"$__let-async",fn:Ln},{name:"$__set-async",fn:qn},{name:"$then",fn:Un},{name:"$resolve-all",fn:Rn},{name:"$resolve-any",fn:Fn},{name:"$resolve-pipe",fn:Bn},{name:"$resolve-fork",fn:Wn}]);var Vn=Kn;const Gn=[{name:"$let-async",fn:(e,t)=>t=>(Object(P["a"])("$let-async",t,3,3),[{symbol:"$__let-async"},Object(o["f"])(e,t[1]),t[2]])},{name:"$set-async",fn:(e,t)=>t=>(Object(P["a"])("$set-async",t,3,3),[{symbol:"$__set-async"},Object(o["f"])(e,t[1]),t[2]])}];var Jn=Gn;const Hn=[];var zn=Hn;function Yn(e){return e.funcs=(e.funcs||[]).concat(Vn),e.macros=(e.macros||[]).concat(Jn),e.symbols=(e.symbols||[]).concat(zn),e}const Qn=(()=>{let e=Object.assign({},s);return e.reservedNames=Object.assign({},e.reservedNames,{Template:"$concat"}),e=St(e),e=Zt(e),e=Dn(e),e=Yn(e),e.stripComments=!0,T(e)})(),Zn=Qn,er=Qn,tr=(()=>{let e=Object.assign({},s);return e.reservedNames=Object.assign({},e.reservedNames,{Template:"$concat"}),e=St(e),e=Zt(e),e=Dn(e),e=Yn(e),e.stripComments=!0,X(e)})(),nr=tr,rr=tr,or=(()=>{let e=Object.assign({},s);return e.reservedNames=Object.assign({},e.reservedNames,{Template:"$concat"}),e=St(e),e=Zt(e),e=Dn(e),e=Yn(e),e.stripComments=!0,e.returnMultipleRoot=!0,T(e)})(),ar=(()=>{let e=Object.assign({},s);return e.reservedNames=Object.assign({},e.reservedNames,{Template:"$concat"}),e=St(e),e=Zt(e),e=Dn(e),e=Yn(e),e.stripComments=!0,e.returnMultipleRoot=!0,X(e)})(),sr=(e,t)=>(...t)=>{Object(P["a"])("$__outputIf",t,2);const{car:n,cdr:r}=G(...t);let o=[];return Se(n)&&(2<t.length?(o.push({symbol:e.config.reservedNames.Template},...t.slice(1)),o=Object(E["b"])(e,o)):o=Object(E["b"])(e,r)),o},lr=(e,t)=>(...n)=>{Object(P["a"])("$__outputForOf",n,2);const r=R(...n),a=[];if(!Array.isArray(r))throw new Error("[SX] $__outputForOf: Invalid argument(s): args[0] is not array.");for(let s=0;s<r.length;s++){const l=r[s],c=Q(e,t)(!0,!0,[["$data",Object(o["f"])(e,l)],["$index",s],["$array",Object(o["f"])(e,r)],["$parent",Object(o["f"])(e,Object(E["e"])(e).scope)]],...n.slice(1));2<n.length&&Array.isArray(c)?a.push(...c):a.push(c)}return Object(E["b"])(e,[{symbol:e.config.reservedNames.Template}].concat(a.map(t=>[{symbol:e.config.reservedNames.quote},t])))},cr=(e,t)=>(...t)=>{const n={};for(const r of t){if(!(Array.isArray(r)&&0<r.length))throw new Error("[SX] $jsxProps: Invalid argument(s): args[?] is not array.");{const t=Object(o["d"])(r[0]),a=t?t.symbol:String(Object(E["b"])(e,r[0]));switch(a){case"style":if(1===r.length)n[a]="";else if(r.length>=2){const t={};for(const n of r.slice(1))if(Array.isArray(n)&&1<n.length){const r=String(Object(E["b"])(e,n[0]));Object(P["c"])("$jsxProps",t,r),t[r]=String(Object(E["b"])(e,n[1]))}else if("string"===typeof n)for(const e of n.split(";")){const n=/^\s*(\S+)\s*:\s*(.*?)\s*$/.exec(e);n&&(Object(P["c"])("$jsxProps",t,n[1]),t[n[1]]=n[2])}n[a]=t}break;case"class":case"styleClass":if(1===r.length)n[a]=[];else if(r.length>=2){let t=[];for(const n of r.slice(1))Array.isArray(n)?t=t.concat(n.map(t=>Object(E["b"])(e,t))):"string"===typeof n&&(t=t.concat(n.split(" ")));const o=[],s=e=>e.forEach(e=>null===e||void 0===e?void 0:Array.isArray(e)?s(e):o.push(String(e)));s(t),n[a]=o}break;case"className":if(1===r.length)n[a]="";else if(r.length>=2){let t="";for(const n of r.slice(1)){let r="";if(Array.isArray(n)){const t=[],o=n=>n.map(t=>Object(E["b"])(e,t)).forEach(e=>null===e||void 0===e?void 0:Array.isArray(e)?o(e):t.push(String(e)));o(n),r=t.join(" ")}else"string"===typeof n&&(r=n);0<t.length?t+=" "+r:t=r}n[a]=t}break;case"dangerouslySetInnerHTML":1===r.length?n[a]={__html:""}:r.length>=2?n[a]={__html:Object(E["b"])(e,r[1])}:n[a]={__html:Object(E["b"])(e,[{symbol:e.config.reservedNames.list}].concat(r.slice(1)))};break;case"setInnerText":1===r.length?n[a]={__text:""}:r.length>=2?n[a]={__text:Object(E["b"])(e,r[1])}:n[a]={__text:Object(E["b"])(e,[{symbol:e.config.reservedNames.list}].concat(r.slice(1)))};break;default:Object(P["c"])("$jsxProps",n,a),1===r.length?n[a]=!0:2===r.length?n[a]=Object(E["b"])(e,r[1]):n[a]=Object(E["b"])(e,[{symbol:e.config.reservedNames.list}].concat(r.slice(1)));break}}}return n};function ir(e,...t){let n=t,r={};if(0<t.length&&Array.isArray(t[0])){const a=Object(o["d"])(t[0][0],"@");a&&(r=cr(e,"")(...t[0].slice(1)),n=n.slice(1))}return{props:r,children:n}}const ur=(e,t)=>(...n)=>{const{props:r,children:o}=ir(e,...n);return e.config.jsx(t,r,...o)},fr=e=>(t,n)=>(...n)=>{const{props:r,children:o}=ir(t,...n);return t.config.jsx(e,r,...o)},mr=[{name:"$=__if",fn:sr},{name:"$=__for",fn:lr}];var br=mr;const $r=[{name:"@",fn:(e,t)=>t=>Object(o["f"])(e,t)},{name:"$=if",fn:(e,t)=>t=>[{symbol:"$=__if"},t[1],...t.slice(2).map(t=>Object(o["f"])(e,t))]},{name:"$=for",fn:(e,t)=>t=>[{symbol:"$=__for"},t[1],...t.slice(2).map(t=>Object(o["f"])(e,t))]}];var pr=$r;const _r=[];var dr=_r;function yr(e,t){e.funcs=(e.funcs||[]).concat(br),e.macros=(e.macros||[]).concat(pr),e.symbols=(e.symbols||[]).concat(dr);const n=Object.entries(t.components).map(e=>({name:e[0],fn:fr(e[1])}));return e.funcs=e.funcs.concat({name:e.reservedNames.Template,fn:fr(t.jsxFlagment)},...n),e.funcSymbolResolverFallback=ur,e.jsx=t.jsx,e.JsxFragment=t.jsxFlagment,e}function gr(e){let t=Object.assign({},s);return t=St(t),t=Zt(t),t=Dn(t),t=Yn(t),t=yr(t,e),t.stripComments=!0,T(t)}function jr(e){let t=Object.assign({},s);return t=St(t),t=Zt(t),t=Dn(t),t=Yn(t),t=yr(t,e),t.stripComments=!0,X(t)}function Or(e,t,n="text/lisp"){const r=document.querySelectorAll(`script[type="${n}"]`),o=[];for(let a=0;a<r.length;a++)o.push(r[a].innerHTML);return e=e.appendGlobals(t||{}),e(o.join("\n"))}const hr={core:gt,arithmetic:Jt,sequence:Xn,jsx:br,concurrent:Vn},vr={core:ht,arithmetic:zt,sequence:Cn,jsx:pr,concurrent:Jn},wr={core:wt,arithmetic:Qt,sequence:Pn,jsx:dr,concurrent:zn}}}]);
|
2 |
+
//# sourceMappingURL=chunk-a06ef50c.1caef24f.js.map
|
dist/js/chunk-ae402692.003457bc.js
ADDED
The diff for this file is too large to render.
See raw diff
|
|
dist/js/chunk-vendors.20f7f886.js
ADDED
@@ -0,0 +1,8 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
(window["webpackJsonp"]=window["webpackJsonp"]||[]).push([["chunk-vendors"],{"24fb":function(t,e,n){"use strict";function r(t,e){var n=t[1]||"",r=t[3];if(!r)return n;if(e&&"function"===typeof btoa){var i=o(r),a=r.sources.map((function(t){return"/*# sourceURL=".concat(r.sourceRoot||"").concat(t," */")}));return[n].concat(a).concat([i]).join("\n")}return[n].join("\n")}function o(t){var e=btoa(unescape(encodeURIComponent(JSON.stringify(t)))),n="sourceMappingURL=data:application/json;charset=utf-8;base64,".concat(e);return"/*# ".concat(n," */")}t.exports=function(t){var e=[];return e.toString=function(){return this.map((function(e){var n=r(e,t);return e[2]?"@media ".concat(e[2]," {").concat(n,"}"):n})).join("")},e.i=function(t,n,r){"string"===typeof t&&(t=[[null,t,""]]);var o={};if(r)for(var i=0;i<this.length;i++){var a=this[i][0];null!=a&&(o[a]=!0)}for(var s=0;s<t.length;s++){var c=[].concat(t[s]);r&&o[c[0]]||(n&&(c[2]?c[2]="".concat(n," and ").concat(c[2]):c[2]=n),e.push(c))}},e}},2877:function(t,e,n){"use strict";function r(t,e,n,r,o,i,a,s){var c,u="function"===typeof t?t.options:t;if(e&&(u.render=e,u.staticRenderFns=n,u._compiled=!0),r&&(u.functional=!0),i&&(u._scopeId="data-v-"+i),a?(c=function(t){t=t||this.$vnode&&this.$vnode.ssrContext||this.parent&&this.parent.$vnode&&this.parent.$vnode.ssrContext,t||"undefined"===typeof __VUE_SSR_CONTEXT__||(t=__VUE_SSR_CONTEXT__),o&&o.call(this,t),t&&t._registeredComponents&&t._registeredComponents.add(a)},u._ssrRegister=c):o&&(c=s?function(){o.call(this,(u.functional?this.parent:this).$root.$options.shadowRoot)}:o),c)if(u.functional){u._injectStyles=c;var l=u.render;u.render=function(t,e){return c.call(e),l(t,e)}}else{var f=u.beforeCreate;u.beforeCreate=f?[].concat(f,c):[c]}return{exports:t,options:u}}n.d(e,"a",(function(){return r}))},"2b0e":function(t,e,n){"use strict";(function(t){n.d(e,"a",(function(){return Qr}));
|
2 |
+
/*!
|
3 |
+
* Vue.js v2.7.16
|
4 |
+
* (c) 2014-2023 Evan You
|
5 |
+
* Released under the MIT License.
|
6 |
+
*/
|
7 |
+
var r=Object.freeze({}),o=Array.isArray;function i(t){return void 0===t||null===t}function a(t){return void 0!==t&&null!==t}function s(t){return!0===t}function c(t){return!1===t}function u(t){return"string"===typeof t||"number"===typeof t||"symbol"===typeof t||"boolean"===typeof t}function l(t){return"function"===typeof t}function f(t){return null!==t&&"object"===typeof t}var d=Object.prototype.toString;function p(t){return"[object Object]"===d.call(t)}function v(t){return"[object RegExp]"===d.call(t)}function h(t){var e=parseFloat(String(t));return e>=0&&Math.floor(e)===e&&isFinite(t)}function m(t){return a(t)&&"function"===typeof t.then&&"function"===typeof t.catch}function _(t){return null==t?"":Array.isArray(t)||p(t)&&t.toString===d?JSON.stringify(t,y,2):String(t)}function y(t,e){return e&&e.__v_isRef?e.value:e}function g(t){var e=parseFloat(t);return isNaN(e)?t:e}function b(t,e){for(var n=Object.create(null),r=t.split(","),o=0;o<r.length;o++)n[r[o]]=!0;return e?function(t){return n[t.toLowerCase()]}:function(t){return n[t]}}b("slot,component",!0);var w=b("key,ref,slot,slot-scope,is");function C(t,e){var n=t.length;if(n){if(e===t[n-1])return void(t.length=n-1);var r=t.indexOf(e);if(r>-1)return t.splice(r,1)}}var $=Object.prototype.hasOwnProperty;function x(t,e){return $.call(t,e)}function k(t){var e=Object.create(null);return function(n){var r=e[n];return r||(e[n]=t(n))}}var O=/-(\w)/g,S=k((function(t){return t.replace(O,(function(t,e){return e?e.toUpperCase():""}))})),j=k((function(t){return t.charAt(0).toUpperCase()+t.slice(1)})),T=/\B([A-Z])/g,A=k((function(t){return t.replace(T,"-$1").toLowerCase()}));function E(t,e){function n(n){var r=arguments.length;return r?r>1?t.apply(e,arguments):t.call(e,n):t.call(e)}return n._length=t.length,n}function I(t,e){return t.bind(e)}var P=Function.prototype.bind?I:E;function N(t,e){e=e||0;var n=t.length-e,r=new Array(n);while(n--)r[n]=t[n+e];return r}function D(t,e){for(var n in e)t[n]=e[n];return t}function M(t){for(var e={},n=0;n<t.length;n++)t[n]&&D(e,t[n]);return e}function L(t,e,n){}var F=function(t,e,n){return!1},R=function(t){return t};function U(t,e){if(t===e)return!0;var n=f(t),r=f(e);if(!n||!r)return!n&&!r&&String(t)===String(e);try{var o=Array.isArray(t),i=Array.isArray(e);if(o&&i)return t.length===e.length&&t.every((function(t,n){return U(t,e[n])}));if(t instanceof Date&&e instanceof Date)return t.getTime()===e.getTime();if(o||i)return!1;var a=Object.keys(t),s=Object.keys(e);return a.length===s.length&&a.every((function(n){return U(t[n],e[n])}))}catch(c){return!1}}function B(t,e){for(var n=0;n<t.length;n++)if(U(t[n],e))return n;return-1}function V(t){var e=!1;return function(){e||(e=!0,t.apply(this,arguments))}}function H(t,e){return t===e?0===t&&1/t!==1/e:t===t||e===e}var z="data-server-rendered",W=["component","directive","filter"],q=["beforeCreate","created","beforeMount","mounted","beforeUpdate","updated","beforeDestroy","destroyed","activated","deactivated","errorCaptured","serverPrefetch","renderTracked","renderTriggered"],K={optionMergeStrategies:Object.create(null),silent:!1,productionTip:!1,devtools:!1,performance:!1,errorHandler:null,warnHandler:null,ignoredElements:[],keyCodes:Object.create(null),isReservedTag:F,isReservedAttr:F,isUnknownElement:F,getTagNamespace:L,parsePlatformTagName:R,mustUseProp:F,async:!0,_lifecycleHooks:q},G=/a-zA-Z\u00B7\u00C0-\u00D6\u00D8-\u00F6\u00F8-\u037D\u037F-\u1FFF\u200C-\u200D\u203F-\u2040\u2070-\u218F\u2C00-\u2FEF\u3001-\uD7FF\uF900-\uFDCF\uFDF0-\uFFFD/;function J(t){var e=(t+"").charCodeAt(0);return 36===e||95===e}function X(t,e,n,r){Object.defineProperty(t,e,{value:n,enumerable:!!r,writable:!0,configurable:!0})}var Z=new RegExp("[^".concat(G.source,".$_\\d]"));function Q(t){if(!Z.test(t)){var e=t.split(".");return function(t){for(var n=0;n<e.length;n++){if(!t)return;t=t[e[n]]}return t}}}var Y="__proto__"in{},tt="undefined"!==typeof window,et=tt&&window.navigator.userAgent.toLowerCase(),nt=et&&/msie|trident/.test(et),rt=et&&et.indexOf("msie 9.0")>0,ot=et&&et.indexOf("edge/")>0;et&&et.indexOf("android");var it=et&&/iphone|ipad|ipod|ios/.test(et);et&&/chrome\/\d+/.test(et),et&&/phantomjs/.test(et);var at,st=et&&et.match(/firefox\/(\d+)/),ct={}.watch,ut=!1;if(tt)try{var lt={};Object.defineProperty(lt,"passive",{get:function(){ut=!0}}),window.addEventListener("test-passive",null,lt)}catch(Ya){}var ft=function(){return void 0===at&&(at=!tt&&"undefined"!==typeof t&&(t["process"]&&"server"===t["process"].env.VUE_ENV)),at},dt=tt&&window.__VUE_DEVTOOLS_GLOBAL_HOOK__;function pt(t){return"function"===typeof t&&/native code/.test(t.toString())}var vt,ht="undefined"!==typeof Symbol&&pt(Symbol)&&"undefined"!==typeof Reflect&&pt(Reflect.ownKeys);vt="undefined"!==typeof Set&&pt(Set)?Set:function(){function t(){this.set=Object.create(null)}return t.prototype.has=function(t){return!0===this.set[t]},t.prototype.add=function(t){this.set[t]=!0},t.prototype.clear=function(){this.set=Object.create(null)},t}();var mt=null;function _t(t){void 0===t&&(t=null),t||mt&&mt._scope.off(),mt=t,t&&t._scope.on()}var yt=function(){function t(t,e,n,r,o,i,a,s){this.tag=t,this.data=e,this.children=n,this.text=r,this.elm=o,this.ns=void 0,this.context=i,this.fnContext=void 0,this.fnOptions=void 0,this.fnScopeId=void 0,this.key=e&&e.key,this.componentOptions=a,this.componentInstance=void 0,this.parent=void 0,this.raw=!1,this.isStatic=!1,this.isRootInsert=!0,this.isComment=!1,this.isCloned=!1,this.isOnce=!1,this.asyncFactory=s,this.asyncMeta=void 0,this.isAsyncPlaceholder=!1}return Object.defineProperty(t.prototype,"child",{get:function(){return this.componentInstance},enumerable:!1,configurable:!0}),t}(),gt=function(t){void 0===t&&(t="");var e=new yt;return e.text=t,e.isComment=!0,e};function bt(t){return new yt(void 0,void 0,void 0,String(t))}function wt(t){var e=new yt(t.tag,t.data,t.children&&t.children.slice(),t.text,t.elm,t.context,t.componentOptions,t.asyncFactory);return e.ns=t.ns,e.isStatic=t.isStatic,e.key=t.key,e.isComment=t.isComment,e.fnContext=t.fnContext,e.fnOptions=t.fnOptions,e.fnScopeId=t.fnScopeId,e.asyncMeta=t.asyncMeta,e.isCloned=!0,e}"function"===typeof SuppressedError&&SuppressedError;var Ct=0,$t=[],xt=function(){for(var t=0;t<$t.length;t++){var e=$t[t];e.subs=e.subs.filter((function(t){return t})),e._pending=!1}$t.length=0},kt=function(){function t(){this._pending=!1,this.id=Ct++,this.subs=[]}return t.prototype.addSub=function(t){this.subs.push(t)},t.prototype.removeSub=function(t){this.subs[this.subs.indexOf(t)]=null,this._pending||(this._pending=!0,$t.push(this))},t.prototype.depend=function(e){t.target&&t.target.addDep(this)},t.prototype.notify=function(t){var e=this.subs.filter((function(t){return t}));for(var n=0,r=e.length;n<r;n++){var o=e[n];0,o.update()}},t}();kt.target=null;var Ot=[];function St(t){Ot.push(t),kt.target=t}function jt(){Ot.pop(),kt.target=Ot[Ot.length-1]}var Tt=Array.prototype,At=Object.create(Tt),Et=["push","pop","shift","unshift","splice","sort","reverse"];Et.forEach((function(t){var e=Tt[t];X(At,t,(function(){for(var n=[],r=0;r<arguments.length;r++)n[r]=arguments[r];var o,i=e.apply(this,n),a=this.__ob__;switch(t){case"push":case"unshift":o=n;break;case"splice":o=n.slice(2);break}return o&&a.observeArray(o),a.dep.notify(),i}))}));var It=Object.getOwnPropertyNames(At),Pt={},Nt=!0;function Dt(t){Nt=t}var Mt={notify:L,depend:L,addSub:L,removeSub:L},Lt=function(){function t(t,e,n){if(void 0===e&&(e=!1),void 0===n&&(n=!1),this.value=t,this.shallow=e,this.mock=n,this.dep=n?Mt:new kt,this.vmCount=0,X(t,"__ob__",this),o(t)){if(!n)if(Y)t.__proto__=At;else for(var r=0,i=It.length;r<i;r++){var a=It[r];X(t,a,At[a])}e||this.observeArray(t)}else{var s=Object.keys(t);for(r=0;r<s.length;r++){a=s[r];Rt(t,a,Pt,void 0,e,n)}}}return t.prototype.observeArray=function(t){for(var e=0,n=t.length;e<n;e++)Ft(t[e],!1,this.mock)},t}();function Ft(t,e,n){return t&&x(t,"__ob__")&&t.__ob__ instanceof Lt?t.__ob__:!Nt||!n&&ft()||!o(t)&&!p(t)||!Object.isExtensible(t)||t.__v_skip||qt(t)||t instanceof yt?void 0:new Lt(t,e,n)}function Rt(t,e,n,r,i,a,s){void 0===s&&(s=!1);var c=new kt,u=Object.getOwnPropertyDescriptor(t,e);if(!u||!1!==u.configurable){var l=u&&u.get,f=u&&u.set;l&&!f||n!==Pt&&2!==arguments.length||(n=t[e]);var d=i?n&&n.__ob__:Ft(n,!1,a);return Object.defineProperty(t,e,{enumerable:!0,configurable:!0,get:function(){var e=l?l.call(t):n;return kt.target&&(c.depend(),d&&(d.dep.depend(),o(e)&&Vt(e))),qt(e)&&!i?e.value:e},set:function(e){var r=l?l.call(t):n;if(H(r,e)){if(f)f.call(t,e);else{if(l)return;if(!i&&qt(r)&&!qt(e))return void(r.value=e);n=e}d=i?e&&e.__ob__:Ft(e,!1,a),c.notify()}}}),c}}function Ut(t,e,n){if(!Wt(t)){var r=t.__ob__;return o(t)&&h(e)?(t.length=Math.max(t.length,e),t.splice(e,1,n),r&&!r.shallow&&r.mock&&Ft(n,!1,!0),n):e in t&&!(e in Object.prototype)?(t[e]=n,n):t._isVue||r&&r.vmCount?n:r?(Rt(r.value,e,n,void 0,r.shallow,r.mock),r.dep.notify(),n):(t[e]=n,n)}}function Bt(t,e){if(o(t)&&h(e))t.splice(e,1);else{var n=t.__ob__;t._isVue||n&&n.vmCount||Wt(t)||x(t,e)&&(delete t[e],n&&n.dep.notify())}}function Vt(t){for(var e=void 0,n=0,r=t.length;n<r;n++)e=t[n],e&&e.__ob__&&e.__ob__.dep.depend(),o(e)&&Vt(e)}function Ht(t){return zt(t,!0),X(t,"__v_isShallow",!0),t}function zt(t,e){if(!Wt(t)){Ft(t,e,ft());0}}function Wt(t){return!(!t||!t.__v_isReadonly)}function qt(t){return!(!t||!0!==t.__v_isRef)}function Kt(t,e,n){Object.defineProperty(t,n,{enumerable:!0,configurable:!0,get:function(){var t=e[n];if(qt(t))return t.value;var r=t&&t.__ob__;return r&&r.dep.depend(),t},set:function(t){var r=e[n];qt(r)&&!qt(t)?r.value=t:e[n]=t}})}var Gt="watcher";"".concat(Gt," callback"),"".concat(Gt," getter"),"".concat(Gt," cleanup");var Jt;var Xt=function(){function t(t){void 0===t&&(t=!1),this.detached=t,this.active=!0,this.effects=[],this.cleanups=[],this.parent=Jt,!t&&Jt&&(this.index=(Jt.scopes||(Jt.scopes=[])).push(this)-1)}return t.prototype.run=function(t){if(this.active){var e=Jt;try{return Jt=this,t()}finally{Jt=e}}else 0},t.prototype.on=function(){Jt=this},t.prototype.off=function(){Jt=this.parent},t.prototype.stop=function(t){if(this.active){var e=void 0,n=void 0;for(e=0,n=this.effects.length;e<n;e++)this.effects[e].teardown();for(e=0,n=this.cleanups.length;e<n;e++)this.cleanups[e]();if(this.scopes)for(e=0,n=this.scopes.length;e<n;e++)this.scopes[e].stop(!0);if(!this.detached&&this.parent&&!t){var r=this.parent.scopes.pop();r&&r!==this&&(this.parent.scopes[this.index]=r,r.index=this.index)}this.parent=void 0,this.active=!1}},t}();function Zt(t,e){void 0===e&&(e=Jt),e&&e.active&&e.effects.push(t)}function Qt(){return Jt}function Yt(t){var e=t._provided,n=t.$parent&&t.$parent._provided;return n===e?t._provided=Object.create(n):e}var te=k((function(t){var e="&"===t.charAt(0);t=e?t.slice(1):t;var n="~"===t.charAt(0);t=n?t.slice(1):t;var r="!"===t.charAt(0);return t=r?t.slice(1):t,{name:t,once:n,capture:r,passive:e}}));function ee(t,e){function n(){var t=n.fns;if(!o(t))return Ze(t,null,arguments,e,"v-on handler");for(var r=t.slice(),i=0;i<r.length;i++)Ze(r[i],null,arguments,e,"v-on handler")}return n.fns=t,n}function ne(t,e,n,r,o,a){var c,u,l,f;for(c in t)u=t[c],l=e[c],f=te(c),i(u)||(i(l)?(i(u.fns)&&(u=t[c]=ee(u,a)),s(f.once)&&(u=t[c]=o(f.name,u,f.capture)),n(f.name,u,f.capture,f.passive,f.params)):u!==l&&(l.fns=u,t[c]=l));for(c in e)i(t[c])&&(f=te(c),r(f.name,e[c],f.capture))}function re(t,e,n){var r;t instanceof yt&&(t=t.data.hook||(t.data.hook={}));var o=t[e];function c(){n.apply(this,arguments),C(r.fns,c)}i(o)?r=ee([c]):a(o.fns)&&s(o.merged)?(r=o,r.fns.push(c)):r=ee([o,c]),r.merged=!0,t[e]=r}function oe(t,e,n){var r=e.options.props;if(!i(r)){var o={},s=t.attrs,c=t.props;if(a(s)||a(c))for(var u in r){var l=A(u);ie(o,c,u,l,!0)||ie(o,s,u,l,!1)}return o}}function ie(t,e,n,r,o){if(a(e)){if(x(e,n))return t[n]=e[n],o||delete e[n],!0;if(x(e,r))return t[n]=e[r],o||delete e[r],!0}return!1}function ae(t){for(var e=0;e<t.length;e++)if(o(t[e]))return Array.prototype.concat.apply([],t);return t}function se(t){return u(t)?[bt(t)]:o(t)?ue(t):void 0}function ce(t){return a(t)&&a(t.text)&&c(t.isComment)}function ue(t,e){var n,r,c,l,f=[];for(n=0;n<t.length;n++)r=t[n],i(r)||"boolean"===typeof r||(c=f.length-1,l=f[c],o(r)?r.length>0&&(r=ue(r,"".concat(e||"","_").concat(n)),ce(r[0])&&ce(l)&&(f[c]=bt(l.text+r[0].text),r.shift()),f.push.apply(f,r)):u(r)?ce(l)?f[c]=bt(l.text+r):""!==r&&f.push(bt(r)):ce(r)&&ce(l)?f[c]=bt(l.text+r.text):(s(t._isVList)&&a(r.tag)&&i(r.key)&&a(e)&&(r.key="__vlist".concat(e,"_").concat(n,"__")),f.push(r)));return f}function le(t,e){var n,r,i,s,c=null;if(o(t)||"string"===typeof t)for(c=new Array(t.length),n=0,r=t.length;n<r;n++)c[n]=e(t[n],n);else if("number"===typeof t)for(c=new Array(t),n=0;n<t;n++)c[n]=e(n+1,n);else if(f(t))if(ht&&t[Symbol.iterator]){c=[];var u=t[Symbol.iterator](),l=u.next();while(!l.done)c.push(e(l.value,c.length)),l=u.next()}else for(i=Object.keys(t),c=new Array(i.length),n=0,r=i.length;n<r;n++)s=i[n],c[n]=e(t[s],s,n);return a(c)||(c=[]),c._isVList=!0,c}function fe(t,e,n,r){var o,i=this.$scopedSlots[t];i?(n=n||{},r&&(n=D(D({},r),n)),o=i(n)||(l(e)?e():e)):o=this.$slots[t]||(l(e)?e():e);var a=n&&n.slot;return a?this.$createElement("template",{slot:a},o):o}function de(t){return kr(this.$options,"filters",t,!0)||R}function pe(t,e){return o(t)?-1===t.indexOf(e):t!==e}function ve(t,e,n,r,o){var i=K.keyCodes[e]||n;return o&&r&&!K.keyCodes[e]?pe(o,r):i?pe(i,t):r?A(r)!==e:void 0===t}function he(t,e,n,r,i){if(n)if(f(n)){o(n)&&(n=M(n));var a=void 0,s=function(o){if("class"===o||"style"===o||w(o))a=t;else{var s=t.attrs&&t.attrs.type;a=r||K.mustUseProp(e,s,o)?t.domProps||(t.domProps={}):t.attrs||(t.attrs={})}var c=S(o),u=A(o);if(!(c in a)&&!(u in a)&&(a[o]=n[o],i)){var l=t.on||(t.on={});l["update:".concat(o)]=function(t){n[o]=t}}};for(var c in n)s(c)}else;return t}function me(t,e){var n=this._staticTrees||(this._staticTrees=[]),r=n[t];return r&&!e||(r=n[t]=this.$options.staticRenderFns[t].call(this._renderProxy,this._c,this),ye(r,"__static__".concat(t),!1)),r}function _e(t,e,n){return ye(t,"__once__".concat(e).concat(n?"_".concat(n):""),!0),t}function ye(t,e,n){if(o(t))for(var r=0;r<t.length;r++)t[r]&&"string"!==typeof t[r]&&ge(t[r],"".concat(e,"_").concat(r),n);else ge(t,e,n)}function ge(t,e,n){t.isStatic=!0,t.key=e,t.isOnce=n}function be(t,e){if(e)if(p(e)){var n=t.on=t.on?D({},t.on):{};for(var r in e){var o=n[r],i=e[r];n[r]=o?[].concat(o,i):i}}else;return t}function we(t,e,n,r){e=e||{$stable:!n};for(var i=0;i<t.length;i++){var a=t[i];o(a)?we(a,e,n):a&&(a.proxy&&(a.fn.proxy=!0),e[a.key]=a.fn)}return r&&(e.$key=r),e}function Ce(t,e){for(var n=0;n<e.length;n+=2){var r=e[n];"string"===typeof r&&r&&(t[e[n]]=e[n+1])}return t}function $e(t,e){return"string"===typeof t?e+t:t}function xe(t){t._o=_e,t._n=g,t._s=_,t._l=le,t._t=fe,t._q=U,t._i=B,t._m=me,t._f=de,t._k=ve,t._b=he,t._v=bt,t._e=gt,t._u=we,t._g=be,t._d=Ce,t._p=$e}function ke(t,e){if(!t||!t.length)return{};for(var n={},r=0,o=t.length;r<o;r++){var i=t[r],a=i.data;if(a&&a.attrs&&a.attrs.slot&&delete a.attrs.slot,i.context!==e&&i.fnContext!==e||!a||null==a.slot)(n.default||(n.default=[])).push(i);else{var s=a.slot,c=n[s]||(n[s]=[]);"template"===i.tag?c.push.apply(c,i.children||[]):c.push(i)}}for(var u in n)n[u].every(Oe)&&delete n[u];return n}function Oe(t){return t.isComment&&!t.asyncFactory||" "===t.text}function Se(t){return t.isComment&&t.asyncFactory}function je(t,e,n,o){var i,a=Object.keys(n).length>0,s=e?!!e.$stable:!a,c=e&&e.$key;if(e){if(e._normalized)return e._normalized;if(s&&o&&o!==r&&c===o.$key&&!a&&!o.$hasNormal)return o;for(var u in i={},e)e[u]&&"$"!==u[0]&&(i[u]=Te(t,n,u,e[u]))}else i={};for(var l in n)l in i||(i[l]=Ae(n,l));return e&&Object.isExtensible(e)&&(e._normalized=i),X(i,"$stable",s),X(i,"$key",c),X(i,"$hasNormal",a),i}function Te(t,e,n,r){var i=function(){var e=mt;_t(t);var n=arguments.length?r.apply(null,arguments):r({});n=n&&"object"===typeof n&&!o(n)?[n]:se(n);var i=n&&n[0];return _t(e),n&&(!i||1===n.length&&i.isComment&&!Se(i))?void 0:n};return r.proxy&&Object.defineProperty(e,n,{get:i,enumerable:!0,configurable:!0}),i}function Ae(t,e){return function(){return t[e]}}function Ee(t){var e=t.$options,n=e.setup;if(n){var r=t._setupContext=Ie(t);_t(t),St();var o=Ze(n,null,[t._props||Ht({}),r],t,"setup");if(jt(),_t(),l(o))e.render=o;else if(f(o))if(t._setupState=o,o.__sfc){var i=t._setupProxy={};for(var a in o)"__sfc"!==a&&Kt(i,o,a)}else for(var a in o)J(a)||Kt(t,o,a);else 0}}function Ie(t){return{get attrs(){if(!t._attrsProxy){var e=t._attrsProxy={};X(e,"_v_attr_proxy",!0),Pe(e,t.$attrs,r,t,"$attrs")}return t._attrsProxy},get listeners(){if(!t._listenersProxy){var e=t._listenersProxy={};Pe(e,t.$listeners,r,t,"$listeners")}return t._listenersProxy},get slots(){return De(t)},emit:P(t.$emit,t),expose:function(e){e&&Object.keys(e).forEach((function(n){return Kt(t,e,n)}))}}}function Pe(t,e,n,r,o){var i=!1;for(var a in e)a in t?e[a]!==n[a]&&(i=!0):(i=!0,Ne(t,a,r,o));for(var a in t)a in e||(i=!0,delete t[a]);return i}function Ne(t,e,n,r){Object.defineProperty(t,e,{enumerable:!0,configurable:!0,get:function(){return n[r][e]}})}function De(t){return t._slotsProxy||Me(t._slotsProxy={},t.$scopedSlots),t._slotsProxy}function Me(t,e){for(var n in e)t[n]=e[n];for(var n in t)n in e||delete t[n]}function Le(t){t._vnode=null,t._staticTrees=null;var e=t.$options,n=t.$vnode=e._parentVnode,o=n&&n.context;t.$slots=ke(e._renderChildren,o),t.$scopedSlots=n?je(t.$parent,n.data.scopedSlots,t.$slots):r,t._c=function(e,n,r,o){return qe(t,e,n,r,o,!1)},t.$createElement=function(e,n,r,o){return qe(t,e,n,r,o,!0)};var i=n&&n.data;Rt(t,"$attrs",i&&i.attrs||r,null,!0),Rt(t,"$listeners",e._parentListeners||r,null,!0)}var Fe=null;function Re(t){xe(t.prototype),t.prototype.$nextTick=function(t){return ln(t,this)},t.prototype._render=function(){var t=this,e=t.$options,n=e.render,r=e._parentVnode;r&&t._isMounted&&(t.$scopedSlots=je(t.$parent,r.data.scopedSlots,t.$slots,t.$scopedSlots),t._slotsProxy&&Me(t._slotsProxy,t.$scopedSlots)),t.$vnode=r;var i,a=mt,s=Fe;try{_t(t),Fe=t,i=n.call(t._renderProxy,t.$createElement)}catch(Ya){Xe(Ya,t,"render"),i=t._vnode}finally{Fe=s,_t(a)}return o(i)&&1===i.length&&(i=i[0]),i instanceof yt||(i=gt()),i.parent=r,i}}function Ue(t,e){return(t.__esModule||ht&&"Module"===t[Symbol.toStringTag])&&(t=t.default),f(t)?e.extend(t):t}function Be(t,e,n,r,o){var i=gt();return i.asyncFactory=t,i.asyncMeta={data:e,context:n,children:r,tag:o},i}function Ve(t,e){if(s(t.error)&&a(t.errorComp))return t.errorComp;if(a(t.resolved))return t.resolved;var n=Fe;if(n&&a(t.owners)&&-1===t.owners.indexOf(n)&&t.owners.push(n),s(t.loading)&&a(t.loadingComp))return t.loadingComp;if(n&&!a(t.owners)){var r=t.owners=[n],o=!0,c=null,u=null;n.$on("hook:destroyed",(function(){return C(r,n)}));var l=function(t){for(var e=0,n=r.length;e<n;e++)r[e].$forceUpdate();t&&(r.length=0,null!==c&&(clearTimeout(c),c=null),null!==u&&(clearTimeout(u),u=null))},d=V((function(n){t.resolved=Ue(n,e),o?r.length=0:l(!0)})),p=V((function(e){a(t.errorComp)&&(t.error=!0,l(!0))})),v=t(d,p);return f(v)&&(m(v)?i(t.resolved)&&v.then(d,p):m(v.component)&&(v.component.then(d,p),a(v.error)&&(t.errorComp=Ue(v.error,e)),a(v.loading)&&(t.loadingComp=Ue(v.loading,e),0===v.delay?t.loading=!0:c=setTimeout((function(){c=null,i(t.resolved)&&i(t.error)&&(t.loading=!0,l(!1))}),v.delay||200)),a(v.timeout)&&(u=setTimeout((function(){u=null,i(t.resolved)&&p(null)}),v.timeout)))),o=!1,t.loading?t.loadingComp:t.resolved}}function He(t){if(o(t))for(var e=0;e<t.length;e++){var n=t[e];if(a(n)&&(a(n.componentOptions)||Se(n)))return n}}var ze=1,We=2;function qe(t,e,n,r,i,a){return(o(n)||u(n))&&(i=r,r=n,n=void 0),s(a)&&(i=We),Ke(t,e,n,r,i)}function Ke(t,e,n,r,i){if(a(n)&&a(n.__ob__))return gt();if(a(n)&&a(n.is)&&(e=n.is),!e)return gt();var s,c;if(o(r)&&l(r[0])&&(n=n||{},n.scopedSlots={default:r[0]},r.length=0),i===We?r=se(r):i===ze&&(r=ae(r)),"string"===typeof e){var u=void 0;c=t.$vnode&&t.$vnode.ns||K.getTagNamespace(e),s=K.isReservedTag(e)?new yt(K.parsePlatformTagName(e),n,r,void 0,void 0,t):n&&n.pre||!a(u=kr(t.$options,"components",e))?new yt(e,n,r,void 0,void 0,t):cr(u,n,t,r,e)}else s=cr(e,n,t,r);return o(s)?s:a(s)?(a(c)&&Ge(s,c),a(n)&&Je(n),s):gt()}function Ge(t,e,n){if(t.ns=e,"foreignObject"===t.tag&&(e=void 0,n=!0),a(t.children))for(var r=0,o=t.children.length;r<o;r++){var c=t.children[r];a(c.tag)&&(i(c.ns)||s(n)&&"svg"!==c.tag)&&Ge(c,e,n)}}function Je(t){f(t.style)&&hn(t.style),f(t.class)&&hn(t.class)}function Xe(t,e,n){St();try{if(e){var r=e;while(r=r.$parent){var o=r.$options.errorCaptured;if(o)for(var i=0;i<o.length;i++)try{var a=!1===o[i].call(r,t,e,n);if(a)return}catch(Ya){Qe(Ya,r,"errorCaptured hook")}}}Qe(t,e,n)}finally{jt()}}function Ze(t,e,n,r,o){var i;try{i=n?t.apply(e,n):t.call(e),i&&!i._isVue&&m(i)&&!i._handled&&(i.catch((function(t){return Xe(t,r,o+" (Promise/async)")})),i._handled=!0)}catch(Ya){Xe(Ya,r,o)}return i}function Qe(t,e,n){if(K.errorHandler)try{return K.errorHandler.call(null,t,e,n)}catch(Ya){Ya!==t&&Ye(Ya,null,"config.errorHandler")}Ye(t,e,n)}function Ye(t,e,n){if(!tt||"undefined"===typeof console)throw t;console.error(t)}var tn,en=!1,nn=[],rn=!1;function on(){rn=!1;var t=nn.slice(0);nn.length=0;for(var e=0;e<t.length;e++)t[e]()}if("undefined"!==typeof Promise&&pt(Promise)){var an=Promise.resolve();tn=function(){an.then(on),it&&setTimeout(L)},en=!0}else if(nt||"undefined"===typeof MutationObserver||!pt(MutationObserver)&&"[object MutationObserverConstructor]"!==MutationObserver.toString())tn="undefined"!==typeof setImmediate&&pt(setImmediate)?function(){setImmediate(on)}:function(){setTimeout(on,0)};else{var sn=1,cn=new MutationObserver(on),un=document.createTextNode(String(sn));cn.observe(un,{characterData:!0}),tn=function(){sn=(sn+1)%2,un.data=String(sn)},en=!0}function ln(t,e){var n;if(nn.push((function(){if(t)try{t.call(e)}catch(Ya){Xe(Ya,e,"nextTick")}else n&&n(e)})),rn||(rn=!0,tn()),!t&&"undefined"!==typeof Promise)return new Promise((function(t){n=t}))}function fn(t){return function(e,n){if(void 0===n&&(n=mt),n)return dn(n,t,e)}}function dn(t,e,n){var r=t.$options;r[e]=_r(r[e],n)}fn("beforeMount"),fn("mounted"),fn("beforeUpdate"),fn("updated"),fn("beforeDestroy"),fn("destroyed"),fn("activated"),fn("deactivated"),fn("serverPrefetch"),fn("renderTracked"),fn("renderTriggered"),fn("errorCaptured");var pn="2.7.16";var vn=new vt;function hn(t){return mn(t,vn),vn.clear(),t}function mn(t,e){var n,r,i=o(t);if(!(!i&&!f(t)||t.__v_skip||Object.isFrozen(t)||t instanceof yt)){if(t.__ob__){var a=t.__ob__.dep.id;if(e.has(a))return;e.add(a)}if(i){n=t.length;while(n--)mn(t[n],e)}else if(qt(t))mn(t.value,e);else{r=Object.keys(t),n=r.length;while(n--)mn(t[r[n]],e)}}}var _n,yn=0,gn=function(){function t(t,e,n,r,o){Zt(this,Jt&&!Jt._vm?Jt:t?t._scope:void 0),(this.vm=t)&&o&&(t._watcher=this),r?(this.deep=!!r.deep,this.user=!!r.user,this.lazy=!!r.lazy,this.sync=!!r.sync,this.before=r.before):this.deep=this.user=this.lazy=this.sync=!1,this.cb=n,this.id=++yn,this.active=!0,this.post=!1,this.dirty=this.lazy,this.deps=[],this.newDeps=[],this.depIds=new vt,this.newDepIds=new vt,this.expression="",l(e)?this.getter=e:(this.getter=Q(e),this.getter||(this.getter=L)),this.value=this.lazy?void 0:this.get()}return t.prototype.get=function(){var t;St(this);var e=this.vm;try{t=this.getter.call(e,e)}catch(Ya){if(!this.user)throw Ya;Xe(Ya,e,'getter for watcher "'.concat(this.expression,'"'))}finally{this.deep&&hn(t),jt(),this.cleanupDeps()}return t},t.prototype.addDep=function(t){var e=t.id;this.newDepIds.has(e)||(this.newDepIds.add(e),this.newDeps.push(t),this.depIds.has(e)||t.addSub(this))},t.prototype.cleanupDeps=function(){var t=this.deps.length;while(t--){var e=this.deps[t];this.newDepIds.has(e.id)||e.removeSub(this)}var n=this.depIds;this.depIds=this.newDepIds,this.newDepIds=n,this.newDepIds.clear(),n=this.deps,this.deps=this.newDeps,this.newDeps=n,this.newDeps.length=0},t.prototype.update=function(){this.lazy?this.dirty=!0:this.sync?this.run():Zn(this)},t.prototype.run=function(){if(this.active){var t=this.get();if(t!==this.value||f(t)||this.deep){var e=this.value;if(this.value=t,this.user){var n='callback for watcher "'.concat(this.expression,'"');Ze(this.cb,this.vm,[t,e],this.vm,n)}else this.cb.call(this.vm,t,e)}}},t.prototype.evaluate=function(){this.value=this.get(),this.dirty=!1},t.prototype.depend=function(){var t=this.deps.length;while(t--)this.deps[t].depend()},t.prototype.teardown=function(){if(this.vm&&!this.vm._isBeingDestroyed&&C(this.vm._scope.effects,this),this.active){var t=this.deps.length;while(t--)this.deps[t].removeSub(this);this.active=!1,this.onStop&&this.onStop()}},t}();function bn(t){t._events=Object.create(null),t._hasHookEvent=!1;var e=t.$options._parentListeners;e&&xn(t,e)}function wn(t,e){_n.$on(t,e)}function Cn(t,e){_n.$off(t,e)}function $n(t,e){var n=_n;return function r(){var o=e.apply(null,arguments);null!==o&&n.$off(t,r)}}function xn(t,e,n){_n=t,ne(e,n||{},wn,Cn,$n,t),_n=void 0}function kn(t){var e=/^hook:/;t.prototype.$on=function(t,n){var r=this;if(o(t))for(var i=0,a=t.length;i<a;i++)r.$on(t[i],n);else(r._events[t]||(r._events[t]=[])).push(n),e.test(t)&&(r._hasHookEvent=!0);return r},t.prototype.$once=function(t,e){var n=this;function r(){n.$off(t,r),e.apply(n,arguments)}return r.fn=e,n.$on(t,r),n},t.prototype.$off=function(t,e){var n=this;if(!arguments.length)return n._events=Object.create(null),n;if(o(t)){for(var r=0,i=t.length;r<i;r++)n.$off(t[r],e);return n}var a,s=n._events[t];if(!s)return n;if(!e)return n._events[t]=null,n;var c=s.length;while(c--)if(a=s[c],a===e||a.fn===e){s.splice(c,1);break}return n},t.prototype.$emit=function(t){var e=this,n=e._events[t];if(n){n=n.length>1?N(n):n;for(var r=N(arguments,1),o='event handler for "'.concat(t,'"'),i=0,a=n.length;i<a;i++)Ze(n[i],e,r,e,o)}return e}}var On=null;function Sn(t){var e=On;return On=t,function(){On=e}}function jn(t){var e=t.$options,n=e.parent;if(n&&!e.abstract){while(n.$options.abstract&&n.$parent)n=n.$parent;n.$children.push(t)}t.$parent=n,t.$root=n?n.$root:t,t.$children=[],t.$refs={},t._provided=n?n._provided:Object.create(null),t._watcher=null,t._inactive=null,t._directInactive=!1,t._isMounted=!1,t._isDestroyed=!1,t._isBeingDestroyed=!1}function Tn(t){t.prototype._update=function(t,e){var n=this,r=n.$el,o=n._vnode,i=Sn(n);n._vnode=t,n.$el=o?n.__patch__(o,t):n.__patch__(n.$el,t,e,!1),i(),r&&(r.__vue__=null),n.$el&&(n.$el.__vue__=n);var a=n;while(a&&a.$vnode&&a.$parent&&a.$vnode===a.$parent._vnode)a.$parent.$el=a.$el,a=a.$parent},t.prototype.$forceUpdate=function(){var t=this;t._watcher&&t._watcher.update()},t.prototype.$destroy=function(){var t=this;if(!t._isBeingDestroyed){Dn(t,"beforeDestroy"),t._isBeingDestroyed=!0;var e=t.$parent;!e||e._isBeingDestroyed||t.$options.abstract||C(e.$children,t),t._scope.stop(),t._data.__ob__&&t._data.__ob__.vmCount--,t._isDestroyed=!0,t.__patch__(t._vnode,null),Dn(t,"destroyed"),t.$off(),t.$el&&(t.$el.__vue__=null),t.$vnode&&(t.$vnode.parent=null)}}}function An(t,e,n){var r;t.$el=e,t.$options.render||(t.$options.render=gt),Dn(t,"beforeMount"),r=function(){t._update(t._render(),n)};var o={before:function(){t._isMounted&&!t._isDestroyed&&Dn(t,"beforeUpdate")}};new gn(t,r,L,o,!0),n=!1;var i=t._preWatchers;if(i)for(var a=0;a<i.length;a++)i[a].run();return null==t.$vnode&&(t._isMounted=!0,Dn(t,"mounted")),t}function En(t,e,n,o,i){var a=o.data.scopedSlots,s=t.$scopedSlots,c=!!(a&&!a.$stable||s!==r&&!s.$stable||a&&t.$scopedSlots.$key!==a.$key||!a&&t.$scopedSlots.$key),u=!!(i||t.$options._renderChildren||c),l=t.$vnode;t.$options._parentVnode=o,t.$vnode=o,t._vnode&&(t._vnode.parent=o),t.$options._renderChildren=i;var f=o.data.attrs||r;t._attrsProxy&&Pe(t._attrsProxy,f,l.data&&l.data.attrs||r,t,"$attrs")&&(u=!0),t.$attrs=f,n=n||r;var d=t.$options._parentListeners;if(t._listenersProxy&&Pe(t._listenersProxy,n,d||r,t,"$listeners"),t.$listeners=t.$options._parentListeners=n,xn(t,n,d),e&&t.$options.props){Dt(!1);for(var p=t._props,v=t.$options._propKeys||[],h=0;h<v.length;h++){var m=v[h],_=t.$options.props;p[m]=Or(m,_,e,t)}Dt(!0),t.$options.propsData=e}u&&(t.$slots=ke(i,o.context),t.$forceUpdate())}function In(t){while(t&&(t=t.$parent))if(t._inactive)return!0;return!1}function Pn(t,e){if(e){if(t._directInactive=!1,In(t))return}else if(t._directInactive)return;if(t._inactive||null===t._inactive){t._inactive=!1;for(var n=0;n<t.$children.length;n++)Pn(t.$children[n]);Dn(t,"activated")}}function Nn(t,e){if((!e||(t._directInactive=!0,!In(t)))&&!t._inactive){t._inactive=!0;for(var n=0;n<t.$children.length;n++)Nn(t.$children[n]);Dn(t,"deactivated")}}function Dn(t,e,n,r){void 0===r&&(r=!0),St();var o=mt,i=Qt();r&&_t(t);var a=t.$options[e],s="".concat(e," hook");if(a)for(var c=0,u=a.length;c<u;c++)Ze(a[c],t,n||null,t,s);t._hasHookEvent&&t.$emit("hook:"+e),r&&(_t(o),i&&i.on()),jt()}var Mn=[],Ln=[],Fn={},Rn=!1,Un=!1,Bn=0;function Vn(){Bn=Mn.length=Ln.length=0,Fn={},Rn=Un=!1}var Hn=0,zn=Date.now;if(tt&&!nt){var Wn=window.performance;Wn&&"function"===typeof Wn.now&&zn()>document.createEvent("Event").timeStamp&&(zn=function(){return Wn.now()})}var qn=function(t,e){if(t.post){if(!e.post)return 1}else if(e.post)return-1;return t.id-e.id};function Kn(){var t,e;for(Hn=zn(),Un=!0,Mn.sort(qn),Bn=0;Bn<Mn.length;Bn++)t=Mn[Bn],t.before&&t.before(),e=t.id,Fn[e]=null,t.run();var n=Ln.slice(),r=Mn.slice();Vn(),Xn(n),Gn(r),xt(),dt&&K.devtools&&dt.emit("flush")}function Gn(t){var e=t.length;while(e--){var n=t[e],r=n.vm;r&&r._watcher===n&&r._isMounted&&!r._isDestroyed&&Dn(r,"updated")}}function Jn(t){t._inactive=!1,Ln.push(t)}function Xn(t){for(var e=0;e<t.length;e++)t[e]._inactive=!0,Pn(t[e],!0)}function Zn(t){var e=t.id;if(null==Fn[e]&&(t!==kt.target||!t.noRecurse)){if(Fn[e]=!0,Un){var n=Mn.length-1;while(n>Bn&&Mn[n].id>t.id)n--;Mn.splice(n+1,0,t)}else Mn.push(t);Rn||(Rn=!0,ln(Kn))}}function Qn(t){var e=t.$options.provide;if(e){var n=l(e)?e.call(t):e;if(!f(n))return;for(var r=Yt(t),o=ht?Reflect.ownKeys(n):Object.keys(n),i=0;i<o.length;i++){var a=o[i];Object.defineProperty(r,a,Object.getOwnPropertyDescriptor(n,a))}}}function Yn(t){var e=tr(t.$options.inject,t);e&&(Dt(!1),Object.keys(e).forEach((function(n){Rt(t,n,e[n])})),Dt(!0))}function tr(t,e){if(t){for(var n=Object.create(null),r=ht?Reflect.ownKeys(t):Object.keys(t),o=0;o<r.length;o++){var i=r[o];if("__ob__"!==i){var a=t[i].from;if(a in e._provided)n[i]=e._provided[a];else if("default"in t[i]){var s=t[i].default;n[i]=l(s)?s.call(e):s}else 0}}return n}}function er(t,e,n,i,a){var c,u=this,l=a.options;x(i,"_uid")?(c=Object.create(i),c._original=i):(c=i,i=i._original);var f=s(l._compiled),d=!f;this.data=t,this.props=e,this.children=n,this.parent=i,this.listeners=t.on||r,this.injections=tr(l.inject,i),this.slots=function(){return u.$slots||je(i,t.scopedSlots,u.$slots=ke(n,i)),u.$slots},Object.defineProperty(this,"scopedSlots",{enumerable:!0,get:function(){return je(i,t.scopedSlots,this.slots())}}),f&&(this.$options=l,this.$slots=this.slots(),this.$scopedSlots=je(i,t.scopedSlots,this.$slots)),l._scopeId?this._c=function(t,e,n,r){var a=qe(c,t,e,n,r,d);return a&&!o(a)&&(a.fnScopeId=l._scopeId,a.fnContext=i),a}:this._c=function(t,e,n,r){return qe(c,t,e,n,r,d)}}function nr(t,e,n,i,s){var c=t.options,u={},l=c.props;if(a(l))for(var f in l)u[f]=Or(f,l,e||r);else a(n.attrs)&&or(u,n.attrs),a(n.props)&&or(u,n.props);var d=new er(n,u,s,i,t),p=c.render.call(null,d._c,d);if(p instanceof yt)return rr(p,n,d.parent,c,d);if(o(p)){for(var v=se(p)||[],h=new Array(v.length),m=0;m<v.length;m++)h[m]=rr(v[m],n,d.parent,c,d);return h}}function rr(t,e,n,r,o){var i=wt(t);return i.fnContext=n,i.fnOptions=r,e.slot&&((i.data||(i.data={})).slot=e.slot),i}function or(t,e){for(var n in e)t[S(n)]=e[n]}function ir(t){return t.name||t.__name||t._componentTag}xe(er.prototype);var ar={init:function(t,e){if(t.componentInstance&&!t.componentInstance._isDestroyed&&t.data.keepAlive){var n=t;ar.prepatch(n,n)}else{var r=t.componentInstance=ur(t,On);r.$mount(e?t.elm:void 0,e)}},prepatch:function(t,e){var n=e.componentOptions,r=e.componentInstance=t.componentInstance;En(r,n.propsData,n.listeners,e,n.children)},insert:function(t){var e=t.context,n=t.componentInstance;n._isMounted||(n._isMounted=!0,Dn(n,"mounted")),t.data.keepAlive&&(e._isMounted?Jn(n):Pn(n,!0))},destroy:function(t){var e=t.componentInstance;e._isDestroyed||(t.data.keepAlive?Nn(e,!0):e.$destroy())}},sr=Object.keys(ar);function cr(t,e,n,r,o){if(!i(t)){var c=n.$options._base;if(f(t)&&(t=c.extend(t)),"function"===typeof t){var u;if(i(t.cid)&&(u=t,t=Ve(u,c),void 0===t))return Be(u,e,n,r,o);e=e||{},Xr(t),a(e.model)&&dr(t.options,e);var l=oe(e,t,o);if(s(t.options.functional))return nr(t,l,e,n,r);var d=e.on;if(e.on=e.nativeOn,s(t.options.abstract)){var p=e.slot;e={},p&&(e.slot=p)}lr(e);var v=ir(t.options)||o,h=new yt("vue-component-".concat(t.cid).concat(v?"-".concat(v):""),e,void 0,void 0,void 0,n,{Ctor:t,propsData:l,listeners:d,tag:o,children:r},u);return h}}}function ur(t,e){var n={_isComponent:!0,_parentVnode:t,parent:e},r=t.data.inlineTemplate;return a(r)&&(n.render=r.render,n.staticRenderFns=r.staticRenderFns),new t.componentOptions.Ctor(n)}function lr(t){for(var e=t.hook||(t.hook={}),n=0;n<sr.length;n++){var r=sr[n],o=e[r],i=ar[r];o===i||o&&o._merged||(e[r]=o?fr(i,o):i)}}function fr(t,e){var n=function(n,r){t(n,r),e(n,r)};return n._merged=!0,n}function dr(t,e){var n=t.model&&t.model.prop||"value",r=t.model&&t.model.event||"input";(e.attrs||(e.attrs={}))[n]=e.model.value;var i=e.on||(e.on={}),s=i[r],c=e.model.callback;a(s)?(o(s)?-1===s.indexOf(c):s!==c)&&(i[r]=[c].concat(s)):i[r]=c}var pr=L,vr=K.optionMergeStrategies;function hr(t,e,n){if(void 0===n&&(n=!0),!e)return t;for(var r,o,i,a=ht?Reflect.ownKeys(e):Object.keys(e),s=0;s<a.length;s++)r=a[s],"__ob__"!==r&&(o=t[r],i=e[r],n&&x(t,r)?o!==i&&p(o)&&p(i)&&hr(o,i):Ut(t,r,i));return t}function mr(t,e,n){return n?function(){var r=l(e)?e.call(n,n):e,o=l(t)?t.call(n,n):t;return r?hr(r,o):o}:e?t?function(){return hr(l(e)?e.call(this,this):e,l(t)?t.call(this,this):t)}:e:t}function _r(t,e){var n=e?t?t.concat(e):o(e)?e:[e]:t;return n?yr(n):n}function yr(t){for(var e=[],n=0;n<t.length;n++)-1===e.indexOf(t[n])&&e.push(t[n]);return e}function gr(t,e,n,r){var o=Object.create(t||null);return e?D(o,e):o}vr.data=function(t,e,n){return n?mr(t,e,n):e&&"function"!==typeof e?t:mr(t,e)},q.forEach((function(t){vr[t]=_r})),W.forEach((function(t){vr[t+"s"]=gr})),vr.watch=function(t,e,n,r){if(t===ct&&(t=void 0),e===ct&&(e=void 0),!e)return Object.create(t||null);if(!t)return e;var i={};for(var a in D(i,t),e){var s=i[a],c=e[a];s&&!o(s)&&(s=[s]),i[a]=s?s.concat(c):o(c)?c:[c]}return i},vr.props=vr.methods=vr.inject=vr.computed=function(t,e,n,r){if(!t)return e;var o=Object.create(null);return D(o,t),e&&D(o,e),o},vr.provide=function(t,e){return t?function(){var n=Object.create(null);return hr(n,l(t)?t.call(this):t),e&&hr(n,l(e)?e.call(this):e,!1),n}:e};var br=function(t,e){return void 0===e?t:e};function wr(t,e){var n=t.props;if(n){var r,i,a,s={};if(o(n)){r=n.length;while(r--)i=n[r],"string"===typeof i&&(a=S(i),s[a]={type:null})}else if(p(n))for(var c in n)i=n[c],a=S(c),s[a]=p(i)?i:{type:i};else 0;t.props=s}}function Cr(t,e){var n=t.inject;if(n){var r=t.inject={};if(o(n))for(var i=0;i<n.length;i++)r[n[i]]={from:n[i]};else if(p(n))for(var a in n){var s=n[a];r[a]=p(s)?D({from:a},s):{from:s}}else 0}}function $r(t){var e=t.directives;if(e)for(var n in e){var r=e[n];l(r)&&(e[n]={bind:r,update:r})}}function xr(t,e,n){if(l(e)&&(e=e.options),wr(e,n),Cr(e,n),$r(e),!e._base&&(e.extends&&(t=xr(t,e.extends,n)),e.mixins))for(var r=0,o=e.mixins.length;r<o;r++)t=xr(t,e.mixins[r],n);var i,a={};for(i in t)s(i);for(i in e)x(t,i)||s(i);function s(r){var o=vr[r]||br;a[r]=o(t[r],e[r],n,r)}return a}function kr(t,e,n,r){if("string"===typeof n){var o=t[e];if(x(o,n))return o[n];var i=S(n);if(x(o,i))return o[i];var a=j(i);if(x(o,a))return o[a];var s=o[n]||o[i]||o[a];return s}}function Or(t,e,n,r){var o=e[t],i=!x(n,t),a=n[t],s=Er(Boolean,o.type);if(s>-1)if(i&&!x(o,"default"))a=!1;else if(""===a||a===A(t)){var c=Er(String,o.type);(c<0||s<c)&&(a=!0)}if(void 0===a){a=Sr(r,o,t);var u=Nt;Dt(!0),Ft(a),Dt(u)}return a}function Sr(t,e,n){if(x(e,"default")){var r=e.default;return t&&t.$options.propsData&&void 0===t.$options.propsData[n]&&void 0!==t._props[n]?t._props[n]:l(r)&&"Function"!==Tr(e.type)?r.call(t):r}}var jr=/^\s*function (\w+)/;function Tr(t){var e=t&&t.toString().match(jr);return e?e[1]:""}function Ar(t,e){return Tr(t)===Tr(e)}function Er(t,e){if(!o(e))return Ar(e,t)?0:-1;for(var n=0,r=e.length;n<r;n++)if(Ar(e[n],t))return n;return-1}var Ir={enumerable:!0,configurable:!0,get:L,set:L};function Pr(t,e,n){Ir.get=function(){return this[e][n]},Ir.set=function(t){this[e][n]=t},Object.defineProperty(t,n,Ir)}function Nr(t){var e=t.$options;if(e.props&&Dr(t,e.props),Ee(t),e.methods&&Hr(t,e.methods),e.data)Mr(t);else{var n=Ft(t._data={});n&&n.vmCount++}e.computed&&Rr(t,e.computed),e.watch&&e.watch!==ct&&zr(t,e.watch)}function Dr(t,e){var n=t.$options.propsData||{},r=t._props=Ht({}),o=t.$options._propKeys=[],i=!t.$parent;i||Dt(!1);var a=function(i){o.push(i);var a=Or(i,e,n,t);Rt(r,i,a,void 0,!0),i in t||Pr(t,"_props",i)};for(var s in e)a(s);Dt(!0)}function Mr(t){var e=t.$options.data;e=t._data=l(e)?Lr(e,t):e||{},p(e)||(e={});var n=Object.keys(e),r=t.$options.props,o=(t.$options.methods,n.length);while(o--){var i=n[o];0,r&&x(r,i)||J(i)||Pr(t,"_data",i)}var a=Ft(e);a&&a.vmCount++}function Lr(t,e){St();try{return t.call(e,e)}catch(Ya){return Xe(Ya,e,"data()"),{}}finally{jt()}}var Fr={lazy:!0};function Rr(t,e){var n=t._computedWatchers=Object.create(null),r=ft();for(var o in e){var i=e[o],a=l(i)?i:i.get;0,r||(n[o]=new gn(t,a||L,L,Fr)),o in t||Ur(t,o,i)}}function Ur(t,e,n){var r=!ft();l(n)?(Ir.get=r?Br(e):Vr(n),Ir.set=L):(Ir.get=n.get?r&&!1!==n.cache?Br(e):Vr(n.get):L,Ir.set=n.set||L),Object.defineProperty(t,e,Ir)}function Br(t){return function(){var e=this._computedWatchers&&this._computedWatchers[t];if(e)return e.dirty&&e.evaluate(),kt.target&&e.depend(),e.value}}function Vr(t){return function(){return t.call(this,this)}}function Hr(t,e){t.$options.props;for(var n in e)t[n]="function"!==typeof e[n]?L:P(e[n],t)}function zr(t,e){for(var n in e){var r=e[n];if(o(r))for(var i=0;i<r.length;i++)Wr(t,n,r[i]);else Wr(t,n,r)}}function Wr(t,e,n,r){return p(n)&&(r=n,n=n.handler),"string"===typeof n&&(n=t[n]),t.$watch(e,n,r)}function qr(t){var e={get:function(){return this._data}},n={get:function(){return this._props}};Object.defineProperty(t.prototype,"$data",e),Object.defineProperty(t.prototype,"$props",n),t.prototype.$set=Ut,t.prototype.$delete=Bt,t.prototype.$watch=function(t,e,n){var r=this;if(p(e))return Wr(r,t,e,n);n=n||{},n.user=!0;var o=new gn(r,t,e,n);if(n.immediate){var i='callback for immediate watcher "'.concat(o.expression,'"');St(),Ze(e,r,[o.value],r,i),jt()}return function(){o.teardown()}}}var Kr=0;function Gr(t){t.prototype._init=function(t){var e=this;e._uid=Kr++,e._isVue=!0,e.__v_skip=!0,e._scope=new Xt(!0),e._scope.parent=void 0,e._scope._vm=!0,t&&t._isComponent?Jr(e,t):e.$options=xr(Xr(e.constructor),t||{},e),e._renderProxy=e,e._self=e,jn(e),bn(e),Le(e),Dn(e,"beforeCreate",void 0,!1),Yn(e),Nr(e),Qn(e),Dn(e,"created"),e.$options.el&&e.$mount(e.$options.el)}}function Jr(t,e){var n=t.$options=Object.create(t.constructor.options),r=e._parentVnode;n.parent=e.parent,n._parentVnode=r;var o=r.componentOptions;n.propsData=o.propsData,n._parentListeners=o.listeners,n._renderChildren=o.children,n._componentTag=o.tag,e.render&&(n.render=e.render,n.staticRenderFns=e.staticRenderFns)}function Xr(t){var e=t.options;if(t.super){var n=Xr(t.super),r=t.superOptions;if(n!==r){t.superOptions=n;var o=Zr(t);o&&D(t.extendOptions,o),e=t.options=xr(n,t.extendOptions),e.name&&(e.components[e.name]=t)}}return e}function Zr(t){var e,n=t.options,r=t.sealedOptions;for(var o in n)n[o]!==r[o]&&(e||(e={}),e[o]=n[o]);return e}function Qr(t){this._init(t)}function Yr(t){t.use=function(t){var e=this._installedPlugins||(this._installedPlugins=[]);if(e.indexOf(t)>-1)return this;var n=N(arguments,1);return n.unshift(this),l(t.install)?t.install.apply(t,n):l(t)&&t.apply(null,n),e.push(t),this}}function to(t){t.mixin=function(t){return this.options=xr(this.options,t),this}}function eo(t){t.cid=0;var e=1;t.extend=function(t){t=t||{};var n=this,r=n.cid,o=t._Ctor||(t._Ctor={});if(o[r])return o[r];var i=ir(t)||ir(n.options);var a=function(t){this._init(t)};return a.prototype=Object.create(n.prototype),a.prototype.constructor=a,a.cid=e++,a.options=xr(n.options,t),a["super"]=n,a.options.props&&no(a),a.options.computed&&ro(a),a.extend=n.extend,a.mixin=n.mixin,a.use=n.use,W.forEach((function(t){a[t]=n[t]})),i&&(a.options.components[i]=a),a.superOptions=n.options,a.extendOptions=t,a.sealedOptions=D({},a.options),o[r]=a,a}}function no(t){var e=t.options.props;for(var n in e)Pr(t.prototype,"_props",n)}function ro(t){var e=t.options.computed;for(var n in e)Ur(t.prototype,n,e[n])}function oo(t){W.forEach((function(e){t[e]=function(t,n){return n?("component"===e&&p(n)&&(n.name=n.name||t,n=this.options._base.extend(n)),"directive"===e&&l(n)&&(n={bind:n,update:n}),this.options[e+"s"][t]=n,n):this.options[e+"s"][t]}}))}function io(t){return t&&(ir(t.Ctor.options)||t.tag)}function ao(t,e){return o(t)?t.indexOf(e)>-1:"string"===typeof t?t.split(",").indexOf(e)>-1:!!v(t)&&t.test(e)}function so(t,e){var n=t.cache,r=t.keys,o=t._vnode,i=t.$vnode;for(var a in n){var s=n[a];if(s){var c=s.name;c&&!e(c)&&co(n,a,r,o)}}i.componentOptions.children=void 0}function co(t,e,n,r){var o=t[e];!o||r&&o.tag===r.tag||o.componentInstance.$destroy(),t[e]=null,C(n,e)}Gr(Qr),qr(Qr),kn(Qr),Tn(Qr),Re(Qr);var uo=[String,RegExp,Array],lo={name:"keep-alive",abstract:!0,props:{include:uo,exclude:uo,max:[String,Number]},methods:{cacheVNode:function(){var t=this,e=t.cache,n=t.keys,r=t.vnodeToCache,o=t.keyToCache;if(r){var i=r.tag,a=r.componentInstance,s=r.componentOptions;e[o]={name:io(s),tag:i,componentInstance:a},n.push(o),this.max&&n.length>parseInt(this.max)&&co(e,n[0],n,this._vnode),this.vnodeToCache=null}}},created:function(){this.cache=Object.create(null),this.keys=[]},destroyed:function(){for(var t in this.cache)co(this.cache,t,this.keys)},mounted:function(){var t=this;this.cacheVNode(),this.$watch("include",(function(e){so(t,(function(t){return ao(e,t)}))})),this.$watch("exclude",(function(e){so(t,(function(t){return!ao(e,t)}))}))},updated:function(){this.cacheVNode()},render:function(){var t=this.$slots.default,e=He(t),n=e&&e.componentOptions;if(n){var r=io(n),o=this,i=o.include,a=o.exclude;if(i&&(!r||!ao(i,r))||a&&r&&ao(a,r))return e;var s=this,c=s.cache,u=s.keys,l=null==e.key?n.Ctor.cid+(n.tag?"::".concat(n.tag):""):e.key;c[l]?(e.componentInstance=c[l].componentInstance,C(u,l),u.push(l)):(this.vnodeToCache=e,this.keyToCache=l),e.data.keepAlive=!0}return e||t&&t[0]}},fo={KeepAlive:lo};function po(t){var e={get:function(){return K}};Object.defineProperty(t,"config",e),t.util={warn:pr,extend:D,mergeOptions:xr,defineReactive:Rt},t.set=Ut,t.delete=Bt,t.nextTick=ln,t.observable=function(t){return Ft(t),t},t.options=Object.create(null),W.forEach((function(e){t.options[e+"s"]=Object.create(null)})),t.options._base=t,D(t.options.components,fo),Yr(t),to(t),eo(t),oo(t)}po(Qr),Object.defineProperty(Qr.prototype,"$isServer",{get:ft}),Object.defineProperty(Qr.prototype,"$ssrContext",{get:function(){return this.$vnode&&this.$vnode.ssrContext}}),Object.defineProperty(Qr,"FunctionalRenderContext",{value:er}),Qr.version=pn;var vo=b("style,class"),ho=b("input,textarea,option,select,progress"),mo=function(t,e,n){return"value"===n&&ho(t)&&"button"!==e||"selected"===n&&"option"===t||"checked"===n&&"input"===t||"muted"===n&&"video"===t},_o=b("contenteditable,draggable,spellcheck"),yo=b("events,caret,typing,plaintext-only"),go=function(t,e){return xo(e)||"false"===e?"false":"contenteditable"===t&&yo(e)?e:"true"},bo=b("allowfullscreen,async,autofocus,autoplay,checked,compact,controls,declare,default,defaultchecked,defaultmuted,defaultselected,defer,disabled,enabled,formnovalidate,hidden,indeterminate,inert,ismap,itemscope,loop,multiple,muted,nohref,noresize,noshade,novalidate,nowrap,open,pauseonexit,readonly,required,reversed,scoped,seamless,selected,sortable,truespeed,typemustmatch,visible"),wo="http://www.w3.org/1999/xlink",Co=function(t){return":"===t.charAt(5)&&"xlink"===t.slice(0,5)},$o=function(t){return Co(t)?t.slice(6,t.length):""},xo=function(t){return null==t||!1===t};function ko(t){var e=t.data,n=t,r=t;while(a(r.componentInstance))r=r.componentInstance._vnode,r&&r.data&&(e=Oo(r.data,e));while(a(n=n.parent))n&&n.data&&(e=Oo(e,n.data));return So(e.staticClass,e.class)}function Oo(t,e){return{staticClass:jo(t.staticClass,e.staticClass),class:a(t.class)?[t.class,e.class]:e.class}}function So(t,e){return a(t)||a(e)?jo(t,To(e)):""}function jo(t,e){return t?e?t+" "+e:t:e||""}function To(t){return Array.isArray(t)?Ao(t):f(t)?Eo(t):"string"===typeof t?t:""}function Ao(t){for(var e,n="",r=0,o=t.length;r<o;r++)a(e=To(t[r]))&&""!==e&&(n&&(n+=" "),n+=e);return n}function Eo(t){var e="";for(var n in t)t[n]&&(e&&(e+=" "),e+=n);return e}var Io={svg:"http://www.w3.org/2000/svg",math:"http://www.w3.org/1998/Math/MathML"},Po=b("html,body,base,head,link,meta,style,title,address,article,aside,footer,header,h1,h2,h3,h4,h5,h6,hgroup,nav,section,div,dd,dl,dt,figcaption,figure,picture,hr,img,li,main,ol,p,pre,ul,a,b,abbr,bdi,bdo,br,cite,code,data,dfn,em,i,kbd,mark,q,rp,rt,rtc,ruby,s,samp,small,span,strong,sub,sup,time,u,var,wbr,area,audio,map,track,video,embed,object,param,source,canvas,script,noscript,del,ins,caption,col,colgroup,table,thead,tbody,td,th,tr,button,datalist,fieldset,form,input,label,legend,meter,optgroup,option,output,progress,select,textarea,details,dialog,menu,menuitem,summary,content,element,shadow,template,blockquote,iframe,tfoot"),No=b("svg,animate,circle,clippath,cursor,defs,desc,ellipse,filter,font-face,foreignobject,g,glyph,image,line,marker,mask,missing-glyph,path,pattern,polygon,polyline,rect,switch,symbol,text,textpath,tspan,use,view",!0),Do=function(t){return Po(t)||No(t)};function Mo(t){return No(t)?"svg":"math"===t?"math":void 0}var Lo=Object.create(null);function Fo(t){if(!tt)return!0;if(Do(t))return!1;if(t=t.toLowerCase(),null!=Lo[t])return Lo[t];var e=document.createElement(t);return t.indexOf("-")>-1?Lo[t]=e.constructor===window.HTMLUnknownElement||e.constructor===window.HTMLElement:Lo[t]=/HTMLUnknownElement/.test(e.toString())}var Ro=b("text,number,password,search,email,tel,url");function Uo(t){if("string"===typeof t){var e=document.querySelector(t);return e||document.createElement("div")}return t}function Bo(t,e){var n=document.createElement(t);return"select"!==t||e.data&&e.data.attrs&&void 0!==e.data.attrs.multiple&&n.setAttribute("multiple","multiple"),n}function Vo(t,e){return document.createElementNS(Io[t],e)}function Ho(t){return document.createTextNode(t)}function zo(t){return document.createComment(t)}function Wo(t,e,n){t.insertBefore(e,n)}function qo(t,e){t.removeChild(e)}function Ko(t,e){t.appendChild(e)}function Go(t){return t.parentNode}function Jo(t){return t.nextSibling}function Xo(t){return t.tagName}function Zo(t,e){t.textContent=e}function Qo(t,e){t.setAttribute(e,"")}var Yo=Object.freeze({__proto__:null,createElement:Bo,createElementNS:Vo,createTextNode:Ho,createComment:zo,insertBefore:Wo,removeChild:qo,appendChild:Ko,parentNode:Go,nextSibling:Jo,tagName:Xo,setTextContent:Zo,setStyleScope:Qo}),ti={create:function(t,e){ei(e)},update:function(t,e){t.data.ref!==e.data.ref&&(ei(t,!0),ei(e))},destroy:function(t){ei(t,!0)}};function ei(t,e){var n=t.data.ref;if(a(n)){var r=t.context,i=t.componentInstance||t.elm,s=e?null:i,c=e?void 0:i;if(l(n))Ze(n,r,[s],r,"template ref function");else{var u=t.data.refInFor,f="string"===typeof n||"number"===typeof n,d=qt(n),p=r.$refs;if(f||d)if(u){var v=f?p[n]:n.value;e?o(v)&&C(v,i):o(v)?v.includes(i)||v.push(i):f?(p[n]=[i],ni(r,n,p[n])):n.value=[i]}else if(f){if(e&&p[n]!==i)return;p[n]=c,ni(r,n,s)}else if(d){if(e&&n.value!==i)return;n.value=s}else 0}}}function ni(t,e,n){var r=t._setupState;r&&x(r,e)&&(qt(r[e])?r[e].value=n:r[e]=n)}var ri=new yt("",{},[]),oi=["create","activate","update","remove","destroy"];function ii(t,e){return t.key===e.key&&t.asyncFactory===e.asyncFactory&&(t.tag===e.tag&&t.isComment===e.isComment&&a(t.data)===a(e.data)&&ai(t,e)||s(t.isAsyncPlaceholder)&&i(e.asyncFactory.error))}function ai(t,e){if("input"!==t.tag)return!0;var n,r=a(n=t.data)&&a(n=n.attrs)&&n.type,o=a(n=e.data)&&a(n=n.attrs)&&n.type;return r===o||Ro(r)&&Ro(o)}function si(t,e,n){var r,o,i={};for(r=e;r<=n;++r)o=t[r].key,a(o)&&(i[o]=r);return i}function ci(t){var e,n,r={},c=t.modules,l=t.nodeOps;for(e=0;e<oi.length;++e)for(r[oi[e]]=[],n=0;n<c.length;++n)a(c[n][oi[e]])&&r[oi[e]].push(c[n][oi[e]]);function f(t){return new yt(l.tagName(t).toLowerCase(),{},[],void 0,t)}function d(t,e){function n(){0===--n.listeners&&p(t)}return n.listeners=e,n}function p(t){var e=l.parentNode(t);a(e)&&l.removeChild(e,t)}function v(t,e,n,r,o,i,c){if(a(t.elm)&&a(i)&&(t=i[c]=wt(t)),t.isRootInsert=!o,!h(t,e,n,r)){var u=t.data,f=t.children,d=t.tag;a(d)?(t.elm=t.ns?l.createElementNS(t.ns,d):l.createElement(d,t),$(t),g(t,f,e),a(u)&&C(t,e),y(n,t.elm,r)):s(t.isComment)?(t.elm=l.createComment(t.text),y(n,t.elm,r)):(t.elm=l.createTextNode(t.text),y(n,t.elm,r))}}function h(t,e,n,r){var o=t.data;if(a(o)){var i=a(t.componentInstance)&&o.keepAlive;if(a(o=o.hook)&&a(o=o.init)&&o(t,!1),a(t.componentInstance))return m(t,e),y(n,t.elm,r),s(i)&&_(t,e,n,r),!0}}function m(t,e){a(t.data.pendingInsert)&&(e.push.apply(e,t.data.pendingInsert),t.data.pendingInsert=null),t.elm=t.componentInstance.$el,w(t)?(C(t,e),$(t)):(ei(t),e.push(t))}function _(t,e,n,o){var i,s=t;while(s.componentInstance)if(s=s.componentInstance._vnode,a(i=s.data)&&a(i=i.transition)){for(i=0;i<r.activate.length;++i)r.activate[i](ri,s);e.push(s);break}y(n,t.elm,o)}function y(t,e,n){a(t)&&(a(n)?l.parentNode(n)===t&&l.insertBefore(t,e,n):l.appendChild(t,e))}function g(t,e,n){if(o(e)){0;for(var r=0;r<e.length;++r)v(e[r],n,t.elm,null,!0,e,r)}else u(t.text)&&l.appendChild(t.elm,l.createTextNode(String(t.text)))}function w(t){while(t.componentInstance)t=t.componentInstance._vnode;return a(t.tag)}function C(t,n){for(var o=0;o<r.create.length;++o)r.create[o](ri,t);e=t.data.hook,a(e)&&(a(e.create)&&e.create(ri,t),a(e.insert)&&n.push(t))}function $(t){var e;if(a(e=t.fnScopeId))l.setStyleScope(t.elm,e);else{var n=t;while(n)a(e=n.context)&&a(e=e.$options._scopeId)&&l.setStyleScope(t.elm,e),n=n.parent}a(e=On)&&e!==t.context&&e!==t.fnContext&&a(e=e.$options._scopeId)&&l.setStyleScope(t.elm,e)}function x(t,e,n,r,o,i){for(;r<=o;++r)v(n[r],i,t,e,!1,n,r)}function k(t){var e,n,o=t.data;if(a(o))for(a(e=o.hook)&&a(e=e.destroy)&&e(t),e=0;e<r.destroy.length;++e)r.destroy[e](t);if(a(e=t.children))for(n=0;n<t.children.length;++n)k(t.children[n])}function O(t,e,n){for(;e<=n;++e){var r=t[e];a(r)&&(a(r.tag)?(S(r),k(r)):p(r.elm))}}function S(t,e){if(a(e)||a(t.data)){var n,o=r.remove.length+1;for(a(e)?e.listeners+=o:e=d(t.elm,o),a(n=t.componentInstance)&&a(n=n._vnode)&&a(n.data)&&S(n,e),n=0;n<r.remove.length;++n)r.remove[n](t,e);a(n=t.data.hook)&&a(n=n.remove)?n(t,e):e()}else p(t.elm)}function j(t,e,n,r,o){var s,c,u,f,d=0,p=0,h=e.length-1,m=e[0],_=e[h],y=n.length-1,g=n[0],b=n[y],w=!o;while(d<=h&&p<=y)i(m)?m=e[++d]:i(_)?_=e[--h]:ii(m,g)?(A(m,g,r,n,p),m=e[++d],g=n[++p]):ii(_,b)?(A(_,b,r,n,y),_=e[--h],b=n[--y]):ii(m,b)?(A(m,b,r,n,y),w&&l.insertBefore(t,m.elm,l.nextSibling(_.elm)),m=e[++d],b=n[--y]):ii(_,g)?(A(_,g,r,n,p),w&&l.insertBefore(t,_.elm,m.elm),_=e[--h],g=n[++p]):(i(s)&&(s=si(e,d,h)),c=a(g.key)?s[g.key]:T(g,e,d,h),i(c)?v(g,r,t,m.elm,!1,n,p):(u=e[c],ii(u,g)?(A(u,g,r,n,p),e[c]=void 0,w&&l.insertBefore(t,u.elm,m.elm)):v(g,r,t,m.elm,!1,n,p)),g=n[++p]);d>h?(f=i(n[y+1])?null:n[y+1].elm,x(t,f,n,p,y,r)):p>y&&O(e,d,h)}function T(t,e,n,r){for(var o=n;o<r;o++){var i=e[o];if(a(i)&&ii(t,i))return o}}function A(t,e,n,o,c,u){if(t!==e){a(e.elm)&&a(o)&&(e=o[c]=wt(e));var f=e.elm=t.elm;if(s(t.isAsyncPlaceholder))a(e.asyncFactory.resolved)?P(t.elm,e,n):e.isAsyncPlaceholder=!0;else if(s(e.isStatic)&&s(t.isStatic)&&e.key===t.key&&(s(e.isCloned)||s(e.isOnce)))e.componentInstance=t.componentInstance;else{var d,p=e.data;a(p)&&a(d=p.hook)&&a(d=d.prepatch)&&d(t,e);var v=t.children,h=e.children;if(a(p)&&w(e)){for(d=0;d<r.update.length;++d)r.update[d](t,e);a(d=p.hook)&&a(d=d.update)&&d(t,e)}i(e.text)?a(v)&&a(h)?v!==h&&j(f,v,h,n,u):a(h)?(a(t.text)&&l.setTextContent(f,""),x(f,null,h,0,h.length-1,n)):a(v)?O(v,0,v.length-1):a(t.text)&&l.setTextContent(f,""):t.text!==e.text&&l.setTextContent(f,e.text),a(p)&&a(d=p.hook)&&a(d=d.postpatch)&&d(t,e)}}}function E(t,e,n){if(s(n)&&a(t.parent))t.parent.data.pendingInsert=e;else for(var r=0;r<e.length;++r)e[r].data.hook.insert(e[r])}var I=b("attrs,class,staticClass,staticStyle,key");function P(t,e,n,r){var o,i=e.tag,c=e.data,u=e.children;if(r=r||c&&c.pre,e.elm=t,s(e.isComment)&&a(e.asyncFactory))return e.isAsyncPlaceholder=!0,!0;if(a(c)&&(a(o=c.hook)&&a(o=o.init)&&o(e,!0),a(o=e.componentInstance)))return m(e,n),!0;if(a(i)){if(a(u))if(t.hasChildNodes())if(a(o=c)&&a(o=o.domProps)&&a(o=o.innerHTML)){if(o!==t.innerHTML)return!1}else{for(var l=!0,f=t.firstChild,d=0;d<u.length;d++){if(!f||!P(f,u[d],n,r)){l=!1;break}f=f.nextSibling}if(!l||f)return!1}else g(e,u,n);if(a(c)){var p=!1;for(var v in c)if(!I(v)){p=!0,C(e,n);break}!p&&c["class"]&&hn(c["class"])}}else t.data!==e.text&&(t.data=e.text);return!0}return function(t,e,n,o){if(!i(e)){var c=!1,u=[];if(i(t))c=!0,v(e,u);else{var d=a(t.nodeType);if(!d&&ii(t,e))A(t,e,u,null,null,o);else{if(d){if(1===t.nodeType&&t.hasAttribute(z)&&(t.removeAttribute(z),n=!0),s(n)&&P(t,e,u))return E(e,u,!0),t;t=f(t)}var p=t.elm,h=l.parentNode(p);if(v(e,u,p._leaveCb?null:h,l.nextSibling(p)),a(e.parent)){var m=e.parent,_=w(e);while(m){for(var y=0;y<r.destroy.length;++y)r.destroy[y](m);if(m.elm=e.elm,_){for(var g=0;g<r.create.length;++g)r.create[g](ri,m);var b=m.data.hook.insert;if(b.merged)for(var C=b.fns.slice(1),$=0;$<C.length;$++)C[$]()}else ei(m);m=m.parent}}a(h)?O([t],0,0):a(t.tag)&&k(t)}}return E(e,u,c),e.elm}a(t)&&k(t)}}var ui={create:li,update:li,destroy:function(t){li(t,ri)}};function li(t,e){(t.data.directives||e.data.directives)&&fi(t,e)}function fi(t,e){var n,r,o,i=t===ri,a=e===ri,s=pi(t.data.directives,t.context),c=pi(e.data.directives,e.context),u=[],l=[];for(n in c)r=s[n],o=c[n],r?(o.oldValue=r.value,o.oldArg=r.arg,hi(o,"update",e,t),o.def&&o.def.componentUpdated&&l.push(o)):(hi(o,"bind",e,t),o.def&&o.def.inserted&&u.push(o));if(u.length){var f=function(){for(var n=0;n<u.length;n++)hi(u[n],"inserted",e,t)};i?re(e,"insert",f):f()}if(l.length&&re(e,"postpatch",(function(){for(var n=0;n<l.length;n++)hi(l[n],"componentUpdated",e,t)})),!i)for(n in s)c[n]||hi(s[n],"unbind",t,t,a)}var di=Object.create(null);function pi(t,e){var n,r,o=Object.create(null);if(!t)return o;for(n=0;n<t.length;n++){if(r=t[n],r.modifiers||(r.modifiers=di),o[vi(r)]=r,e._setupState&&e._setupState.__sfc){var i=r.def||kr(e,"_setupState","v-"+r.name);r.def="function"===typeof i?{bind:i,update:i}:i}r.def=r.def||kr(e.$options,"directives",r.name,!0)}return o}function vi(t){return t.rawName||"".concat(t.name,".").concat(Object.keys(t.modifiers||{}).join("."))}function hi(t,e,n,r,o){var i=t.def&&t.def[e];if(i)try{i(n.elm,t,n,r,o)}catch(Ya){Xe(Ya,n.context,"directive ".concat(t.name," ").concat(e," hook"))}}var mi=[ti,ui];function _i(t,e){var n=e.componentOptions;if((!a(n)||!1!==n.Ctor.options.inheritAttrs)&&(!i(t.data.attrs)||!i(e.data.attrs))){var r,o,c,u=e.elm,l=t.data.attrs||{},f=e.data.attrs||{};for(r in(a(f.__ob__)||s(f._v_attr_proxy))&&(f=e.data.attrs=D({},f)),f)o=f[r],c=l[r],c!==o&&yi(u,r,o,e.data.pre);for(r in(nt||ot)&&f.value!==l.value&&yi(u,"value",f.value),l)i(f[r])&&(Co(r)?u.removeAttributeNS(wo,$o(r)):_o(r)||u.removeAttribute(r))}}function yi(t,e,n,r){r||t.tagName.indexOf("-")>-1?gi(t,e,n):bo(e)?xo(n)?t.removeAttribute(e):(n="allowfullscreen"===e&&"EMBED"===t.tagName?"true":e,t.setAttribute(e,n)):_o(e)?t.setAttribute(e,go(e,n)):Co(e)?xo(n)?t.removeAttributeNS(wo,$o(e)):t.setAttributeNS(wo,e,n):gi(t,e,n)}function gi(t,e,n){if(xo(n))t.removeAttribute(e);else{if(nt&&!rt&&"TEXTAREA"===t.tagName&&"placeholder"===e&&""!==n&&!t.__ieph){var r=function(e){e.stopImmediatePropagation(),t.removeEventListener("input",r)};t.addEventListener("input",r),t.__ieph=!0}t.setAttribute(e,n)}}var bi={create:_i,update:_i};function wi(t,e){var n=e.elm,r=e.data,o=t.data;if(!(i(r.staticClass)&&i(r.class)&&(i(o)||i(o.staticClass)&&i(o.class)))){var s=ko(e),c=n._transitionClasses;a(c)&&(s=jo(s,To(c))),s!==n._prevClass&&(n.setAttribute("class",s),n._prevClass=s)}}var Ci,$i={create:wi,update:wi},xi="__r",ki="__c";function Oi(t){if(a(t[xi])){var e=nt?"change":"input";t[e]=[].concat(t[xi],t[e]||[]),delete t[xi]}a(t[ki])&&(t.change=[].concat(t[ki],t.change||[]),delete t[ki])}function Si(t,e,n){var r=Ci;return function o(){var i=e.apply(null,arguments);null!==i&&Ai(t,o,n,r)}}var ji=en&&!(st&&Number(st[1])<=53);function Ti(t,e,n,r){if(ji){var o=Hn,i=e;e=i._wrapper=function(t){if(t.target===t.currentTarget||t.timeStamp>=o||t.timeStamp<=0||t.target.ownerDocument!==document)return i.apply(this,arguments)}}Ci.addEventListener(t,e,ut?{capture:n,passive:r}:n)}function Ai(t,e,n,r){(r||Ci).removeEventListener(t,e._wrapper||e,n)}function Ei(t,e){if(!i(t.data.on)||!i(e.data.on)){var n=e.data.on||{},r=t.data.on||{};Ci=e.elm||t.elm,Oi(n),ne(n,r,Ti,Ai,Si,e.context),Ci=void 0}}var Ii,Pi={create:Ei,update:Ei,destroy:function(t){return Ei(t,ri)}};function Ni(t,e){if(!i(t.data.domProps)||!i(e.data.domProps)){var n,r,o=e.elm,c=t.data.domProps||{},u=e.data.domProps||{};for(n in(a(u.__ob__)||s(u._v_attr_proxy))&&(u=e.data.domProps=D({},u)),c)n in u||(o[n]="");for(n in u){if(r=u[n],"textContent"===n||"innerHTML"===n){if(e.children&&(e.children.length=0),r===c[n])continue;1===o.childNodes.length&&o.removeChild(o.childNodes[0])}if("value"===n&&"PROGRESS"!==o.tagName){o._value=r;var l=i(r)?"":String(r);Di(o,l)&&(o.value=l)}else if("innerHTML"===n&&No(o.tagName)&&i(o.innerHTML)){Ii=Ii||document.createElement("div"),Ii.innerHTML="<svg>".concat(r,"</svg>");var f=Ii.firstChild;while(o.firstChild)o.removeChild(o.firstChild);while(f.firstChild)o.appendChild(f.firstChild)}else if(r!==c[n])try{o[n]=r}catch(Ya){}}}}function Di(t,e){return!t.composing&&("OPTION"===t.tagName||Mi(t,e)||Li(t,e))}function Mi(t,e){var n=!0;try{n=document.activeElement!==t}catch(Ya){}return n&&t.value!==e}function Li(t,e){var n=t.value,r=t._vModifiers;if(a(r)){if(r.number)return g(n)!==g(e);if(r.trim)return n.trim()!==e.trim()}return n!==e}var Fi={create:Ni,update:Ni},Ri=k((function(t){var e={},n=/;(?![^(]*\))/g,r=/:(.+)/;return t.split(n).forEach((function(t){if(t){var n=t.split(r);n.length>1&&(e[n[0].trim()]=n[1].trim())}})),e}));function Ui(t){var e=Bi(t.style);return t.staticStyle?D(t.staticStyle,e):e}function Bi(t){return Array.isArray(t)?M(t):"string"===typeof t?Ri(t):t}function Vi(t,e){var n,r={};if(e){var o=t;while(o.componentInstance)o=o.componentInstance._vnode,o&&o.data&&(n=Ui(o.data))&&D(r,n)}(n=Ui(t.data))&&D(r,n);var i=t;while(i=i.parent)i.data&&(n=Ui(i.data))&&D(r,n);return r}var Hi,zi=/^--/,Wi=/\s*!important$/,qi=function(t,e,n){if(zi.test(e))t.style.setProperty(e,n);else if(Wi.test(n))t.style.setProperty(A(e),n.replace(Wi,""),"important");else{var r=Gi(e);if(Array.isArray(n))for(var o=0,i=n.length;o<i;o++)t.style[r]=n[o];else t.style[r]=n}},Ki=["Webkit","Moz","ms"],Gi=k((function(t){if(Hi=Hi||document.createElement("div").style,t=S(t),"filter"!==t&&t in Hi)return t;for(var e=t.charAt(0).toUpperCase()+t.slice(1),n=0;n<Ki.length;n++){var r=Ki[n]+e;if(r in Hi)return r}}));function Ji(t,e){var n=e.data,r=t.data;if(!(i(n.staticStyle)&&i(n.style)&&i(r.staticStyle)&&i(r.style))){var o,s,c=e.elm,u=r.staticStyle,l=r.normalizedStyle||r.style||{},f=u||l,d=Bi(e.data.style)||{};e.data.normalizedStyle=a(d.__ob__)?D({},d):d;var p=Vi(e,!0);for(s in f)i(p[s])&&qi(c,s,"");for(s in p)o=p[s],qi(c,s,null==o?"":o)}}var Xi={create:Ji,update:Ji},Zi=/\s+/;function Qi(t,e){if(e&&(e=e.trim()))if(t.classList)e.indexOf(" ")>-1?e.split(Zi).forEach((function(e){return t.classList.add(e)})):t.classList.add(e);else{var n=" ".concat(t.getAttribute("class")||""," ");n.indexOf(" "+e+" ")<0&&t.setAttribute("class",(n+e).trim())}}function Yi(t,e){if(e&&(e=e.trim()))if(t.classList)e.indexOf(" ")>-1?e.split(Zi).forEach((function(e){return t.classList.remove(e)})):t.classList.remove(e),t.classList.length||t.removeAttribute("class");else{var n=" ".concat(t.getAttribute("class")||""," "),r=" "+e+" ";while(n.indexOf(r)>=0)n=n.replace(r," ");n=n.trim(),n?t.setAttribute("class",n):t.removeAttribute("class")}}function ta(t){if(t){if("object"===typeof t){var e={};return!1!==t.css&&D(e,ea(t.name||"v")),D(e,t),e}return"string"===typeof t?ea(t):void 0}}var ea=k((function(t){return{enterClass:"".concat(t,"-enter"),enterToClass:"".concat(t,"-enter-to"),enterActiveClass:"".concat(t,"-enter-active"),leaveClass:"".concat(t,"-leave"),leaveToClass:"".concat(t,"-leave-to"),leaveActiveClass:"".concat(t,"-leave-active")}})),na=tt&&!rt,ra="transition",oa="animation",ia="transition",aa="transitionend",sa="animation",ca="animationend";na&&(void 0===window.ontransitionend&&void 0!==window.onwebkittransitionend&&(ia="WebkitTransition",aa="webkitTransitionEnd"),void 0===window.onanimationend&&void 0!==window.onwebkitanimationend&&(sa="WebkitAnimation",ca="webkitAnimationEnd"));var ua=tt?window.requestAnimationFrame?window.requestAnimationFrame.bind(window):setTimeout:function(t){return t()};function la(t){ua((function(){ua(t)}))}function fa(t,e){var n=t._transitionClasses||(t._transitionClasses=[]);n.indexOf(e)<0&&(n.push(e),Qi(t,e))}function da(t,e){t._transitionClasses&&C(t._transitionClasses,e),Yi(t,e)}function pa(t,e,n){var r=ha(t,e),o=r.type,i=r.timeout,a=r.propCount;if(!o)return n();var s=o===ra?aa:ca,c=0,u=function(){t.removeEventListener(s,l),n()},l=function(e){e.target===t&&++c>=a&&u()};setTimeout((function(){c<a&&u()}),i+1),t.addEventListener(s,l)}var va=/\b(transform|all)(,|$)/;function ha(t,e){var n,r=window.getComputedStyle(t),o=(r[ia+"Delay"]||"").split(", "),i=(r[ia+"Duration"]||"").split(", "),a=ma(o,i),s=(r[sa+"Delay"]||"").split(", "),c=(r[sa+"Duration"]||"").split(", "),u=ma(s,c),l=0,f=0;e===ra?a>0&&(n=ra,l=a,f=i.length):e===oa?u>0&&(n=oa,l=u,f=c.length):(l=Math.max(a,u),n=l>0?a>u?ra:oa:null,f=n?n===ra?i.length:c.length:0);var d=n===ra&&va.test(r[ia+"Property"]);return{type:n,timeout:l,propCount:f,hasTransform:d}}function ma(t,e){while(t.length<e.length)t=t.concat(t);return Math.max.apply(null,e.map((function(e,n){return _a(e)+_a(t[n])})))}function _a(t){return 1e3*Number(t.slice(0,-1).replace(",","."))}function ya(t,e){var n=t.elm;a(n._leaveCb)&&(n._leaveCb.cancelled=!0,n._leaveCb());var r=ta(t.data.transition);if(!i(r)&&!a(n._enterCb)&&1===n.nodeType){var o=r.css,s=r.type,c=r.enterClass,u=r.enterToClass,d=r.enterActiveClass,p=r.appearClass,v=r.appearToClass,h=r.appearActiveClass,m=r.beforeEnter,_=r.enter,y=r.afterEnter,b=r.enterCancelled,w=r.beforeAppear,C=r.appear,$=r.afterAppear,x=r.appearCancelled,k=r.duration,O=On,S=On.$vnode;while(S&&S.parent)O=S.context,S=S.parent;var j=!O._isMounted||!t.isRootInsert;if(!j||C||""===C){var T=j&&p?p:c,A=j&&h?h:d,E=j&&v?v:u,I=j&&w||m,P=j&&l(C)?C:_,N=j&&$||y,D=j&&x||b,M=g(f(k)?k.enter:k);0;var L=!1!==o&&!rt,F=wa(P),R=n._enterCb=V((function(){L&&(da(n,E),da(n,A)),R.cancelled?(L&&da(n,T),D&&D(n)):N&&N(n),n._enterCb=null}));t.data.show||re(t,"insert",(function(){var e=n.parentNode,r=e&&e._pending&&e._pending[t.key];r&&r.tag===t.tag&&r.elm._leaveCb&&r.elm._leaveCb(),P&&P(n,R)})),I&&I(n),L&&(fa(n,T),fa(n,A),la((function(){da(n,T),R.cancelled||(fa(n,E),F||(ba(M)?setTimeout(R,M):pa(n,s,R)))}))),t.data.show&&(e&&e(),P&&P(n,R)),L||F||R()}}}function ga(t,e){var n=t.elm;a(n._enterCb)&&(n._enterCb.cancelled=!0,n._enterCb());var r=ta(t.data.transition);if(i(r)||1!==n.nodeType)return e();if(!a(n._leaveCb)){var o=r.css,s=r.type,c=r.leaveClass,u=r.leaveToClass,l=r.leaveActiveClass,d=r.beforeLeave,p=r.leave,v=r.afterLeave,h=r.leaveCancelled,m=r.delayLeave,_=r.duration,y=!1!==o&&!rt,b=wa(p),w=g(f(_)?_.leave:_);0;var C=n._leaveCb=V((function(){n.parentNode&&n.parentNode._pending&&(n.parentNode._pending[t.key]=null),y&&(da(n,u),da(n,l)),C.cancelled?(y&&da(n,c),h&&h(n)):(e(),v&&v(n)),n._leaveCb=null}));m?m($):$()}function $(){C.cancelled||(!t.data.show&&n.parentNode&&((n.parentNode._pending||(n.parentNode._pending={}))[t.key]=t),d&&d(n),y&&(fa(n,c),fa(n,l),la((function(){da(n,c),C.cancelled||(fa(n,u),b||(ba(w)?setTimeout(C,w):pa(n,s,C)))}))),p&&p(n,C),y||b||C())}}function ba(t){return"number"===typeof t&&!isNaN(t)}function wa(t){if(i(t))return!1;var e=t.fns;return a(e)?wa(Array.isArray(e)?e[0]:e):(t._length||t.length)>1}function Ca(t,e){!0!==e.data.show&&ya(e)}var $a=tt?{create:Ca,activate:Ca,remove:function(t,e){!0!==t.data.show?ga(t,e):e()}}:{},xa=[bi,$i,Pi,Fi,Xi,$a],ka=xa.concat(mi),Oa=ci({nodeOps:Yo,modules:ka});rt&&document.addEventListener("selectionchange",(function(){var t=document.activeElement;t&&t.vmodel&&Na(t,"input")}));var Sa={inserted:function(t,e,n,r){"select"===n.tag?(r.elm&&!r.elm._vOptions?re(n,"postpatch",(function(){Sa.componentUpdated(t,e,n)})):ja(t,e,n.context),t._vOptions=[].map.call(t.options,Ea)):("textarea"===n.tag||Ro(t.type))&&(t._vModifiers=e.modifiers,e.modifiers.lazy||(t.addEventListener("compositionstart",Ia),t.addEventListener("compositionend",Pa),t.addEventListener("change",Pa),rt&&(t.vmodel=!0)))},componentUpdated:function(t,e,n){if("select"===n.tag){ja(t,e,n.context);var r=t._vOptions,o=t._vOptions=[].map.call(t.options,Ea);if(o.some((function(t,e){return!U(t,r[e])}))){var i=t.multiple?e.value.some((function(t){return Aa(t,o)})):e.value!==e.oldValue&&Aa(e.value,o);i&&Na(t,"change")}}}};function ja(t,e,n){Ta(t,e,n),(nt||ot)&&setTimeout((function(){Ta(t,e,n)}),0)}function Ta(t,e,n){var r=e.value,o=t.multiple;if(!o||Array.isArray(r)){for(var i,a,s=0,c=t.options.length;s<c;s++)if(a=t.options[s],o)i=B(r,Ea(a))>-1,a.selected!==i&&(a.selected=i);else if(U(Ea(a),r))return void(t.selectedIndex!==s&&(t.selectedIndex=s));o||(t.selectedIndex=-1)}}function Aa(t,e){return e.every((function(e){return!U(e,t)}))}function Ea(t){return"_value"in t?t._value:t.value}function Ia(t){t.target.composing=!0}function Pa(t){t.target.composing&&(t.target.composing=!1,Na(t.target,"input"))}function Na(t,e){var n=document.createEvent("HTMLEvents");n.initEvent(e,!0,!0),t.dispatchEvent(n)}function Da(t){return!t.componentInstance||t.data&&t.data.transition?t:Da(t.componentInstance._vnode)}var Ma={bind:function(t,e,n){var r=e.value;n=Da(n);var o=n.data&&n.data.transition,i=t.__vOriginalDisplay="none"===t.style.display?"":t.style.display;r&&o?(n.data.show=!0,ya(n,(function(){t.style.display=i}))):t.style.display=r?i:"none"},update:function(t,e,n){var r=e.value,o=e.oldValue;if(!r!==!o){n=Da(n);var i=n.data&&n.data.transition;i?(n.data.show=!0,r?ya(n,(function(){t.style.display=t.__vOriginalDisplay})):ga(n,(function(){t.style.display="none"}))):t.style.display=r?t.__vOriginalDisplay:"none"}},unbind:function(t,e,n,r,o){o||(t.style.display=t.__vOriginalDisplay)}},La={model:Sa,show:Ma},Fa={name:String,appear:Boolean,css:Boolean,mode:String,type:String,enterClass:String,leaveClass:String,enterToClass:String,leaveToClass:String,enterActiveClass:String,leaveActiveClass:String,appearClass:String,appearActiveClass:String,appearToClass:String,duration:[Number,String,Object]};function Ra(t){var e=t&&t.componentOptions;return e&&e.Ctor.options.abstract?Ra(He(e.children)):t}function Ua(t){var e={},n=t.$options;for(var r in n.propsData)e[r]=t[r];var o=n._parentListeners;for(var r in o)e[S(r)]=o[r];return e}function Ba(t,e){if(/\d-keep-alive$/.test(e.tag))return t("keep-alive",{props:e.componentOptions.propsData})}function Va(t){while(t=t.parent)if(t.data.transition)return!0}function Ha(t,e){return e.key===t.key&&e.tag===t.tag}var za=function(t){return t.tag||Se(t)},Wa=function(t){return"show"===t.name},qa={name:"transition",props:Fa,abstract:!0,render:function(t){var e=this,n=this.$slots.default;if(n&&(n=n.filter(za),n.length)){0;var r=this.mode;0;var o=n[0];if(Va(this.$vnode))return o;var i=Ra(o);if(!i)return o;if(this._leaving)return Ba(t,o);var a="__transition-".concat(this._uid,"-");i.key=null==i.key?i.isComment?a+"comment":a+i.tag:u(i.key)?0===String(i.key).indexOf(a)?i.key:a+i.key:i.key;var s=(i.data||(i.data={})).transition=Ua(this),c=this._vnode,l=Ra(c);if(i.data.directives&&i.data.directives.some(Wa)&&(i.data.show=!0),l&&l.data&&!Ha(i,l)&&!Se(l)&&(!l.componentInstance||!l.componentInstance._vnode.isComment)){var f=l.data.transition=D({},s);if("out-in"===r)return this._leaving=!0,re(f,"afterLeave",(function(){e._leaving=!1,e.$forceUpdate()})),Ba(t,o);if("in-out"===r){if(Se(i))return c;var d,p=function(){d()};re(s,"afterEnter",p),re(s,"enterCancelled",p),re(f,"delayLeave",(function(t){d=t}))}}return o}}},Ka=D({tag:String,moveClass:String},Fa);delete Ka.mode;var Ga={props:Ka,beforeMount:function(){var t=this,e=this._update;this._update=function(n,r){var o=Sn(t);t.__patch__(t._vnode,t.kept,!1,!0),t._vnode=t.kept,o(),e.call(t,n,r)}},render:function(t){for(var e=this.tag||this.$vnode.data.tag||"span",n=Object.create(null),r=this.prevChildren=this.children,o=this.$slots.default||[],i=this.children=[],a=Ua(this),s=0;s<o.length;s++){var c=o[s];if(c.tag)if(null!=c.key&&0!==String(c.key).indexOf("__vlist"))i.push(c),n[c.key]=c,(c.data||(c.data={})).transition=a;else;}if(r){var u=[],l=[];for(s=0;s<r.length;s++){c=r[s];c.data.transition=a,c.data.pos=c.elm.getBoundingClientRect(),n[c.key]?u.push(c):l.push(c)}this.kept=t(e,null,u),this.removed=l}return t(e,null,i)},updated:function(){var t=this.prevChildren,e=this.moveClass||(this.name||"v")+"-move";t.length&&this.hasMove(t[0].elm,e)&&(t.forEach(Ja),t.forEach(Xa),t.forEach(Za),this._reflow=document.body.offsetHeight,t.forEach((function(t){if(t.data.moved){var n=t.elm,r=n.style;fa(n,e),r.transform=r.WebkitTransform=r.transitionDuration="",n.addEventListener(aa,n._moveCb=function t(r){r&&r.target!==n||r&&!/transform$/.test(r.propertyName)||(n.removeEventListener(aa,t),n._moveCb=null,da(n,e))})}})))},methods:{hasMove:function(t,e){if(!na)return!1;if(this._hasMove)return this._hasMove;var n=t.cloneNode();t._transitionClasses&&t._transitionClasses.forEach((function(t){Yi(n,t)})),Qi(n,e),n.style.display="none",this.$el.appendChild(n);var r=ha(n);return this.$el.removeChild(n),this._hasMove=r.hasTransform}}};function Ja(t){t.elm._moveCb&&t.elm._moveCb(),t.elm._enterCb&&t.elm._enterCb()}function Xa(t){t.data.newPos=t.elm.getBoundingClientRect()}function Za(t){var e=t.data.pos,n=t.data.newPos,r=e.left-n.left,o=e.top-n.top;if(r||o){t.data.moved=!0;var i=t.elm.style;i.transform=i.WebkitTransform="translate(".concat(r,"px,").concat(o,"px)"),i.transitionDuration="0s"}}var Qa={Transition:qa,TransitionGroup:Ga};Qr.config.mustUseProp=mo,Qr.config.isReservedTag=Do,Qr.config.isReservedAttr=vo,Qr.config.getTagNamespace=Mo,Qr.config.isUnknownElement=Fo,D(Qr.options.directives,La),D(Qr.options.components,Qa),Qr.prototype.__patch__=tt?Oa:L,Qr.prototype.$mount=function(t,e){return t=t&&tt?Uo(t):void 0,An(this,t,e)},tt&&setTimeout((function(){K.devtools&&dt&&dt.emit("init",Qr)}),0)}).call(this,n("c8ba"))},"499e":function(t,e,n){"use strict";function r(t,e){for(var n=[],r={},o=0;o<e.length;o++){var i=e[o],a=i[0],s=i[1],c=i[2],u=i[3],l={id:t+":"+o,css:s,media:c,sourceMap:u};r[a]?r[a].parts.push(l):n.push(r[a]={id:a,parts:[l]})}return n}n.r(e),n.d(e,"default",(function(){return v}));var o="undefined"!==typeof document;if("undefined"!==typeof DEBUG&&DEBUG&&!o)throw new Error("vue-style-loader cannot be used in a non-browser environment. Use { target: 'node' } in your Webpack config to indicate a server-rendering environment.");var i={},a=o&&(document.head||document.getElementsByTagName("head")[0]),s=null,c=0,u=!1,l=function(){},f=null,d="data-vue-ssr-id",p="undefined"!==typeof navigator&&/msie [6-9]\b/.test(navigator.userAgent.toLowerCase());function v(t,e,n,o){u=n,f=o||{};var a=r(t,e);return h(a),function(e){for(var n=[],o=0;o<a.length;o++){var s=a[o],c=i[s.id];c.refs--,n.push(c)}e?(a=r(t,e),h(a)):a=[];for(o=0;o<n.length;o++){c=n[o];if(0===c.refs){for(var u=0;u<c.parts.length;u++)c.parts[u]();delete i[c.id]}}}}function h(t){for(var e=0;e<t.length;e++){var n=t[e],r=i[n.id];if(r){r.refs++;for(var o=0;o<r.parts.length;o++)r.parts[o](n.parts[o]);for(;o<n.parts.length;o++)r.parts.push(_(n.parts[o]));r.parts.length>n.parts.length&&(r.parts.length=n.parts.length)}else{var a=[];for(o=0;o<n.parts.length;o++)a.push(_(n.parts[o]));i[n.id]={id:n.id,refs:1,parts:a}}}}function m(){var t=document.createElement("style");return t.type="text/css",a.appendChild(t),t}function _(t){var e,n,r=document.querySelector("style["+d+'~="'+t.id+'"]');if(r){if(u)return l;r.parentNode.removeChild(r)}if(p){var o=c++;r=s||(s=m()),e=g.bind(null,r,o,!1),n=g.bind(null,r,o,!0)}else r=m(),e=b.bind(null,r),n=function(){r.parentNode.removeChild(r)};return e(t),function(r){if(r){if(r.css===t.css&&r.media===t.media&&r.sourceMap===t.sourceMap)return;e(t=r)}else n()}}var y=function(){var t=[];return function(e,n){return t[e]=n,t.filter(Boolean).join("\n")}}();function g(t,e,n,r){var o=n?"":r.css;if(t.styleSheet)t.styleSheet.cssText=y(e,o);else{var i=document.createTextNode(o),a=t.childNodes;a[e]&&t.removeChild(a[e]),a.length?t.insertBefore(i,a[e]):t.appendChild(i)}}function b(t,e){var n=e.css,r=e.media,o=e.sourceMap;if(r&&t.setAttribute("media",r),f.ssrId&&t.setAttribute(d,e.id),o&&(n+="\n/*# sourceURL="+o.sources[0]+" */",n+="\n/*# sourceMappingURL=data:application/json;base64,"+btoa(unescape(encodeURIComponent(JSON.stringify(o))))+" */"),t.styleSheet)t.styleSheet.cssText=n;else{while(t.firstChild)t.removeChild(t.firstChild);t.appendChild(document.createTextNode(n))}}},c8ba:function(t,e){var n;n=function(){return this}();try{n=n||new Function("return this")()}catch(r){"object"===typeof window&&(n=window)}t.exports=n}}]);
|
8 |
+
//# sourceMappingURL=chunk-vendors.20f7f886.js.map
|
dist/js/index.cbb15892.js
ADDED
@@ -0,0 +1,2 @@
|
|
|
|
|
|
|
1 |
+
(function(e){function n(n){for(var t,u,c=n[0],i=n[1],l=n[2],s=0,h=[];s<c.length;s++)u=c[s],Object.prototype.hasOwnProperty.call(o,u)&&o[u]&&h.push(o[u][0]),o[u]=0;for(t in i)Object.prototype.hasOwnProperty.call(i,t)&&(e[t]=i[t]);f&&f(n);while(h.length)h.shift()();return a.push.apply(a,l||[]),r()}function r(){for(var e,n=0;n<a.length;n++){for(var r=a[n],t=!0,u=1;u<r.length;u++){var i=r[u];0!==o[i]&&(t=!1)}t&&(a.splice(n--,1),e=c(c.s=r[0]))}return e}var t={},o={index:0},a=[];function u(e){return c.p+"js/"+({}[e]||e)+"."+{"chunk-ae402692":"003457bc","chunk-0c4e36c8":"95d70738","chunk-117382e0":"d47336d3","chunk-40965e1a":"74707226","chunk-04395031":"dbff2f6b","chunk-0cbfe13e":"73856287","chunk-2d0db258":"a4804a7a","chunk-2d0c53c7":"d24941b8","chunk-48b5b2a0":"3db5a0aa","chunk-a06ef50c":"1caef24f"}[e]+".js"}function c(n){if(t[n])return t[n].exports;var r=t[n]={i:n,l:!1,exports:{}};return e[n].call(r.exports,r,r.exports,c),r.l=!0,r.exports}c.e=function(e){var n=[],r=o[e];if(0!==r)if(r)n.push(r[2]);else{var t=new Promise((function(n,t){r=o[e]=[n,t]}));n.push(r[2]=t);var a,i=document.createElement("script");i.charset="utf-8",i.timeout=120,c.nc&&i.setAttribute("nonce",c.nc),i.src=u(e);var l=new Error;a=function(n){i.onerror=i.onload=null,clearTimeout(s);var r=o[e];if(0!==r){if(r){var t=n&&("load"===n.type?"missing":n.type),a=n&&n.target&&n.target.src;l.message="Loading chunk "+e+" failed.\n("+t+": "+a+")",l.name="ChunkLoadError",l.type=t,l.request=a,r[1](l)}o[e]=void 0}};var s=setTimeout((function(){a({type:"timeout",target:i})}),12e4);i.onerror=i.onload=a,document.head.appendChild(i)}return Promise.all(n)},c.m=e,c.c=t,c.d=function(e,n,r){c.o(e,n)||Object.defineProperty(e,n,{enumerable:!0,get:r})},c.r=function(e){"undefined"!==typeof Symbol&&Symbol.toStringTag&&Object.defineProperty(e,Symbol.toStringTag,{value:"Module"}),Object.defineProperty(e,"__esModule",{value:!0})},c.t=function(e,n){if(1&n&&(e=c(e)),8&n)return e;if(4&n&&"object"===typeof e&&e&&e.__esModule)return e;var r=Object.create(null);if(c.r(r),Object.defineProperty(r,"default",{enumerable:!0,value:e}),2&n&&"string"!=typeof e)for(var t in e)c.d(r,t,function(n){return e[n]}.bind(null,t));return r},c.n=function(e){var n=e&&e.__esModule?function(){return e["default"]}:function(){return e};return c.d(n,"a",n),n},c.o=function(e,n){return Object.prototype.hasOwnProperty.call(e,n)},c.p="",c.oe=function(e){throw console.error(e),e};var i=window["webpackJsonp"]=window["webpackJsonp"]||[],l=i.push.bind(i);i.push=n,i=i.slice();for(var s=0;s<i.length;s++)n(i[s]);var f=l;a.push([0,"chunk-vendors"]),r()})({0:function(e,n,r){e.exports=r("c7c3")},6786:function(e,n,r){"use strict";r("97cc")},"828f":function(e,n,r){var t=r("24fb");n=t(!1),n.push([e.i,"body{margin:0;overflow:hidden}",""]),e.exports=n},"97cc":function(e,n,r){var t=r("828f");t.__esModule&&(t=t.default),"string"===typeof t&&(t=[[e.i,t,""]]),t.locals&&(e.exports=t.locals);var o=r("499e").default;o("a0b5d278",t,!0,{sourceMap:!1,shadowMode:!1})},aa4a:function(e,n,r){var t={"./flex-engraver.vue":["e168","chunk-ae402692","chunk-0c4e36c8","chunk-117382e0"],"./playground.vue":["1140","chunk-ae402692","chunk-0c4e36c8","chunk-40965e1a","chunk-04395031"],"./profiler.vue":["5748","chunk-ae402692","chunk-40965e1a","chunk-0cbfe13e"]};function o(e){if(!r.o(t,e))return Promise.resolve().then((function(){var n=new Error("Cannot find module '"+e+"'");throw n.code="MODULE_NOT_FOUND",n}));var n=t[e],o=n[0];return Promise.all(n.slice(1).map(r.e)).then((function(){return r(o)}))}o.keys=function(){return Object.keys(t)},o.id="aa4a",e.exports=o},c7c3:function(e,n,r){"use strict";r.r(n);var t=r("2b0e"),o=function(){var e=this,n=e._self._c;return n("body",[e.view?n(e.view,{tag:"component"}):e._e()],1)},a=[];const u=["playground","profiler","flex-engraver"];u.forEach(e=>t["a"].component(e,()=>r("aa4a")(`./${e}.vue`)));var c={name:"lotus",data(){return{view:null}},mounted(){this.onHashChange(),window.onhashchange=()=>this.onHashChange()},methods:{onHashChange(){this.view=(location.hash.substr(1)||"playground").replace(/\?.*/,"")}}},i=c,l=(r("6786"),r("2877")),s=Object(l["a"])(i,o,a,!1,null,null,null),f=s.exports;new t["a"]({render:e=>e(f)}).$mount("body")}});
|
2 |
+
//# sourceMappingURL=index.cbb15892.js.map
|
dist/soundfont/acoustic_grand_piano-mp3.js
ADDED
The diff for this file is too large to render.
See raw diff
|
|
dist/soundfont/acoustic_grand_piano-ogg.js
ADDED
The diff for this file is too large to render.
See raw diff
|
|
inc/DictArray.ts
ADDED
@@ -0,0 +1,35 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
|
2 |
+
class DictArray extends Array {
|
3 |
+
constructor (jsonObj?: object) {
|
4 |
+
super();
|
5 |
+
|
6 |
+
if (jsonObj) {
|
7 |
+
console.assert(typeof jsonObj === "object", "invalid input type:", jsonObj);
|
8 |
+
|
9 |
+
if (Array.isArray(jsonObj))
|
10 |
+
Object.keys(jsonObj).forEach(index => this[index] = jsonObj[index]);
|
11 |
+
else
|
12 |
+
Object.entries(jsonObj).forEach(([key, value]) => this[key] = value);
|
13 |
+
}
|
14 |
+
}
|
15 |
+
|
16 |
+
|
17 |
+
toJSON () {
|
18 |
+
return Object.keys(this).reduce((dict, index) => (dict[index] = this[index], dict), {__prototype: "DictArray"});
|
19 |
+
}
|
20 |
+
|
21 |
+
|
22 |
+
clear () {
|
23 |
+
Object.keys(this).forEach(index => delete this[index]);
|
24 |
+
this.length = 0;
|
25 |
+
}
|
26 |
+
|
27 |
+
|
28 |
+
clone () {
|
29 |
+
return new DictArray(this);
|
30 |
+
}
|
31 |
+
};
|
32 |
+
|
33 |
+
|
34 |
+
|
35 |
+
export default DictArray;
|
inc/asyncCall.ts
ADDED
@@ -0,0 +1,11 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
|
2 |
+
const asyncCall = (func, ...args): Promise<any> => new Promise((resolve, reject) => func(...args, (err, data) => {
|
3 |
+
if (err)
|
4 |
+
reject(err);
|
5 |
+
else
|
6 |
+
resolve(data);
|
7 |
+
}));
|
8 |
+
|
9 |
+
|
10 |
+
|
11 |
+
export default asyncCall;
|
inc/constants.ts
ADDED
@@ -0,0 +1,38 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
|
2 |
+
const LILYPOND_VERSION = "2.20.0";
|
3 |
+
|
4 |
+
|
5 |
+
const CM_TO_PX = 37.794;
|
6 |
+
|
7 |
+
const LILY_TOP_MARGIN_DEFAULT = 0.5;
|
8 |
+
const LILY_BOTTOM_MARGIN_DEFAULT = 0.6;
|
9 |
+
const LILY_HORIZONTAL_MARGIN_DEFAULT = 1;
|
10 |
+
const LILY_STAFF_SIZE_DEFAULT = 20;
|
11 |
+
|
12 |
+
//const SYSTEM_SYSTEM_SPACING = 0.08;
|
13 |
+
|
14 |
+
const STAFF_HEAD_DEDUCTION = 0.06; // extra width augment when system wrap added, cm per staff size
|
15 |
+
|
16 |
+
|
17 |
+
const SVG_DTD = `<?xml version="1.0" standalone="no"?>
|
18 |
+
<!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.1//EN" "http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd"
|
19 |
+
[<!ENTITY quot """> <!ENTITY amp "&"> <!ENTITY apos "'"> <!ENTITY lt "<"> <!ENTITY gt ">"> <!ENTITY nbsp " "> <!ENTITY iexcl "¡"> <!ENTITY cent "¢"> <!ENTITY pound "£"> <!ENTITY curren "¤"> <!ENTITY yen "¥"> <!ENTITY brvbar "¦"> <!ENTITY sect "§"> <!ENTITY uml "¨"> <!ENTITY copy "©"> <!ENTITY ordf "ª"> <!ENTITY laquo "«"> <!ENTITY not "¬"> <!ENTITY shy "­"> <!ENTITY reg "®"> <!ENTITY macr "¯"> <!ENTITY deg "°"> <!ENTITY plusmn "±"> <!ENTITY sup2 "²"> <!ENTITY sup3 "³"> <!ENTITY acute "´"> <!ENTITY micro "µ"> <!ENTITY para "¶"> <!ENTITY middot "·"> <!ENTITY cedil "¸"> <!ENTITY sup1 "¹"> <!ENTITY ordm "º"> <!ENTITY raquo "»"> <!ENTITY frac14 "¼"> <!ENTITY frac12 "½"> <!ENTITY frac34 "¾"> <!ENTITY iquest "¿"> <!ENTITY Agrave "À"> <!ENTITY Aacute "Á"> <!ENTITY Acirc "Â"> <!ENTITY Atilde "Ã"> <!ENTITY Auml "Ä"> <!ENTITY Aring "Å"> <!ENTITY AElig "Æ"> <!ENTITY Ccedil "Ç"> <!ENTITY Egrave "È"> <!ENTITY Eacute "É"> <!ENTITY Ecirc "Ê"> <!ENTITY Euml "Ë"> <!ENTITY Igrave "Ì"> <!ENTITY Iacute "Í"> <!ENTITY Icirc "Î"> <!ENTITY Iuml "Ï"> <!ENTITY ETH "Ð"> <!ENTITY Ntilde "Ñ"> <!ENTITY Ograve "Ò"> <!ENTITY Oacute "Ó"> <!ENTITY Ocirc "Ô"> <!ENTITY Otilde "Õ"> <!ENTITY Ouml "Ö"> <!ENTITY times "×"> <!ENTITY Oslash "Ø"> <!ENTITY Ugrave "Ù"> <!ENTITY Uacute "Ú"> <!ENTITY Ucirc "Û"> <!ENTITY Uuml "Ü"> <!ENTITY Yacute "Ý"> <!ENTITY THORN "Þ"> <!ENTITY szlig "ß"> <!ENTITY agrave "à"> <!ENTITY aacute "á"> <!ENTITY acirc "â"> <!ENTITY atilde "ã"> <!ENTITY auml "ä"> <!ENTITY aring "å"> <!ENTITY aelig "æ"> <!ENTITY ccedil "ç"> <!ENTITY egrave "è"> <!ENTITY eacute "é"> <!ENTITY ecirc "ê"> <!ENTITY euml "ë"> <!ENTITY igrave "ì"> <!ENTITY iacute "í"> <!ENTITY icirc "î"> <!ENTITY iuml "ï"> <!ENTITY eth "ð"> <!ENTITY ntilde "ñ"> <!ENTITY ograve "ò"> <!ENTITY oacute "ó"> <!ENTITY ocirc "ô"> <!ENTITY otilde "õ"> <!ENTITY ouml "ö"> <!ENTITY divide "÷"> <!ENTITY oslash "ø"> <!ENTITY ugrave "ù"> <!ENTITY uacute "ú"> <!ENTITY ucirc "û"> <!ENTITY uuml "ü"> <!ENTITY yacute "ý"> <!ENTITY thorn "þ"> <!ENTITY yuml "ÿ"> <!ENTITY OElig "Œ"> <!ENTITY oelig "œ"> <!ENTITY Scaron "Š"> <!ENTITY scaron "š"> <!ENTITY Yuml "Ÿ"> <!ENTITY fnof "ƒ"> <!ENTITY circ "ˆ"> <!ENTITY tilde "˜"> <!ENTITY Alpha "Α"> <!ENTITY Beta "Β"> <!ENTITY Gamma "Γ"> <!ENTITY Delta "Δ"> <!ENTITY Epsilon "Ε"> <!ENTITY Zeta "Ζ"> <!ENTITY Eta "Η"> <!ENTITY Theta "Θ"> <!ENTITY Iota "Ι"> <!ENTITY Kappa "Κ"> <!ENTITY Lambda "Λ"> <!ENTITY Mu "Μ"> <!ENTITY Nu "Ν"> <!ENTITY Xi "Ξ"> <!ENTITY Omicron "Ο"> <!ENTITY Pi "Π"> <!ENTITY Rho "Ρ"> <!ENTITY Sigma "Σ"> <!ENTITY Tau "Τ"> <!ENTITY Upsilon "Υ"> <!ENTITY Phi "Φ"> <!ENTITY Chi "Χ"> <!ENTITY Psi "Ψ"> <!ENTITY Omega "Ω"> <!ENTITY alpha "α"> <!ENTITY beta "β"> <!ENTITY gamma "γ"> <!ENTITY delta "δ"> <!ENTITY epsilon "ε"> <!ENTITY zeta "ζ"> <!ENTITY eta "η"> <!ENTITY theta "θ"> <!ENTITY iota "ι"> <!ENTITY kappa "κ"> <!ENTITY lambda "λ"> <!ENTITY mu "μ"> <!ENTITY nu "ν"> <!ENTITY xi "ξ"> <!ENTITY omicron "ο"> <!ENTITY pi "π"> <!ENTITY rho "ρ"> <!ENTITY sigmaf "ς"> <!ENTITY sigma "σ"> <!ENTITY tau "τ"> <!ENTITY upsilon "υ"> <!ENTITY phi "φ"> <!ENTITY chi "χ"> <!ENTITY psi "ψ"> <!ENTITY omega "ω"> <!ENTITY thetasym "ϑ"> <!ENTITY upsih "ϒ"> <!ENTITY piv "ϖ"> <!ENTITY ensp " "> <!ENTITY emsp " "> <!ENTITY thinsp " "> <!ENTITY zwnj "‌"> <!ENTITY zwj "‍"> <!ENTITY lrm "‎"> <!ENTITY rlm "‏"> <!ENTITY ndash "–"> <!ENTITY mdash "—"> <!ENTITY lsquo "‘"> <!ENTITY rsquo "’"> <!ENTITY sbquo "‚"> <!ENTITY ldquo "“"> <!ENTITY rdquo "”"> <!ENTITY bdquo "„"> <!ENTITY dagger "†"> <!ENTITY Dagger "‡"> <!ENTITY bull "•"> <!ENTITY hellip "…"> <!ENTITY permil "‰"> <!ENTITY prime "′"> <!ENTITY Prime "″"> <!ENTITY lsaquo "‹"> <!ENTITY rsaquo "›"> <!ENTITY oline "‾"> <!ENTITY frasl "⁄"> <!ENTITY euro "€"> <!ENTITY image "ℑ"> <!ENTITY weierp "℘"> <!ENTITY real "ℜ"> <!ENTITY trade "™"> <!ENTITY alefsym "ℵ"> <!ENTITY larr "←"> <!ENTITY uarr "↑"> <!ENTITY rarr "→"> <!ENTITY darr "↓"> <!ENTITY harr "↔"> <!ENTITY crarr "↵"> <!ENTITY lArr "⇐"> <!ENTITY uArr "⇑"> <!ENTITY rArr "⇒"> <!ENTITY dArr "⇓"> <!ENTITY hArr "⇔"> <!ENTITY forall "∀"> <!ENTITY part "∂"> <!ENTITY exist "∃"> <!ENTITY empty "∅"> <!ENTITY nabla "∇"> <!ENTITY isin "∈"> <!ENTITY notin "∉"> <!ENTITY ni "∋"> <!ENTITY prod "∏"> <!ENTITY sum "∑"> <!ENTITY minus "−"> <!ENTITY lowast "∗"> <!ENTITY radic "√"> <!ENTITY prop "∝"> <!ENTITY infin "∞"> <!ENTITY ang "∠"> <!ENTITY and "∧"> <!ENTITY or "∨"> <!ENTITY cap "∩"> <!ENTITY cup "∪"> <!ENTITY int "∫"> <!ENTITY there4 "∴"> <!ENTITY sim "∼"> <!ENTITY cong "≅"> <!ENTITY asymp "≈"> <!ENTITY ne "≠"> <!ENTITY equiv "≡"> <!ENTITY le "≤"> <!ENTITY ge "≥"> <!ENTITY sub "⊂"> <!ENTITY sup "⊃"> <!ENTITY nsub "⊄"> <!ENTITY sube "⊆"> <!ENTITY supe "⊇"> <!ENTITY oplus "⊕"> <!ENTITY otimes "⊗"> <!ENTITY perp "⊥"> <!ENTITY sdot "⋅"> <!ENTITY lceil "⌈"> <!ENTITY rceil "⌉"> <!ENTITY lfloor "⌊"> <!ENTITY rfloor "⌋"> <!ENTITY lang "〈"> <!ENTITY rang "〉"> <!ENTITY loz "◊"> <!ENTITY spades "♠"> <!ENTITY clubs "♣"> <!ENTITY hearts "♥"> <!ENTITY diams "♦">]>
|
20 |
+
`;
|
21 |
+
|
22 |
+
|
23 |
+
|
24 |
+
export {
|
25 |
+
LILYPOND_VERSION,
|
26 |
+
|
27 |
+
CM_TO_PX,
|
28 |
+
|
29 |
+
LILY_TOP_MARGIN_DEFAULT,
|
30 |
+
LILY_BOTTOM_MARGIN_DEFAULT,
|
31 |
+
LILY_HORIZONTAL_MARGIN_DEFAULT,
|
32 |
+
LILY_STAFF_SIZE_DEFAULT,
|
33 |
+
//SYSTEM_SYSTEM_SPACING,
|
34 |
+
|
35 |
+
STAFF_HEAD_DEDUCTION,
|
36 |
+
|
37 |
+
SVG_DTD,
|
38 |
+
};
|
inc/domUtils.ts
ADDED
@@ -0,0 +1,49 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
|
2 |
+
const traverse = (node, handle) => {
|
3 |
+
handle(node);
|
4 |
+
|
5 |
+
if (node.childNodes) {
|
6 |
+
for (let i = 0; i < node.childNodes.length; ++i)
|
7 |
+
traverse(node.childNodes[i], handle);
|
8 |
+
}
|
9 |
+
};
|
10 |
+
|
11 |
+
|
12 |
+
const childrenWithTag = (node: any, tagName: string): any[] => {
|
13 |
+
const children = Array.from(node.childNodes);
|
14 |
+
return children.filter((node: any) => node.tagName === tagName);
|
15 |
+
};
|
16 |
+
|
17 |
+
|
18 |
+
const hasChildrenWithTag = (node: any, tagName: string): boolean => {
|
19 |
+
const children = Array.from(node.childNodes);
|
20 |
+
return children.some((node: any) => node.tagName === tagName);
|
21 |
+
};
|
22 |
+
|
23 |
+
|
24 |
+
const findPreviousSibling = (node: any, tagName: string): any => {
|
25 |
+
let sibling = node.previousSibling;
|
26 |
+
while (sibling && sibling.tagName !== tagName)
|
27 |
+
sibling = sibling.previousSibling;
|
28 |
+
|
29 |
+
return sibling;
|
30 |
+
};
|
31 |
+
|
32 |
+
|
33 |
+
const findNextSibling = (node: any, tagName: string): any => {
|
34 |
+
let sibling = node.nextSibling;
|
35 |
+
while (sibling && sibling.tagName !== tagName)
|
36 |
+
sibling = sibling.nextSibling;
|
37 |
+
|
38 |
+
return sibling;
|
39 |
+
};
|
40 |
+
|
41 |
+
|
42 |
+
|
43 |
+
export {
|
44 |
+
traverse,
|
45 |
+
childrenWithTag,
|
46 |
+
hasChildrenWithTag,
|
47 |
+
findPreviousSibling,
|
48 |
+
findNextSibling,
|
49 |
+
};
|
inc/jisonWrapper.ts
ADDED
@@ -0,0 +1,34 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
|
2 |
+
import jison from "jison";
|
3 |
+
|
4 |
+
|
5 |
+
|
6 |
+
export class Parser {
|
7 |
+
parser: any;
|
8 |
+
|
9 |
+
|
10 |
+
constructor (grammar: string) {
|
11 |
+
// mute jison logs during grammar loading
|
12 |
+
const logs = [];
|
13 |
+
jison.print = log => logs.push(log);
|
14 |
+
|
15 |
+
this.parser = jison.Parser(grammar);
|
16 |
+
if (!this.parser) {
|
17 |
+
console.warn("jison logs:", logs);
|
18 |
+
throw new Error("jison parser loading failed");
|
19 |
+
}
|
20 |
+
|
21 |
+
jison.print = (...args) => console.log("[JISON]", ...args);
|
22 |
+
}
|
23 |
+
|
24 |
+
|
25 |
+
parse (source: string): object {
|
26 |
+
//console.log("parse:", source.length);
|
27 |
+
return this.parser.parse(source);
|
28 |
+
}
|
29 |
+
|
30 |
+
|
31 |
+
generate (): string {
|
32 |
+
return this.parser.generate();
|
33 |
+
}
|
34 |
+
};
|
inc/jsonRecovery.ts
ADDED
@@ -0,0 +1,41 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
|
2 |
+
|
3 |
+
const recoverJSON = (json: string | object, classDict) => {
|
4 |
+
if (typeof json === "object")
|
5 |
+
json = JSON.stringify(json);
|
6 |
+
|
7 |
+
return JSON.parse(json, (_, value) => {
|
8 |
+
if (value && (typeof value === "object") && value.__prototype) {
|
9 |
+
const Class = classDict[value.__prototype];
|
10 |
+
if (Class) {
|
11 |
+
const {__prototype, ...fields} = value;
|
12 |
+
return new Class(fields);
|
13 |
+
}
|
14 |
+
}
|
15 |
+
|
16 |
+
return value;
|
17 |
+
});
|
18 |
+
};
|
19 |
+
|
20 |
+
|
21 |
+
class SimpleClass {
|
22 |
+
constructor (data?: object) {
|
23 |
+
if (data)
|
24 |
+
Object.assign(this, data);
|
25 |
+
}
|
26 |
+
|
27 |
+
|
28 |
+
toJSON () {
|
29 |
+
return {
|
30 |
+
__prototype: (this.constructor as any).className,
|
31 |
+
...this,
|
32 |
+
};
|
33 |
+
}
|
34 |
+
};
|
35 |
+
|
36 |
+
|
37 |
+
|
38 |
+
export {
|
39 |
+
recoverJSON,
|
40 |
+
SimpleClass,
|
41 |
+
};
|
inc/lilyNotation/fuzzyMatch.ts
ADDED
@@ -0,0 +1,125 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
|
2 |
+
// eslint-disable-next-line
|
3 |
+
import {MusicNotation} from "@k-l-lambda/music-widgets";
|
4 |
+
|
5 |
+
|
6 |
+
|
7 |
+
interface Note {
|
8 |
+
index: number;
|
9 |
+
pitch: number;
|
10 |
+
softIndex: number;
|
11 |
+
baseCsi?: number;
|
12 |
+
};
|
13 |
+
|
14 |
+
|
15 |
+
const notePitchDistance = (n1: Note, n2: Note): number => {
|
16 |
+
let differ = Math.abs(n1.pitch - n2.pitch);
|
17 |
+
if (differ > 7)
|
18 |
+
differ -= 11.5;
|
19 |
+
|
20 |
+
return differ * 2;
|
21 |
+
};
|
22 |
+
|
23 |
+
|
24 |
+
const fuzzyMatchNotes = (path: number[], cnotes: Note[], snotes: Note[], pitchTolerance: number, offsetTolerance = 2): {fixed: number, matched: number} => {
|
25 |
+
const candidates: {[key: number]: {si: number, distance: number}} = {};
|
26 |
+
|
27 |
+
const matched = snotes.reduce((count, snote) => {
|
28 |
+
const pcan = cnotes.filter(cnote => notePitchDistance(cnote, snote) <= pitchTolerance
|
29 |
+
&& Math.abs(snote.baseCsi - cnote.softIndex) < offsetTolerance)
|
30 |
+
.sort((n1, n2) => Math.abs(snote.baseCsi - n1.softIndex) - Math.abs(snote.baseCsi - n2.softIndex));
|
31 |
+
|
32 |
+
if (pcan.length) {
|
33 |
+
const bestDistance = notePitchDistance(pcan[0], snote);
|
34 |
+
const last = candidates[pcan[0].index];
|
35 |
+
if (!last || bestDistance < last.distance)
|
36 |
+
candidates[pcan[0].index] = {si: snote.index, distance: bestDistance};
|
37 |
+
|
38 |
+
++count;
|
39 |
+
}
|
40 |
+
|
41 |
+
return count;
|
42 |
+
}, 0);
|
43 |
+
|
44 |
+
const fixed = Object.entries(candidates).map(([ci, {si}]) => {
|
45 |
+
path[si] = Number(ci);
|
46 |
+
|
47 |
+
return [si, Number(ci)];
|
48 |
+
});
|
49 |
+
//console.debug("fuzzyMatch.fixed:", fixed.map(pair => pair.join()));
|
50 |
+
|
51 |
+
return {fixed: fixed.length, matched};
|
52 |
+
};
|
53 |
+
|
54 |
+
|
55 |
+
const fuzzyMatchNotations = (path: number[], criterion: MusicNotation.NotationData, sample: MusicNotation.NotationData, {pitchToleranceMax = 4} = {}) => {
|
56 |
+
// unmatch overlapped indices
|
57 |
+
const overlapped = new Set();
|
58 |
+
path.reduce((set, ci) => {
|
59 |
+
if (ci >= 0) {
|
60 |
+
if (set.has(ci))
|
61 |
+
overlapped.add(ci);
|
62 |
+
else
|
63 |
+
set.add(ci);
|
64 |
+
}
|
65 |
+
|
66 |
+
return set;
|
67 |
+
}, new Set());
|
68 |
+
//console.debug("overlapped:", overlapped, [...path]);
|
69 |
+
path.forEach((ci, si) => {
|
70 |
+
if (overlapped.has(ci))
|
71 |
+
path[si] = -1;
|
72 |
+
});
|
73 |
+
|
74 |
+
// assign base offset on sample notes
|
75 |
+
let offset = null;
|
76 |
+
for (let i = 0; i < sample.notes.length; ++i) {
|
77 |
+
const note = sample.notes[i];
|
78 |
+
const ci = path[i];
|
79 |
+
|
80 |
+
if (ci < 0)
|
81 |
+
(note as any).baseOffset = offset;
|
82 |
+
else
|
83 |
+
offset = note.softIndex - criterion.notes[ci].softIndex;
|
84 |
+
}
|
85 |
+
for (let i = sample.notes.length - 1; i >= 0; --i) {
|
86 |
+
const note = sample.notes[i];
|
87 |
+
const ci = path[i];
|
88 |
+
|
89 |
+
if (ci < 0) {
|
90 |
+
const lastOffset = (note as any).baseOffset;
|
91 |
+
(note as any).baseOffset = Number.isFinite(lastOffset) ? (lastOffset + offset) / 2 : offset;
|
92 |
+
}
|
93 |
+
else
|
94 |
+
offset = note.softIndex - criterion.notes[ci].softIndex;
|
95 |
+
}
|
96 |
+
|
97 |
+
let pitchTolerance = 0;
|
98 |
+
while (true) {
|
99 |
+
const cnotes = criterion.notes.filter(note => !path.some(ci => ci === note.index)).map(note => ({index: note.index, pitch: note.pitch, softIndex: note.softIndex}));
|
100 |
+
const snotes = sample.notes.filter(note => path[note.index] < 0 && Number.isFinite((note as any).baseOffset))
|
101 |
+
.map(note => ({index: note.index, pitch: note.pitch, softIndex: note.softIndex, baseCsi: note.softIndex - (note as any).baseOffset}));
|
102 |
+
|
103 |
+
//console.debug("fuzzyMatch.notes:", cnotes.map(note => note.index), snotes.map(note => note.index));
|
104 |
+
|
105 |
+
if (!cnotes.length || !snotes.length)
|
106 |
+
break;
|
107 |
+
|
108 |
+
const {fixed, matched} = fuzzyMatchNotes(path, cnotes, snotes, pitchTolerance);
|
109 |
+
if (matched)
|
110 |
+
console.debug("fuzzyMatch.pass:", `c:${cnotes.length}, s:${snotes.length},`, pitchTolerance, `${fixed}/${matched}`);
|
111 |
+
|
112 |
+
if (fixed >= matched) {
|
113 |
+
++pitchTolerance;
|
114 |
+
if (pitchTolerance > pitchToleranceMax)
|
115 |
+
break;
|
116 |
+
}
|
117 |
+
}
|
118 |
+
//console.debug("fuzzyMatch.path:", path);
|
119 |
+
};
|
120 |
+
|
121 |
+
|
122 |
+
|
123 |
+
export {
|
124 |
+
fuzzyMatchNotations,
|
125 |
+
};
|
inc/lilyNotation/implicitType.ts
ADDED
@@ -0,0 +1,16 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
|
2 |
+
// implicit note (from expressive marks) types
|
3 |
+
enum ImplicitType {
|
4 |
+
None = 0,
|
5 |
+
|
6 |
+
Mordent = "mordent",
|
7 |
+
Prall = "prall",
|
8 |
+
Turn = "turn",
|
9 |
+
Trill = "trill",
|
10 |
+
Tremolo = "tremolo",
|
11 |
+
Arpeggio = "arpeggio",
|
12 |
+
};
|
13 |
+
|
14 |
+
|
15 |
+
|
16 |
+
export default ImplicitType;
|
inc/lilyNotation/index.ts
ADDED
@@ -0,0 +1,27 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
|
2 |
+
import {Note, Notation, assignNotationEventsIds} from "./notation";
|
3 |
+
import {fuzzyMatchNotations} from "./fuzzyMatch";
|
4 |
+
import {matchWithMIDI, matchWithExactMIDI} from "./matcher";
|
5 |
+
import Scheduler from "./scheduler";
|
6 |
+
import ImplicitType from "./implicitType";
|
7 |
+
import * as measureLayout from "../measureLayout";
|
8 |
+
import {LayoutType} from "../measureLayout";
|
9 |
+
|
10 |
+
|
11 |
+
|
12 |
+
const {...MLayoutClasses} = measureLayout;
|
13 |
+
|
14 |
+
|
15 |
+
|
16 |
+
export {
|
17 |
+
Note,
|
18 |
+
Notation,
|
19 |
+
fuzzyMatchNotations,
|
20 |
+
assignNotationEventsIds,
|
21 |
+
matchWithMIDI,
|
22 |
+
matchWithExactMIDI,
|
23 |
+
Scheduler,
|
24 |
+
ImplicitType,
|
25 |
+
LayoutType,
|
26 |
+
MLayoutClasses,
|
27 |
+
};
|
inc/lilyNotation/matcher.ts
ADDED
@@ -0,0 +1,191 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
|
2 |
+
import {MusicNotation, Matcher} from "@k-l-lambda/music-widgets";
|
3 |
+
import {MIDI} from "@k-l-lambda/music-widgets";
|
4 |
+
|
5 |
+
import {WHOLE_DURATION_MAGNITUDE} from "../lilyParser/utils";
|
6 |
+
import {Notation} from "./notation";
|
7 |
+
import ImplicitType from "./implicitType";
|
8 |
+
|
9 |
+
|
10 |
+
|
11 |
+
export interface MatcherResult {
|
12 |
+
criterion: MusicNotation.NotationData,
|
13 |
+
sample: MusicNotation.NotationData,
|
14 |
+
path: number[], // S to C
|
15 |
+
trackMap?: number[], // S to C
|
16 |
+
};
|
17 |
+
|
18 |
+
|
19 |
+
const matchWithMIDI = async (lilyNotation: Notation, target: MIDI.MidiData): Promise<MatcherResult> => {
|
20 |
+
const midiTickFactor = (WHOLE_DURATION_MAGNITUDE / 4) / target.header.ticksPerBeat;
|
21 |
+
|
22 |
+
const midiNotation = MusicNotation.Notation.parseMidi(target);
|
23 |
+
|
24 |
+
const criterion = lilyNotation.toPerformingNotation();
|
25 |
+
|
26 |
+
Matcher.genNotationContext(criterion, {softIndexFactor: 1e3});
|
27 |
+
Matcher.genNotationContext(midiNotation, {softIndexFactor: midiTickFactor * 1e3});
|
28 |
+
//console.debug("notations:", criterion, midiNotation);
|
29 |
+
|
30 |
+
for (const note of midiNotation.notes)
|
31 |
+
Matcher.makeMatchNodes(note, criterion);
|
32 |
+
|
33 |
+
const navigator = await Matcher.runNavigation(criterion, midiNotation);
|
34 |
+
const path = navigator.path();
|
35 |
+
|
36 |
+
const matcher = {criterion, sample: midiNotation, path};
|
37 |
+
lilyNotation.assignMatcher(matcher);
|
38 |
+
|
39 |
+
return matcher;
|
40 |
+
};
|
41 |
+
|
42 |
+
|
43 |
+
const IMPLICIT_PITCH_BIAS = {
|
44 |
+
[ImplicitType.Mordent]: [0, -1, -2],
|
45 |
+
[ImplicitType.Prall]: [0, 1, 2],
|
46 |
+
[ImplicitType.Turn]: [-2, -1, 0, 1, 2],
|
47 |
+
[ImplicitType.Trill]: [0, 1, 2],
|
48 |
+
};
|
49 |
+
|
50 |
+
|
51 |
+
const alignNotationTicks = (source: MusicNotation.Notation, criterion: MusicNotation.Notation, {midiTrackTickBias}: {midiTrackTickBias?: number[]}) => {
|
52 |
+
const midiTickFactor = criterion.ticksPerBeat / source.ticksPerBeat;
|
53 |
+
|
54 |
+
source.ticksPerBeat = criterion.ticksPerBeat;
|
55 |
+
source.notes.forEach(note => {
|
56 |
+
note.startTick *= midiTickFactor;
|
57 |
+
note.endTick *= midiTickFactor;
|
58 |
+
|
59 |
+
const bias = midiTrackTickBias && midiTrackTickBias[note.track];
|
60 |
+
if (bias) {
|
61 |
+
note.startTick += bias;
|
62 |
+
note.endTick += bias;
|
63 |
+
}
|
64 |
+
});
|
65 |
+
|
66 |
+
source.events.forEach(event => {
|
67 |
+
event.ticks *= midiTickFactor;
|
68 |
+
|
69 |
+
const bias = midiTrackTickBias && midiTrackTickBias[event.track];
|
70 |
+
if (bias)
|
71 |
+
event.ticks += bias;
|
72 |
+
});
|
73 |
+
source.events.sort((e1, e2) => e1.ticks - e2.ticks);
|
74 |
+
};
|
75 |
+
|
76 |
+
|
77 |
+
const matchWithExactMIDI = async (lilyNotation: Notation, target: MIDI.MidiData): Promise<MatcherResult> => {
|
78 |
+
const criterion = lilyNotation.toPerformingNotation();
|
79 |
+
|
80 |
+
const trackTickBiasMap = lilyNotation.trackTickBias;
|
81 |
+
const targetTrackNames = target.tracks.map(events => {
|
82 |
+
const nameEvent = events.find(event => event.subtype === "trackName");
|
83 |
+
return nameEvent ? nameEvent.text : null;
|
84 |
+
});
|
85 |
+
const trackIndexC2S = lilyNotation.trackNames.map(name => targetTrackNames.indexOf(name));
|
86 |
+
//if (trackIndexC2S.includes(-1))
|
87 |
+
// console.debug("mismatched track found:", trackIndexC2S, lilyNotation.trackNames, targetTrackNames);
|
88 |
+
|
89 |
+
const midiTrackTickBias = targetTrackNames.map(name => name ? (trackTickBiasMap[name] || 0) : 0);
|
90 |
+
//console.log("midiTrackTickBias:", midiTrackTickBias);
|
91 |
+
|
92 |
+
const midiNotation = MusicNotation.Notation.parseMidi(target, {fixOverlap: false});
|
93 |
+
alignNotationTicks(midiNotation, criterion, {midiTrackTickBias});
|
94 |
+
|
95 |
+
// 1st pass: ordinary notes exact matching
|
96 |
+
const noteKey = note => `${note.track}|${Math.round(note.startTick)}|${note.pitch}`;
|
97 |
+
|
98 |
+
const snoteMap: {[key: string]: MusicNotation.Note} = {};
|
99 |
+
midiNotation.notes.forEach(note => {
|
100 |
+
snoteMap[noteKey(note)] = note;
|
101 |
+
});
|
102 |
+
|
103 |
+
const path = Array(midiNotation.notes.length).fill(-1);
|
104 |
+
|
105 |
+
const restCNotes = [];
|
106 |
+
criterion.notes.forEach(note => {
|
107 |
+
const implicit = !!(note as any).implicitType;
|
108 |
+
|
109 |
+
const key = noteKey({...note, track: trackIndexC2S[note.track - 1]});
|
110 |
+
const sn = snoteMap[key];
|
111 |
+
if (sn) {
|
112 |
+
path[sn.index] = note.index;
|
113 |
+
delete snoteMap[key];
|
114 |
+
|
115 |
+
if (!implicit)
|
116 |
+
return;
|
117 |
+
}
|
118 |
+
|
119 |
+
restCNotes.push(note);
|
120 |
+
});
|
121 |
+
|
122 |
+
// 2nd pass: implicit notes matching
|
123 |
+
const restCNotes2 = [];
|
124 |
+
const restSNotes = Object.values(snoteMap);
|
125 |
+
restCNotes.forEach(note => {
|
126 |
+
if (note.implicitType) {
|
127 |
+
const pbs = IMPLICIT_PITCH_BIAS[note.implicitType];
|
128 |
+
if (pbs) {
|
129 |
+
const matches = restSNotes.filter(sn => {
|
130 |
+
const tick = Math.round(sn.startTick);
|
131 |
+
const pb = sn.pitch - note.pitch;
|
132 |
+
return tick >= note.startTick && tick <= note.endTick && pbs.includes(pb);
|
133 |
+
});
|
134 |
+
|
135 |
+
if (matches.length) {
|
136 |
+
matches.forEach(sn => {
|
137 |
+
path[sn.index] = note.index;
|
138 |
+
delete snoteMap[noteKey(sn)];
|
139 |
+
});
|
140 |
+
|
141 |
+
return;
|
142 |
+
}
|
143 |
+
}
|
144 |
+
}
|
145 |
+
|
146 |
+
restCNotes2.push(note);
|
147 |
+
});
|
148 |
+
|
149 |
+
// 3rd pass: rest fuzzy matching
|
150 |
+
const restCNotes3 = [];
|
151 |
+
restCNotes2.forEach(note => {
|
152 |
+
const sn = restSNotes.find(sn => sn.pitch === note.pitch && Math.abs(sn.startTick - note.startTick) < 4);
|
153 |
+
if (sn) {
|
154 |
+
path[sn.index] = note.index;
|
155 |
+
delete snoteMap[noteKey(sn)];
|
156 |
+
}
|
157 |
+
else
|
158 |
+
restCNotes3.push(note);
|
159 |
+
});
|
160 |
+
|
161 |
+
// 4th pass: nearest matching
|
162 |
+
const restSNotes3 = Object.values(snoteMap);
|
163 |
+
//console.log("restCNotes3:", restCNotes3.map(n => n.index));
|
164 |
+
//console.log("restSNotes3:", restSNotes3.map(n => n.index));
|
165 |
+
restSNotes3.forEach(note => {
|
166 |
+
const cn = restCNotes3.reduce((best, cn) => {
|
167 |
+
if (cn.pitch === note.pitch) {
|
168 |
+
if (!best || Math.abs(cn.startTick - note.startTick) < Math.abs(best.startTick - note.startTick))
|
169 |
+
return cn;
|
170 |
+
}
|
171 |
+
|
172 |
+
return best;
|
173 |
+
}, null);
|
174 |
+
if (cn)
|
175 |
+
path[note.index] = cn.index;
|
176 |
+
});
|
177 |
+
|
178 |
+
const trackMap = Object.entries(trackIndexC2S).reduce((map, [ct, st]) => ((st >= 0 && (map[st] = Number(ct) + 1)), map), []);
|
179 |
+
|
180 |
+
const matcher = {criterion, sample: midiNotation, path, trackMap};
|
181 |
+
lilyNotation.assignMatcher(matcher);
|
182 |
+
|
183 |
+
return matcher;
|
184 |
+
};
|
185 |
+
|
186 |
+
|
187 |
+
|
188 |
+
export {
|
189 |
+
matchWithMIDI,
|
190 |
+
matchWithExactMIDI,
|
191 |
+
};
|
inc/lilyNotation/notation.ts
ADDED
@@ -0,0 +1,675 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
|
2 |
+
import {MusicNotation, MIDI} from "@k-l-lambda/music-widgets";
|
3 |
+
|
4 |
+
import {WHOLE_DURATION_MAGNITUDE} from "../lilyParser/utils";
|
5 |
+
import {PitchContextTable} from "../pitchContext";
|
6 |
+
import {MatcherResult} from "./matcher";
|
7 |
+
import {MeasureLayout, LayoutType} from "../measureLayout";
|
8 |
+
import ImplicitType from "./implicitType";
|
9 |
+
import npmPackage from "../../package.json";
|
10 |
+
import pick from "../pick";
|
11 |
+
|
12 |
+
|
13 |
+
|
14 |
+
const TICKS_PER_BEAT = WHOLE_DURATION_MAGNITUDE / 4;
|
15 |
+
|
16 |
+
const ARPEGGIO_REFERENCE_DURATION = 240;
|
17 |
+
|
18 |
+
// import WHOLE_DURATION_MAGNITUDE from "../lilyParser" may result in null error in nodejs
|
19 |
+
console.assert(Number.isFinite(TICKS_PER_BEAT), "TICKS_PER_BEAT is invalid:", TICKS_PER_BEAT);
|
20 |
+
|
21 |
+
|
22 |
+
export interface ChordPosition {
|
23 |
+
index: number;
|
24 |
+
count: number;
|
25 |
+
};
|
26 |
+
|
27 |
+
|
28 |
+
interface StaffNoteProperties {
|
29 |
+
rest: boolean;
|
30 |
+
tied: boolean;
|
31 |
+
overlapped: boolean;
|
32 |
+
implicitType: ImplicitType;
|
33 |
+
afterGrace: boolean;
|
34 |
+
chordPosition: ChordPosition;
|
35 |
+
division: number;
|
36 |
+
|
37 |
+
contextIndex: number;
|
38 |
+
staffTrack: number;
|
39 |
+
};
|
40 |
+
|
41 |
+
|
42 |
+
export interface Note extends MusicNotation.Note, Partial<StaffNoteProperties> {
|
43 |
+
id: string;
|
44 |
+
measure: number;
|
45 |
+
endTick: number;
|
46 |
+
outMeasure?: boolean;
|
47 |
+
};
|
48 |
+
|
49 |
+
|
50 |
+
interface SubNote {
|
51 |
+
startTick: number;
|
52 |
+
endTick: number;
|
53 |
+
pitch: number;
|
54 |
+
velocity?: number;
|
55 |
+
};
|
56 |
+
|
57 |
+
|
58 |
+
interface MeasureNote extends Partial<StaffNoteProperties> {
|
59 |
+
tick: number;
|
60 |
+
pitch: number;
|
61 |
+
duration: number;
|
62 |
+
chordPosition: ChordPosition;
|
63 |
+
|
64 |
+
track: number;
|
65 |
+
channel: number;
|
66 |
+
id: string;
|
67 |
+
ids: string[];
|
68 |
+
|
69 |
+
subNotes: SubNote[];
|
70 |
+
};
|
71 |
+
|
72 |
+
|
73 |
+
interface MeasureEvent {
|
74 |
+
data: any;
|
75 |
+
track: number;
|
76 |
+
ticks?: number;
|
77 |
+
};
|
78 |
+
|
79 |
+
|
80 |
+
interface Measure {
|
81 |
+
tick: number;
|
82 |
+
duration: number;
|
83 |
+
|
84 |
+
notes: MeasureNote[];
|
85 |
+
events?: MeasureEvent[];
|
86 |
+
};
|
87 |
+
|
88 |
+
|
89 |
+
interface PerformOptions {
|
90 |
+
withRestTied?: boolean;
|
91 |
+
};
|
92 |
+
|
93 |
+
|
94 |
+
/*interface NotationData {
|
95 |
+
pitchContextGroup: PitchContextTable[];
|
96 |
+
measures: Measure[];
|
97 |
+
};*/
|
98 |
+
|
99 |
+
|
100 |
+
const EXTRA_NOTE_FIELDS = ["rest", "tied", "overlapped", "implicitType", "afterGrace", "contextIndex", "staffTrack", "chordPosition", "division"];
|
101 |
+
const COMMON_NOTE_FIELDS = ["id", "ids", "pitch", "velocity", "track", "channel", ...EXTRA_NOTE_FIELDS];
|
102 |
+
|
103 |
+
|
104 |
+
export class Notation {
|
105 |
+
pitchContextGroup: PitchContextTable[];
|
106 |
+
measureLayout: MeasureLayout;
|
107 |
+
measures: Measure[];
|
108 |
+
|
109 |
+
trackNames: string[];
|
110 |
+
idTrackMap: {[key: string]: number};
|
111 |
+
|
112 |
+
ripe: boolean = false;
|
113 |
+
|
114 |
+
|
115 |
+
static fromAbsoluteNotes (notes: Note[], measureHeads: number[], data?: Partial<Notation>): Notation {
|
116 |
+
const notation = new Notation(data);
|
117 |
+
|
118 |
+
notation.measures = Array(measureHeads.length).fill(null).map((__, i) => {
|
119 |
+
const tick = measureHeads[i];
|
120 |
+
const duration = measureHeads[i + 1] ? (measureHeads[i + 1] - tick) : 0;
|
121 |
+
|
122 |
+
const mnotes = notes.filter(note => note.measure === i + 1).map(note => ({
|
123 |
+
tick: note.startTick - tick,
|
124 |
+
duration: note.endTick - note.startTick,
|
125 |
+
...pick(note, COMMON_NOTE_FIELDS),
|
126 |
+
subNotes: [],
|
127 |
+
} as MeasureNote));
|
128 |
+
|
129 |
+
// reduce note data size
|
130 |
+
mnotes.forEach(mn => ["rest", "tied", "implicitType", "afterGrace"].forEach(field => {
|
131 |
+
if (!mn[field])
|
132 |
+
delete mn[field];
|
133 |
+
}));
|
134 |
+
|
135 |
+
return {
|
136 |
+
tick,
|
137 |
+
duration,
|
138 |
+
notes: mnotes,
|
139 |
+
};
|
140 |
+
});
|
141 |
+
|
142 |
+
notation.idTrackMap = notes.reduce((map, note) => {
|
143 |
+
if (note.id)
|
144 |
+
map[note.id] = note.track;
|
145 |
+
|
146 |
+
return map;
|
147 |
+
}, {});
|
148 |
+
|
149 |
+
return notation;
|
150 |
+
}
|
151 |
+
|
152 |
+
|
153 |
+
static performAbsoluteNotes (abNotes: Note[], {withRestTied = false}: PerformOptions = {}): MusicNotation.Note[] {
|
154 |
+
const notes = abNotes.filter(note => (withRestTied || (!note.rest && !note.tied)) && !note.overlapped).map(note => ({
|
155 |
+
measure: note.measure,
|
156 |
+
channel: note.channel,
|
157 |
+
track: note.track,
|
158 |
+
start: note.start,
|
159 |
+
startTick: note.startTick,
|
160 |
+
endTick: note.endTick,
|
161 |
+
pitch: note.pitch,
|
162 |
+
duration: note.duration,
|
163 |
+
velocity: note.velocity || 127,
|
164 |
+
id: note.id,
|
165 |
+
ids: note.ids,
|
166 |
+
staffTrack: note.staffTrack,
|
167 |
+
contextIndex: note.contextIndex,
|
168 |
+
implicitType: note.implicitType,
|
169 |
+
chordPosition: note.chordPosition,
|
170 |
+
outMeasure: note.outMeasure,
|
171 |
+
}));
|
172 |
+
|
173 |
+
const noteMap = notes.reduce((map, note) => {
|
174 |
+
const key = `${note.channel}|${note.start}|${note.pitch}`;
|
175 |
+
const priorNote = map[key];
|
176 |
+
if (priorNote)
|
177 |
+
priorNote.ids.push(...note.ids);
|
178 |
+
else
|
179 |
+
map[key] = note;
|
180 |
+
|
181 |
+
return map;
|
182 |
+
}, {});
|
183 |
+
|
184 |
+
return Object.values(noteMap);
|
185 |
+
}
|
186 |
+
|
187 |
+
|
188 |
+
constructor (data?: Partial<Notation>) {
|
189 |
+
if (data)
|
190 |
+
Object.assign(this, data);
|
191 |
+
}
|
192 |
+
|
193 |
+
|
194 |
+
get ordinaryMeasureIndices (): number[] {
|
195 |
+
if (this.measureLayout)
|
196 |
+
return this.measureLayout.serialize(LayoutType.Ordinary);
|
197 |
+
|
198 |
+
return Array(this.measures.length).fill(null).map((_, i) => i + 1);
|
199 |
+
}
|
200 |
+
|
201 |
+
|
202 |
+
// In Lilypond 2.20.0, minus tick value at the head of a track result in MIDI event time bias,
|
203 |
+
// So store the bias values to correct MIDI time from lilyond.
|
204 |
+
get trackTickBias (): {[key: string]: number} {
|
205 |
+
const headMeasure = this.measures[0];
|
206 |
+
return this.trackNames.reduce((map, name, track) => {
|
207 |
+
map[name] = 0;
|
208 |
+
if (headMeasure) {
|
209 |
+
const note = headMeasure.notes.find(note => note.track === track);
|
210 |
+
if (note)
|
211 |
+
map[name] = Math.min(note.tick, 0);
|
212 |
+
}
|
213 |
+
|
214 |
+
return map;
|
215 |
+
}, {});
|
216 |
+
}
|
217 |
+
|
218 |
+
|
219 |
+
get idSet (): Set<string> {
|
220 |
+
return this.measures.reduce((set, measure) =>
|
221 |
+
(measure.notes.filter(note => !note.rest).forEach(note => note.ids.forEach(id => set.add(id))), set), new Set<string>());
|
222 |
+
}
|
223 |
+
|
224 |
+
|
225 |
+
toJSON () {
|
226 |
+
return {
|
227 |
+
__prototype: "LilyNotation",
|
228 |
+
pitchContextGroup: this.pitchContextGroup,
|
229 |
+
measureLayout: this.measureLayout,
|
230 |
+
measures: this.measures,
|
231 |
+
idTrackMap: this.idTrackMap,
|
232 |
+
trackNames: this.trackNames,
|
233 |
+
ripe: this.ripe,
|
234 |
+
};
|
235 |
+
}
|
236 |
+
|
237 |
+
|
238 |
+
toAbsoluteNotes (measureIndices: number[] = this.ordinaryMeasureIndices): Note[] {
|
239 |
+
let measureTick = 0;
|
240 |
+
const measureNotes: Note[][] = measureIndices.map(index => {
|
241 |
+
const measure = this.measures[index - 1];
|
242 |
+
console.assert(!!measure, "invalid measure index:", index, this.measures.length);
|
243 |
+
|
244 |
+
const notes = measure.notes.map(mnote => {
|
245 |
+
const outMeasure = mnote.tick < 0 || mnote.tick >= measure.duration;
|
246 |
+
|
247 |
+
return {
|
248 |
+
startTick: measureTick + mnote.tick,
|
249 |
+
endTick: measureTick + mnote.tick + mnote.duration,
|
250 |
+
start: measureTick + mnote.tick,
|
251 |
+
duration: mnote.duration,
|
252 |
+
measure: index,
|
253 |
+
outMeasure,
|
254 |
+
...pick(mnote, COMMON_NOTE_FIELDS),
|
255 |
+
} as Note;
|
256 |
+
});
|
257 |
+
|
258 |
+
measureTick += measure.duration;
|
259 |
+
|
260 |
+
return notes;
|
261 |
+
});
|
262 |
+
|
263 |
+
return [].concat(...measureNotes);
|
264 |
+
}
|
265 |
+
|
266 |
+
|
267 |
+
getMeasureIndices (type: LayoutType) {
|
268 |
+
return this.measureLayout.serialize(type);
|
269 |
+
}
|
270 |
+
|
271 |
+
|
272 |
+
toPerformingNotation (measureIndices: number[] = this.ordinaryMeasureIndices, options: PerformOptions = {}): MusicNotation.Notation {
|
273 |
+
//console.debug("toPerformingNotation:", this, measureIndices);
|
274 |
+
const abNotes = this.toAbsoluteNotes(measureIndices);
|
275 |
+
const notes = Notation.performAbsoluteNotes(abNotes, options);
|
276 |
+
|
277 |
+
//const lastNote = notes[notes.length - 1];
|
278 |
+
const endTime = Math.max(...notes.map(note => note.start + note.duration));
|
279 |
+
|
280 |
+
const endTick = measureIndices.reduce((tick, index) => tick + this.measures[index - 1].duration, 0);
|
281 |
+
|
282 |
+
const notation = new MusicNotation.Notation({
|
283 |
+
ticksPerBeat: TICKS_PER_BEAT,
|
284 |
+
meta: {},
|
285 |
+
tempos: [], // TODO
|
286 |
+
channels: [notes],
|
287 |
+
endTime,
|
288 |
+
endTick,
|
289 |
+
});
|
290 |
+
|
291 |
+
return notation;
|
292 |
+
}
|
293 |
+
|
294 |
+
|
295 |
+
toPerformingMIDI (measureIndices: number[], {trackList}: {trackList?: boolean[]} = {}): MIDI.MidiData & {zeroTick: number} {
|
296 |
+
if (!measureIndices.length)
|
297 |
+
return null;
|
298 |
+
|
299 |
+
// to avoid begin minus tick
|
300 |
+
const zeroTick = -Math.min(0,
|
301 |
+
...(this.measures[0] ? this.measures[0].events.map((e) => e.ticks) : []),
|
302 |
+
...(this.measures[0] ? this.measures[0].notes.map((note) => note.tick) : []));
|
303 |
+
|
304 |
+
let measureTick = zeroTick;
|
305 |
+
const measureEvents: MeasureEvent[][] = measureIndices.map(index => {
|
306 |
+
const measure = this.measures[index - 1];
|
307 |
+
console.assert(!!measure, "invalid measure index:", index, this.measures.length);
|
308 |
+
|
309 |
+
const events = measure.events.map(mevent => ({
|
310 |
+
ticks: measureTick + mevent.ticks,
|
311 |
+
track: mevent.track,
|
312 |
+
data: {
|
313 |
+
...mevent.data,
|
314 |
+
measure: index,
|
315 |
+
},
|
316 |
+
}));
|
317 |
+
|
318 |
+
measureTick += measure.duration;
|
319 |
+
|
320 |
+
return events;
|
321 |
+
});
|
322 |
+
|
323 |
+
interface MidiEvent extends MIDI.MidiEvent {
|
324 |
+
ticks?: number;
|
325 |
+
measure?: number;
|
326 |
+
ids?: string[];
|
327 |
+
staffTrack?: number;
|
328 |
+
};
|
329 |
+
type MidiTrack = MidiEvent[];
|
330 |
+
|
331 |
+
const eventPriority = (event: MidiEvent): number => event.ticks + (event.subtype === "noteOff" ? -1e-4 : 0);
|
332 |
+
|
333 |
+
const tracks: MidiTrack[] = [].concat(...measureEvents).reduce((tracks, mevent) => {
|
334 |
+
tracks[mevent.track] = tracks[mevent.track] || [];
|
335 |
+
tracks[mevent.track].push({
|
336 |
+
ticks: mevent.ticks,
|
337 |
+
...mevent.data,
|
338 |
+
});
|
339 |
+
|
340 |
+
return tracks;
|
341 |
+
}, []);
|
342 |
+
|
343 |
+
tracks[0] = tracks[0] || [];
|
344 |
+
tracks[0].push({
|
345 |
+
ticks: 0,
|
346 |
+
type: "meta",
|
347 |
+
subtype: "text",
|
348 |
+
text: `${npmPackage.name} ${npmPackage.version}`,
|
349 |
+
});
|
350 |
+
|
351 |
+
// append note events
|
352 |
+
measureTick = zeroTick;
|
353 |
+
measureIndices.map(index => {
|
354 |
+
const measure = this.measures[index - 1];
|
355 |
+
console.assert(!!measure, "invalid measure index:", index, this.measures.length);
|
356 |
+
|
357 |
+
measure.notes.forEach(note => {
|
358 |
+
if (trackList && !trackList[note.track])
|
359 |
+
return;
|
360 |
+
|
361 |
+
if (note.rest)
|
362 |
+
return;
|
363 |
+
|
364 |
+
const tick = measureTick + note.tick;
|
365 |
+
|
366 |
+
const track = tracks[note.track] = tracks[note.track] || [];
|
367 |
+
|
368 |
+
note.subNotes.forEach(subnote => {
|
369 |
+
track.push({
|
370 |
+
ticks: tick + subnote.startTick,
|
371 |
+
measure: index,
|
372 |
+
ids: note.ids,
|
373 |
+
type: "channel",
|
374 |
+
subtype: "noteOn",
|
375 |
+
channel: note.channel,
|
376 |
+
noteNumber: subnote.pitch,
|
377 |
+
velocity: subnote.velocity,
|
378 |
+
staffTrack: note.staffTrack,
|
379 |
+
});
|
380 |
+
|
381 |
+
track.push({
|
382 |
+
ticks: tick + subnote.endTick,
|
383 |
+
measure: index,
|
384 |
+
ids: note.ids,
|
385 |
+
type: "channel",
|
386 |
+
subtype: "noteOff",
|
387 |
+
channel: note.channel,
|
388 |
+
noteNumber: subnote.pitch,
|
389 |
+
velocity: 0,
|
390 |
+
staffTrack: note.staffTrack,
|
391 |
+
});
|
392 |
+
});
|
393 |
+
});
|
394 |
+
|
395 |
+
measureTick += measure.duration;
|
396 |
+
});
|
397 |
+
|
398 |
+
const finalTick = measureTick;
|
399 |
+
|
400 |
+
// ensure no empty track
|
401 |
+
for (let t = 0; t < tracks.length; ++t)
|
402 |
+
tracks[t] = tracks[t] || [];
|
403 |
+
|
404 |
+
// sort & make deltaTime
|
405 |
+
tracks.forEach(events => {
|
406 |
+
events.sort((e1, e2) => eventPriority(e1) - eventPriority(e2));
|
407 |
+
|
408 |
+
let ticks = 0;
|
409 |
+
events.forEach(event => {
|
410 |
+
event.deltaTime = event.ticks - ticks;
|
411 |
+
ticks = event.ticks;
|
412 |
+
});
|
413 |
+
|
414 |
+
events.push({deltaTime: Math.max(finalTick - ticks, 0), type: "meta", subtype: "endOfTrack"});
|
415 |
+
});
|
416 |
+
|
417 |
+
return {
|
418 |
+
header: {
|
419 |
+
formatType: 0,
|
420 |
+
ticksPerBeat: TICKS_PER_BEAT,
|
421 |
+
},
|
422 |
+
tracks,
|
423 |
+
zeroTick,
|
424 |
+
};
|
425 |
+
}
|
426 |
+
|
427 |
+
|
428 |
+
toPerformingNotationWithEvents (measureIndices: number[], options: {trackList?: boolean[]} = {}): MusicNotation.Notation {
|
429 |
+
if (!measureIndices.length)
|
430 |
+
return null;
|
431 |
+
|
432 |
+
const {zeroTick, ...midi} = this.toPerformingMIDI(measureIndices, options);
|
433 |
+
const notation = MusicNotation.Notation.parseMidi(midi);
|
434 |
+
|
435 |
+
assignNotationNoteDataFromEvents(notation);
|
436 |
+
|
437 |
+
let tick = zeroTick;
|
438 |
+
|
439 |
+
notation.measures = measureIndices.map(index => {
|
440 |
+
const startTick = tick;
|
441 |
+
tick += this.measures[index - 1].duration;
|
442 |
+
|
443 |
+
return {
|
444 |
+
index,
|
445 |
+
startTick,
|
446 |
+
endTick: tick,
|
447 |
+
};
|
448 |
+
});
|
449 |
+
|
450 |
+
return notation;
|
451 |
+
}
|
452 |
+
|
453 |
+
|
454 |
+
getContextGroup (measureIndices: number[]): PitchContextTable[] {
|
455 |
+
const contextGroup = this.pitchContextGroup.map(table => table.items.map(item => item.context));
|
456 |
+
const midiNotation = this.toPerformingNotation(measureIndices);
|
457 |
+
|
458 |
+
return PitchContextTable.createPitchContextGroup(contextGroup, midiNotation);
|
459 |
+
}
|
460 |
+
|
461 |
+
|
462 |
+
assignMatcher (matcher: MatcherResult): this {
|
463 |
+
const tickFactor = TICKS_PER_BEAT / matcher.sample.ticksPerBeat;
|
464 |
+
|
465 |
+
matcher.path.forEach((ci, si) => {
|
466 |
+
if (ci >= 0) {
|
467 |
+
const cn = matcher.criterion.notes[ci] as Note;
|
468 |
+
const sn = matcher.sample.notes[si] as Note;
|
469 |
+
sn.ids = cn.ids || [cn.id];
|
470 |
+
sn.measure = cn.measure;
|
471 |
+
}
|
472 |
+
});
|
473 |
+
assignNotationEventsIds(matcher.sample, ["ids", "measure"]);
|
474 |
+
|
475 |
+
// assign sub notes
|
476 |
+
const c2sIndices = Array(matcher.criterion.notes.length).fill(null).map(() => []);
|
477 |
+
matcher.path.forEach((ci, si) => ci >= 0 && c2sIndices[ci].push(si));
|
478 |
+
|
479 |
+
const velocities: {[key: number]: number} = {};
|
480 |
+
c2sIndices.forEach((indices, ci) => {
|
481 |
+
const cn = matcher.criterion.notes[ci] as any;
|
482 |
+
const measure = this.measures[cn.measure - 1];
|
483 |
+
console.assert(!!measure, "cannot find measure for c note:", cn, this.measures.length);
|
484 |
+
|
485 |
+
//const mn = measure.notes.find(note => note.tick === cn.startTick - measure.tick && note.pitch === cn.pitch);
|
486 |
+
const mn = measure.notes.find(note => note.id === cn.id && note.pitch === cn.pitch);
|
487 |
+
console.assert(!!mn, "cannot find measure note for c note:", cn, measure);
|
488 |
+
|
489 |
+
let bias = 0;
|
490 |
+
|
491 |
+
// arpeggio time bias
|
492 |
+
if (mn.implicitType === "arpeggio" && mn.chordPosition) {
|
493 |
+
const referenceDuration = Math.min(mn.duration * 0.5, ARPEGGIO_REFERENCE_DURATION);
|
494 |
+
bias = Math.round(mn.chordPosition.index * referenceDuration / mn.chordPosition.count);
|
495 |
+
}
|
496 |
+
|
497 |
+
const snotes = indices
|
498 |
+
.map(si => matcher.sample.notes[si])
|
499 |
+
.filter((sn, i) => Math.abs(sn.startTick - cn.startTick) < WHOLE_DURATION_MAGNITUDE
|
500 |
+
&& (!mn.afterGrace || i < 1)); // lilypond's afterGrace MIDI parsing is ill, clip notes to avoid error matching
|
501 |
+
|
502 |
+
if (!snotes.length) {
|
503 |
+
//const track = matcher.trackMap[mn.track];
|
504 |
+
|
505 |
+
mn.subNotes = [{
|
506 |
+
//track,
|
507 |
+
startTick: bias,
|
508 |
+
endTick: mn.duration,
|
509 |
+
pitch: mn.pitch,
|
510 |
+
velocity: velocities[mn.track] || 90,
|
511 |
+
}];
|
512 |
+
}
|
513 |
+
else {
|
514 |
+
const referenceTick = measure.tick + mn.tick;
|
515 |
+
//const referenceTick = snotes[0].startTick;
|
516 |
+
//console.log("mn bias:", mn.id, measure.tick + mn.tick - snotes[0].startTick);
|
517 |
+
|
518 |
+
mn.subNotes = snotes.map(sn => {
|
519 |
+
velocities[sn.track] = sn.velocity;
|
520 |
+
|
521 |
+
console.assert(sn.endTick > sn.startTick, "midi note duration is zero or negative:", sn);
|
522 |
+
|
523 |
+
// fix bad subnote duration
|
524 |
+
const duration = sn.endTick - sn.startTick;
|
525 |
+
if (duration > mn.duration * 2)
|
526 |
+
sn.endTick = sn.startTick + mn.duration;
|
527 |
+
|
528 |
+
return {
|
529 |
+
//track: sn.track,
|
530 |
+
startTick: Math.round(sn.startTick - referenceTick + bias),
|
531 |
+
endTick: Math.round(sn.endTick - referenceTick),
|
532 |
+
pitch: sn.pitch,
|
533 |
+
velocity: sn.velocity,
|
534 |
+
};
|
535 |
+
});
|
536 |
+
}
|
537 |
+
});
|
538 |
+
|
539 |
+
// assign events
|
540 |
+
this.measures.forEach(measure => measure.events = []);
|
541 |
+
|
542 |
+
//console.debug("matcher.trackMap:", matcher.trackMap);
|
543 |
+
(matcher.sample.events as MeasureEvent[]).forEach(event => {
|
544 |
+
// exclude note events
|
545 |
+
if (["noteOn", "noteOff"].includes(event.data.subtype))
|
546 |
+
return;
|
547 |
+
|
548 |
+
let track = matcher.trackMap[event.track];
|
549 |
+
if (!Number.isFinite(track)) {
|
550 |
+
const id = event.data.ids && event.data.ids[0];
|
551 |
+
track = id ? (this.idTrackMap[id] || 0) : 0;
|
552 |
+
}
|
553 |
+
|
554 |
+
if (Number.isInteger(event.data.measure)) {
|
555 |
+
const measure = this.measures[event.data.measure - 1];
|
556 |
+
console.assert(!!measure, "measure index is invalid:", event.data.measure, this.measures.length);
|
557 |
+
|
558 |
+
measure.events.push({
|
559 |
+
data: event.data,
|
560 |
+
track,
|
561 |
+
ticks: Math.round(event.ticks * tickFactor - measure.tick),
|
562 |
+
});
|
563 |
+
}
|
564 |
+
else {
|
565 |
+
switch (event.data.subtype) {
|
566 |
+
case "setTempo":
|
567 |
+
case "timeSignature":
|
568 |
+
case "keySignature": {
|
569 |
+
// find container measure by tick range
|
570 |
+
const tick = event.ticks * tickFactor;
|
571 |
+
const measure = this.measures.find(measure => tick >= measure.tick && tick < measure.tick + measure.duration);
|
572 |
+
if (measure) {
|
573 |
+
measure.events.push({
|
574 |
+
data: event.data,
|
575 |
+
track,
|
576 |
+
ticks: Math.round(tick - measure.tick),
|
577 |
+
});
|
578 |
+
}
|
579 |
+
}
|
580 |
+
|
581 |
+
break;
|
582 |
+
}
|
583 |
+
}
|
584 |
+
});
|
585 |
+
|
586 |
+
this.ripe = true;
|
587 |
+
|
588 |
+
return this;
|
589 |
+
}
|
590 |
+
|
591 |
+
|
592 |
+
// find the MIDI event of setTempo in measures data, and change the value of microsecondsPerBeat
|
593 |
+
setTempo (bpm: number): boolean {
|
594 |
+
let found = false;
|
595 |
+
for (const measure of this.measures) {
|
596 |
+
for (const event of measure.events) {
|
597 |
+
if (event.data.subtype === "setTempo") {
|
598 |
+
event.data.microsecondsPerBeat = 60e+6 / bpm;
|
599 |
+
found = true;
|
600 |
+
}
|
601 |
+
}
|
602 |
+
}
|
603 |
+
|
604 |
+
return found;
|
605 |
+
}
|
606 |
+
};
|
607 |
+
|
608 |
+
|
609 |
+
export const assignNotationEventsIds = (midiNotation: MusicNotation.NotationData, fields = ["ids"]) => {
|
610 |
+
const events = midiNotation.notes.reduce((events, note) => {
|
611 |
+
const dict = pick(note, fields);
|
612 |
+
events.push({ticks: note.startTick, subtype: "noteOn", channel: note.channel, pitch: note.pitch, dict});
|
613 |
+
events.push({ticks: note.endTick, subtype: "noteOff", channel: note.channel, pitch: note.pitch, dict});
|
614 |
+
|
615 |
+
["setTempo", "timeSignature", "keySignature"].forEach(subtype => events.push({ticks: note.startTick, subtype, dict}));
|
616 |
+
|
617 |
+
return events;
|
618 |
+
}, []).sort((e1, e2) => e1.ticks - e2.ticks);
|
619 |
+
|
620 |
+
let index = -1;
|
621 |
+
let ticks = -Infinity;
|
622 |
+
for (const event of midiNotation.events) {
|
623 |
+
console.assert(Number.isFinite(event.ticks), "invalid event ticks:", event);
|
624 |
+
|
625 |
+
while (event.ticks > ticks && index < events.length) {
|
626 |
+
++index;
|
627 |
+
ticks = events[index] && events[index].ticks;
|
628 |
+
}
|
629 |
+
|
630 |
+
if (index >= events.length)
|
631 |
+
break;
|
632 |
+
|
633 |
+
if (event.ticks < ticks)
|
634 |
+
continue;
|
635 |
+
|
636 |
+
for (let i = index; i < events.length; ++i) {
|
637 |
+
const ne = events[i];
|
638 |
+
if (!ne) {
|
639 |
+
console.warn("null event:", i, events.length);
|
640 |
+
break;
|
641 |
+
}
|
642 |
+
|
643 |
+
if (ne.ticks > ticks)
|
644 |
+
break;
|
645 |
+
else {
|
646 |
+
if (event.data.subtype === ne.subtype && event.data.channel === ne.channel && event.data.noteNumber === ne.pitch) {
|
647 |
+
Object.assign(event.data, ne.dict);
|
648 |
+
break;
|
649 |
+
}
|
650 |
+
}
|
651 |
+
}
|
652 |
+
}
|
653 |
+
};
|
654 |
+
|
655 |
+
|
656 |
+
const assignNotationNoteDataFromEvents = (midiNotation: MusicNotation.NotationData, fields = ["ids", "measure", "staffTrack"]) => {
|
657 |
+
const noteId = (channel: number, pitch: number, tick: number): string => `${channel}|${pitch}|${tick}`;
|
658 |
+
|
659 |
+
const noteMap = midiNotation.notes.reduce((map, note) => {
|
660 |
+
map[noteId(note.channel, note.pitch, note.startTick)] = note;
|
661 |
+
|
662 |
+
return map;
|
663 |
+
}, {});
|
664 |
+
|
665 |
+
midiNotation.events.forEach(event => {
|
666 |
+
if (event.data.subtype === "noteOn") {
|
667 |
+
const id = noteId(event.data.channel, event.data.noteNumber, event.ticks);
|
668 |
+
const note = noteMap[id];
|
669 |
+
console.assert(!!note, "cannot find note of", id);
|
670 |
+
|
671 |
+
if (note)
|
672 |
+
Object.assign(note, pick(event.data, fields));
|
673 |
+
}
|
674 |
+
});
|
675 |
+
};
|
inc/lilyNotation/scheduler.ts
ADDED
@@ -0,0 +1,167 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
|
2 |
+
import {MusicNotation} from "@k-l-lambda/music-widgets";
|
3 |
+
|
4 |
+
import {ChordPosition} from "./notation";
|
5 |
+
|
6 |
+
|
7 |
+
|
8 |
+
declare class StaffToken {
|
9 |
+
system?: number;
|
10 |
+
measure?: number;
|
11 |
+
x: number;
|
12 |
+
endX?: number;
|
13 |
+
};
|
14 |
+
|
15 |
+
|
16 |
+
interface TickItem {
|
17 |
+
tick: number;
|
18 |
+
endTick?: number;
|
19 |
+
system: number;
|
20 |
+
measure: number;
|
21 |
+
x: number;
|
22 |
+
endX: number;
|
23 |
+
};
|
24 |
+
|
25 |
+
|
26 |
+
interface SheetPosition {
|
27 |
+
system: number,
|
28 |
+
x: number,
|
29 |
+
};
|
30 |
+
|
31 |
+
|
32 |
+
interface Note extends MusicNotation.Note {
|
33 |
+
chordPosition?: ChordPosition;
|
34 |
+
measure?: number;
|
35 |
+
outMeasure?: boolean;
|
36 |
+
};
|
37 |
+
|
38 |
+
|
39 |
+
interface Notation {
|
40 |
+
notes: Note[];
|
41 |
+
endTick: number;
|
42 |
+
};
|
43 |
+
|
44 |
+
|
45 |
+
|
46 |
+
export default class Scheduler {
|
47 |
+
tickTable: TickItem[];
|
48 |
+
|
49 |
+
|
50 |
+
static createFromNotation (midiNotation: Notation, tokenMap: Map<string, StaffToken>): Scheduler {
|
51 |
+
const tokenTable: {[key: number]: StaffToken[]} = {};
|
52 |
+
const idSet = new Set<string>();
|
53 |
+
let measureIndex = 0;
|
54 |
+
|
55 |
+
midiNotation.notes.filter(note => !note.outMeasure && (!note.chordPosition || note.chordPosition.index === 0)).forEach(note => {
|
56 |
+
if (note.ids) {
|
57 |
+
if (note.measure !== measureIndex) {
|
58 |
+
idSet.clear();
|
59 |
+
measureIndex = note.measure;
|
60 |
+
}
|
61 |
+
else if (note.ids.some(id => idSet.has(id)))
|
62 |
+
return;
|
63 |
+
note.ids.forEach(idSet.add.bind(idSet));
|
64 |
+
|
65 |
+
tokenTable[note.startTick] = tokenTable[note.startTick] || [];
|
66 |
+
const tokens = note.ids.map(id => tokenMap.get(id)).filter(t => t);
|
67 |
+
tokenTable[note.startTick].push(...tokens);
|
68 |
+
}
|
69 |
+
});
|
70 |
+
//console.log("tokenTable:", tokenTable);
|
71 |
+
|
72 |
+
return Scheduler.createFromTokenTable(tokenTable, midiNotation.endTick);
|
73 |
+
}
|
74 |
+
|
75 |
+
|
76 |
+
static createFromTokenTable (tokenTable: {[key: number]: StaffToken[]}, endTick: number): Scheduler {
|
77 |
+
const tickTable: TickItem[] = Object.entries(tokenTable).map(([tick, tokens]) => {
|
78 |
+
if (!tokens.length)
|
79 |
+
return null;
|
80 |
+
|
81 |
+
tokens.sort((t1, t2) => t1.system === t2.system ? t1.x - t2.x : t1.system - t2.system);
|
82 |
+
const token = tokens[0];
|
83 |
+
|
84 |
+
return {
|
85 |
+
tick: Number(tick),
|
86 |
+
system: token.system,
|
87 |
+
measure: token.measure,
|
88 |
+
x: token.x,
|
89 |
+
endX: token.endX,
|
90 |
+
};
|
91 |
+
}).filter(Boolean).sort((i1, i2) => i1.tick - i2.tick);
|
92 |
+
//console.log("sequence:", sequence);
|
93 |
+
|
94 |
+
tickTable.forEach((item, i) => {
|
95 |
+
const nextItem = tickTable[i + 1];
|
96 |
+
|
97 |
+
item.endTick = nextItem ? nextItem.tick : endTick;
|
98 |
+
console.assert(item.endTick > item.tick, "invalid tick item, tick span is non-positive:", item, tokenTable[item.tick]);
|
99 |
+
|
100 |
+
if (nextItem && item.system === nextItem.system && [0, 1].includes(nextItem.measure - item.measure))
|
101 |
+
item.endX = nextItem.x;
|
102 |
+
});
|
103 |
+
|
104 |
+
return new Scheduler({
|
105 |
+
tickTable,
|
106 |
+
});
|
107 |
+
}
|
108 |
+
|
109 |
+
|
110 |
+
constructor ({tickTable}: {tickTable: TickItem[]}) {
|
111 |
+
console.assert(tickTable.length > 0, "invalid tick table:", tickTable);
|
112 |
+
|
113 |
+
this.tickTable = tickTable;
|
114 |
+
}
|
115 |
+
|
116 |
+
|
117 |
+
get startTick (): number {
|
118 |
+
if (!this.tickTable[0])
|
119 |
+
return null;
|
120 |
+
|
121 |
+
return this.tickTable[0].tick;
|
122 |
+
}
|
123 |
+
|
124 |
+
|
125 |
+
get endTick (): number {
|
126 |
+
if (!this.tickTable[0])
|
127 |
+
return null;
|
128 |
+
|
129 |
+
return this.tickTable[this.tickTable.length - 1].endTick;
|
130 |
+
}
|
131 |
+
|
132 |
+
|
133 |
+
lookupPosition (tick: number): SheetPosition {
|
134 |
+
tick = Math.max(Math.min(tick, this.endTick), this.startTick);
|
135 |
+
|
136 |
+
const item = this.tickTable.find(item => item.tick <= tick && item.endTick > tick) || this.tickTable[this.tickTable.length - 1];
|
137 |
+
if (!item) {
|
138 |
+
console.warn("cannot find tick item:", tick, this.tickTable);
|
139 |
+
return null;
|
140 |
+
}
|
141 |
+
|
142 |
+
const x = item.x + (tick - item.tick) * (item.endX - item.x) / (item.endTick - item.tick);
|
143 |
+
|
144 |
+
return {
|
145 |
+
system: item.system,
|
146 |
+
x,
|
147 |
+
};
|
148 |
+
}
|
149 |
+
|
150 |
+
|
151 |
+
lookupTick (position: SheetPosition): number {
|
152 |
+
const rowItems = this.tickTable.filter(item => item.system === position.system).sort((i1, i2) => i1.x - i2.x);
|
153 |
+
let item = rowItems.find(item => item.x <= position.x && item.endX >= position.x);
|
154 |
+
if (!item) {
|
155 |
+
const firstItem = rowItems[0];
|
156 |
+
if (firstItem && position.x < firstItem.endX)
|
157 |
+
item = firstItem;
|
158 |
+
else
|
159 |
+
//console.warn("lookup position out of range:", position, this.tickTable);
|
160 |
+
return null;
|
161 |
+
}
|
162 |
+
|
163 |
+
const tick = item.tick + Math.max(position.x - item.x, 0) * (item.endTick - item.tick) / (item.endX - item.x);
|
164 |
+
|
165 |
+
return tick;
|
166 |
+
}
|
167 |
+
};
|
inc/lilyParser/idioms.ts
ADDED
@@ -0,0 +1,51 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
|
2 |
+
const PHONETS = "cdefgab";
|
3 |
+
const PHONETS_ALIAS = {
|
4 |
+
h: "b",
|
5 |
+
};
|
6 |
+
|
7 |
+
const phonetDifferToShift = differ => differ > 3 ? -1 : (differ < -3 ? 1 : 0);
|
8 |
+
|
9 |
+
const PHONET_VALUES = {
|
10 |
+
c: 0, d: 2, e: 4, f: 5, g: 7, a: 9, b: 11,
|
11 |
+
};
|
12 |
+
|
13 |
+
|
14 |
+
const SHARP_ALTERS = [
|
15 |
+
"is", "d", "s", "iss", "k", "sharp",
|
16 |
+
];
|
17 |
+
const FLAT_ALTERS = [
|
18 |
+
"es", "b", "f", "ess", "flat",
|
19 |
+
];
|
20 |
+
// in Dutch, 'as', 'es' means a flat/e flat, that's not supported here
|
21 |
+
|
22 |
+
const DOUBLE_SHARP_ALTERS = [
|
23 |
+
...SHARP_ALTERS.map(word => word + word), "x",
|
24 |
+
];
|
25 |
+
const DOUBLE_FLAT_ALTERS = [
|
26 |
+
...FLAT_ALTERS.map(word => word + word), "ses",
|
27 |
+
];
|
28 |
+
|
29 |
+
const alterTablize = (list, value) => list.reduce((table, word) => ((table[word] = value), table), {});
|
30 |
+
|
31 |
+
|
32 |
+
const ALTER_VALUES = {
|
33 |
+
...alterTablize(SHARP_ALTERS, 1),
|
34 |
+
...alterTablize(FLAT_ALTERS, -1),
|
35 |
+
...alterTablize(DOUBLE_SHARP_ALTERS, 2),
|
36 |
+
...alterTablize(DOUBLE_FLAT_ALTERS, -2),
|
37 |
+
};
|
38 |
+
|
39 |
+
|
40 |
+
const FIFTH_PHONETS = "fcgdaeb";
|
41 |
+
|
42 |
+
|
43 |
+
|
44 |
+
export {
|
45 |
+
PHONETS,
|
46 |
+
PHONETS_ALIAS,
|
47 |
+
phonetDifferToShift,
|
48 |
+
PHONET_VALUES,
|
49 |
+
ALTER_VALUES,
|
50 |
+
FIFTH_PHONETS,
|
51 |
+
};
|
inc/lilyParser/index.ts
ADDED
@@ -0,0 +1,28 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
|
2 |
+
import {termDictionary, getDurationSubdivider, BaseTerm, MusicEvent} from "./lilyTerms";
|
3 |
+
import LilyDocument from "./lilyDocument";
|
4 |
+
import LilyInterpreter, {MusicTrack, TrackContext, TremoloType} from "./lilyInterpreter";
|
5 |
+
export * from "./utils";
|
6 |
+
|
7 |
+
import * as measures from "./measures";
|
8 |
+
import {createPianoRhythm} from "./pianoRhythm";
|
9 |
+
|
10 |
+
|
11 |
+
|
12 |
+
const LilyTerms = termDictionary;
|
13 |
+
|
14 |
+
|
15 |
+
|
16 |
+
export {
|
17 |
+
LilyDocument,
|
18 |
+
LilyInterpreter,
|
19 |
+
MusicTrack,
|
20 |
+
TrackContext,
|
21 |
+
TremoloType,
|
22 |
+
BaseTerm,
|
23 |
+
MusicEvent,
|
24 |
+
LilyTerms,
|
25 |
+
getDurationSubdivider,
|
26 |
+
measures,
|
27 |
+
createPianoRhythm,
|
28 |
+
};
|
inc/lilyParser/lilyDocument.ts
ADDED
@@ -0,0 +1,1062 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
|
2 |
+
import TextSource from "../textSource";
|
3 |
+
import {LILY_STAFF_SIZE_DEFAULT} from "../constants";
|
4 |
+
import {
|
5 |
+
parseRaw,
|
6 |
+
BaseTerm, Assignment, LiteralString, Command, Variable, MarkupCommand, Grace, AfterGrace, Include, Version, Block, InlineBlock,
|
7 |
+
Scheme, Chord, BriefChord, Lyric, MusicBlock, SimultaneousList, ContextedMusic, Divide, Tempo, PostEvent, Primitive, ChordElement, MusicEvent,
|
8 |
+
SchemePointer, Comment, Language, StemDirection,
|
9 |
+
} from "./lilyTerms";
|
10 |
+
import LilyInterpreter from "./lilyInterpreter";
|
11 |
+
import {MAIN_SCORE_NAME, DocLocation} from "./utils";
|
12 |
+
|
13 |
+
// eslint-disable-next-line
|
14 |
+
import {Root} from "./lilyTerms";
|
15 |
+
|
16 |
+
|
17 |
+
|
18 |
+
type AttributeValue = number | boolean | string | BaseTerm;
|
19 |
+
|
20 |
+
interface AttributeValueHandle {
|
21 |
+
value: AttributeValue;
|
22 |
+
};
|
23 |
+
|
24 |
+
|
25 |
+
export interface LilyDocumentAttribute {
|
26 |
+
[key: string]: AttributeValueHandle
|
27 |
+
};
|
28 |
+
|
29 |
+
export interface LilyDocumentAttributeReadOnly {
|
30 |
+
staffSize: number;
|
31 |
+
[key: string]: AttributeValue
|
32 |
+
};
|
33 |
+
|
34 |
+
|
35 |
+
|
36 |
+
export default class LilyDocument {
|
37 |
+
root: Root;
|
38 |
+
|
39 |
+
cacheInterpreter?: LilyInterpreter;
|
40 |
+
|
41 |
+
reservedVariables?: Set<string>;
|
42 |
+
|
43 |
+
|
44 |
+
constructor (data: object) {
|
45 |
+
//console.log("raw data:", data);
|
46 |
+
this.root = parseRaw(data);
|
47 |
+
}
|
48 |
+
|
49 |
+
|
50 |
+
toString () {
|
51 |
+
return this.root.join();
|
52 |
+
//return this.root.serialize();
|
53 |
+
}
|
54 |
+
|
55 |
+
|
56 |
+
interpret ({useCached = true} = {}): LilyInterpreter {
|
57 |
+
if (!useCached || !this.cacheInterpreter) {
|
58 |
+
this.cacheInterpreter = new LilyInterpreter();
|
59 |
+
this.cacheInterpreter.interpretDocument(this);
|
60 |
+
}
|
61 |
+
|
62 |
+
return this.cacheInterpreter;
|
63 |
+
}
|
64 |
+
|
65 |
+
|
66 |
+
globalAttributes ({readonly = false} = {}): LilyDocumentAttribute | LilyDocumentAttributeReadOnly {
|
67 |
+
const globalStaffSize = this.root.getField("set-global-staff-size");
|
68 |
+
const header = this.root.getBlock("header");
|
69 |
+
let paper = this.root.getBlock("paper");
|
70 |
+
const layoutStaffSize = paper && paper.getField("layout-set-staff-size");
|
71 |
+
let staffSize = globalStaffSize || layoutStaffSize;
|
72 |
+
|
73 |
+
if (!readonly) {
|
74 |
+
let sectionsDirty = false;
|
75 |
+
|
76 |
+
if (!staffSize) {
|
77 |
+
this.root.sections.push(new Scheme({exp: {proto: "SchemeFunction", func: "set-global-staff-size", args: [LILY_STAFF_SIZE_DEFAULT]}}));
|
78 |
+
staffSize = this.root.getField("set-global-staff-size");
|
79 |
+
|
80 |
+
sectionsDirty = true;
|
81 |
+
}
|
82 |
+
|
83 |
+
// A4 paper size
|
84 |
+
const DEFAULT_PAPER_WIDTH = {
|
85 |
+
proto: "Assignment",
|
86 |
+
key: "paper-width",
|
87 |
+
value: {proto: "NumberUnit", number: 21, unit: "\\cm"},
|
88 |
+
};
|
89 |
+
const DEFAULT_PAPER_HEIGHT = {
|
90 |
+
proto: "Assignment",
|
91 |
+
key: "paper-height",
|
92 |
+
value: {proto: "NumberUnit", number: 29.71, unit: "\\cm"},
|
93 |
+
};
|
94 |
+
|
95 |
+
if (!paper) {
|
96 |
+
paper = new Block({
|
97 |
+
block: "score",
|
98 |
+
head: "\\paper",
|
99 |
+
body: [DEFAULT_PAPER_WIDTH, DEFAULT_PAPER_HEIGHT],
|
100 |
+
});
|
101 |
+
this.root.sections.push(paper);
|
102 |
+
|
103 |
+
sectionsDirty = true;
|
104 |
+
}
|
105 |
+
|
106 |
+
if (!paper.getField("paper-width"))
|
107 |
+
paper.body.push(parseRaw(DEFAULT_PAPER_WIDTH));
|
108 |
+
|
109 |
+
if (!paper.getField("paper-height"))
|
110 |
+
paper.body.push(parseRaw(DEFAULT_PAPER_HEIGHT));
|
111 |
+
|
112 |
+
if (sectionsDirty)
|
113 |
+
this.root.reorderSections();
|
114 |
+
}
|
115 |
+
else
|
116 |
+
staffSize = staffSize || {value: LILY_STAFF_SIZE_DEFAULT};
|
117 |
+
|
118 |
+
const blockPropertyCommon = (block: Block, key: string) => ({
|
119 |
+
get value () {
|
120 |
+
if (!block)
|
121 |
+
return undefined;
|
122 |
+
|
123 |
+
const item = block.getField(key);
|
124 |
+
if (!item)
|
125 |
+
return undefined;
|
126 |
+
|
127 |
+
return item.value;
|
128 |
+
},
|
129 |
+
|
130 |
+
set value (value) {
|
131 |
+
console.assert(!!block, "block is null.");
|
132 |
+
|
133 |
+
if (value === undefined) // delete field
|
134 |
+
block.body = block.body.filter(assign => !(assign instanceof Assignment) || assign.key !== key);
|
135 |
+
else {
|
136 |
+
const item = block.getField(key);
|
137 |
+
if (item)
|
138 |
+
item.value = parseRaw(value);
|
139 |
+
else
|
140 |
+
block.body.push(new Assignment({key, value}));
|
141 |
+
}
|
142 |
+
},
|
143 |
+
});
|
144 |
+
const paperPropertyCommon = key => blockPropertyCommon(paper, key);
|
145 |
+
|
146 |
+
const paperPropertySchemeToken = key => ({
|
147 |
+
get value () {
|
148 |
+
if (!paper)
|
149 |
+
return undefined;
|
150 |
+
|
151 |
+
const item = paper.getField(key);
|
152 |
+
if (!item)
|
153 |
+
return undefined;
|
154 |
+
|
155 |
+
return item.value.exp;
|
156 |
+
},
|
157 |
+
|
158 |
+
set value (value) {
|
159 |
+
console.assert(!!paper, "paper is null.");
|
160 |
+
|
161 |
+
const item = paper.getField(key);
|
162 |
+
if (item)
|
163 |
+
item.value.exp = value;
|
164 |
+
else
|
165 |
+
paper.body.push(new Assignment({key, value: {proto: "Scheme", exp: value}}));
|
166 |
+
},
|
167 |
+
});
|
168 |
+
|
169 |
+
let midiBlock = null;
|
170 |
+
const scores = this.root.sections.filter(section => section instanceof Block && section.head === "\\score") as Block[];
|
171 |
+
for (const score of scores) {
|
172 |
+
midiBlock = score.body.find(term => term instanceof Block && term.head === "\\midi");
|
173 |
+
if (midiBlock)
|
174 |
+
break;
|
175 |
+
}
|
176 |
+
|
177 |
+
const midiTempo = {
|
178 |
+
get value (): Tempo {
|
179 |
+
return midiBlock && midiBlock.body.find(term => term instanceof Tempo);
|
180 |
+
},
|
181 |
+
|
182 |
+
set value (value: Tempo) {
|
183 |
+
if (!midiBlock) {
|
184 |
+
const score = this.root.getBlock("score");
|
185 |
+
if (score) {
|
186 |
+
midiBlock = new Block({block: "score", head: "\\midi", body: []});
|
187 |
+
score.body.push(midiBlock);
|
188 |
+
}
|
189 |
+
else
|
190 |
+
console.warn("no score block, midiTempo assign failed.");
|
191 |
+
}
|
192 |
+
|
193 |
+
if (midiBlock) {
|
194 |
+
midiBlock.body = midiBlock.body.filter(term => !(term instanceof Tempo));
|
195 |
+
midiBlock.body.push(value);
|
196 |
+
}
|
197 |
+
},
|
198 |
+
};
|
199 |
+
|
200 |
+
const assignments = this.root.entries.filter(term => term instanceof Assignment) as Assignment[];
|
201 |
+
const assignmentTable = assignments.reduce((table, assign) => ((table[assign.key.toString()] = assign.query(assign.key)), table), {});
|
202 |
+
|
203 |
+
const headerFields = [
|
204 |
+
"title", "subtitle", "subsubtitle", "composer", "poet", "arranger", "opus", "copyright", "instrument", "dedication", "tagline",
|
205 |
+
].reduce((dict, field) => ((dict[field] = blockPropertyCommon(header, field)), dict), {});
|
206 |
+
|
207 |
+
const attributes = {
|
208 |
+
staffSize,
|
209 |
+
midiTempo,
|
210 |
+
...headerFields,
|
211 |
+
paperWidth: paperPropertyCommon("paper-width"),
|
212 |
+
paperHeight: paperPropertyCommon("paper-height"),
|
213 |
+
topMargin: paperPropertyCommon("top-margin"),
|
214 |
+
bottomMargin: paperPropertyCommon("bottom-margin"),
|
215 |
+
leftMargin: paperPropertyCommon("left-margin"),
|
216 |
+
rightMargin: paperPropertyCommon("right-margin"),
|
217 |
+
systemSpacing: paperPropertySchemeToken("system-system-spacing.basic-distance"),
|
218 |
+
topMarkupSpacing: paperPropertySchemeToken("top-markup-spacing.basic-distance"),
|
219 |
+
raggedLast: paperPropertySchemeToken("ragged-last"),
|
220 |
+
raggedBottom: paperPropertySchemeToken("ragged-bottom"),
|
221 |
+
raggedLastBottom: paperPropertySchemeToken("ragged-last-bottom"),
|
222 |
+
printPageNumber: paperPropertySchemeToken("print-page-number"),
|
223 |
+
...assignmentTable,
|
224 |
+
};
|
225 |
+
|
226 |
+
if (readonly)
|
227 |
+
Object.keys(attributes).forEach(key => attributes[key] = attributes[key] && attributes[key].value);
|
228 |
+
|
229 |
+
return attributes;
|
230 |
+
}
|
231 |
+
|
232 |
+
|
233 |
+
globalAttributesReadOnly (): LilyDocumentAttributeReadOnly {
|
234 |
+
const attributes = this.globalAttributes() as any;
|
235 |
+
Object.keys(attributes).forEach(key => attributes[key] = attributes[key] && attributes[key].value);
|
236 |
+
|
237 |
+
return attributes;
|
238 |
+
}
|
239 |
+
|
240 |
+
|
241 |
+
markup (docMarkup: LilyDocument) {
|
242 |
+
// copy attributes
|
243 |
+
const attrS = this.globalAttributes() as LilyDocumentAttribute;
|
244 |
+
const attrM = docMarkup.globalAttributesReadOnly();
|
245 |
+
|
246 |
+
[
|
247 |
+
"staffSize", "midiTempo", "paperWidth", "paperHeight",
|
248 |
+
"topMargin", "bottomMargin", "leftMargin", "rightMargin",
|
249 |
+
"systemSpacing", "topMarkupSpacing", "raggedLast", "raggedBottom", "raggedLastBottom",
|
250 |
+
"printPageNumber",
|
251 |
+
].forEach(field => {
|
252 |
+
if (attrM[field] !== undefined) {
|
253 |
+
if (typeof attrS[field].value === "object" && attrS[field].value && (attrS[field].value as any).set)
|
254 |
+
(attrS[field].value as any).set(attrM[field]);
|
255 |
+
else
|
256 |
+
attrS[field].value = attrM[field];
|
257 |
+
}
|
258 |
+
});
|
259 |
+
|
260 |
+
// execute commands list
|
261 |
+
const commands = docMarkup.root.getField("LotusCommands");
|
262 |
+
const cmdList = commands && commands.value && commands.value.args && commands.value.args[0].body;
|
263 |
+
if (cmdList && Array.isArray(cmdList)) {
|
264 |
+
for (const command of cmdList) {
|
265 |
+
if (command.exp && this[command.exp])
|
266 |
+
this[command.exp]();
|
267 |
+
else
|
268 |
+
console.warn("unexpected markup command:", command);
|
269 |
+
}
|
270 |
+
}
|
271 |
+
|
272 |
+
// copy LotusOption assignments
|
273 |
+
const assignments = docMarkup.root.entries.filter(term => term instanceof Assignment && /^LotusOption\..+/.test(term.key.toString()));
|
274 |
+
assignments.forEach(assignment => this.root.sections.push(assignment.clone()));
|
275 |
+
|
276 |
+
// copy score blocks
|
277 |
+
const layoutBody = [];
|
278 |
+
const midiBody = [];
|
279 |
+
const score = docMarkup.root.getBlock("score");
|
280 |
+
if (score) {
|
281 |
+
const layout = score.body.find(term => term instanceof Block && term.head === "\\layout") as Block;
|
282 |
+
if (layout)
|
283 |
+
layout.body.forEach(term => layoutBody.push(term.clone()));
|
284 |
+
|
285 |
+
const midi = score.body.find(term => term instanceof Block && term.head === "\\midi") as Block;
|
286 |
+
if (midi)
|
287 |
+
midi.body.forEach(term => midiBody.push(term.clone()));
|
288 |
+
}
|
289 |
+
if (layoutBody.length || midiBody.length) {
|
290 |
+
const thisScore = this.root.getBlock("score");
|
291 |
+
if (thisScore) {
|
292 |
+
const layout = thisScore.body.find(term => term instanceof Block && term.head === "\\layout") as Block;
|
293 |
+
if (layout)
|
294 |
+
layout.body.push(...layoutBody);
|
295 |
+
|
296 |
+
const midi = thisScore.body.find(term => term instanceof Block && term.head === "\\midi") as Block;
|
297 |
+
if (midi)
|
298 |
+
midi.body.push(...midiBody);
|
299 |
+
}
|
300 |
+
}
|
301 |
+
}
|
302 |
+
|
303 |
+
|
304 |
+
getVariables (): Set<string> {
|
305 |
+
return new Set(this.root.findAll(Variable).map(variable => variable.name));
|
306 |
+
}
|
307 |
+
|
308 |
+
|
309 |
+
// deprecated
|
310 |
+
getMusicTracks ({expand = false} = {}): MusicBlock[] {
|
311 |
+
const score = this.root.getBlock("score");
|
312 |
+
if (!score)
|
313 |
+
return null;
|
314 |
+
|
315 |
+
let tracks = [];
|
316 |
+
|
317 |
+
// extract sequential music blocks from score block
|
318 |
+
score.forEachTopTerm(MusicBlock, block => {
|
319 |
+
tracks.push(block);
|
320 |
+
});
|
321 |
+
|
322 |
+
// expand variables in tracks
|
323 |
+
if (expand)
|
324 |
+
tracks = tracks.map(track => track.clone().expandVariables(this.root));
|
325 |
+
|
326 |
+
return tracks;
|
327 |
+
}
|
328 |
+
|
329 |
+
|
330 |
+
getLocationTickTable (): {[key: string]: number} {
|
331 |
+
const notes = this.root.findAll(term => (term instanceof ChordElement) || (term instanceof MusicEvent));
|
332 |
+
|
333 |
+
return notes.reduce((table, note) => {
|
334 |
+
if (note._location && Number.isFinite(note._tick))
|
335 |
+
table[`${note._location.lines[0]}:${note._location.columns[0]}`] = note._tick;
|
336 |
+
|
337 |
+
return table;
|
338 |
+
}, {});
|
339 |
+
}
|
340 |
+
|
341 |
+
|
342 |
+
// update terms' _location data according to a serialized source
|
343 |
+
relocate (source: string = this.toString()) {
|
344 |
+
this.root.relocate(source);
|
345 |
+
}
|
346 |
+
|
347 |
+
|
348 |
+
appendIncludeFile (filename: string) {
|
349 |
+
if (!this.root.includeFiles.includes(filename)) {
|
350 |
+
const versionPos = this.root.sections.findIndex(term => term instanceof Version);
|
351 |
+
this.root.sections.splice(versionPos + 1, 0,
|
352 |
+
new Include({cmd: "include", args: [LiteralString.fromString(filename)]}));
|
353 |
+
}
|
354 |
+
}
|
355 |
+
|
356 |
+
|
357 |
+
removeStaffGroup () {
|
358 |
+
const score = this.root.getBlock("score");
|
359 |
+
if (score) {
|
360 |
+
score.body.forEach(item => {
|
361 |
+
if (item instanceof SimultaneousList)
|
362 |
+
item.removeStaffGroup();
|
363 |
+
});
|
364 |
+
}
|
365 |
+
}
|
366 |
+
|
367 |
+
|
368 |
+
fixTinyTrillSpans () {
|
369 |
+
// TODO: replace successive \startTrillSpan & \stopTrillSpan with ^\trill
|
370 |
+
}
|
371 |
+
|
372 |
+
|
373 |
+
removeMusicCommands (cmds: string | string[]) {
|
374 |
+
cmds = Array.isArray(cmds) ? cmds : [cmds];
|
375 |
+
|
376 |
+
const isToRemoved = item => (item instanceof Command) && cmds.includes(item.cmd);
|
377 |
+
|
378 |
+
this.root.forEachTerm(MusicBlock, block => {
|
379 |
+
block.body = block.body.filter(item => !isToRemoved(item));
|
380 |
+
});
|
381 |
+
}
|
382 |
+
|
383 |
+
|
384 |
+
removeTrillSpans () {
|
385 |
+
this.removeMusicCommands(["startTrillSpan", "stopTrillSpan"]);
|
386 |
+
}
|
387 |
+
|
388 |
+
|
389 |
+
removeBreaks () {
|
390 |
+
this.removeMusicCommands("break");
|
391 |
+
}
|
392 |
+
|
393 |
+
|
394 |
+
removePageBreaks () {
|
395 |
+
this.removeMusicCommands("pageBreak");
|
396 |
+
}
|
397 |
+
|
398 |
+
|
399 |
+
scoreBreakBefore (enabled = true) {
|
400 |
+
const score = this.root.getBlock("score");
|
401 |
+
if (score) {
|
402 |
+
let header = score.entries.find((entry: any) => entry.head === "\\header") as Block;
|
403 |
+
if (!header) {
|
404 |
+
header = new Block({head: "\\header", body: []});
|
405 |
+
score.body.push(header);
|
406 |
+
}
|
407 |
+
|
408 |
+
let breakbefore = header.getField("breakbefore");
|
409 |
+
if (breakbefore)
|
410 |
+
breakbefore = breakbefore.value;
|
411 |
+
|
412 |
+
else {
|
413 |
+
breakbefore = new Scheme({exp: true});
|
414 |
+
header.body.push(new Assignment({key: "breakbefore", value: breakbefore}));
|
415 |
+
}
|
416 |
+
|
417 |
+
breakbefore.exp = enabled;
|
418 |
+
}
|
419 |
+
else
|
420 |
+
console.warn("no score block");
|
421 |
+
}
|
422 |
+
|
423 |
+
|
424 |
+
unfoldRepeats () {
|
425 |
+
const score = this.root.getBlock("score");
|
426 |
+
const musicList = score ? score.body : this.root.sections;
|
427 |
+
|
428 |
+
let count = 0;
|
429 |
+
|
430 |
+
musicList.forEach((term, i) => {
|
431 |
+
if (term.isMusic && (term as Command).cmd !== "unfoldRepeats") {
|
432 |
+
const unfold = new Command({cmd: "unfoldRepeats", args: [term]});
|
433 |
+
musicList.splice(i, 1, unfold);
|
434 |
+
|
435 |
+
++count;
|
436 |
+
}
|
437 |
+
});
|
438 |
+
|
439 |
+
if (!count)
|
440 |
+
console.warn("no music term to unfold");
|
441 |
+
|
442 |
+
return count;
|
443 |
+
}
|
444 |
+
|
445 |
+
|
446 |
+
containsRepeat (): boolean {
|
447 |
+
const termContainsRepeat = (term: BaseTerm): boolean => {
|
448 |
+
if (!term.entries)
|
449 |
+
return false;
|
450 |
+
|
451 |
+
const subTerms = term.entries.filter(term => term instanceof BaseTerm);
|
452 |
+
|
453 |
+
for (const term of subTerms) {
|
454 |
+
if ((term as Command).cmd === "repeat")
|
455 |
+
return true;
|
456 |
+
}
|
457 |
+
|
458 |
+
for (const term of subTerms) {
|
459 |
+
if (termContainsRepeat(term))
|
460 |
+
return true;
|
461 |
+
}
|
462 |
+
|
463 |
+
return false;
|
464 |
+
};
|
465 |
+
|
466 |
+
return termContainsRepeat(this.root);
|
467 |
+
}
|
468 |
+
|
469 |
+
|
470 |
+
removeEmptySubMusicBlocks () {
|
471 |
+
this.root.forEachTerm(MusicBlock, block => {
|
472 |
+
block.body = block.body.filter(term => !(term instanceof MusicBlock && term.body.length === 0));
|
473 |
+
});
|
474 |
+
}
|
475 |
+
|
476 |
+
|
477 |
+
mergeContinuousGraces () {
|
478 |
+
this.removeEmptySubMusicBlocks();
|
479 |
+
|
480 |
+
const isGraceCommand = term => term instanceof Grace;
|
481 |
+
const isGraceInnerTerm = term => isGraceCommand(term) || term instanceof Divide || term instanceof PostEvent;
|
482 |
+
|
483 |
+
this.root.forEachTerm(MusicBlock, block => {
|
484 |
+
const groups = [];
|
485 |
+
let currentGroup = null;
|
486 |
+
|
487 |
+
block.body.forEach((term, i) => {
|
488 |
+
if (currentGroup) {
|
489 |
+
if (isGraceInnerTerm(term)) {
|
490 |
+
currentGroup.count++;
|
491 |
+
|
492 |
+
if (currentGroup.count === 2)
|
493 |
+
groups.push(currentGroup);
|
494 |
+
}
|
495 |
+
else
|
496 |
+
currentGroup = null;
|
497 |
+
}
|
498 |
+
else {
|
499 |
+
if (isGraceCommand(term))
|
500 |
+
currentGroup = {start: i, count: 1};
|
501 |
+
}
|
502 |
+
});
|
503 |
+
|
504 |
+
let offset = 0;
|
505 |
+
groups.forEach(group => {
|
506 |
+
const startIndex = group.start + offset;
|
507 |
+
const mainBody = new MusicBlock({body: []});
|
508 |
+
|
509 |
+
for (let i = startIndex; i < startIndex + group.count; ++ i) {
|
510 |
+
const term = block.body[i];
|
511 |
+
const music = isGraceCommand(term) ? term.args[0] : term;
|
512 |
+
if (music instanceof MusicBlock)
|
513 |
+
mainBody.body.push(...music.body);
|
514 |
+
else
|
515 |
+
mainBody.body.push(music);
|
516 |
+
}
|
517 |
+
|
518 |
+
block.body[startIndex].args[0] = mainBody;
|
519 |
+
block.body.splice(startIndex + 1, group.count - 1);
|
520 |
+
|
521 |
+
offset -= group.count - 1;
|
522 |
+
});
|
523 |
+
});
|
524 |
+
}
|
525 |
+
|
526 |
+
|
527 |
+
mergeContinuousEmptyAfterGraces () {
|
528 |
+
const isEmptyAfterGrace = term => term instanceof AfterGrace && term.args[0] instanceof MusicBlock && term.args[0].body.length === 0;
|
529 |
+
const isGraceInnerTerm = term => isEmptyAfterGrace(term) || term instanceof Divide || term instanceof PostEvent;
|
530 |
+
|
531 |
+
this.root.forEachTerm(MusicBlock, block => {
|
532 |
+
const groups = [];
|
533 |
+
let currentGroup = null;
|
534 |
+
|
535 |
+
block.body.forEach((term, i) => {
|
536 |
+
if (currentGroup) {
|
537 |
+
if (isGraceInnerTerm(term)) {
|
538 |
+
currentGroup.count++;
|
539 |
+
|
540 |
+
if (currentGroup.count === 2)
|
541 |
+
groups.push(currentGroup);
|
542 |
+
}
|
543 |
+
else
|
544 |
+
currentGroup = null;
|
545 |
+
}
|
546 |
+
else {
|
547 |
+
if (isEmptyAfterGrace(term))
|
548 |
+
currentGroup = {start: i, count: 1};
|
549 |
+
}
|
550 |
+
});
|
551 |
+
|
552 |
+
let offset = 0;
|
553 |
+
groups.forEach(group => {
|
554 |
+
const startIndex = group.start + offset;
|
555 |
+
const mainBody = new MusicBlock({body: []});
|
556 |
+
|
557 |
+
for (let i = startIndex; i < startIndex + group.count; ++ i) {
|
558 |
+
const term = block.body[i];
|
559 |
+
const music = isEmptyAfterGrace(term) ? term.args[1] : term;
|
560 |
+
if (music instanceof MusicBlock)
|
561 |
+
mainBody.body.push(...music.body);
|
562 |
+
else
|
563 |
+
mainBody.body.push(music);
|
564 |
+
}
|
565 |
+
|
566 |
+
block.body[startIndex].args[1] = mainBody;
|
567 |
+
block.body.splice(startIndex + 1, group.count - 1);
|
568 |
+
|
569 |
+
offset -= group.count - 1;
|
570 |
+
});
|
571 |
+
});
|
572 |
+
}
|
573 |
+
|
574 |
+
|
575 |
+
fixInvalidKeys (mode = "major") {
|
576 |
+
this.root.forEachTerm(Command, cmd => {
|
577 |
+
if (cmd.cmd === "key") {
|
578 |
+
if (cmd.args[1] === "\\none")
|
579 |
+
cmd.args[1] = "\\" + mode;
|
580 |
+
}
|
581 |
+
});
|
582 |
+
}
|
583 |
+
|
584 |
+
|
585 |
+
fixInvalidBriefChords () {
|
586 |
+
this.root.forEachTerm(BriefChord, chord => {
|
587 |
+
const items = chord.body.items;
|
588 |
+
if (items) {
|
589 |
+
// merge multiple ^ items
|
590 |
+
while (items.filter(item => item === "^").length > 1) {
|
591 |
+
const index = items.lastIndexOf("^");
|
592 |
+
items.splice(index, 1, ".");
|
593 |
+
}
|
594 |
+
}
|
595 |
+
});
|
596 |
+
}
|
597 |
+
|
598 |
+
|
599 |
+
fixInvalidMarkupWords () {
|
600 |
+
this.root.forEachTerm(MarkupCommand, cmd => {
|
601 |
+
//console.log("markup:", cmd);
|
602 |
+
cmd.forEachTerm(InlineBlock, block => {
|
603 |
+
// replace scheme expression by literal string
|
604 |
+
block.body = block.body.map(term => {
|
605 |
+
if (term instanceof Scheme)
|
606 |
+
return LiteralString.fromString(term.join().replace(/\s+$/, ""));
|
607 |
+
|
608 |
+
if (typeof term === "string" && term.includes("$"))
|
609 |
+
return LiteralString.fromString(term);
|
610 |
+
|
611 |
+
return term;
|
612 |
+
});
|
613 |
+
});
|
614 |
+
});
|
615 |
+
}
|
616 |
+
|
617 |
+
|
618 |
+
fixNestedRepeat () {
|
619 |
+
// \repeat { \repeat { P1 } \alternative { {P2} } } \alternative { {P3} }
|
620 |
+
// ->
|
621 |
+
// \repeat { P1 } \alternative { {P2} {P3} }
|
622 |
+
this.root.forEachTerm(Command, cmd => {
|
623 |
+
if (cmd.isRepeatWithAlternative) {
|
624 |
+
const block = cmd.args[2];
|
625 |
+
const alternative = cmd.args[3].args[0];
|
626 |
+
const lastMusic = block.body[block.body.length - 1];
|
627 |
+
if (lastMusic && lastMusic.isRepeatWithAlternative) {
|
628 |
+
block.body.splice(block.body.length - 1, 1, ...lastMusic.args[2].body);
|
629 |
+
alternative.body = [...lastMusic.args[3].args[0].body, ...alternative.body];
|
630 |
+
}
|
631 |
+
}
|
632 |
+
});
|
633 |
+
}
|
634 |
+
|
635 |
+
|
636 |
+
fixEmptyContextedStaff () {
|
637 |
+
// staff.1 << >> staff.2 << voice.1 {} voice.2 {} >>
|
638 |
+
// ->
|
639 |
+
// staff.1 << voice.1 {} >> staff.2 << voice.2 {} >>
|
640 |
+
const subMusics = (simul: SimultaneousList) => simul.list.filter(term => term instanceof ContextedMusic);
|
641 |
+
|
642 |
+
const score = this.root.getBlock("score");
|
643 |
+
score.forEachTerm(SimultaneousList, simul => {
|
644 |
+
const staves = simul.list.filter(term => term instanceof ContextedMusic && term.body instanceof SimultaneousList);
|
645 |
+
if (staves.length > 1) {
|
646 |
+
const staff1 = staves[0].body;
|
647 |
+
const staff2 = staves[1].body;
|
648 |
+
|
649 |
+
if (subMusics(staff1).length === 0 && subMusics(staff2).length > 1) {
|
650 |
+
const index = staff2.list.findIndex(term => term instanceof ContextedMusic);
|
651 |
+
const [music] = staff2.list.splice(index, 1);
|
652 |
+
staff1.list.push(music);
|
653 |
+
}
|
654 |
+
}
|
655 |
+
});
|
656 |
+
}
|
657 |
+
|
658 |
+
|
659 |
+
removeEmptyContextedStaff () {
|
660 |
+
const subMusics = (simul: SimultaneousList) => simul.list.filter(term => term instanceof ContextedMusic);
|
661 |
+
|
662 |
+
const score = this.root.getBlock("score");
|
663 |
+
score.forEachTerm(SimultaneousList, simul => {
|
664 |
+
simul.list = simul.list.filter(term => !(term instanceof ContextedMusic) || !(term.body instanceof SimultaneousList)
|
665 |
+
|| subMusics(term.body).length > 0);
|
666 |
+
});
|
667 |
+
}
|
668 |
+
|
669 |
+
|
670 |
+
redivide () {
|
671 |
+
this.root.forEachTopTerm(MusicBlock, (block: MusicBlock) => block.redivide());
|
672 |
+
}
|
673 |
+
|
674 |
+
|
675 |
+
makeMIDIDedicatedScore (): Block {
|
676 |
+
const block = this.root.findFirst(term => term instanceof Block && term.head === "\\score" && term.isMIDIDedicated) as Block;
|
677 |
+
if (block)
|
678 |
+
return block;
|
679 |
+
|
680 |
+
const score = this.root.getBlock("score");
|
681 |
+
const newScore = score.clone();
|
682 |
+
|
683 |
+
newScore.body = newScore.body.filter(term => !(term instanceof Block && term.head === "\\layout"));
|
684 |
+
score.body = score.body.filter(term => !(term instanceof Block && term.head === "\\midi"));
|
685 |
+
|
686 |
+
this.root.sections.push(newScore);
|
687 |
+
|
688 |
+
return newScore;
|
689 |
+
}
|
690 |
+
|
691 |
+
|
692 |
+
excludeChordTracksFromMIDI () {
|
693 |
+
// if there is chord mode music in score, duplicate score block as a dedicated MIDI score which excludes chord mode music.
|
694 |
+
let contains = false;
|
695 |
+
|
696 |
+
const isChordMusic = term => term instanceof ContextedMusic
|
697 |
+
&& term.head instanceof Command && term.head.args[0] === "ChordNames";
|
698 |
+
|
699 |
+
const isBlock = (head, term) => term instanceof Block && term.head === head;
|
700 |
+
|
701 |
+
// TODO: midiMusic forked in interpreter issue
|
702 |
+
//this.abstractMainScore();
|
703 |
+
|
704 |
+
const score = this.root.getBlock("score");
|
705 |
+
const newScore = score.clone() as Block;
|
706 |
+
newScore.forEachTerm(SimultaneousList, simul => {
|
707 |
+
const trimmedList = simul.list.filter(term => !isChordMusic(term));
|
708 |
+
if (trimmedList.length < simul.list.length) {
|
709 |
+
simul.list = trimmedList;
|
710 |
+
contains = true;
|
711 |
+
}
|
712 |
+
});
|
713 |
+
newScore._headComment = Comment.createSingle(" midi output");
|
714 |
+
|
715 |
+
if (contains) {
|
716 |
+
const trimmedBody = score.body.filter(term => !isBlock("\\midi", term));
|
717 |
+
if (trimmedBody.length < score.body.length) {
|
718 |
+
score.body = trimmedBody;
|
719 |
+
|
720 |
+
newScore.body = newScore.body.filter(term => !isBlock("\\layout", term));
|
721 |
+
this.root.sections.push(newScore);
|
722 |
+
}
|
723 |
+
}
|
724 |
+
}
|
725 |
+
|
726 |
+
|
727 |
+
// [deprecated]
|
728 |
+
// generate tied notehead location candidates
|
729 |
+
getTiedNoteLocations (source: TextSource): DocLocation[] {
|
730 |
+
const chordPairs: [Chord, Chord][] = [];
|
731 |
+
|
732 |
+
const hasMusicBlock = term => {
|
733 |
+
if (term instanceof MusicBlock)
|
734 |
+
return true;
|
735 |
+
|
736 |
+
if (term instanceof Command)
|
737 |
+
return term.args.filter(arg => arg instanceof MusicBlock).length > 0;
|
738 |
+
|
739 |
+
return false;
|
740 |
+
};
|
741 |
+
|
742 |
+
this.root.forEachTerm(MusicBlock, (block: MusicBlock) => {
|
743 |
+
for (const voice of block.voices) {
|
744 |
+
let lastChord: Chord = null;
|
745 |
+
let tying = false;
|
746 |
+
let afterBlock = false;
|
747 |
+
let atHead = true;
|
748 |
+
|
749 |
+
for (const chunk of voice.body) {
|
750 |
+
for (const term of chunk.terms) {
|
751 |
+
if (term instanceof Primitive && term.exp === "~") {
|
752 |
+
tying = true;
|
753 |
+
afterBlock = false;
|
754 |
+
}
|
755 |
+
else if (hasMusicBlock(term)) {
|
756 |
+
afterBlock = true;
|
757 |
+
tying = false;
|
758 |
+
//console.log("afterBlock:", term);
|
759 |
+
}
|
760 |
+
else if (term instanceof Chord) {
|
761 |
+
if (tying && lastChord)
|
762 |
+
chordPairs.push([lastChord, term]);
|
763 |
+
// maybe there is a tie at tail of the last block
|
764 |
+
else if (afterBlock)
|
765 |
+
chordPairs.push([null, term]);
|
766 |
+
// maybe there is a tie before the current block
|
767 |
+
else if (atHead)
|
768 |
+
chordPairs.push([null, term]);
|
769 |
+
|
770 |
+
// PENDING: maybe some user-defined command block contains tie at tail.
|
771 |
+
|
772 |
+
atHead = false;
|
773 |
+
afterBlock = false;
|
774 |
+
tying = false;
|
775 |
+
lastChord = term;
|
776 |
+
|
777 |
+
if (term.post_events) {
|
778 |
+
for (const event of term.post_events) {
|
779 |
+
if (event instanceof PostEvent && event.arg === "~")
|
780 |
+
tying = true;
|
781 |
+
}
|
782 |
+
}
|
783 |
+
}
|
784 |
+
}
|
785 |
+
}
|
786 |
+
}
|
787 |
+
});
|
788 |
+
|
789 |
+
//console.log("chordPairs:", chordPairs);
|
790 |
+
|
791 |
+
const locations = [];
|
792 |
+
|
793 |
+
chordPairs.forEach(pair => {
|
794 |
+
const forePitches = pair[0] && new Set(pair[0].pitchNames);
|
795 |
+
|
796 |
+
const chordSource = source.slice(pair[1]._location.lines, pair[1]._location.columns);
|
797 |
+
const pitchColumns = TextSource.matchPositions(/\w+/g, chordSource);
|
798 |
+
|
799 |
+
pair[1].pitchNames
|
800 |
+
.map((pitch, index) => ({pitch, index}))
|
801 |
+
.filter(({pitch}) => !forePitches || forePitches.has(pitch) || pitch === "q")
|
802 |
+
.forEach(({index}) => locations.push([
|
803 |
+
pair[1]._location.lines[0], // line
|
804 |
+
pair[1]._location.columns[0] + pitchColumns[index], // column
|
805 |
+
]));
|
806 |
+
});
|
807 |
+
|
808 |
+
return locations;
|
809 |
+
}
|
810 |
+
|
811 |
+
|
812 |
+
// generate tied notehead location candidates
|
813 |
+
getTiedNoteLocations2 (): DocLocation[] {
|
814 |
+
const locations = [];
|
815 |
+
|
816 |
+
this.root.forEachTerm(Chord, chord => chord.pitches.forEach(pitch => {
|
817 |
+
if (pitch._tied)
|
818 |
+
locations.push([pitch._location.lines[0], pitch._location.columns[0]]);
|
819 |
+
}));
|
820 |
+
|
821 |
+
return locations;
|
822 |
+
}
|
823 |
+
|
824 |
+
|
825 |
+
getBriefChordLocations (): DocLocation[] {
|
826 |
+
const locations = [];
|
827 |
+
|
828 |
+
this.root.forEachTerm(BriefChord,
|
829 |
+
chord => locations.push([chord._location.lines[0], chord._location.columns[0]]));
|
830 |
+
|
831 |
+
return locations;
|
832 |
+
}
|
833 |
+
|
834 |
+
|
835 |
+
getLyricLocations (): DocLocation[] {
|
836 |
+
const locations = [];
|
837 |
+
|
838 |
+
this.root.forEachTerm(Lyric,
|
839 |
+
lyric => locations.push([lyric._location.lines[0], lyric._location.columns[0]]));
|
840 |
+
|
841 |
+
return locations;
|
842 |
+
}
|
843 |
+
|
844 |
+
|
845 |
+
/*removeAloneSpacer () {
|
846 |
+
this.root.forEachTopTerm(MusicBlock, block => {
|
847 |
+
const aloneSpacers = cc(block.musicChunks.filter(chunk => chunk.size === 1 && chunk.terms[0].isSpacer).map(chunk => chunk.terms));
|
848 |
+
//console.log("aloneSpacers:", aloneSpacers.map(s => s._location));
|
849 |
+
|
850 |
+
if (aloneSpacers.length) {
|
851 |
+
const removeInBlock = block => block.body = block.body.filter(term => !aloneSpacers.includes(term));
|
852 |
+
|
853 |
+
removeInBlock(block);
|
854 |
+
block.forEachTerm(MusicBlock, removeInBlock);
|
855 |
+
}
|
856 |
+
});
|
857 |
+
}*/
|
858 |
+
|
859 |
+
|
860 |
+
unfoldDurationMultipliers () {
|
861 |
+
this.root.forEachTerm(MusicBlock, block => {
|
862 |
+
block.unfoldDurationMultipliers();
|
863 |
+
});
|
864 |
+
}
|
865 |
+
|
866 |
+
|
867 |
+
appendMIDIInstrumentsFromName () {
|
868 |
+
const isSet = (term: BaseTerm, keyPattern: RegExp): boolean => term instanceof Command && term.cmd === "set" && keyPattern.test((term.args[0] as Assignment).key.toString());
|
869 |
+
const append = (body: BaseTerm[]) => {
|
870 |
+
const ntIndex = body.findIndex(term => isSet(term, /\.instrumentName/));
|
871 |
+
if (ntIndex >= 0 && !body.some(term => isSet(term, /\.midiInstrument/))) {
|
872 |
+
const nameAssign = (body[ntIndex] as Command).args[0] as Assignment;
|
873 |
+
const key = nameAssign.key.toString().replace(/\.instrumentName/, ".midiInstrument");
|
874 |
+
body.splice(ntIndex + 1, 0, Command.createSet(key, nameAssign.value));
|
875 |
+
}
|
876 |
+
};
|
877 |
+
|
878 |
+
this.root.forEachTopTerm(Block, block => {
|
879 |
+
if (block.head === "\\score") {
|
880 |
+
block.forEachTerm(SimultaneousList, simu => append(simu.list));
|
881 |
+
block.forEachTerm(MusicBlock, musicBlock => append(musicBlock.body));
|
882 |
+
}
|
883 |
+
});
|
884 |
+
}
|
885 |
+
|
886 |
+
|
887 |
+
useMidiInstrumentChannelMapping () {
|
888 |
+
this.appendMIDIInstrumentsFromName();
|
889 |
+
|
890 |
+
const midiBlock = this.root.findFirst(term => term instanceof Block && term.head === "\\midi") as Block;
|
891 |
+
if (!midiBlock) {
|
892 |
+
console.warn("no MIDI block found.");
|
893 |
+
return;
|
894 |
+
}
|
895 |
+
|
896 |
+
const channelMapping = midiBlock.findFirst(term => term instanceof Assignment && term.key === "midiChannelMapping") as Assignment;
|
897 |
+
if (channelMapping)
|
898 |
+
channelMapping.value = new Scheme({exp: new SchemePointer({value: "instrument"})});
|
899 |
+
else {
|
900 |
+
midiBlock.body.push(parseRaw({
|
901 |
+
proto: "Block",
|
902 |
+
block: "context",
|
903 |
+
head: "\\context",
|
904 |
+
body: [
|
905 |
+
{proto: "Command",cmd: "Score",args: []},
|
906 |
+
{proto: "Assignment", key: "midiChannelMapping", value: {proto: "Scheme", exp: {proto: "SchemePointer", value: "instrument"}}},
|
907 |
+
],
|
908 |
+
}));
|
909 |
+
}
|
910 |
+
}
|
911 |
+
|
912 |
+
|
913 |
+
formalize () {
|
914 |
+
if (!this.root.findFirst(Version))
|
915 |
+
this.root.sections.unshift(Version.default);
|
916 |
+
|
917 |
+
if (!this.root.findFirst(Language))
|
918 |
+
this.root.sections.splice(1, 0, Language.make("english"));
|
919 |
+
|
920 |
+
if (!this.root.getBlock("header"))
|
921 |
+
this.root.sections.splice(2, 0, new Block({block: "header", head: "\\header", body:[]}));
|
922 |
+
|
923 |
+
if (!this.root.getBlock("score")) {
|
924 |
+
const topMusics = this.root.sections.filter(section => section.isMusic);
|
925 |
+
this.root.sections = this.root.sections.filter(section => !section.isMusic);
|
926 |
+
|
927 |
+
const score = new Block({block: "score", head: "\\score", body: [
|
928 |
+
...topMusics,
|
929 |
+
new Block({block: "score", head: "\\layout", body: []}),
|
930 |
+
new Block({block: "score", head: "\\midi", body: []}),
|
931 |
+
]});
|
932 |
+
|
933 |
+
this.root.sections.push(score);
|
934 |
+
}
|
935 |
+
}
|
936 |
+
|
937 |
+
|
938 |
+
convertStaffToPianoStaff () {
|
939 |
+
const score = this.root.getBlock("score");
|
940 |
+
if (score) {
|
941 |
+
const pstaff = score.findFirst(term => term instanceof ContextedMusic && term.head.cmd === "new" && term.head.args[0] === "Staff") as ContextedMusic;
|
942 |
+
if (pstaff) {
|
943 |
+
pstaff.head.args[0] = "PianoStaff";
|
944 |
+
|
945 |
+
if (pstaff.body instanceof SimultaneousList) {
|
946 |
+
pstaff.body.list = [].concat(...pstaff.body.list.map(term => {
|
947 |
+
if (term instanceof ContextedMusic) {
|
948 |
+
const subMusics = term.list.filter(sub => sub instanceof ContextedMusic);
|
949 |
+
|
950 |
+
return subMusics.map(music => {
|
951 |
+
const staff = term.clone();
|
952 |
+
staff.list = [
|
953 |
+
...term.list.filter(sub => !(sub instanceof ContextedMusic)),
|
954 |
+
music,
|
955 |
+
];
|
956 |
+
|
957 |
+
staff.head.cmd = "new";
|
958 |
+
|
959 |
+
return staff;
|
960 |
+
});
|
961 |
+
}
|
962 |
+
else
|
963 |
+
return [term];
|
964 |
+
}));
|
965 |
+
}
|
966 |
+
}
|
967 |
+
}
|
968 |
+
}
|
969 |
+
|
970 |
+
|
971 |
+
pruneStemDirections () {
|
972 |
+
this.root.forEachTerm(MusicBlock, block => {
|
973 |
+
let direction = null;
|
974 |
+
const redundants = [];
|
975 |
+
|
976 |
+
block.body.forEach(term => {
|
977 |
+
if (term instanceof StemDirection) {
|
978 |
+
if (term.direction === direction)
|
979 |
+
redundants.push(term);
|
980 |
+
else
|
981 |
+
direction = term.direction;
|
982 |
+
}
|
983 |
+
else if (term instanceof Command && term.findFirst(MusicBlock))
|
984 |
+
direction = null;
|
985 |
+
});
|
986 |
+
|
987 |
+
block.body = block.body.filter(term => !redundants.includes(term));
|
988 |
+
});
|
989 |
+
}
|
990 |
+
|
991 |
+
|
992 |
+
removeRepeats () {
|
993 |
+
this.root.forEachTerm(MusicBlock, block => block.spreadRepeatBlocks());
|
994 |
+
}
|
995 |
+
|
996 |
+
|
997 |
+
articulateMIDIOutput () {
|
998 |
+
const ARTICULATE_FILENAME = "articulate-lotus.ly";
|
999 |
+
|
1000 |
+
this.abstractMainScore();
|
1001 |
+
const midiScore = this.makeMIDIDedicatedScore();
|
1002 |
+
|
1003 |
+
if (!this.root.includeFiles.includes(ARTICULATE_FILENAME)) {
|
1004 |
+
let pos = this.root.sections.indexOf(midiScore);
|
1005 |
+
if (pos < 0)
|
1006 |
+
pos = Math.min(this.root.sections.length, 3);
|
1007 |
+
this.root.sections.splice(pos, 0, Include.create(ARTICULATE_FILENAME));
|
1008 |
+
}
|
1009 |
+
|
1010 |
+
midiScore.body = midiScore.body.map(term => {
|
1011 |
+
if (term.isMusic && !(term instanceof Command && term.cmd === "articulate"))
|
1012 |
+
return new Command({cmd: "articulate", args: [term]});
|
1013 |
+
|
1014 |
+
return term;
|
1015 |
+
});
|
1016 |
+
}
|
1017 |
+
|
1018 |
+
|
1019 |
+
removeInvalidExpressionsOnRests (): number {
|
1020 |
+
const isInvalidPostEvent = (event: PostEvent | string): boolean =>
|
1021 |
+
[".", "!", "_"].includes(event instanceof PostEvent ? event.arg as string : event);
|
1022 |
+
|
1023 |
+
let count = 0;
|
1024 |
+
this.root.forEachTerm(MusicEvent, (term: MusicEvent) => {
|
1025 |
+
if (term.isRest) {
|
1026 |
+
if (term.post_events.some(isInvalidPostEvent)) {
|
1027 |
+
term.post_events = term.post_events.filter(event => !isInvalidPostEvent(event));
|
1028 |
+
++count;
|
1029 |
+
}
|
1030 |
+
}
|
1031 |
+
});
|
1032 |
+
|
1033 |
+
return count;
|
1034 |
+
}
|
1035 |
+
|
1036 |
+
|
1037 |
+
abstractMainScore () {
|
1038 |
+
const score = this.root.getBlock("score");
|
1039 |
+
const music = score.body.find(term => term.isMusic);
|
1040 |
+
if (music && !(music instanceof Variable)) {
|
1041 |
+
const sectionIndex = this.root.sections.indexOf(score);
|
1042 |
+
const assignment = new Assignment({
|
1043 |
+
key: MAIN_SCORE_NAME,
|
1044 |
+
value: music,
|
1045 |
+
});
|
1046 |
+
this.root.sections.splice(sectionIndex, 0, assignment);
|
1047 |
+
|
1048 |
+
score.body = score.body.map(term => term === music ? new Variable({name: MAIN_SCORE_NAME}) : term);
|
1049 |
+
}
|
1050 |
+
}
|
1051 |
+
|
1052 |
+
|
1053 |
+
absoluteBlocksToRelative () {
|
1054 |
+
this.root.forEachTopTerm(Assignment, assignment => {
|
1055 |
+
if (assignment.value instanceof MusicBlock) {
|
1056 |
+
const relative = assignment.value.absoluteToRelative();
|
1057 |
+
if (relative)
|
1058 |
+
assignment.value = relative;
|
1059 |
+
}
|
1060 |
+
});
|
1061 |
+
}
|
1062 |
+
};
|
inc/lilyParser/lilyInterpreter.ts
ADDED
@@ -0,0 +1,1476 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
|
2 |
+
import {romanize} from "../romanNumeral";
|
3 |
+
import {WHOLE_DURATION_MAGNITUDE, GRACE_DURATION_FACTOR, FUNCTIONAL_VARIABLE_NAME_PATTERN, MAIN_SCORE_NAME, lcmMulti, lcm} from "./utils";
|
4 |
+
import {parseRaw, getDurationSubdivider, MusicChunk, constructMusicFromMeasureLayout, StemDirection} from "./lilyTerms";
|
5 |
+
import LogRecorder from "../logRecorder";
|
6 |
+
import {StaffContext, PitchContextTable} from "../pitchContext";
|
7 |
+
import * as idioms from "./idioms";
|
8 |
+
import LilyDocument from "./lilyDocument";
|
9 |
+
import * as LilyNotation from "../lilyNotation";
|
10 |
+
|
11 |
+
import {
|
12 |
+
BaseTerm,
|
13 |
+
Root, Block, MusicEvent, Repeat, Relative, TimeSignature, Partial, Times, Tuplet, Grace, AfterGrace, Clef, Scheme, Include, Rest,
|
14 |
+
KeySignature, OctaveShift, Duration, Chord, MusicBlock, Assignment, Variable, Command, SimultaneousList, ContextedMusic, Primitive, Version,
|
15 |
+
ChordMode, LyricMode, ChordElement, Language, PostEvent, Transposition, ParallelMusic,
|
16 |
+
} from "./lilyTerms";
|
17 |
+
import {MeasureLayout, BlockMLayout, SingleMLayout} from "../measureLayout";
|
18 |
+
|
19 |
+
|
20 |
+
|
21 |
+
interface DurationContextStackStatus {
|
22 |
+
factor?: {value: number};
|
23 |
+
tickBias?: number;
|
24 |
+
tremoloDuration?: Duration;
|
25 |
+
};
|
26 |
+
|
27 |
+
|
28 |
+
type MusicTransformer = (music: BaseTerm, context: TrackContext) => BaseTerm[];
|
29 |
+
type MusicListener = (music: BaseTerm, context: TrackContext) => void;
|
30 |
+
|
31 |
+
|
32 |
+
type ContextDict = {[key: string]: string};
|
33 |
+
|
34 |
+
|
35 |
+
export enum TremoloType {
|
36 |
+
None,
|
37 |
+
Single,
|
38 |
+
Pitcher,
|
39 |
+
Catcher,
|
40 |
+
};
|
41 |
+
|
42 |
+
|
43 |
+
interface PitchContextTerm {
|
44 |
+
staffName: string;
|
45 |
+
track?: number;
|
46 |
+
//voiceName?: string;
|
47 |
+
|
48 |
+
tick?: number;
|
49 |
+
event: MusicEvent;
|
50 |
+
|
51 |
+
clef?: {
|
52 |
+
y: number,
|
53 |
+
value: number,
|
54 |
+
};
|
55 |
+
octaveShift?: number;
|
56 |
+
key?: number;
|
57 |
+
newMeasure?: boolean;
|
58 |
+
pitches?: ChordElement[];
|
59 |
+
tickBias?: number;
|
60 |
+
rest?: boolean;
|
61 |
+
tremoloType?: TremoloType;
|
62 |
+
};
|
63 |
+
|
64 |
+
|
65 |
+
class LilyStaffContext extends StaffContext {
|
66 |
+
staffTrack: number;
|
67 |
+
notes: LilyNotation.Note[] = [];
|
68 |
+
|
69 |
+
channelMap: number[] = [];
|
70 |
+
|
71 |
+
|
72 |
+
executeTerm (term: PitchContextTerm) {
|
73 |
+
//console.log("executeTerm:", term);
|
74 |
+
|
75 |
+
if (term.newMeasure)
|
76 |
+
this.resetAlters();
|
77 |
+
|
78 |
+
if (term.clef)
|
79 |
+
this.setClef(term.clef.y, term.clef.value);
|
80 |
+
if (Number.isFinite(term.octaveShift))
|
81 |
+
this.setOctaveShift(-term.octaveShift);
|
82 |
+
if (Number.isFinite(term.key)) {
|
83 |
+
this.resetKeyAlters();
|
84 |
+
|
85 |
+
if (term.key) {
|
86 |
+
const step = term.key > 0 ? 1 : -1;
|
87 |
+
for (let p = step; p / term.key <= 1; p += step) {
|
88 |
+
const index = ((step > 0 ? p - 1 : p) + 70) % 7;
|
89 |
+
const note = idioms.PHONETS.indexOf(idioms.FIFTH_PHONETS[index]);
|
90 |
+
|
91 |
+
this.keyAlters[note] = (this.keyAlters[note] || 0) + step;
|
92 |
+
}
|
93 |
+
this.dirty = true;
|
94 |
+
}
|
95 |
+
}
|
96 |
+
|
97 |
+
if (term.pitches) {
|
98 |
+
// accidental alters
|
99 |
+
term.pitches.forEach(pitch => {
|
100 |
+
const note = pitch.absoluteNotePosition;
|
101 |
+
const alter = this.alterOnNote(note);
|
102 |
+
if (pitch.alterValue !== alter) {
|
103 |
+
this.alters[note] = pitch.alterValue;
|
104 |
+
this.dirty = true;
|
105 |
+
}
|
106 |
+
});
|
107 |
+
|
108 |
+
const event = term.event;
|
109 |
+
const contextIndex = this.snapshot({tick: event._tick});
|
110 |
+
|
111 |
+
const implicitType = event.implicitType || (term.tremoloType ? LilyNotation.ImplicitType.Tremolo : LilyNotation.ImplicitType.None);
|
112 |
+
|
113 |
+
this.notes.push(...term.pitches.map((pitch, index) => ({
|
114 |
+
track: term.track,
|
115 |
+
channel: this.channelMap[term.track] || 0,
|
116 |
+
measure: event._measure,
|
117 |
+
start: event._tick,
|
118 |
+
duration: event.durationMagnitude,
|
119 |
+
division: event.division,
|
120 |
+
startTick: event._tick,
|
121 |
+
endTick: event._tick + event.durationMagnitude,
|
122 |
+
pitch: pitch.absolutePitchValue + (pitch._transposition || 0),
|
123 |
+
velocity: 127,
|
124 |
+
id: pitch.href,
|
125 |
+
ids: [pitch.href],
|
126 |
+
tied: !!pitch._tied,
|
127 |
+
rest: event.isRest,
|
128 |
+
afterGrace: !!term.tickBias,
|
129 |
+
implicitType,
|
130 |
+
staffTrack: this.staffTrack,
|
131 |
+
contextIndex,
|
132 |
+
// TODO: consider connected arpeggio & downward arpeggio
|
133 |
+
chordPosition: {
|
134 |
+
index,
|
135 |
+
count: term.pitches.length,
|
136 |
+
},
|
137 |
+
})));
|
138 |
+
|
139 |
+
term.pitches.forEach(pitch => {
|
140 |
+
const tiedParent = pitch.tiedParent;
|
141 |
+
if (tiedParent) {
|
142 |
+
const note = this.notes.find(note => note.id === tiedParent.href);
|
143 |
+
if (note)
|
144 |
+
note.ids.push(pitch.href);
|
145 |
+
}
|
146 |
+
});
|
147 |
+
}
|
148 |
+
else if (term.rest) {
|
149 |
+
const event = term.event;
|
150 |
+
const contextIndex = this.snapshot({tick: event._tick});
|
151 |
+
|
152 |
+
this.notes.push({
|
153 |
+
track: term.track,
|
154 |
+
channel: this.channelMap[term.track] || 0,
|
155 |
+
measure: event._measure,
|
156 |
+
start: event._tick,
|
157 |
+
duration: event.durationMagnitude,
|
158 |
+
startTick: event._tick,
|
159 |
+
endTick: event._tick + event.durationMagnitude,
|
160 |
+
pitch: null,
|
161 |
+
velocity: 0,
|
162 |
+
id: event.href,
|
163 |
+
ids: [event.href],
|
164 |
+
tied: false,
|
165 |
+
rest: true,
|
166 |
+
afterGrace: !!term.tickBias,
|
167 |
+
implicitType: event.implicitType,
|
168 |
+
staffTrack: this.staffTrack,
|
169 |
+
contextIndex,
|
170 |
+
});
|
171 |
+
}
|
172 |
+
}
|
173 |
+
|
174 |
+
|
175 |
+
get pitchContextTable (): PitchContextTable {
|
176 |
+
const items = this.track.contexts.map(context => ({
|
177 |
+
tick: context.tick,
|
178 |
+
endTick: null,
|
179 |
+
context,
|
180 |
+
}));
|
181 |
+
items.forEach((item, i) => {
|
182 |
+
item.endTick = (i + 1 < items.length ? items[i + 1].tick : Infinity);
|
183 |
+
});
|
184 |
+
|
185 |
+
return new PitchContextTable({items});
|
186 |
+
}
|
187 |
+
};
|
188 |
+
|
189 |
+
|
190 |
+
|
191 |
+
export class MusicTrack {
|
192 |
+
block: MusicBlock;
|
193 |
+
anchorPitch: ChordElement;
|
194 |
+
contextDict?: ContextDict = null;
|
195 |
+
|
196 |
+
name?: string;
|
197 |
+
measureHeads: number[];
|
198 |
+
|
199 |
+
|
200 |
+
static fromBlockAnchor (block: MusicBlock, anchorPitch: ChordElement): MusicTrack {
|
201 |
+
const track = new MusicTrack;
|
202 |
+
|
203 |
+
track.block = block;
|
204 |
+
track.anchorPitch = anchorPitch;
|
205 |
+
|
206 |
+
const context = new TrackContext(track);
|
207 |
+
context.execute(track.music);
|
208 |
+
|
209 |
+
return track;
|
210 |
+
}
|
211 |
+
|
212 |
+
|
213 |
+
get music (): BaseTerm {
|
214 |
+
if (!this.block._parent) {
|
215 |
+
this.block._parent = new Relative({cmd: "relative", args: this.anchorPitch ? [this.anchorPitch.clone(), this.block] : [this.block]});
|
216 |
+
this.block.updateChordAnchors();
|
217 |
+
}
|
218 |
+
|
219 |
+
return this.block._parent;
|
220 |
+
}
|
221 |
+
|
222 |
+
|
223 |
+
get noteDurationSubdivider (): number {
|
224 |
+
return getDurationSubdivider(this.block);
|
225 |
+
}
|
226 |
+
|
227 |
+
|
228 |
+
get durationMagnitude (): number {
|
229 |
+
return this.block && this.block.durationMagnitude;
|
230 |
+
}
|
231 |
+
|
232 |
+
|
233 |
+
get isLyricMode (): boolean {
|
234 |
+
return (this.music instanceof LyricMode) || !!this.block.findFirst(term => term instanceof LyricMode);
|
235 |
+
}
|
236 |
+
|
237 |
+
|
238 |
+
get isChordMode (): boolean {
|
239 |
+
return (this.music instanceof ChordMode) || !!this.block.findFirst(term => term instanceof ChordMode);
|
240 |
+
}
|
241 |
+
|
242 |
+
|
243 |
+
get measureLayoutCode (): string {
|
244 |
+
let code = this.block.measureLayout.code;
|
245 |
+
if (/^\[.*\]$/.test(code))
|
246 |
+
code = code.match(/\[(.*)\]/)[1];
|
247 |
+
|
248 |
+
return code;
|
249 |
+
}
|
250 |
+
|
251 |
+
|
252 |
+
transform (transformer: MusicTransformer) {
|
253 |
+
new TrackContext(this, {transformer}).execute(this.music);
|
254 |
+
}
|
255 |
+
|
256 |
+
|
257 |
+
clarifyDurations () {
|
258 |
+
this.transform(term => {
|
259 |
+
if (term instanceof MusicEvent) {
|
260 |
+
if (!term.duration)
|
261 |
+
term.duration = term.durationValue;
|
262 |
+
}
|
263 |
+
|
264 |
+
return [term];
|
265 |
+
});
|
266 |
+
}
|
267 |
+
|
268 |
+
|
269 |
+
splitLongRests () {
|
270 |
+
this.clarifyDurations();
|
271 |
+
|
272 |
+
this.transform((term, context) => {
|
273 |
+
if (!(term instanceof MusicEvent) || (!term.withMultiplier && !(term instanceof Rest)))
|
274 |
+
return [term];
|
275 |
+
|
276 |
+
const timeDenominator = context.time ? context.time.value.denominator : 4;
|
277 |
+
const duration = term.durationValue;
|
278 |
+
const denominator = Math.max(duration.denominator, timeDenominator);
|
279 |
+
|
280 |
+
const isR = !(term as Rest).isSpacer;
|
281 |
+
|
282 |
+
if (term.withMultiplier) {
|
283 |
+
const factor = duration.multipliers.reduce((factor, multiplier) => factor * Number(multiplier), 1);
|
284 |
+
if (!Number.isInteger(factor) || factor <= 0) {
|
285 |
+
console.warn("invalid multiplier:", factor, duration.multipliers);
|
286 |
+
return [term];
|
287 |
+
}
|
288 |
+
|
289 |
+
const event = term.clone() as MusicEvent;
|
290 |
+
event.duration.multipliers = [];
|
291 |
+
|
292 |
+
// break duration into multiple rest events
|
293 |
+
const restCount = (event.duration.magnitude / WHOLE_DURATION_MAGNITUDE) * (factor - 1) * denominator;
|
294 |
+
if (!Number.isInteger(restCount))
|
295 |
+
console.warn("Rest count is not integear:", restCount, denominator, event.duration.magnitude, factor);
|
296 |
+
|
297 |
+
const rests = Array(Math.floor(restCount)).fill(null).map(() =>
|
298 |
+
new Rest({name: "s", duration: new Duration({number: denominator, dots: 0})}));
|
299 |
+
|
300 |
+
return [event, ...rests];
|
301 |
+
}
|
302 |
+
else {
|
303 |
+
const divider = lcm(duration.subdivider, denominator);
|
304 |
+
const restCount = term.durationMagnitude * divider / WHOLE_DURATION_MAGNITUDE;
|
305 |
+
console.assert(Number.isInteger(restCount), "rest count is not an integer:", restCount);
|
306 |
+
|
307 |
+
if (isR && restCount > 1)
|
308 |
+
console.warn("splitLongRests: 'r' was splitted into", restCount, "parts.", term._location);
|
309 |
+
|
310 |
+
const list = Array(restCount).fill(null).map(() =>
|
311 |
+
new Rest({name: isR ? "r" : "s", duration: new Duration({number: divider, dots: 0})}));
|
312 |
+
|
313 |
+
if (term.post_events)
|
314 |
+
list[list.length - 1].post_events = term.post_events.map(e => e instanceof BaseTerm ? e.clone() : e);
|
315 |
+
|
316 |
+
return list;
|
317 |
+
}
|
318 |
+
});
|
319 |
+
}
|
320 |
+
|
321 |
+
|
322 |
+
spreadMusicBlocks (): boolean {
|
323 |
+
let has = false;
|
324 |
+
|
325 |
+
this.transform((term) => {
|
326 |
+
if (term instanceof MusicBlock) {
|
327 |
+
has = true;
|
328 |
+
return term.body;
|
329 |
+
}
|
330 |
+
else
|
331 |
+
return [term];
|
332 |
+
});
|
333 |
+
|
334 |
+
return has;
|
335 |
+
}
|
336 |
+
|
337 |
+
|
338 |
+
spreadRelativeBlocks (): boolean {
|
339 |
+
// check if traverse is nessary
|
340 |
+
if (!this.block.findFirst(Relative))
|
341 |
+
return false;
|
342 |
+
|
343 |
+
this.transform((term, context) => {
|
344 |
+
if (term instanceof Relative) {
|
345 |
+
if (term.music instanceof MusicBlock)
|
346 |
+
term.music.updateChordAnchors();
|
347 |
+
|
348 |
+
const terms = term.shiftBody(context.pitch);
|
349 |
+
|
350 |
+
// initialize anchor pitch for track head chord
|
351 |
+
if (!context.event || !context.event.getPreviousT(Chord)) {
|
352 |
+
const tempBlock = new MusicBlock({body: []});
|
353 |
+
tempBlock.body = terms;
|
354 |
+
const head = tempBlock.findFirst(Chord);
|
355 |
+
if (head)
|
356 |
+
//head._anchorPitch = this.anchorPitch;
|
357 |
+
head._anchorPitch = context.pitch;
|
358 |
+
}
|
359 |
+
|
360 |
+
return terms;
|
361 |
+
}
|
362 |
+
else
|
363 |
+
return [term];
|
364 |
+
});
|
365 |
+
|
366 |
+
return true;
|
367 |
+
}
|
368 |
+
|
369 |
+
|
370 |
+
spreadRepeatBlocks ({ignoreRepeat = true, keepTailPass = false} = {}): boolean {
|
371 |
+
// check if traverse is nessary
|
372 |
+
if (!this.block.findFirst(Repeat))
|
373 |
+
return false;
|
374 |
+
|
375 |
+
this.transform(term => {
|
376 |
+
if (term instanceof Repeat) {
|
377 |
+
if (!ignoreRepeat)
|
378 |
+
return term.getUnfoldTerms();
|
379 |
+
else if (keepTailPass)
|
380 |
+
return term.getTailPassTerms();
|
381 |
+
else
|
382 |
+
return term.getPlainTerms();
|
383 |
+
}
|
384 |
+
else if (term instanceof Variable && term.name === "lotusRepeatABA")
|
385 |
+
return [];
|
386 |
+
else
|
387 |
+
return [term];
|
388 |
+
});
|
389 |
+
|
390 |
+
return true;
|
391 |
+
}
|
392 |
+
|
393 |
+
|
394 |
+
flatten ({spreadRepeats = false} = {}) {
|
395 |
+
this.splitLongRests();
|
396 |
+
this.spreadRelativeBlocks();
|
397 |
+
|
398 |
+
if (spreadRepeats) {
|
399 |
+
while (this.spreadRepeatBlocks())
|
400 |
+
;
|
401 |
+
|
402 |
+
// expand all music blocks
|
403 |
+
while (this.spreadMusicBlocks());
|
404 |
+
}
|
405 |
+
}
|
406 |
+
|
407 |
+
|
408 |
+
sliceMeasures (start: number, count: number): MusicTrack {
|
409 |
+
this.flatten({spreadRepeats: true});
|
410 |
+
|
411 |
+
const context = new TrackContext(this);
|
412 |
+
context.pitch = this.anchorPitch;
|
413 |
+
this.block.updateChordAnchors();
|
414 |
+
|
415 |
+
for (const term of this.block.body) {
|
416 |
+
if (Number.isInteger(term._measure)) {
|
417 |
+
if (term._measure < start)
|
418 |
+
context.execute(term);
|
419 |
+
else
|
420 |
+
break;
|
421 |
+
}
|
422 |
+
}
|
423 |
+
|
424 |
+
const terms = context.declarations.concat(this.block.body.filter(term => term._measure >= start && term._measure < start + count));
|
425 |
+
const newBlock = MusicBlock.fromTerms(terms);
|
426 |
+
|
427 |
+
return MusicTrack.fromBlockAnchor(newBlock, context.pitch);
|
428 |
+
}
|
429 |
+
|
430 |
+
|
431 |
+
redivide () {
|
432 |
+
this.block.redivide({measureHeads: this.measureHeads});
|
433 |
+
}
|
434 |
+
|
435 |
+
|
436 |
+
applyMeasureLayout (layout: MeasureLayout, {flatten = true} = {}) {
|
437 |
+
//console.log("applyMeasureLayout:", this, layout);
|
438 |
+
|
439 |
+
if (flatten)
|
440 |
+
this.flatten({spreadRepeats: true});
|
441 |
+
|
442 |
+
const chunks = this.block.measureChunkMap;
|
443 |
+
|
444 |
+
// validate layout value
|
445 |
+
const indices = layout.serialize(LilyNotation.LayoutType.Ordinary);
|
446 |
+
indices.forEach(index => {
|
447 |
+
if (!chunks.get(index))
|
448 |
+
throw new Error(`applyMeasureLayout: measure[${index}] missed in chunk map.`);
|
449 |
+
});
|
450 |
+
|
451 |
+
// append zero-duration tail chunk, e.g. \bar "|."
|
452 |
+
const tailIndex = Math.max(...indices) + 1;
|
453 |
+
const tailChunk = chunks.get(tailIndex);
|
454 |
+
if (tailChunk && !tailChunk.durationMagnitude && layout instanceof BlockMLayout)
|
455 |
+
//layout.seq.push(SingleMLayout.from(tailIndex));
|
456 |
+
layout = BlockMLayout.fromSeq([...layout.seq, SingleMLayout.from(tailIndex)]);
|
457 |
+
|
458 |
+
this.block.body = constructMusicFromMeasureLayout(layout, chunks).terms;
|
459 |
+
|
460 |
+
this.redivide();
|
461 |
+
}
|
462 |
+
|
463 |
+
|
464 |
+
generateStaffTracks (): PitchContextTerm[] {
|
465 |
+
const pcTerms: PitchContextTerm[] = [];
|
466 |
+
|
467 |
+
let currentTerm = null;
|
468 |
+
const commitTerm = () => {
|
469 |
+
if (currentTerm) {
|
470 |
+
pcTerms.push(currentTerm);
|
471 |
+
currentTerm = null;
|
472 |
+
}
|
473 |
+
};
|
474 |
+
const getCurrentTerm = (staffName: string): PitchContextTerm => {
|
475 |
+
if (currentTerm && currentTerm.staffName !== staffName)
|
476 |
+
commitTerm();
|
477 |
+
|
478 |
+
if (!currentTerm)
|
479 |
+
currentTerm = {staffName};
|
480 |
+
|
481 |
+
return currentTerm;
|
482 |
+
};
|
483 |
+
|
484 |
+
let measureIndex = 0;
|
485 |
+
|
486 |
+
const listener = (term: BaseTerm, track: TrackContext) => {
|
487 |
+
getCurrentTerm(track.staffName).tick = term._tick;
|
488 |
+
|
489 |
+
if (term._measure !== measureIndex) {
|
490 |
+
getCurrentTerm(track.staffName).newMeasure = true;
|
491 |
+
commitTerm();
|
492 |
+
|
493 |
+
measureIndex = term._measure;
|
494 |
+
}
|
495 |
+
|
496 |
+
if (term instanceof Chord) {
|
497 |
+
const pcTerm = getCurrentTerm(track.staffName);
|
498 |
+
pcTerm.event = term;
|
499 |
+
pcTerm.pitches = term.pitchesValue.filter(pitch => pitch instanceof ChordElement) as ChordElement[];
|
500 |
+
pcTerm.pitches = [...pcTerm.pitches].sort((p1, p2) => p1.absolutePitchValue - p2.absolutePitchValue);
|
501 |
+
|
502 |
+
if (track.tickBias)
|
503 |
+
pcTerm.tickBias = track.tickBias;
|
504 |
+
|
505 |
+
pcTerm.tremoloType = track.tremoloType;
|
506 |
+
|
507 |
+
commitTerm();
|
508 |
+
}
|
509 |
+
else if (term instanceof Rest && term.name !== "s") {
|
510 |
+
const pcTerm = getCurrentTerm(track.staffName);
|
511 |
+
pcTerm.event = term;
|
512 |
+
pcTerm.rest = true;
|
513 |
+
|
514 |
+
commitTerm();
|
515 |
+
}
|
516 |
+
else if (term instanceof Clef) {
|
517 |
+
//console.log("clef:", term.clefName);
|
518 |
+
switch (term.clefName) {
|
519 |
+
case "treble":
|
520 |
+
// a treble (G4) on the 2nd staff line
|
521 |
+
getCurrentTerm(track.staffName).clef = {y: 1, value: 4};
|
522 |
+
|
523 |
+
break;
|
524 |
+
case "bass":
|
525 |
+
// a bass (F3) on the 4th staff line
|
526 |
+
getCurrentTerm(track.staffName).clef = {y: -1, value: -4};
|
527 |
+
|
528 |
+
break;
|
529 |
+
case "tenor":
|
530 |
+
// a tenor (C4) on the 3rd staff line
|
531 |
+
getCurrentTerm(track.staffName).clef = {y: 0, value: 0};
|
532 |
+
|
533 |
+
break;
|
534 |
+
}
|
535 |
+
}
|
536 |
+
else if (term instanceof KeySignature)
|
537 |
+
getCurrentTerm(track.staffName).key = term.key;
|
538 |
+
else if (term instanceof OctaveShift)
|
539 |
+
getCurrentTerm(track.staffName).octaveShift = term.value;
|
540 |
+
};
|
541 |
+
new TrackContext(this, {listener}).execute(this.music);
|
542 |
+
|
543 |
+
return pcTerms;
|
544 |
+
}
|
545 |
+
};
|
546 |
+
|
547 |
+
|
548 |
+
export class TrackContext {
|
549 |
+
track: MusicTrack;
|
550 |
+
transformer?: MusicTransformer;
|
551 |
+
listener?: MusicListener;
|
552 |
+
|
553 |
+
stack: DurationContextStackStatus[] = [];
|
554 |
+
|
555 |
+
// declarations
|
556 |
+
staff: Command = null;
|
557 |
+
clef: Clef = null;
|
558 |
+
key: KeySignature = null;
|
559 |
+
time: TimeSignature = null;
|
560 |
+
octave: OctaveShift = null;
|
561 |
+
|
562 |
+
pitch: ChordElement = null;
|
563 |
+
|
564 |
+
staffName: string = null;
|
565 |
+
voiceName: string = null;
|
566 |
+
transposition: number = 0;
|
567 |
+
|
568 |
+
// time status
|
569 |
+
tick: number = 0;
|
570 |
+
tickInMeasure: number = 0;
|
571 |
+
measureSpan: number = WHOLE_DURATION_MAGNITUDE;
|
572 |
+
measureIndex: number = 1;
|
573 |
+
partialDuration: Duration = null;
|
574 |
+
measureHeads: number[] = [0];
|
575 |
+
|
576 |
+
event: MusicEvent = null;
|
577 |
+
tying: MusicEvent = null;
|
578 |
+
staccato: boolean = false;
|
579 |
+
inGrace: boolean = false;
|
580 |
+
stemDirection: string = null;
|
581 |
+
beamOn: boolean = false;
|
582 |
+
|
583 |
+
tremoloType: TremoloType = TremoloType.None;
|
584 |
+
|
585 |
+
|
586 |
+
constructor (track = new MusicTrack, {transformer = null, listener = null, contextDict = null}:
|
587 |
+
{
|
588 |
+
transformer?: MusicTransformer,
|
589 |
+
listener?: MusicListener,
|
590 |
+
contextDict?: ContextDict,
|
591 |
+
} = {}) {
|
592 |
+
this.track = track;
|
593 |
+
this.track.contextDict = contextDict || this.track.contextDict;
|
594 |
+
this.track.measureHeads = this.measureHeads;
|
595 |
+
|
596 |
+
this.transformer = transformer;
|
597 |
+
this.listener = listener;
|
598 |
+
|
599 |
+
if (this.track.contextDict) {
|
600 |
+
this.staffName = this.track.contextDict.Staff;
|
601 |
+
this.voiceName = this.track.contextDict.Voice;
|
602 |
+
}
|
603 |
+
//console.debug("contextDict:", contextDict);
|
604 |
+
}
|
605 |
+
|
606 |
+
|
607 |
+
clone (): this {
|
608 |
+
const ctx = {...this};
|
609 |
+
Object.setPrototypeOf(ctx, Object.getPrototypeOf(this));
|
610 |
+
|
611 |
+
return ctx;
|
612 |
+
}
|
613 |
+
|
614 |
+
|
615 |
+
mergeParallelClones (contexts: TrackContext[]) {
|
616 |
+
const frontContext = contexts.reduce((front, context) => {
|
617 |
+
const next = !front || context.tick > front.tick ? context : front;
|
618 |
+
next.tying = next.tying || context.tying;
|
619 |
+
next.staccato = next.staccato || context.staccato;
|
620 |
+
|
621 |
+
return next;
|
622 |
+
}, null);
|
623 |
+
const lastContext = contexts[contexts.length - 1];
|
624 |
+
|
625 |
+
this.tick = frontContext.tick;
|
626 |
+
this.tickInMeasure = frontContext.tickInMeasure;
|
627 |
+
this.measureIndex = frontContext.measureIndex;
|
628 |
+
this.partialDuration = frontContext.partialDuration;
|
629 |
+
this.tying = frontContext.tying;
|
630 |
+
this.staccato = frontContext.staccato;
|
631 |
+
|
632 |
+
this.pitch = lastContext.pitch;
|
633 |
+
this.event = lastContext.event;
|
634 |
+
}
|
635 |
+
|
636 |
+
|
637 |
+
get factor (): {value: number} {
|
638 |
+
for (let i = this.stack.length - 1; i >= 0; i--) {
|
639 |
+
const status = this.stack[i];
|
640 |
+
if (status.factor)
|
641 |
+
return status.factor;
|
642 |
+
}
|
643 |
+
|
644 |
+
return null;
|
645 |
+
}
|
646 |
+
|
647 |
+
|
648 |
+
get tremoloDuration (): Duration {
|
649 |
+
for (let i = this.stack.length - 1; i >= 0; i--) {
|
650 |
+
const status = this.stack[i];
|
651 |
+
if (status.tremoloDuration)
|
652 |
+
return status.tremoloDuration;
|
653 |
+
}
|
654 |
+
|
655 |
+
return null;
|
656 |
+
}
|
657 |
+
|
658 |
+
|
659 |
+
get tickBias (): number {
|
660 |
+
for (let i = this.stack.length - 1; i >= 0; i--) {
|
661 |
+
const status = this.stack[i];
|
662 |
+
if (status.tickBias)
|
663 |
+
return status.tickBias;
|
664 |
+
}
|
665 |
+
|
666 |
+
return 0;
|
667 |
+
}
|
668 |
+
|
669 |
+
|
670 |
+
get measureIndexBias (): number {
|
671 |
+
if (this.tickInMeasure + this.tickBias < -1)
|
672 |
+
return -1;
|
673 |
+
|
674 |
+
return 0;
|
675 |
+
}
|
676 |
+
|
677 |
+
|
678 |
+
get factorValue (): number {
|
679 |
+
return this.factor && Number.isFinite(this.factor.value) ? this.factor.value : 1;
|
680 |
+
}
|
681 |
+
|
682 |
+
|
683 |
+
get currentMeasureSpan (): number {
|
684 |
+
return Math.round(this.partialDuration ? this.partialDuration.magnitude : this.measureSpan);
|
685 |
+
}
|
686 |
+
|
687 |
+
|
688 |
+
setPitch (pitch: ChordElement) {
|
689 |
+
if (!this.track.anchorPitch)
|
690 |
+
this.track.anchorPitch = pitch;
|
691 |
+
|
692 |
+
this.pitch = pitch;
|
693 |
+
}
|
694 |
+
|
695 |
+
|
696 |
+
newMeasure (measureSpan: number) {
|
697 |
+
console.assert(Number.isFinite(this.measureHeads[this.measureIndex - 1]), "invalid measureHeads at", this.measureIndex - 1, this.measureHeads);
|
698 |
+
this.measureHeads[this.measureIndex] = this.measureHeads[this.measureIndex - 1] + measureSpan;
|
699 |
+
|
700 |
+
++this.measureIndex;
|
701 |
+
this.tickInMeasure -= measureSpan;
|
702 |
+
|
703 |
+
this.partialDuration = null;
|
704 |
+
}
|
705 |
+
|
706 |
+
|
707 |
+
checkIncompleteMeasure () {
|
708 |
+
if (this.tickInMeasure) {
|
709 |
+
console.warn("incomplete measure trunated:", this.measureIndex, `${this.tickInMeasure}/${this.currentMeasureSpan}`);
|
710 |
+
this.newMeasure(this.tickInMeasure);
|
711 |
+
}
|
712 |
+
}
|
713 |
+
|
714 |
+
|
715 |
+
elapse (duration: number) {
|
716 |
+
const increment = duration * this.factorValue;
|
717 |
+
|
718 |
+
this.tick += increment;
|
719 |
+
|
720 |
+
this.tickInMeasure += increment;
|
721 |
+
while (Math.round(this.tickInMeasure) >= this.currentMeasureSpan)
|
722 |
+
this.newMeasure(this.currentMeasureSpan);
|
723 |
+
}
|
724 |
+
|
725 |
+
|
726 |
+
push (status: DurationContextStackStatus) {
|
727 |
+
this.stack.push(status);
|
728 |
+
}
|
729 |
+
|
730 |
+
|
731 |
+
pop () {
|
732 |
+
this.stack.pop();
|
733 |
+
}
|
734 |
+
|
735 |
+
|
736 |
+
processGrace (music: BaseTerm, factor = GRACE_DURATION_FACTOR) {
|
737 |
+
// pull back grace notes' ticks
|
738 |
+
let events = [music];
|
739 |
+
if (!(music instanceof MusicEvent))
|
740 |
+
events = music.findAll(MusicEvent);
|
741 |
+
|
742 |
+
let tick = this.tick;
|
743 |
+
events.reverse().forEach(event => {
|
744 |
+
tick -= Math.round(event.durationMagnitude * factor * this.factorValue);
|
745 |
+
event._tick = tick;
|
746 |
+
|
747 |
+
event.findAll(ChordElement).forEach(note => note._tick = tick);
|
748 |
+
});
|
749 |
+
}
|
750 |
+
|
751 |
+
|
752 |
+
execute (term: BaseTerm) {
|
753 |
+
if (!term) {
|
754 |
+
console.warn("null term:", term);
|
755 |
+
return;
|
756 |
+
}
|
757 |
+
|
758 |
+
if (!(term instanceof BaseTerm))
|
759 |
+
return;
|
760 |
+
|
761 |
+
term._measure = this.measureIndex + this.measureIndexBias;
|
762 |
+
term._tick = this.tick;
|
763 |
+
|
764 |
+
if (term instanceof MusicEvent) {
|
765 |
+
term._previous = this.event;
|
766 |
+
|
767 |
+
if (term instanceof Chord) {
|
768 |
+
if (!this.track.anchorPitch)
|
769 |
+
this.track.anchorPitch = ChordElement.default.clone();
|
770 |
+
|
771 |
+
this.setPitch(term.absolutePitch);
|
772 |
+
|
773 |
+
term.pitches.forEach(pitch => {
|
774 |
+
this.execute(pitch);
|
775 |
+
|
776 |
+
if (pitch instanceof ChordElement)
|
777 |
+
pitch._transposition = this.transposition;
|
778 |
+
});
|
779 |
+
|
780 |
+
// update tied for ChordElement
|
781 |
+
// TODO: staccato trigger condition?
|
782 |
+
if (this.tying /*&& !this.staccato*/ && this.event && this.event instanceof Chord) {
|
783 |
+
const pitches = new Set(this.event.pitchElements.map(pitch => pitch.absolutePitch.pitch));
|
784 |
+
term.pitchElements.forEach(pitch => {
|
785 |
+
if (pitches.has(pitch.absolutePitch.pitch))
|
786 |
+
pitch._tied = this.tying;
|
787 |
+
//else
|
788 |
+
// console.log("missed tie:", `${pitch._location.lines[0]}:${pitch._location.columns[0]}`, pitch.absolutePitch.pitch, pitches);
|
789 |
+
});
|
790 |
+
|
791 |
+
if (this.staccato)
|
792 |
+
console.warn("tie on staccato note:", term.href);
|
793 |
+
}
|
794 |
+
//console.log("chord:", term.pitches[0]);
|
795 |
+
}
|
796 |
+
|
797 |
+
if (term.beamOn)
|
798 |
+
this.beamOn = true;
|
799 |
+
else if (term.beamOff)
|
800 |
+
this.beamOn = false;
|
801 |
+
|
802 |
+
this.event = term;
|
803 |
+
|
804 |
+
this.elapse(term.durationMagnitude);
|
805 |
+
|
806 |
+
term._lastMeasure = this.tickInMeasure > 0 ? this.measureIndex : this.measureIndex - 1;
|
807 |
+
|
808 |
+
this.tying = null;
|
809 |
+
this.staccato = false;
|
810 |
+
|
811 |
+
if (term.isTying)
|
812 |
+
this.tying = term;
|
813 |
+
if (term.isStaccato)
|
814 |
+
this.staccato = true;
|
815 |
+
}
|
816 |
+
else if (term instanceof ChordElement) {
|
817 |
+
// ignore
|
818 |
+
}
|
819 |
+
else if (term instanceof MusicBlock) {
|
820 |
+
if (!this.track.block)
|
821 |
+
this.track.block = term;
|
822 |
+
|
823 |
+
term.updateChordAnchors();
|
824 |
+
|
825 |
+
if (this.transformer) {
|
826 |
+
const body = [];
|
827 |
+
for (const subterm of term.body) {
|
828 |
+
const terms = this.transformer(subterm, this);
|
829 |
+
terms.forEach(t => this.execute(t));
|
830 |
+
|
831 |
+
body.push(...terms);
|
832 |
+
}
|
833 |
+
|
834 |
+
term.body = body;
|
835 |
+
}
|
836 |
+
else {
|
837 |
+
for (const subterm of term.body)
|
838 |
+
this.execute(subterm);
|
839 |
+
}
|
840 |
+
}
|
841 |
+
else if (term instanceof Command && term.cmd === "numericTimeSignature")
|
842 |
+
this.execute(term.args[0]);
|
843 |
+
else if (term instanceof TimeSignature) {
|
844 |
+
this.time = term;
|
845 |
+
this.measureSpan = term.value.value * WHOLE_DURATION_MAGNITUDE;
|
846 |
+
}
|
847 |
+
else if (term instanceof Partial)
|
848 |
+
this.partialDuration = term.duration;
|
849 |
+
else if (term instanceof Repeat) {
|
850 |
+
switch (term.type) {
|
851 |
+
case "volta":
|
852 |
+
this.checkIncompleteMeasure();
|
853 |
+
|
854 |
+
this.execute(term.bodyBlock);
|
855 |
+
|
856 |
+
this.checkIncompleteMeasure();
|
857 |
+
|
858 |
+
if (term.alternativeBlocks) {
|
859 |
+
for (const block of term.alternativeBlocks) {
|
860 |
+
this.execute(block);
|
861 |
+
|
862 |
+
this.checkIncompleteMeasure();
|
863 |
+
}
|
864 |
+
}
|
865 |
+
|
866 |
+
break;
|
867 |
+
case "tremolo":
|
868 |
+
this.push({factor: {value: term.times}, tremoloDuration: term.sumDuration});
|
869 |
+
this.tremoloType = term.singleTremolo ? TremoloType.Single : TremoloType.Pitcher;
|
870 |
+
this.execute(term.bodyBlock);
|
871 |
+
this.tremoloType = TremoloType.None;
|
872 |
+
this.pop();
|
873 |
+
|
874 |
+
break;
|
875 |
+
default:
|
876 |
+
console.warn("unsupported repeat type:", term.type);
|
877 |
+
}
|
878 |
+
}
|
879 |
+
else if (term instanceof Relative) {
|
880 |
+
if (term.anchor)
|
881 |
+
this.setPitch(term.anchor);
|
882 |
+
|
883 |
+
this.execute(term.music);
|
884 |
+
}
|
885 |
+
else if (term instanceof LyricMode) {
|
886 |
+
// ignore lyric mode
|
887 |
+
}
|
888 |
+
else if (term instanceof Command && term.cmd === "lyricsto") {
|
889 |
+
// ignore lyric mode
|
890 |
+
}
|
891 |
+
else if (term instanceof ChordMode) {
|
892 |
+
// ignore chord mode
|
893 |
+
}
|
894 |
+
else if (term instanceof Transposition)
|
895 |
+
this.transposition = term.transposition;
|
896 |
+
else if (term instanceof Times) {
|
897 |
+
this.push({factor: term.factor});
|
898 |
+
this.execute(term.music);
|
899 |
+
this.pop();
|
900 |
+
}
|
901 |
+
else if (term instanceof Tuplet) {
|
902 |
+
this.push({factor: term.divider.reciprocal});
|
903 |
+
this.execute(term.music);
|
904 |
+
this.pop();
|
905 |
+
}
|
906 |
+
else if (term instanceof Grace) {
|
907 |
+
this.inGrace = true;
|
908 |
+
this.push({factor: {value: 0}});
|
909 |
+
this.execute(term.music);
|
910 |
+
this.pop();
|
911 |
+
this.inGrace = false;
|
912 |
+
|
913 |
+
this.processGrace(term.music);
|
914 |
+
}
|
915 |
+
else if (term instanceof AfterGrace) {
|
916 |
+
this.execute(term.body);
|
917 |
+
|
918 |
+
this.inGrace = true;
|
919 |
+
this.push({factor: {value: 0}, tickBias: -term.body.durationMagnitude});
|
920 |
+
this.execute(term.grace);
|
921 |
+
this.pop();
|
922 |
+
this.inGrace = false;
|
923 |
+
|
924 |
+
this.processGrace(term.grace);
|
925 |
+
}
|
926 |
+
else if (term instanceof Clef)
|
927 |
+
this.clef = term;
|
928 |
+
else if (term instanceof KeySignature)
|
929 |
+
this.key = term;
|
930 |
+
else if (term instanceof OctaveShift)
|
931 |
+
this.octave = term;
|
932 |
+
else if (term instanceof Command && term.cmd === "change") {
|
933 |
+
const pair = term.getAssignmentPair();
|
934 |
+
if (pair) {
|
935 |
+
switch (pair.key) {
|
936 |
+
case "Staff":
|
937 |
+
this.staffName = pair.value.toString();
|
938 |
+
this.staff = term;
|
939 |
+
|
940 |
+
break;
|
941 |
+
case "Voice":
|
942 |
+
this.voiceName = pair.value.toString();
|
943 |
+
|
944 |
+
break;
|
945 |
+
}
|
946 |
+
}
|
947 |
+
}
|
948 |
+
else if (term instanceof Primitive) {
|
949 |
+
if (term.exp === "~")
|
950 |
+
this.tying = this.event;
|
951 |
+
}
|
952 |
+
else if (term instanceof PostEvent) {
|
953 |
+
if (term.isStaccato)
|
954 |
+
this.staccato = true;
|
955 |
+
}
|
956 |
+
else if (term instanceof SimultaneousList) {
|
957 |
+
const contexts: TrackContext[] = [];
|
958 |
+
let lastContext = this;
|
959 |
+
for (const subterm of term.list) {
|
960 |
+
const context = this.clone();
|
961 |
+
context.pitch = lastContext.pitch;
|
962 |
+
context.event = lastContext.event;
|
963 |
+
|
964 |
+
context.execute(subterm);
|
965 |
+
contexts.push(context);
|
966 |
+
lastContext = context;
|
967 |
+
}
|
968 |
+
|
969 |
+
this.mergeParallelClones(contexts);
|
970 |
+
}
|
971 |
+
else if (term instanceof ContextedMusic) {
|
972 |
+
// TODO: process contextDict
|
973 |
+
|
974 |
+
this.execute(term.body);
|
975 |
+
}
|
976 |
+
else if (term instanceof StemDirection)
|
977 |
+
this.stemDirection = term.direction;
|
978 |
+
else {
|
979 |
+
if (term.isMusic)
|
980 |
+
console.warn("[TrackContext] unexpected music term:", term);
|
981 |
+
}
|
982 |
+
|
983 |
+
if (this.listener)
|
984 |
+
this.listener(term, this);
|
985 |
+
|
986 |
+
if (term instanceof MusicEvent && this.tremoloType === TremoloType.Pitcher)
|
987 |
+
this.tremoloType = TremoloType.Catcher;
|
988 |
+
}
|
989 |
+
|
990 |
+
|
991 |
+
get declarations (): BaseTerm[] {
|
992 |
+
return [this.staff, this.clef, this.key, this.time, this.octave].filter(term => term);
|
993 |
+
}
|
994 |
+
};
|
995 |
+
|
996 |
+
|
997 |
+
class MusicPerformance {
|
998 |
+
staffNames: string[] = [];
|
999 |
+
musicTracks: MusicTrack[] = [];
|
1000 |
+
|
1001 |
+
|
1002 |
+
get mainTrack (): MusicTrack {
|
1003 |
+
// find the longest track
|
1004 |
+
const trackPrior = (track: MusicTrack, index: number): number => -track.block.durationMagnitude + index * 1e-3;
|
1005 |
+
const priorTracks = this.musicTracks
|
1006 |
+
.filter(track => track.block._parent && !track.isChordMode && !track.isLyricMode)
|
1007 |
+
.map((track, index) => ({track, index}))
|
1008 |
+
.sort((t1, t2) => trackPrior(t1.track, t1.index) - trackPrior(t2.track, t2.index));
|
1009 |
+
|
1010 |
+
return priorTracks[0] ? priorTracks[0].track : null;
|
1011 |
+
}
|
1012 |
+
|
1013 |
+
|
1014 |
+
get trackNames (): string[] {
|
1015 |
+
return this.musicTracks.map(track => `${track.contextDict.Staff}:${track.contextDict.Voice}`);
|
1016 |
+
}
|
1017 |
+
|
1018 |
+
|
1019 |
+
get trackContextDicts (): ContextDict[] {
|
1020 |
+
const dicts = this.musicTracks.map(track => track.contextDict);
|
1021 |
+
dicts.unshift(undefined); // zero placeholder for track index from 1 in notation & SheetDocument
|
1022 |
+
|
1023 |
+
return dicts;
|
1024 |
+
}
|
1025 |
+
|
1026 |
+
|
1027 |
+
get trackInstruments (): string[] {
|
1028 |
+
return this.musicTracks.map(track => {
|
1029 |
+
const instrumentKey = Object.keys(track.contextDict).find(key => /\.instrumentName/.test(key));
|
1030 |
+
if (instrumentKey)
|
1031 |
+
return track.contextDict[instrumentKey];
|
1032 |
+
|
1033 |
+
return null;
|
1034 |
+
});
|
1035 |
+
}
|
1036 |
+
|
1037 |
+
|
1038 |
+
get instrumentList (): string[] {
|
1039 |
+
return Array.from(new Set(this.trackInstruments));
|
1040 |
+
}
|
1041 |
+
|
1042 |
+
|
1043 |
+
get channelMap (): number[] {
|
1044 |
+
const instrumentList = this.instrumentList;
|
1045 |
+
const channels = this.trackInstruments.map(instrument => instrumentList.indexOf(instrument) + 1);
|
1046 |
+
channels.unshift(0);
|
1047 |
+
|
1048 |
+
return channels;
|
1049 |
+
}
|
1050 |
+
|
1051 |
+
|
1052 |
+
get measureLayoutCode (): string {
|
1053 |
+
return this.mainTrack && this.mainTrack.measureLayoutCode;
|
1054 |
+
}
|
1055 |
+
|
1056 |
+
|
1057 |
+
applyMeasureLayout (layout: MeasureLayout) {
|
1058 |
+
this.musicTracks.forEach(track => track.applyMeasureLayout(layout));
|
1059 |
+
}
|
1060 |
+
|
1061 |
+
|
1062 |
+
getNotation ({logger = new LogRecorder()} = {}): LilyNotation.Notation {
|
1063 |
+
const pcTerms: PitchContextTerm[] = [].concat(...this.musicTracks.map((track, i) =>
|
1064 |
+
track.generateStaffTracks().map(term => ({track: i + 1, ...term}))));
|
1065 |
+
//console.log("pcTerms:", pcTerms);
|
1066 |
+
|
1067 |
+
const termsToContexts = (staffTerms: PitchContextTerm[], trackIndex: number): LilyStaffContext => {
|
1068 |
+
staffTerms.forEach(term => {
|
1069 |
+
if (term.event)
|
1070 |
+
term.tick = term.event._tick;
|
1071 |
+
});
|
1072 |
+
staffTerms.sort((t1, t2) => t1.tick - t2.tick);
|
1073 |
+
|
1074 |
+
const context = new LilyStaffContext({logger});
|
1075 |
+
context.staffTrack = trackIndex;
|
1076 |
+
context.channelMap = this.channelMap;
|
1077 |
+
|
1078 |
+
logger.append("staffTerms", staffTerms);
|
1079 |
+
//console.debug("staffTerms:", staffTerms);
|
1080 |
+
staffTerms.forEach(term => context.executeTerm(term));
|
1081 |
+
|
1082 |
+
return context;
|
1083 |
+
};
|
1084 |
+
|
1085 |
+
const staffContexts: LilyStaffContext[] = [];
|
1086 |
+
if (this.staffNames.length) {
|
1087 |
+
this.staffNames.forEach((name, trackIndex) => {
|
1088 |
+
const staffTerms = pcTerms.filter(term => term.staffName === name);
|
1089 |
+
staffContexts.push(termsToContexts(staffTerms, trackIndex));
|
1090 |
+
});
|
1091 |
+
}
|
1092 |
+
else
|
1093 |
+
staffContexts.push(termsToContexts(pcTerms, 0));
|
1094 |
+
|
1095 |
+
const notes = []
|
1096 |
+
.concat(...staffContexts.map(context => context.notes))
|
1097 |
+
.sort((n1, n2) => n1.startTick - n2.startTick);
|
1098 |
+
|
1099 |
+
// append duration of tied notes to root note
|
1100 |
+
const pitchMap = {};
|
1101 |
+
notes.forEach(note => {
|
1102 |
+
if (note.tied) {
|
1103 |
+
const root = pitchMap[note.pitch];
|
1104 |
+
if (root) {
|
1105 |
+
root.endTick = Math.max(root.endTick, note.endTick);
|
1106 |
+
root.duration = root.endTick - root.startTick;
|
1107 |
+
}
|
1108 |
+
}
|
1109 |
+
else
|
1110 |
+
pitchMap[note.pitch] = note;
|
1111 |
+
});
|
1112 |
+
|
1113 |
+
const pitchContextGroup = staffContexts.map(context => context.pitchContextTable);
|
1114 |
+
|
1115 |
+
const mainTrack = this.mainTrack;
|
1116 |
+
|
1117 |
+
const measureHeads = mainTrack && mainTrack.measureHeads;
|
1118 |
+
const measureLayout = mainTrack && mainTrack.block.measureLayout;
|
1119 |
+
|
1120 |
+
return LilyNotation.Notation.fromAbsoluteNotes(notes, measureHeads, {pitchContextGroup, measureLayout, trackNames: this.trackNames});
|
1121 |
+
}
|
1122 |
+
|
1123 |
+
|
1124 |
+
getNoteDurationSubdivider (): number {
|
1125 |
+
const subdivider = lcmMulti(...this.musicTracks.map(track => track.noteDurationSubdivider));
|
1126 |
+
|
1127 |
+
return subdivider;
|
1128 |
+
}
|
1129 |
+
|
1130 |
+
|
1131 |
+
sliceMeasures (start: number, count: number) {
|
1132 |
+
this.musicTracks = this.musicTracks.map(track => {
|
1133 |
+
const newTrack = track.sliceMeasures(start, count);
|
1134 |
+
newTrack.name = track.name; // inherit name
|
1135 |
+
|
1136 |
+
return newTrack;
|
1137 |
+
});
|
1138 |
+
}
|
1139 |
+
};
|
1140 |
+
|
1141 |
+
|
1142 |
+
|
1143 |
+
export default class LilyInterpreter {
|
1144 |
+
variableTable: Map<string, BaseTerm> = new Map();
|
1145 |
+
|
1146 |
+
// temporary status
|
1147 |
+
musicTracks: MusicTrack[] = [];
|
1148 |
+
staffNames: string[] = [];
|
1149 |
+
musicTrackIndex: number = 0;
|
1150 |
+
musicPerformance: MusicPerformance;
|
1151 |
+
mainPerformance: MusicPerformance;
|
1152 |
+
|
1153 |
+
version: Version = null;
|
1154 |
+
language: Language = null;
|
1155 |
+
header: Block = null;
|
1156 |
+
includeFiles: Set<string> = new Set;
|
1157 |
+
statements: BaseTerm[] = [];
|
1158 |
+
paper: Block = null;
|
1159 |
+
layout: Block = null;
|
1160 |
+
scores: Block[] = [];
|
1161 |
+
|
1162 |
+
layoutMusic: MusicPerformance;
|
1163 |
+
midiMusic: MusicPerformance;
|
1164 |
+
|
1165 |
+
functionalCommand?: Variable;
|
1166 |
+
|
1167 |
+
reservedVariables: Set<string> = new Set();
|
1168 |
+
|
1169 |
+
|
1170 |
+
static trackName (index: number): string {
|
1171 |
+
return `Voice_${romanize(index)}`;
|
1172 |
+
};
|
1173 |
+
|
1174 |
+
|
1175 |
+
/*eval (term: BaseTerm): BaseTerm {
|
1176 |
+
return this.execute(term.clone());
|
1177 |
+
}*/
|
1178 |
+
|
1179 |
+
|
1180 |
+
get mainScore (): BaseTerm {
|
1181 |
+
return this.variableTable.get(MAIN_SCORE_NAME);
|
1182 |
+
};
|
1183 |
+
|
1184 |
+
|
1185 |
+
interpretMusic (music: BaseTerm, contextDict: ContextDict): Variable {
|
1186 |
+
//console.log("interpretMusic:", music);
|
1187 |
+
const context = new TrackContext(undefined, {contextDict});
|
1188 |
+
//context.execute(music.clone());
|
1189 |
+
context.execute(music);
|
1190 |
+
|
1191 |
+
context.track.spreadRelativeBlocks();
|
1192 |
+
this.musicTracks.push(context.track);
|
1193 |
+
|
1194 |
+
const varName = LilyInterpreter.trackName(++this.musicTrackIndex);
|
1195 |
+
|
1196 |
+
context.track.name = varName;
|
1197 |
+
|
1198 |
+
return new Variable({name: varName});
|
1199 |
+
}
|
1200 |
+
|
1201 |
+
|
1202 |
+
interpretDocument (doc: LilyDocument): this {
|
1203 |
+
if (doc.reservedVariables)
|
1204 |
+
this.appendReservedVariables(doc.reservedVariables);
|
1205 |
+
|
1206 |
+
this.execute(doc.root);
|
1207 |
+
|
1208 |
+
return this;
|
1209 |
+
}
|
1210 |
+
|
1211 |
+
|
1212 |
+
createMusicPerformance () {
|
1213 |
+
if (this.musicTracks.length) {
|
1214 |
+
if (!this.musicPerformance)
|
1215 |
+
this.musicPerformance = new MusicPerformance();
|
1216 |
+
|
1217 |
+
this.staffNames.forEach(name => {
|
1218 |
+
if (!this.musicPerformance.staffNames.some(sn => sn === name))
|
1219 |
+
this.musicPerformance.staffNames.push(name);
|
1220 |
+
else if (!name)
|
1221 |
+
console.warn("[LilyInterpreter] Multiple empty context staff name may cause error pitchContextTable:", this.musicPerformance.staffNames);
|
1222 |
+
});
|
1223 |
+
this.musicTracks.forEach(track => this.musicPerformance.musicTracks.push(track));
|
1224 |
+
|
1225 |
+
this.staffNames = [];
|
1226 |
+
this.musicTracks = [];
|
1227 |
+
}
|
1228 |
+
}
|
1229 |
+
|
1230 |
+
|
1231 |
+
execute (term: BaseTerm, {execMusic = false, contextDict = {}}: {execMusic?: boolean, contextDict?: ContextDict} = {}): BaseTerm {
|
1232 |
+
if (!term)
|
1233 |
+
return term;
|
1234 |
+
|
1235 |
+
if (this.functionalCommand && term.isMusic) {
|
1236 |
+
term._functional = this.functionalCommand.name;
|
1237 |
+
this.functionalCommand = null;
|
1238 |
+
}
|
1239 |
+
|
1240 |
+
if (term instanceof Root) {
|
1241 |
+
for (const section of term.sections) {
|
1242 |
+
const sec = this.execute(section, {execMusic: true});
|
1243 |
+
if (sec instanceof Version)
|
1244 |
+
this.version = sec;
|
1245 |
+
else if (sec instanceof Language)
|
1246 |
+
this.language = sec;
|
1247 |
+
else if (sec instanceof Scheme)
|
1248 |
+
this.statements.push(sec);
|
1249 |
+
else if (sec instanceof Block) {
|
1250 |
+
switch (sec.head) {
|
1251 |
+
case "\\header":
|
1252 |
+
this.header = sec;
|
1253 |
+
|
1254 |
+
break;
|
1255 |
+
case "\\paper":
|
1256 |
+
this.paper = sec;
|
1257 |
+
|
1258 |
+
break;
|
1259 |
+
case "\\layout":
|
1260 |
+
this.layout = sec;
|
1261 |
+
|
1262 |
+
break;
|
1263 |
+
case "\\score":
|
1264 |
+
this.scores.push(sec);
|
1265 |
+
|
1266 |
+
break;
|
1267 |
+
}
|
1268 |
+
}
|
1269 |
+
}
|
1270 |
+
|
1271 |
+
this.createMusicPerformance();
|
1272 |
+
|
1273 |
+
if (this.musicPerformance) {
|
1274 |
+
this.layoutMusic = this.musicPerformance;
|
1275 |
+
this.midiMusic = this.musicPerformance;
|
1276 |
+
|
1277 |
+
this.musicPerformance = null;
|
1278 |
+
}
|
1279 |
+
}
|
1280 |
+
else if (term instanceof Assignment) {
|
1281 |
+
if (term.key) {
|
1282 |
+
const name = term.key as string;
|
1283 |
+
const isMainScore = name === MAIN_SCORE_NAME;
|
1284 |
+
if (isMainScore)
|
1285 |
+
this.musicPerformance = null;
|
1286 |
+
|
1287 |
+
const value = this.execute(term.value, {execMusic: isMainScore});
|
1288 |
+
|
1289 |
+
this.variableTable.set(name, value);
|
1290 |
+
|
1291 |
+
if (isMainScore)
|
1292 |
+
this.mainPerformance = this.musicPerformance;
|
1293 |
+
}
|
1294 |
+
}
|
1295 |
+
else if (term instanceof Block) {
|
1296 |
+
switch (term.head) {
|
1297 |
+
case "\\score":
|
1298 |
+
const body = term.body.map(subterm => this.execute(subterm, {execMusic: true}));
|
1299 |
+
|
1300 |
+
this.musicPerformance = null;
|
1301 |
+
|
1302 |
+
return new Block({
|
1303 |
+
block: term.block,
|
1304 |
+
head: term.head,
|
1305 |
+
body,
|
1306 |
+
});
|
1307 |
+
case "\\layout":
|
1308 |
+
this.layoutMusic = this.musicPerformance;
|
1309 |
+
|
1310 |
+
break;
|
1311 |
+
case "\\midi":
|
1312 |
+
this.midiMusic = this.musicPerformance;
|
1313 |
+
|
1314 |
+
break;
|
1315 |
+
}
|
1316 |
+
}
|
1317 |
+
else if (term instanceof Variable) {
|
1318 |
+
const result = this.variableTable.get(term.name);
|
1319 |
+
if (!result) {
|
1320 |
+
if (FUNCTIONAL_VARIABLE_NAME_PATTERN.test(term.name)) {
|
1321 |
+
this.functionalCommand = term;
|
1322 |
+
|
1323 |
+
return term;
|
1324 |
+
}
|
1325 |
+
else if (this.reservedVariables.has(term.name)) {
|
1326 |
+
// ignore reserved variables
|
1327 |
+
}
|
1328 |
+
else
|
1329 |
+
console.warn("uninitialized variable is referred:", term);
|
1330 |
+
}
|
1331 |
+
|
1332 |
+
if (term.name === MAIN_SCORE_NAME) {
|
1333 |
+
this.musicPerformance = this.mainPerformance;
|
1334 |
+
|
1335 |
+
return term;
|
1336 |
+
}
|
1337 |
+
|
1338 |
+
return this.execute(result, {execMusic, contextDict});
|
1339 |
+
}
|
1340 |
+
else if (term instanceof MusicBlock) {
|
1341 |
+
const result = new MusicBlock({
|
1342 |
+
_parent: term._parent,
|
1343 |
+
_functional: term._functional,
|
1344 |
+
body: term.body.map(subterm => this.execute(subterm)).filter(Boolean),
|
1345 |
+
});
|
1346 |
+
|
1347 |
+
this.functionalCommand = null;
|
1348 |
+
|
1349 |
+
if (execMusic) {
|
1350 |
+
const variable = this.interpretMusic(result, contextDict);
|
1351 |
+
return new MusicBlock({body: [variable]});
|
1352 |
+
}
|
1353 |
+
|
1354 |
+
return result;
|
1355 |
+
}
|
1356 |
+
else if (term instanceof SimultaneousList) {
|
1357 |
+
const list = term.list.map(subterm => this.execute(subterm, {execMusic, contextDict})).filter(term => term);
|
1358 |
+
this.createMusicPerformance();
|
1359 |
+
|
1360 |
+
return new SimultaneousList({list});
|
1361 |
+
}
|
1362 |
+
else if (term instanceof ContextedMusic) {
|
1363 |
+
if (term.contextDict && typeof term.contextDict.Staff === "string")
|
1364 |
+
this.staffNames.push(term.contextDict.Staff);
|
1365 |
+
|
1366 |
+
return new ContextedMusic({
|
1367 |
+
head: this.execute(term.head),
|
1368 |
+
lyrics: this.execute(term.lyrics),
|
1369 |
+
body: this.execute(term.body, {execMusic, contextDict: {...contextDict, ...term.contextDict}}),
|
1370 |
+
});
|
1371 |
+
}
|
1372 |
+
else if (term instanceof ParallelMusic) {
|
1373 |
+
term.voices.forEach(voice => {
|
1374 |
+
const block = new MusicBlock({
|
1375 |
+
body: MusicChunk.join(voice.body),
|
1376 |
+
});
|
1377 |
+
|
1378 |
+
const value = this.execute(block);
|
1379 |
+
this.variableTable.set(voice.name, value);
|
1380 |
+
});
|
1381 |
+
}
|
1382 |
+
else if (term instanceof Include)
|
1383 |
+
this.includeFiles.add(term.filename);
|
1384 |
+
else if (term instanceof Command) {
|
1385 |
+
switch (term.cmd) {
|
1386 |
+
case "set":
|
1387 |
+
if (term.args[0] instanceof Assignment) {
|
1388 |
+
const assign = term.args[0];
|
1389 |
+
contextDict[assign.key.toString()] = assign.value.toString();
|
1390 |
+
}
|
1391 |
+
|
1392 |
+
break;
|
1393 |
+
}
|
1394 |
+
|
1395 |
+
return parseRaw({proto: term.proto, cmd: term.cmd, args: term.args.map(arg => this.execute(arg, {execMusic, contextDict}))});
|
1396 |
+
}
|
1397 |
+
|
1398 |
+
return term;
|
1399 |
+
}
|
1400 |
+
|
1401 |
+
|
1402 |
+
updateTrackAssignments () {
|
1403 |
+
if (this.layoutMusic)
|
1404 |
+
this.layoutMusic.musicTracks.forEach(track => this.variableTable.set(track.name, track.music));
|
1405 |
+
|
1406 |
+
if (this.midiMusic && this.midiMusic !== this.layoutMusic)
|
1407 |
+
this.midiMusic.musicTracks.forEach(track => this.variableTable.set(track.name, track.music));
|
1408 |
+
|
1409 |
+
// update main score variable order in table
|
1410 |
+
const mainScore = this.mainScore;
|
1411 |
+
if (mainScore) {
|
1412 |
+
this.variableTable.delete(MAIN_SCORE_NAME);
|
1413 |
+
this.variableTable.set(MAIN_SCORE_NAME, mainScore);
|
1414 |
+
}
|
1415 |
+
}
|
1416 |
+
|
1417 |
+
|
1418 |
+
toDocument (): LilyDocument {
|
1419 |
+
this.updateTrackAssignments();
|
1420 |
+
|
1421 |
+
const variables = [].concat(...[this.paper, this.layout, ...this.scores, this.mainScore].filter(block => block).map(block => block.findAll(Variable).map(variable => variable.name)));
|
1422 |
+
const variablesUnique = Array.from(new Set(variables));
|
1423 |
+
|
1424 |
+
// sort variables by order in variable table
|
1425 |
+
const vars = [...this.variableTable.keys()];
|
1426 |
+
variablesUnique.sort((v1, v2) => vars.indexOf(v1) - vars.indexOf(v2));
|
1427 |
+
|
1428 |
+
const assignments = variablesUnique.filter(name => this.variableTable.get(name)).map(name => new Assignment({key: name, value: this.variableTable.get(name)}));
|
1429 |
+
const includes = Array.from(this.includeFiles).map(filename => Include.create(filename));
|
1430 |
+
|
1431 |
+
const root = new Root({sections: [
|
1432 |
+
this.version,
|
1433 |
+
this.language,
|
1434 |
+
this.header,
|
1435 |
+
...includes,
|
1436 |
+
...this.statements,
|
1437 |
+
this.paper,
|
1438 |
+
this.layout,
|
1439 |
+
...assignments,
|
1440 |
+
...this.scores,
|
1441 |
+
].filter(section => section)});
|
1442 |
+
|
1443 |
+
const doc = new LilyDocument(root);
|
1444 |
+
doc.reservedVariables = this.reservedVariables;
|
1445 |
+
|
1446 |
+
return doc;
|
1447 |
+
}
|
1448 |
+
|
1449 |
+
|
1450 |
+
sliceMeasures (start: number, count: number) {
|
1451 |
+
if (this.layoutMusic)
|
1452 |
+
this.layoutMusic.sliceMeasures(start, count);
|
1453 |
+
|
1454 |
+
if (this.midiMusic && this.midiMusic !== this.layoutMusic)
|
1455 |
+
this.midiMusic.sliceMeasures(start, count);
|
1456 |
+
}
|
1457 |
+
|
1458 |
+
|
1459 |
+
addIncludeFile (filename: string) {
|
1460 |
+
this.includeFiles.add(filename);
|
1461 |
+
}
|
1462 |
+
|
1463 |
+
|
1464 |
+
appendReservedVariables (names: Iterable<string>) {
|
1465 |
+
for (const name of names)
|
1466 |
+
this.reservedVariables.add(name);
|
1467 |
+
}
|
1468 |
+
|
1469 |
+
|
1470 |
+
getNotation ({logger = new LogRecorder()} = {}): LilyNotation.Notation {
|
1471 |
+
if (this.midiMusic)
|
1472 |
+
return this.midiMusic.getNotation({logger});
|
1473 |
+
|
1474 |
+
return null;
|
1475 |
+
}
|
1476 |
+
};
|
inc/lilyParser/lilyTerms.ts
ADDED
@@ -0,0 +1,2991 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
|
2 |
+
import {WHOLE_DURATION_MAGNITUDE, FractionNumber, lcmMulti, gcd, MAIN_SCORE_NAME} from "./utils";
|
3 |
+
import * as idioms from "./idioms";
|
4 |
+
import {LILYPOND_VERSION} from "../constants";
|
5 |
+
import * as measureLayout from "../measureLayout";
|
6 |
+
import ImplicitType from "../lilyNotation/implicitType";
|
7 |
+
import pick from "../pick";
|
8 |
+
|
9 |
+
|
10 |
+
|
11 |
+
interface Location {
|
12 |
+
lines: [number, number];
|
13 |
+
columns: [number, number];
|
14 |
+
};
|
15 |
+
|
16 |
+
|
17 |
+
abstract class Locator {
|
18 |
+
location: Location;
|
19 |
+
|
20 |
+
|
21 |
+
constructor (term: BaseTerm) {
|
22 |
+
term._location = term._location || {lines: [0, 0], columns: [0, 0]};
|
23 |
+
this.location = term._location;
|
24 |
+
}
|
25 |
+
|
26 |
+
abstract set (line: number, column: number): void;
|
27 |
+
};
|
28 |
+
|
29 |
+
|
30 |
+
class OpenLocator extends Locator {
|
31 |
+
set (line: number, column: number) {
|
32 |
+
this.location.lines[0] = line;
|
33 |
+
this.location.columns[0] = column;
|
34 |
+
}
|
35 |
+
};
|
36 |
+
|
37 |
+
|
38 |
+
class CloseLocator extends Locator {
|
39 |
+
set (line: number, column: number) {
|
40 |
+
this.location.lines[1] = line;
|
41 |
+
this.location.columns[1] = column;
|
42 |
+
}
|
43 |
+
};
|
44 |
+
|
45 |
+
|
46 |
+
// concat array of array
|
47 |
+
const cc = <T>(arrays: T[][]): T[] => [].concat(...arrays);
|
48 |
+
|
49 |
+
|
50 |
+
export class MusicChunk {
|
51 |
+
parent: MusicBlock;
|
52 |
+
terms: BaseTerm[];
|
53 |
+
|
54 |
+
|
55 |
+
static join (chunks: MusicChunk[]): BaseTerm[] {
|
56 |
+
return cc(chunks.map((chunk, i) => i === chunks.length - 1 ? chunk.terms : [...chunk.terms, new Divide({})]));
|
57 |
+
}
|
58 |
+
|
59 |
+
|
60 |
+
constructor (parent: MusicBlock, terms: BaseTerm[] = []) {
|
61 |
+
this.parent = parent;
|
62 |
+
this.terms = terms;
|
63 |
+
}
|
64 |
+
|
65 |
+
|
66 |
+
push (term: BaseTerm) {
|
67 |
+
this.terms.push(term);
|
68 |
+
}
|
69 |
+
|
70 |
+
|
71 |
+
get size () {
|
72 |
+
return this.terms.length;
|
73 |
+
}
|
74 |
+
|
75 |
+
|
76 |
+
get durationMagnitude () {
|
77 |
+
return this.terms.reduce((magnitude, term) => magnitude + term.durationMagnitude, 0);
|
78 |
+
}
|
79 |
+
};
|
80 |
+
|
81 |
+
|
82 |
+
interface MusicVoice {
|
83 |
+
name?: string;
|
84 |
+
body: MusicChunk[];
|
85 |
+
};
|
86 |
+
|
87 |
+
|
88 |
+
type MusicChunkMap = Map<number, MusicChunk>;
|
89 |
+
|
90 |
+
|
91 |
+
const isNullItem = item => item === "" || item === undefined || item === null || (Array.isArray(item) && !item.length);
|
92 |
+
const compact = items => cc(items.map((item, index) => isNullItem(item) ? [] : [index > 0 ? "\b" : null, item]));
|
93 |
+
|
94 |
+
|
95 |
+
export const getDurationSubdivider = (term: BaseTerm): number => {
|
96 |
+
if (term instanceof MusicEvent) {
|
97 |
+
if (!(term instanceof Rest) || !term.isSpacer)
|
98 |
+
return term.durationValue.subdivider;
|
99 |
+
}
|
100 |
+
else if (term instanceof MusicBlock)
|
101 |
+
return lcmMulti(...term.body.map(getDurationSubdivider));
|
102 |
+
else if (term instanceof MusicChunk)
|
103 |
+
return lcmMulti(...term.terms.map(getDurationSubdivider));
|
104 |
+
else if ((term instanceof Times) || (term instanceof Tuplet)) {
|
105 |
+
const divider = term instanceof Tuplet ? term.divider : term.factor.reciprocal;
|
106 |
+
divider.numerator *= getDurationSubdivider(term.music);
|
107 |
+
|
108 |
+
return divider.reduced.numerator;
|
109 |
+
}
|
110 |
+
else if (term instanceof Repeat)
|
111 |
+
return getDurationSubdivider(term.bodyBlock);
|
112 |
+
else if (term instanceof Relative)
|
113 |
+
return getDurationSubdivider(term.music);
|
114 |
+
else if (term.isMusic)
|
115 |
+
console.warn("[getDurationSubdivider] unexpected music term:", term);
|
116 |
+
|
117 |
+
return 1;
|
118 |
+
};
|
119 |
+
|
120 |
+
|
121 |
+
export const constructMusicFromMeasureLayout = (layout: measureLayout.MeasureLayout, chunks: MusicChunkMap): MusicChunk => {
|
122 |
+
const joinMeasureSeq = (seq: measureLayout.MeasureSeq): BaseTerm[] => MusicChunk.join(seq.map(sublayout => constructMusicFromMeasureLayout(sublayout, chunks)));
|
123 |
+
|
124 |
+
if (layout instanceof measureLayout.SingleMLayout) {
|
125 |
+
const chunk = chunks.get(layout.measure);
|
126 |
+
console.assert(!!chunk, "no chunk for measure:", layout.measure);
|
127 |
+
|
128 |
+
return chunk;
|
129 |
+
}
|
130 |
+
else if (layout instanceof measureLayout.BlockMLayout) {
|
131 |
+
const terms = joinMeasureSeq(layout.seq);
|
132 |
+
|
133 |
+
return new MusicChunk(null, terms);
|
134 |
+
}
|
135 |
+
else if (layout instanceof measureLayout.VoltaMLayout) {
|
136 |
+
const bodyTerms = joinMeasureSeq(layout.body);
|
137 |
+
const alternative = layout.alternates && layout.alternates.map(alternate => new MusicBlock({body: joinMeasureSeq(alternate)}));
|
138 |
+
|
139 |
+
const repeat = Repeat.createVolta(layout.times.toString(), new MusicBlock({body: bodyTerms}), alternative);
|
140 |
+
|
141 |
+
return new MusicChunk(null, [repeat]);
|
142 |
+
}
|
143 |
+
else if (layout instanceof measureLayout.ABAMLayout) {
|
144 |
+
const mainList = constructMusicFromMeasureLayout(layout.main, chunks);
|
145 |
+
const main = mainList.terms.length === 1 ? mainList.terms[0] : new MusicBlock({body: mainList.terms});
|
146 |
+
const restTerms = joinMeasureSeq(layout.rest);
|
147 |
+
|
148 |
+
const block = new MusicBlock({body: [main, ...restTerms]});
|
149 |
+
|
150 |
+
return new MusicChunk(null, [new Variable({name: "lotusRepeatABA"}), block]);
|
151 |
+
}
|
152 |
+
};
|
153 |
+
|
154 |
+
|
155 |
+
export class BaseTerm {
|
156 |
+
_location?: Location;
|
157 |
+
_measure?: number;
|
158 |
+
_tick?: number;
|
159 |
+
_previous?: BaseTerm;
|
160 |
+
_anchorPitch?: ChordElement;
|
161 |
+
_parent?: BaseTerm;
|
162 |
+
|
163 |
+
_headComment: Comment;
|
164 |
+
_tailComment: Comment;
|
165 |
+
|
166 |
+
// lotus extensional function modifier
|
167 |
+
_functional: string;
|
168 |
+
|
169 |
+
|
170 |
+
constructor (data: object) {
|
171 |
+
//Object.assign(this, data);
|
172 |
+
for (const key in data)
|
173 |
+
this[key] = parseRaw(data[key]);
|
174 |
+
}
|
175 |
+
|
176 |
+
|
177 |
+
serialize (): any[] {
|
178 |
+
console.warn("unimplemented serilization:", this);
|
179 |
+
return [];
|
180 |
+
}
|
181 |
+
|
182 |
+
|
183 |
+
join (): string {
|
184 |
+
let words = this.serialize().filter(word => ["string", "number"].includes(typeof word)).map(word => word.toString()) as string[];
|
185 |
+
words = words.filter((word, i) => !(i && words[i - 1] === "\n" && word === "\n"));
|
186 |
+
|
187 |
+
let indent = 0;
|
188 |
+
const result: string[] = [];
|
189 |
+
|
190 |
+
const pop = char => {
|
191 |
+
if (!char || result[result.length - 1] === char) {
|
192 |
+
result.pop();
|
193 |
+
return true;
|
194 |
+
}
|
195 |
+
};
|
196 |
+
|
197 |
+
for (const word of words) {
|
198 |
+
switch (word) {
|
199 |
+
case "\b":
|
200 |
+
// remove last space
|
201 |
+
pop(" ");
|
202 |
+
continue;
|
203 |
+
|
204 |
+
case "\b\n":
|
205 |
+
// remove last newline
|
206 |
+
while (pop("\t")) {}
|
207 |
+
pop("\n");
|
208 |
+
continue;
|
209 |
+
|
210 |
+
case "\n":
|
211 |
+
// no space at line tail
|
212 |
+
pop(" ");
|
213 |
+
}
|
214 |
+
|
215 |
+
if (/^(\}|>>)/.test(word))
|
216 |
+
pop("\t"); // remove the last tab
|
217 |
+
|
218 |
+
result.push(word);
|
219 |
+
|
220 |
+
if (/\n$/.test(word)) {
|
221 |
+
if (/(\{|<<)\n$/.test(word))
|
222 |
+
++indent;
|
223 |
+
else if (/^(\}|>>)/.test(word))
|
224 |
+
--indent;
|
225 |
+
|
226 |
+
if (indent)
|
227 |
+
result.push(...Array(indent).fill("\t"));
|
228 |
+
}
|
229 |
+
else
|
230 |
+
result.push(" ");
|
231 |
+
}
|
232 |
+
|
233 |
+
return result.join("");
|
234 |
+
}
|
235 |
+
|
236 |
+
|
237 |
+
relocate (source: string = this.join()) {
|
238 |
+
const words = this.serialize()
|
239 |
+
.filter(word => word !== null && word !== undefined
|
240 |
+
&& (typeof word !== "string" || (/\S/.test(word) && !word.includes("\b"))))
|
241 |
+
.map(word => typeof word === "string" ? word.replace(/\n/g, "") : word);
|
242 |
+
|
243 |
+
const chars = source.split("");
|
244 |
+
let line = 1;
|
245 |
+
let column = 0;
|
246 |
+
|
247 |
+
let wordIndex = 0;
|
248 |
+
|
249 |
+
for (let i = 0; i < chars.length; ++i) {
|
250 |
+
if (wordIndex >= words.length)
|
251 |
+
break;
|
252 |
+
|
253 |
+
const char = chars[i];
|
254 |
+
|
255 |
+
switch (char) {
|
256 |
+
case "\n":
|
257 |
+
++line;
|
258 |
+
column = 0;
|
259 |
+
|
260 |
+
break;
|
261 |
+
case " ":
|
262 |
+
case "\t":
|
263 |
+
++column;
|
264 |
+
|
265 |
+
break;
|
266 |
+
default:
|
267 |
+
let word = words[wordIndex];
|
268 |
+
while (word instanceof Locator) {
|
269 |
+
word.set(line, column);
|
270 |
+
|
271 |
+
++wordIndex;
|
272 |
+
word = words[wordIndex];
|
273 |
+
}
|
274 |
+
|
275 |
+
if (wordIndex >= words.length)
|
276 |
+
break;
|
277 |
+
|
278 |
+
word = word.toString();
|
279 |
+
if (char === word[0]) {
|
280 |
+
i += word.length - 1;
|
281 |
+
column += word.length;
|
282 |
+
++wordIndex;
|
283 |
+
}
|
284 |
+
else {
|
285 |
+
//debugger;
|
286 |
+
throw new Error(`unexpected char in source: [${i}]'${char}', expect: ${word}`);
|
287 |
+
}
|
288 |
+
}
|
289 |
+
}
|
290 |
+
}
|
291 |
+
|
292 |
+
|
293 |
+
clone (): this {
|
294 |
+
return parseRaw(JSON.parse(JSON.stringify(this)));
|
295 |
+
}
|
296 |
+
|
297 |
+
|
298 |
+
get entries (): BaseTerm[] {
|
299 |
+
return null;
|
300 |
+
}
|
301 |
+
|
302 |
+
|
303 |
+
get isMusic (): boolean {
|
304 |
+
return false;
|
305 |
+
}
|
306 |
+
|
307 |
+
|
308 |
+
get musicChunks (): MusicChunk[] {
|
309 |
+
if (!this.isMusic || !this.entries)
|
310 |
+
return [];
|
311 |
+
|
312 |
+
return [].concat(...this.entries.map(entry => entry.musicChunks));
|
313 |
+
}
|
314 |
+
|
315 |
+
|
316 |
+
get measures (): number[] {
|
317 |
+
const indices = [this._measure].concat(...(this.entries || []).map(entry => entry.measures)).filter(index => Number.isInteger(index));
|
318 |
+
|
319 |
+
return Array.from(new Set(indices));
|
320 |
+
}
|
321 |
+
|
322 |
+
|
323 |
+
get durationMagnitude (): number {
|
324 |
+
return 0;
|
325 |
+
}
|
326 |
+
|
327 |
+
|
328 |
+
get proto () {
|
329 |
+
return termProtoMap.get(Object.getPrototypeOf(this));
|
330 |
+
}
|
331 |
+
|
332 |
+
|
333 |
+
get href (): string {
|
334 |
+
if (this._location)
|
335 |
+
return `${this._location.lines[0]}:${this._location.columns[0]}:${this._location.columns[1]}`;
|
336 |
+
|
337 |
+
return null;
|
338 |
+
}
|
339 |
+
|
340 |
+
|
341 |
+
get measureLayout (): measureLayout.MeasureLayout {
|
342 |
+
return null;
|
343 |
+
}
|
344 |
+
|
345 |
+
|
346 |
+
getField (key): any {
|
347 |
+
console.assert(!!this.entries, "[BaseTerm.getField] term's entries is null:", this);
|
348 |
+
|
349 |
+
for (const entry of this.entries) {
|
350 |
+
const result = entry.query(key);
|
351 |
+
if (result)
|
352 |
+
return result;
|
353 |
+
}
|
354 |
+
}
|
355 |
+
|
356 |
+
|
357 |
+
query (key: string): any {
|
358 |
+
void(key);
|
359 |
+
//console.warn("term.query not implemented:", this);
|
360 |
+
}
|
361 |
+
|
362 |
+
|
363 |
+
appendAssignment (key, value) {
|
364 |
+
console.assert(!!this.entries, "no entries on this term.");
|
365 |
+
|
366 |
+
const assign = this.getField(key);
|
367 |
+
if (assign)
|
368 |
+
assign.value = value;
|
369 |
+
else {
|
370 |
+
this.entries.push(parseRaw({
|
371 |
+
proto: "Assignment",
|
372 |
+
key,
|
373 |
+
value: value,
|
374 |
+
}));
|
375 |
+
}
|
376 |
+
}
|
377 |
+
|
378 |
+
|
379 |
+
findFirst (condition: Function): BaseTerm {
|
380 |
+
if (!this.entries)
|
381 |
+
return null;
|
382 |
+
|
383 |
+
if (BaseTerm.isPrototypeOf(condition)) {
|
384 |
+
const termClass = condition;
|
385 |
+
condition = term => term instanceof termClass;
|
386 |
+
}
|
387 |
+
|
388 |
+
for (const entry of this.entries) {
|
389 |
+
if (condition(entry))
|
390 |
+
return entry;
|
391 |
+
|
392 |
+
if (entry instanceof BaseTerm) {
|
393 |
+
const result = entry.findFirst(condition);
|
394 |
+
if (result)
|
395 |
+
return result;
|
396 |
+
}
|
397 |
+
}
|
398 |
+
}
|
399 |
+
|
400 |
+
|
401 |
+
findLast (condition: any): BaseTerm {
|
402 |
+
if (!this.entries)
|
403 |
+
return null;
|
404 |
+
|
405 |
+
if (BaseTerm.isPrototypeOf(condition)) {
|
406 |
+
const termClass = condition;
|
407 |
+
condition = term => term instanceof termClass;
|
408 |
+
}
|
409 |
+
|
410 |
+
const reversedEntries = [...this.entries];
|
411 |
+
reversedEntries.reverse();
|
412 |
+
|
413 |
+
for (const entry of reversedEntries) {
|
414 |
+
if (condition(entry))
|
415 |
+
return entry;
|
416 |
+
|
417 |
+
if (entry instanceof BaseTerm) {
|
418 |
+
const result = entry.findLast(condition);
|
419 |
+
if (result)
|
420 |
+
return result;
|
421 |
+
}
|
422 |
+
}
|
423 |
+
}
|
424 |
+
|
425 |
+
|
426 |
+
findAll (condition: any): any[] {
|
427 |
+
if (!this.entries)
|
428 |
+
return [];
|
429 |
+
|
430 |
+
if (BaseTerm.isPrototypeOf(condition)) {
|
431 |
+
const termClass = condition;
|
432 |
+
condition = term => term instanceof termClass;
|
433 |
+
}
|
434 |
+
|
435 |
+
const result = [];
|
436 |
+
|
437 |
+
for (const entry of this.entries) {
|
438 |
+
if (condition(entry))
|
439 |
+
result.push(entry);
|
440 |
+
|
441 |
+
if (entry instanceof BaseTerm)
|
442 |
+
result.push(...entry.findAll(condition));
|
443 |
+
}
|
444 |
+
|
445 |
+
return result;
|
446 |
+
}
|
447 |
+
|
448 |
+
|
449 |
+
forEachTerm (termClass, handle) {
|
450 |
+
if (!this.entries)
|
451 |
+
return;
|
452 |
+
|
453 |
+
for (const entry of this.entries) {
|
454 |
+
if (entry instanceof termClass)
|
455 |
+
handle(entry);
|
456 |
+
|
457 |
+
if (entry instanceof BaseTerm)
|
458 |
+
entry.forEachTerm(termClass, handle);
|
459 |
+
}
|
460 |
+
}
|
461 |
+
|
462 |
+
|
463 |
+
forEachTopTerm (termClass, handle) {
|
464 |
+
if (!this.entries)
|
465 |
+
return;
|
466 |
+
|
467 |
+
for (const entry of this.entries) {
|
468 |
+
if (entry instanceof termClass)
|
469 |
+
handle(entry);
|
470 |
+
else if (entry instanceof BaseTerm)
|
471 |
+
entry.forEachTopTerm(termClass, handle);
|
472 |
+
}
|
473 |
+
}
|
474 |
+
|
475 |
+
|
476 |
+
toJSON () {
|
477 |
+
// exlude meta fields in JSON
|
478 |
+
const fields = Object.keys(this).filter(key => !/^_/.test(key));
|
479 |
+
const data = pick(this, fields);
|
480 |
+
|
481 |
+
Object.entries(data).forEach(([key, value]) => {
|
482 |
+
if (value && typeof value === "object" && !Array.isArray(value) && !(value instanceof BaseTerm))
|
483 |
+
data[key] = {proto: "_PLAIN", ...value};
|
484 |
+
});
|
485 |
+
|
486 |
+
return {
|
487 |
+
proto: this.proto,
|
488 |
+
...data,
|
489 |
+
};
|
490 |
+
}
|
491 |
+
|
492 |
+
|
493 |
+
static isTerm (x): boolean {
|
494 |
+
return typeof x === "object" && x instanceof BaseTerm;
|
495 |
+
}
|
496 |
+
|
497 |
+
|
498 |
+
static optionalSerialize (item: any): any[] {
|
499 |
+
//return BaseTerm.isTerm(item) ? (item as BaseTerm).serialize() : (item === undefined ? [] : [item]);
|
500 |
+
if (!BaseTerm.isTerm(item))
|
501 |
+
return item === undefined ? [] : [item];
|
502 |
+
|
503 |
+
return [
|
504 |
+
...BaseTerm.optionalSerialize(item._headComment),
|
505 |
+
...item.serialize(),
|
506 |
+
...(item._tailComment ? ["\b\n", "\t"] : []),
|
507 |
+
...BaseTerm.optionalSerialize(item._tailComment),
|
508 |
+
];
|
509 |
+
}
|
510 |
+
|
511 |
+
|
512 |
+
static serializeScheme (item: any): any[] {
|
513 |
+
if (typeof item === "boolean")
|
514 |
+
item = item ? "#t" : "#f";
|
515 |
+
|
516 |
+
return BaseTerm.optionalSerialize(item);
|
517 |
+
}
|
518 |
+
}
|
519 |
+
|
520 |
+
|
521 |
+
export class Root extends BaseTerm {
|
522 |
+
sections: BaseTerm[];
|
523 |
+
|
524 |
+
|
525 |
+
serialize () {
|
526 |
+
return cc(this.sections.map(section => [...BaseTerm.optionalSerialize(section), "\n\n"]));
|
527 |
+
}
|
528 |
+
|
529 |
+
|
530 |
+
get entries (): BaseTerm[] {
|
531 |
+
return this.sections;
|
532 |
+
}
|
533 |
+
|
534 |
+
|
535 |
+
getBlock (head): Block {
|
536 |
+
return this.entries.find((entry: any) => entry.head === head || (entry.head === "\\" + head)) as Block;
|
537 |
+
}
|
538 |
+
|
539 |
+
|
540 |
+
get includeFiles (): string[] {
|
541 |
+
return this.sections.filter(section => section instanceof Include).map((include: Include) => include.filename);
|
542 |
+
}
|
543 |
+
|
544 |
+
|
545 |
+
static priorityForSection (term: BaseTerm): number {
|
546 |
+
if (term instanceof Version)
|
547 |
+
return 0;
|
548 |
+
|
549 |
+
if (term instanceof Language)
|
550 |
+
return 1;
|
551 |
+
|
552 |
+
if (term instanceof Scheme)
|
553 |
+
return 3;
|
554 |
+
|
555 |
+
if (term instanceof Assignment)
|
556 |
+
return 7;
|
557 |
+
|
558 |
+
if (term instanceof Block) {
|
559 |
+
switch (term.head) {
|
560 |
+
case "\\header":
|
561 |
+
return 2;
|
562 |
+
|
563 |
+
case "\\paper":
|
564 |
+
return 4;
|
565 |
+
|
566 |
+
case "\\layout":
|
567 |
+
return 5;
|
568 |
+
|
569 |
+
case "\\score":
|
570 |
+
return 10;
|
571 |
+
}
|
572 |
+
}
|
573 |
+
|
574 |
+
return Infinity;
|
575 |
+
}
|
576 |
+
|
577 |
+
|
578 |
+
reorderSections () {
|
579 |
+
this.sections.sort((s1, s2) => Root.priorityForSection(s1) - Root.priorityForSection(s2));
|
580 |
+
}
|
581 |
+
};
|
582 |
+
|
583 |
+
|
584 |
+
export class Primitive extends BaseTerm {
|
585 |
+
exp: string | number;
|
586 |
+
|
587 |
+
|
588 |
+
serialize () {
|
589 |
+
return [this.exp];
|
590 |
+
}
|
591 |
+
};
|
592 |
+
|
593 |
+
|
594 |
+
export class LiteralString extends BaseTerm {
|
595 |
+
exp: string
|
596 |
+
|
597 |
+
|
598 |
+
static fromString (content: string): LiteralString {
|
599 |
+
return new LiteralString({exp: JSON.stringify(content)});
|
600 |
+
}
|
601 |
+
|
602 |
+
|
603 |
+
serialize () {
|
604 |
+
return [this.exp];
|
605 |
+
}
|
606 |
+
|
607 |
+
|
608 |
+
toString () {
|
609 |
+
try {
|
610 |
+
return eval(this.exp);
|
611 |
+
}
|
612 |
+
catch (err) {
|
613 |
+
console.warn("invalid lilypond string exp:", this.exp);
|
614 |
+
return this.exp;
|
615 |
+
}
|
616 |
+
}
|
617 |
+
};
|
618 |
+
|
619 |
+
|
620 |
+
export class Command extends BaseTerm {
|
621 |
+
cmd: string;
|
622 |
+
args: any[];
|
623 |
+
|
624 |
+
|
625 |
+
static createSet (key: string|BaseTerm, value: BaseTerm): Command {
|
626 |
+
return new Command({cmd: "set", args: [new Assignment({key, value})]});
|
627 |
+
}
|
628 |
+
|
629 |
+
|
630 |
+
constructor (data) {
|
631 |
+
super(data);
|
632 |
+
|
633 |
+
this.args.forEach(term => {
|
634 |
+
if (term instanceof MusicBlock || term instanceof Block)
|
635 |
+
term._parent = this;
|
636 |
+
});
|
637 |
+
}
|
638 |
+
|
639 |
+
|
640 |
+
serialize () {
|
641 |
+
return [
|
642 |
+
"\\" + this.cmd,
|
643 |
+
...[].concat(...this.args.map(BaseTerm.optionalSerialize)),
|
644 |
+
["break", "pageBreak", "overrideProperty"].includes(this.cmd) ? "\n" : null,
|
645 |
+
];
|
646 |
+
}
|
647 |
+
|
648 |
+
|
649 |
+
get entries () {
|
650 |
+
return this.args.filter(arg => arg instanceof BaseTerm);
|
651 |
+
}
|
652 |
+
|
653 |
+
|
654 |
+
get isMusic (): boolean {
|
655 |
+
return this.args.some(arg => arg.isMusic);
|
656 |
+
}
|
657 |
+
|
658 |
+
|
659 |
+
get musicChunks (): MusicChunk[] {
|
660 |
+
if (this.cmd === "alternative")
|
661 |
+
return [].concat(...this.args[0].body.map(term => term.musicChunks));
|
662 |
+
|
663 |
+
return [].concat(...this.entries.map(entry => entry.musicChunks));
|
664 |
+
}
|
665 |
+
|
666 |
+
|
667 |
+
get isRepeatWithAlternative () {
|
668 |
+
return this.cmd === "repeat"
|
669 |
+
&& this.args[2] instanceof MusicBlock
|
670 |
+
&& this.args[3]
|
671 |
+
&& this.args[3].cmd === "alternative";
|
672 |
+
}
|
673 |
+
|
674 |
+
|
675 |
+
get durationMagnitude (): number {
|
676 |
+
switch (this.cmd) {
|
677 |
+
// TODO: refine this in Times
|
678 |
+
case "times": {
|
679 |
+
const factor = eval(this.args[0]);
|
680 |
+
return this.args[1].durationMagnitude * factor;
|
681 |
+
}
|
682 |
+
|
683 |
+
// TODO: refine this in Tuplet
|
684 |
+
case "tuplet": {
|
685 |
+
const factor = 1 / eval(this.args[0]);
|
686 |
+
return this.args[this.args.length - 1].durationMagnitude * factor;
|
687 |
+
}
|
688 |
+
|
689 |
+
case "afterGrace":
|
690 |
+
return this.args[0].durationMagnitude;
|
691 |
+
|
692 |
+
default:
|
693 |
+
if (this instanceof Grace)
|
694 |
+
return 0;
|
695 |
+
|
696 |
+
return this.args.filter(arg => arg instanceof BaseTerm).reduce((magnitude, term) => magnitude + term.durationMagnitude, 0);
|
697 |
+
}
|
698 |
+
}
|
699 |
+
|
700 |
+
|
701 |
+
get measureLayout (): measureLayout.MeasureLayout {
|
702 |
+
const args = [...this.args].reverse();
|
703 |
+
for (const arg of args) {
|
704 |
+
const layout = arg instanceof BaseTerm && arg.measureLayout;
|
705 |
+
if (layout)
|
706 |
+
return layout;
|
707 |
+
}
|
708 |
+
|
709 |
+
return null;
|
710 |
+
}
|
711 |
+
|
712 |
+
|
713 |
+
getAssignmentPair (): {key: any, value: any} {
|
714 |
+
if (this.args[0] instanceof Assignment)
|
715 |
+
return {key: this.args[0].key, value: this.args[0].value};
|
716 |
+
|
717 |
+
if (this.args[1] instanceof Assignment)
|
718 |
+
return {key: this.args[0], value: this.args[1].value};
|
719 |
+
|
720 |
+
if (typeof this.args[0] === "string")
|
721 |
+
return {key: this.args[0], value: ""};
|
722 |
+
|
723 |
+
return null;
|
724 |
+
}
|
725 |
+
};
|
726 |
+
|
727 |
+
|
728 |
+
export class Variable extends Command {
|
729 |
+
name: string
|
730 |
+
|
731 |
+
|
732 |
+
constructor ({name}) {
|
733 |
+
super({cmd: name, args: []});
|
734 |
+
|
735 |
+
this.name = name;
|
736 |
+
}
|
737 |
+
|
738 |
+
|
739 |
+
toJSON (): any {
|
740 |
+
return {
|
741 |
+
proto: this.proto,
|
742 |
+
name: this.name,
|
743 |
+
};
|
744 |
+
}
|
745 |
+
|
746 |
+
|
747 |
+
queryValue (dict: BaseTerm): any {
|
748 |
+
const field = dict.getField(this.name);
|
749 |
+
|
750 |
+
return field && field.value;
|
751 |
+
}
|
752 |
+
|
753 |
+
|
754 |
+
get isMusic (): boolean {
|
755 |
+
if ([MAIN_SCORE_NAME].includes(this.name))
|
756 |
+
return true;
|
757 |
+
|
758 |
+
return false;
|
759 |
+
}
|
760 |
+
};
|
761 |
+
|
762 |
+
|
763 |
+
export class MarkupCommand extends Command {
|
764 |
+
toString () {
|
765 |
+
const strs = [];
|
766 |
+
this.forEachTerm(LiteralString, term => strs.push(term.toString()));
|
767 |
+
|
768 |
+
return strs.join("\n");
|
769 |
+
}
|
770 |
+
};
|
771 |
+
|
772 |
+
|
773 |
+
export class Repeat extends Command {
|
774 |
+
static createVolta (times: string, body: MusicBlock, alternative?: MusicBlock[]): Repeat {
|
775 |
+
const args: any[] = [
|
776 |
+
"volta",
|
777 |
+
times,
|
778 |
+
body,
|
779 |
+
];
|
780 |
+
|
781 |
+
if (alternative) {
|
782 |
+
args.push(new Command({
|
783 |
+
cmd: "alternative",
|
784 |
+
args: [new MusicBlock({body: alternative})],
|
785 |
+
}));
|
786 |
+
}
|
787 |
+
|
788 |
+
return new Repeat({cmd: "repeat", args});
|
789 |
+
}
|
790 |
+
|
791 |
+
|
792 |
+
get type (): string {
|
793 |
+
return this.args[0];
|
794 |
+
}
|
795 |
+
|
796 |
+
|
797 |
+
get times () {
|
798 |
+
return Number(this.args[1]);
|
799 |
+
}
|
800 |
+
|
801 |
+
|
802 |
+
get bodyBlock (): MusicBlock {
|
803 |
+
return this.args[2];
|
804 |
+
}
|
805 |
+
|
806 |
+
|
807 |
+
get alternativeBlocks (): MusicBlock[] {
|
808 |
+
return this.args[3] && this.args[3].args[0].body;
|
809 |
+
}
|
810 |
+
|
811 |
+
|
812 |
+
// this result length equal to times, if not null
|
813 |
+
get completeAlternativeBlocks (): MusicBlock[] {
|
814 |
+
if (!this.alternativeBlocks || !this.alternativeBlocks.length)
|
815 |
+
return null;
|
816 |
+
|
817 |
+
if (this.alternativeBlocks.length >= this.times)
|
818 |
+
return this.alternativeBlocks.slice(0, this.times);
|
819 |
+
|
820 |
+
const list = [];
|
821 |
+
for (let i = 0; i < this.times - this.alternativeBlocks.length; ++ i)
|
822 |
+
list.push(this.alternativeBlocks[0]);
|
823 |
+
list.push(...this.alternativeBlocks);
|
824 |
+
|
825 |
+
return list;
|
826 |
+
}
|
827 |
+
|
828 |
+
|
829 |
+
get measureLayout (): measureLayout.MeasureLayout {
|
830 |
+
switch (this.type) {
|
831 |
+
case "volta": {
|
832 |
+
const layout = new measureLayout.VoltaMLayout();
|
833 |
+
layout.times = this.times;
|
834 |
+
layout.body = this.bodyBlock.measureLayout.seq;
|
835 |
+
layout.alternates = this.alternativeBlocks && this.alternativeBlocks.map(block => block.measureLayout.seq);
|
836 |
+
|
837 |
+
return layout;
|
838 |
+
}
|
839 |
+
|
840 |
+
case "tremolo":
|
841 |
+
return this.bodyBlock.measureLayout;
|
842 |
+
|
843 |
+
default:
|
844 |
+
console.warn("unsupported repeat type:", this.type);
|
845 |
+
}
|
846 |
+
|
847 |
+
return null;
|
848 |
+
}
|
849 |
+
|
850 |
+
|
851 |
+
// for tremolo
|
852 |
+
get sumDuration (): Duration {
|
853 |
+
if (this.bodyBlock instanceof MusicEvent)
|
854 |
+
return Duration.fromMagnitude(this.args[2].durationMagnitude * this.times);
|
855 |
+
else if (this.bodyBlock instanceof MusicBlock) {
|
856 |
+
const events = this.bodyBlock.body.filter(term => term instanceof MusicEvent);
|
857 |
+
const magnitude = events.reduce((m, event) => m + event.durationMagnitude, 0) * this.times;
|
858 |
+
|
859 |
+
return Duration.fromMagnitude(magnitude);
|
860 |
+
}
|
861 |
+
|
862 |
+
return null;
|
863 |
+
}
|
864 |
+
|
865 |
+
|
866 |
+
get singleTremolo (): boolean {
|
867 |
+
if (this.type === "tremolo") {
|
868 |
+
if (this.bodyBlock instanceof MusicEvent)
|
869 |
+
return true;
|
870 |
+
|
871 |
+
if (this.bodyBlock instanceof MusicBlock) {
|
872 |
+
const events = this.bodyBlock.body.filter(term => term instanceof MusicEvent);
|
873 |
+
return events.length === 1;
|
874 |
+
}
|
875 |
+
}
|
876 |
+
|
877 |
+
return false;
|
878 |
+
}
|
879 |
+
|
880 |
+
|
881 |
+
// \repeat {body} \alternative {{alter1} {alter2}} => body alter1 body alter2
|
882 |
+
getUnfoldTerms (): BaseTerm[] {
|
883 |
+
const completeAlternativeBlocks = this.completeAlternativeBlocks;
|
884 |
+
|
885 |
+
const list = [];
|
886 |
+
for (let i = 0; i < this.times; ++i) {
|
887 |
+
list.push(...this.bodyBlock.clone().body);
|
888 |
+
|
889 |
+
if (completeAlternativeBlocks)
|
890 |
+
list.push(...completeAlternativeBlocks[i].clone().body);
|
891 |
+
}
|
892 |
+
|
893 |
+
return list;
|
894 |
+
}
|
895 |
+
|
896 |
+
|
897 |
+
// \repeat {body} \alternative {{alter1} {alter2}} => body alter1 alter2
|
898 |
+
getPlainTerms (): BaseTerm[] {
|
899 |
+
const list = [...this.bodyBlock.clone().body];
|
900 |
+
|
901 |
+
const alternativeBlocks = this.alternativeBlocks;
|
902 |
+
if (alternativeBlocks)
|
903 |
+
alternativeBlocks.forEach(block => list.push(...block.clone().body));
|
904 |
+
|
905 |
+
return list;
|
906 |
+
}
|
907 |
+
|
908 |
+
|
909 |
+
// \repeat {body} \alternative {{alter1} {alter2}} => body alter2
|
910 |
+
getTailPassTerms (): BaseTerm[] {
|
911 |
+
const list = [...this.bodyBlock.clone().body];
|
912 |
+
|
913 |
+
const alternativeBlocks = this.alternativeBlocks;
|
914 |
+
if (alternativeBlocks)
|
915 |
+
list.push(...alternativeBlocks[alternativeBlocks.length - 1].clone().body);
|
916 |
+
|
917 |
+
return list;
|
918 |
+
}
|
919 |
+
};
|
920 |
+
|
921 |
+
|
922 |
+
export class Relative extends Command {
|
923 |
+
static makeBlock (block: MusicBlock, {anchor}: {anchor?: ChordElement} = {}): Relative {
|
924 |
+
if (!anchor) {
|
925 |
+
const chord = block.findFirst(Chord) as Chord;
|
926 |
+
anchor = chord && chord.anchorPitch;
|
927 |
+
}
|
928 |
+
|
929 |
+
return new Relative({cmd: "relative", args: [anchor, block].filter(term => term)});
|
930 |
+
}
|
931 |
+
|
932 |
+
|
933 |
+
get anchor (): ChordElement {
|
934 |
+
if (this.args[0] instanceof ChordElement)
|
935 |
+
return this.args[0];
|
936 |
+
|
937 |
+
return null;
|
938 |
+
}
|
939 |
+
|
940 |
+
|
941 |
+
get music (): BaseTerm {
|
942 |
+
return this.args[this.args.length - 1];
|
943 |
+
}
|
944 |
+
|
945 |
+
|
946 |
+
get headChord (): Chord {
|
947 |
+
return this.findFirst(Chord) as Chord;
|
948 |
+
}
|
949 |
+
|
950 |
+
|
951 |
+
get tailPitch (): ChordElement {
|
952 |
+
const tail = this.findLast(Chord) as Chord;
|
953 |
+
|
954 |
+
return tail && tail.absolutePitch;
|
955 |
+
}
|
956 |
+
|
957 |
+
|
958 |
+
// with side effect
|
959 |
+
shiftBody (newAnchor?: ChordElement): BaseTerm[] {
|
960 |
+
const headChord = this.headChord;
|
961 |
+
if (newAnchor && headChord) {
|
962 |
+
headChord.shiftAnchor(newAnchor);
|
963 |
+
headChord._anchorPitch = null;
|
964 |
+
//console.log("shiftAnchor.post:", headChord.join(), headChord);
|
965 |
+
}
|
966 |
+
|
967 |
+
const music = this.music;
|
968 |
+
if (music instanceof MusicBlock) {
|
969 |
+
//music.clearPitchCache();
|
970 |
+
return music.body;
|
971 |
+
}
|
972 |
+
|
973 |
+
return [music];
|
974 |
+
}
|
975 |
+
}
|
976 |
+
|
977 |
+
|
978 |
+
export class ParallelMusic extends Command {
|
979 |
+
get varNames (): string[] {
|
980 |
+
return ((this.args[0].exp as SchemePointer).value as SchemeFunction).asList as string[];
|
981 |
+
}
|
982 |
+
|
983 |
+
|
984 |
+
get body (): MusicBlock {
|
985 |
+
return this.args[1];
|
986 |
+
}
|
987 |
+
|
988 |
+
|
989 |
+
get voices (): MusicVoice[] {
|
990 |
+
const voiceNames = this.varNames;
|
991 |
+
const chunks = this.body.musicChunks;
|
992 |
+
const measureCount = Math.ceil(chunks.length / voiceNames.length);
|
993 |
+
|
994 |
+
return voiceNames.map((name, index) => ({
|
995 |
+
name: name.toString(),
|
996 |
+
body: Array(measureCount).fill(null).map((_, m) => chunks[m * voiceNames.length + index]).filter(Boolean),
|
997 |
+
}));
|
998 |
+
}
|
999 |
+
};
|
1000 |
+
|
1001 |
+
|
1002 |
+
export class TimeSignature extends Command {
|
1003 |
+
get value (): FractionNumber {
|
1004 |
+
return FractionNumber.fromExpression(this.args[0]);
|
1005 |
+
}
|
1006 |
+
};
|
1007 |
+
|
1008 |
+
|
1009 |
+
export class Partial extends Command {
|
1010 |
+
get duration (): Duration {
|
1011 |
+
return this.args[0];
|
1012 |
+
}
|
1013 |
+
};
|
1014 |
+
|
1015 |
+
|
1016 |
+
export class Times extends Command {
|
1017 |
+
get factor (): FractionNumber {
|
1018 |
+
return FractionNumber.fromExpression(this.args[0]);
|
1019 |
+
}
|
1020 |
+
|
1021 |
+
|
1022 |
+
get music (): BaseTerm {
|
1023 |
+
return this.args[this.args.length - 1];
|
1024 |
+
}
|
1025 |
+
};
|
1026 |
+
|
1027 |
+
|
1028 |
+
export class Tuplet extends Command {
|
1029 |
+
get divider (): FractionNumber {
|
1030 |
+
return FractionNumber.fromExpression(this.args[0]);
|
1031 |
+
}
|
1032 |
+
|
1033 |
+
|
1034 |
+
get music (): BaseTerm {
|
1035 |
+
return this.args[this.args.length - 1];
|
1036 |
+
}
|
1037 |
+
};
|
1038 |
+
|
1039 |
+
|
1040 |
+
export class Grace extends Command {
|
1041 |
+
get music (): BaseTerm {
|
1042 |
+
return this.args[this.args.length - 1];
|
1043 |
+
}
|
1044 |
+
};
|
1045 |
+
|
1046 |
+
|
1047 |
+
export class AfterGrace extends Command {
|
1048 |
+
get body (): BaseTerm {
|
1049 |
+
return this.args[0];
|
1050 |
+
}
|
1051 |
+
|
1052 |
+
get grace (): BaseTerm {
|
1053 |
+
return this.args[1];
|
1054 |
+
}
|
1055 |
+
|
1056 |
+
|
1057 |
+
get measureLayout (): measureLayout.MeasureLayout {
|
1058 |
+
return measureLayout.BlockMLayout.fromSeq([
|
1059 |
+
this.body.measureLayout,
|
1060 |
+
this.grace.measureLayout,
|
1061 |
+
]);
|
1062 |
+
}
|
1063 |
+
};
|
1064 |
+
|
1065 |
+
|
1066 |
+
export class Clef extends Command {
|
1067 |
+
get clefName (): string {
|
1068 |
+
return this.args[0].toString();
|
1069 |
+
}
|
1070 |
+
};
|
1071 |
+
|
1072 |
+
|
1073 |
+
export class KeySignature extends Command {
|
1074 |
+
get keyPitch (): ChordElement {
|
1075 |
+
return new ChordElement({pitch: this.args[0], options: {proto: "_PLAIN"}});
|
1076 |
+
}
|
1077 |
+
|
1078 |
+
|
1079 |
+
get key (): number {
|
1080 |
+
const keyPitch = this.keyPitch;
|
1081 |
+
const minor = this.args[1] === "\\minor";
|
1082 |
+
const phonetOrder = idioms.FIFTH_PHONETS.indexOf(keyPitch.phonet);
|
1083 |
+
|
1084 |
+
return phonetOrder + (minor ? -4 : -1) + keyPitch.alterValue * 7;
|
1085 |
+
}
|
1086 |
+
};
|
1087 |
+
|
1088 |
+
|
1089 |
+
export class OctaveShift extends Command {
|
1090 |
+
get value (): number {
|
1091 |
+
return this.args[0].exp;
|
1092 |
+
}
|
1093 |
+
};
|
1094 |
+
|
1095 |
+
|
1096 |
+
export class Include extends Command {
|
1097 |
+
static create (filename: string): Include {
|
1098 |
+
return new Include({cmd: "include", args: [LiteralString.fromString(filename)]});
|
1099 |
+
}
|
1100 |
+
|
1101 |
+
|
1102 |
+
get filename (): string {
|
1103 |
+
return this.args[0].toString();
|
1104 |
+
}
|
1105 |
+
};
|
1106 |
+
|
1107 |
+
|
1108 |
+
export class Version extends Command {
|
1109 |
+
static get default (): Version {
|
1110 |
+
return new Version({cmd: "version", args: [LiteralString.fromString(LILYPOND_VERSION)]});
|
1111 |
+
}
|
1112 |
+
|
1113 |
+
|
1114 |
+
get version (): string {
|
1115 |
+
return this.args[0].toString();
|
1116 |
+
}
|
1117 |
+
};
|
1118 |
+
|
1119 |
+
|
1120 |
+
export class Language extends Command {
|
1121 |
+
static make (language: string): Language {
|
1122 |
+
return new Language({cmd: "language", args: [LiteralString.fromString(language)]});
|
1123 |
+
}
|
1124 |
+
|
1125 |
+
|
1126 |
+
get language (): string {
|
1127 |
+
return this.args[0].toString();
|
1128 |
+
}
|
1129 |
+
};
|
1130 |
+
|
1131 |
+
|
1132 |
+
export class LyricMode extends Command {
|
1133 |
+
get block (): MusicBlock {
|
1134 |
+
return this.args[0];
|
1135 |
+
}
|
1136 |
+
};
|
1137 |
+
|
1138 |
+
|
1139 |
+
export class ChordMode extends Command {
|
1140 |
+
get block (): MusicBlock {
|
1141 |
+
return this.args[0];
|
1142 |
+
}
|
1143 |
+
};
|
1144 |
+
|
1145 |
+
|
1146 |
+
export class Transposition extends Command {
|
1147 |
+
get transposition (): number {
|
1148 |
+
return this.args[0].pitchValue - 60;
|
1149 |
+
}
|
1150 |
+
};
|
1151 |
+
|
1152 |
+
|
1153 |
+
export class StemDirection extends Command {
|
1154 |
+
get direction (): string {
|
1155 |
+
return this.cmd.substr(4);
|
1156 |
+
}
|
1157 |
+
};
|
1158 |
+
|
1159 |
+
|
1160 |
+
export class Change extends Command {
|
1161 |
+
get key (): string {
|
1162 |
+
return this.args[0].toString();
|
1163 |
+
}
|
1164 |
+
|
1165 |
+
|
1166 |
+
get value (): string {
|
1167 |
+
return this.args[1].toString();
|
1168 |
+
}
|
1169 |
+
};
|
1170 |
+
|
1171 |
+
|
1172 |
+
export class Block extends BaseTerm {
|
1173 |
+
block: string;
|
1174 |
+
head: (string|string[]);
|
1175 |
+
body: BaseTerm[];
|
1176 |
+
|
1177 |
+
|
1178 |
+
constructor (data) {
|
1179 |
+
super(data);
|
1180 |
+
|
1181 |
+
this.body = this.body.map(parseRawEnforce);
|
1182 |
+
}
|
1183 |
+
|
1184 |
+
|
1185 |
+
serialize () {
|
1186 |
+
const heads = Array.isArray(this.head) ? this.head : (this.head ? [this.head] : []);
|
1187 |
+
|
1188 |
+
return [
|
1189 |
+
...heads,
|
1190 |
+
"{\n",
|
1191 |
+
...cc(this.body.map(section => [...BaseTerm.optionalSerialize(section), "\n"])),
|
1192 |
+
"}\n",
|
1193 |
+
];
|
1194 |
+
}
|
1195 |
+
|
1196 |
+
|
1197 |
+
get entries () {
|
1198 |
+
return this.body;
|
1199 |
+
}
|
1200 |
+
|
1201 |
+
|
1202 |
+
get isMIDIDedicated () {
|
1203 |
+
const subBlocks = this.body.filter(term => term instanceof Block) as Block[];
|
1204 |
+
|
1205 |
+
return subBlocks.some(term => term.head === "\\midi")
|
1206 |
+
&& !subBlocks.some(term => term.head === "\\layout");
|
1207 |
+
}
|
1208 |
+
|
1209 |
+
|
1210 |
+
get assignmentDict (): {[key: string]: string} {
|
1211 |
+
const assignments = this.body.filter(term => term instanceof Assignment) as Assignment[];
|
1212 |
+
|
1213 |
+
return assignments.reduce((dict, assignment) => ((dict[assignment.key.toString()] = assignment.value.toString()), dict), {});
|
1214 |
+
}
|
1215 |
+
};
|
1216 |
+
|
1217 |
+
|
1218 |
+
export class InlineBlock extends Block {
|
1219 |
+
serialize () {
|
1220 |
+
return [
|
1221 |
+
"{",
|
1222 |
+
...cc(this.body.map(BaseTerm.optionalSerialize)),
|
1223 |
+
"}",
|
1224 |
+
];
|
1225 |
+
}
|
1226 |
+
};
|
1227 |
+
|
1228 |
+
|
1229 |
+
export class MusicBlock extends BaseTerm {
|
1230 |
+
body: BaseTerm[];
|
1231 |
+
|
1232 |
+
|
1233 |
+
static fromTerms (terms: BaseTerm[]): MusicBlock {
|
1234 |
+
const block = new MusicBlock({body: [...terms]});
|
1235 |
+
block.clarifyHead();
|
1236 |
+
|
1237 |
+
return block;
|
1238 |
+
}
|
1239 |
+
|
1240 |
+
|
1241 |
+
constructor (data) {
|
1242 |
+
super(data);
|
1243 |
+
|
1244 |
+
this.body = this.body.map(parseRawEnforce);
|
1245 |
+
}
|
1246 |
+
|
1247 |
+
|
1248 |
+
serialize () {
|
1249 |
+
return [
|
1250 |
+
"{\n",
|
1251 |
+
...cc(this.body.map(BaseTerm.optionalSerialize)),
|
1252 |
+
"\n",
|
1253 |
+
"}\n",
|
1254 |
+
];
|
1255 |
+
}
|
1256 |
+
|
1257 |
+
|
1258 |
+
clone (): this {
|
1259 |
+
if (this._parent) {
|
1260 |
+
const parent = this._parent.clone();
|
1261 |
+
const block = parent.findFirst(MusicBlock);
|
1262 |
+
console.assert(block && block._parent === parent, "invalid block-parent relation", parent, block);
|
1263 |
+
|
1264 |
+
return block as this;
|
1265 |
+
}
|
1266 |
+
|
1267 |
+
return BaseTerm.prototype.clone.call(this) as this;
|
1268 |
+
}
|
1269 |
+
|
1270 |
+
|
1271 |
+
get entries () {
|
1272 |
+
return this.body;
|
1273 |
+
}
|
1274 |
+
|
1275 |
+
|
1276 |
+
get isMusic (): boolean {
|
1277 |
+
return true;
|
1278 |
+
}
|
1279 |
+
|
1280 |
+
|
1281 |
+
get musicChunks (): MusicChunk[] {
|
1282 |
+
const chunks = [];
|
1283 |
+
let currentChunk = new MusicChunk(this);
|
1284 |
+
|
1285 |
+
const dumpChunk = () => {
|
1286 |
+
if (currentChunk.size)
|
1287 |
+
chunks.push(currentChunk);
|
1288 |
+
|
1289 |
+
currentChunk = new MusicChunk(this);
|
1290 |
+
};
|
1291 |
+
|
1292 |
+
for (const term of this.entries) {
|
1293 |
+
if (term instanceof Repeat) {
|
1294 |
+
dumpChunk();
|
1295 |
+
chunks.push(...term.musicChunks);
|
1296 |
+
}
|
1297 |
+
else if (term instanceof Divide)
|
1298 |
+
dumpChunk();
|
1299 |
+
else
|
1300 |
+
currentChunk.push(term);
|
1301 |
+
}
|
1302 |
+
|
1303 |
+
dumpChunk();
|
1304 |
+
|
1305 |
+
return chunks;
|
1306 |
+
}
|
1307 |
+
|
1308 |
+
|
1309 |
+
// [deprecated]
|
1310 |
+
// for parallelMusic only
|
1311 |
+
get voiceNames () {
|
1312 |
+
const header = this._parent as Command;
|
1313 |
+
if (header && header.cmd === "parallelMusic") {
|
1314 |
+
if (header.args[0] instanceof Scheme && header.args[0].exp instanceof SchemePointer && header.args[0].exp.value instanceof SchemeFunction) {
|
1315 |
+
const voices = header.args[0].exp.value.asList;
|
1316 |
+
return voices;
|
1317 |
+
}
|
1318 |
+
}
|
1319 |
+
|
1320 |
+
return null;
|
1321 |
+
}
|
1322 |
+
|
1323 |
+
|
1324 |
+
// [deprecated]
|
1325 |
+
get voices (): MusicVoice[] {
|
1326 |
+
const voiceNames = this.voiceNames;
|
1327 |
+
if (!voiceNames)
|
1328 |
+
return [{body: this.musicChunks}];
|
1329 |
+
|
1330 |
+
const chunks = this.musicChunks;
|
1331 |
+
const measureCount = Math.ceil(chunks.length / voiceNames.length);
|
1332 |
+
|
1333 |
+
return voiceNames.map((name, index) => ({
|
1334 |
+
name: name.toString(),
|
1335 |
+
body: Array(measureCount).fill(null).map((_, m) => chunks[m * voiceNames.length + index]).filter(chunk => chunk),
|
1336 |
+
}));
|
1337 |
+
}
|
1338 |
+
|
1339 |
+
|
1340 |
+
get durationMagnitude (): number {
|
1341 |
+
return this.body.reduce((magnitude, term) => magnitude + term.durationMagnitude, 0);
|
1342 |
+
}
|
1343 |
+
|
1344 |
+
|
1345 |
+
get isRelative (): boolean {
|
1346 |
+
return this._parent instanceof Relative;
|
1347 |
+
}
|
1348 |
+
|
1349 |
+
|
1350 |
+
get anchorPitch (): ChordElement {
|
1351 |
+
if (this.isRelative)
|
1352 |
+
return (this._parent as Relative).anchor;
|
1353 |
+
|
1354 |
+
return null;
|
1355 |
+
}
|
1356 |
+
|
1357 |
+
|
1358 |
+
get measures (): number[] {
|
1359 |
+
// make a continouse indices list
|
1360 |
+
const subterms = this.findAll(term => term.isMusic);
|
1361 |
+
const subIndices = [].concat(...subterms.map(term => term.measures)).filter(index => Number.isInteger(index));
|
1362 |
+
if (!subIndices.length)
|
1363 |
+
return [];
|
1364 |
+
|
1365 |
+
const min = Math.min(...subIndices);
|
1366 |
+
const max = Math.max(...subIndices);
|
1367 |
+
|
1368 |
+
return Array(max + 1 - min).fill(null).map((_, i) => i + min);
|
1369 |
+
}
|
1370 |
+
|
1371 |
+
|
1372 |
+
get notes (): Chord[] {
|
1373 |
+
const notes = this.body.filter(term => term instanceof Chord && !term.isRest) as Chord[];
|
1374 |
+
this.forEachTopTerm(MusicBlock, block => notes.push(...block.notes));
|
1375 |
+
|
1376 |
+
return notes;
|
1377 |
+
}
|
1378 |
+
|
1379 |
+
|
1380 |
+
get sonicNotes (): Chord[] {
|
1381 |
+
return this.notes.filter(note => !note.completeTied);
|
1382 |
+
}
|
1383 |
+
|
1384 |
+
|
1385 |
+
get noteTicks (): number[] {
|
1386 |
+
const ticks = this.sonicNotes.map(note => note._tick);
|
1387 |
+
return Array.from(new Set(ticks)).sort((t1, t2) => t1 - t2);
|
1388 |
+
}
|
1389 |
+
|
1390 |
+
|
1391 |
+
get measureTicks (): [number, number][] {
|
1392 |
+
const tickTable: {[key: string]: number} = {};
|
1393 |
+
|
1394 |
+
this.body.forEach(term => {
|
1395 |
+
if (Number.isFinite(term._measure) && Number.isFinite(term._tick)) {
|
1396 |
+
if (!Number.isFinite(tickTable[term._measure]))
|
1397 |
+
tickTable[term._measure] = term._tick;
|
1398 |
+
}
|
1399 |
+
});
|
1400 |
+
|
1401 |
+
return Object.entries(tickTable).map(([measure, tick]) => [Number(measure), tick]);
|
1402 |
+
}
|
1403 |
+
|
1404 |
+
|
1405 |
+
get measureLayout (): measureLayout.MeasureLayout {
|
1406 |
+
const seq = this.body.filter(term => term.isMusic).map(term => term.measureLayout).filter(Boolean);
|
1407 |
+
|
1408 |
+
if (this._functional === "lotusRepeatABA") {
|
1409 |
+
const [main, ...rest] = seq;
|
1410 |
+
|
1411 |
+
const layout = new measureLayout.ABAMLayout();
|
1412 |
+
layout.main = main;
|
1413 |
+
layout.rest = measureLayout.BlockMLayout.trimSeq(rest);
|
1414 |
+
|
1415 |
+
return layout;
|
1416 |
+
}
|
1417 |
+
|
1418 |
+
return measureLayout.BlockMLayout.fromSeq(seq);
|
1419 |
+
}
|
1420 |
+
|
1421 |
+
|
1422 |
+
get measureChunkMap (): MusicChunkMap {
|
1423 |
+
const map = new Map<number, MusicChunk>();
|
1424 |
+
this.body.forEach(term => {
|
1425 |
+
if (Number.isInteger(term._measure) && !(term instanceof Divide)) {
|
1426 |
+
if (!map.get(term._measure))
|
1427 |
+
map.set(term._measure, new MusicChunk(this));
|
1428 |
+
|
1429 |
+
const chunk = map.get(term._measure);
|
1430 |
+
chunk.terms.push(term);
|
1431 |
+
}
|
1432 |
+
});
|
1433 |
+
|
1434 |
+
return map;
|
1435 |
+
}
|
1436 |
+
|
1437 |
+
|
1438 |
+
clearPitchCache () {
|
1439 |
+
this.forEachTerm(ChordElement, pitch => {
|
1440 |
+
pitch._absolutePitch = null;
|
1441 |
+
//pitch._previous = null;
|
1442 |
+
});
|
1443 |
+
}
|
1444 |
+
|
1445 |
+
|
1446 |
+
updateChordAnchors () {
|
1447 |
+
const chord = this.findFirst(Chord) as Chord;
|
1448 |
+
if (chord)
|
1449 |
+
chord._anchorPitch = chord._anchorPitch || this.anchorPitch;
|
1450 |
+
|
1451 |
+
this.clearPitchCache();
|
1452 |
+
}
|
1453 |
+
|
1454 |
+
|
1455 |
+
// deprecated
|
1456 |
+
updateChordChains () {
|
1457 |
+
let previous: MusicEvent = null;
|
1458 |
+
|
1459 |
+
this.updateChordAnchors();
|
1460 |
+
|
1461 |
+
this.forEachTerm(MusicBlock, block => block.updateChordAnchors());
|
1462 |
+
|
1463 |
+
this.forEachTerm(MusicEvent, event => {
|
1464 |
+
event._previous = previous;
|
1465 |
+
|
1466 |
+
previous = event;
|
1467 |
+
});
|
1468 |
+
}
|
1469 |
+
|
1470 |
+
|
1471 |
+
// with side effect
|
1472 |
+
spreadRepeatBlocks ({ignoreRepeat = true, keepTailPass = false} = {}): this {
|
1473 |
+
this.forEachTerm(MusicBlock, block => block.spreadRepeatBlocks());
|
1474 |
+
|
1475 |
+
this.body = cc(this.body.map(term => {
|
1476 |
+
if (term instanceof Repeat) {
|
1477 |
+
if (!ignoreRepeat)
|
1478 |
+
return term.getUnfoldTerms();
|
1479 |
+
else if (keepTailPass)
|
1480 |
+
return term.getTailPassTerms();
|
1481 |
+
else
|
1482 |
+
return term.getPlainTerms();
|
1483 |
+
}
|
1484 |
+
else
|
1485 |
+
return [term];
|
1486 |
+
}));
|
1487 |
+
|
1488 |
+
return this;
|
1489 |
+
}
|
1490 |
+
|
1491 |
+
|
1492 |
+
/*// with side effect
|
1493 |
+
spreadRelativeBlocks (): this {
|
1494 |
+
this.forEachTerm(MusicBlock, block => block.spreadRelativeBlocks());
|
1495 |
+
|
1496 |
+
let anchorPitch = null;
|
1497 |
+
|
1498 |
+
this.body = cc(this.body.map(term => {
|
1499 |
+
if (term instanceof Relative) {
|
1500 |
+
const list = term.shiftBody(anchorPitch);
|
1501 |
+
|
1502 |
+
anchorPitch = term.tailPitch || anchorPitch;
|
1503 |
+
|
1504 |
+
return list;
|
1505 |
+
}
|
1506 |
+
else
|
1507 |
+
return [term];
|
1508 |
+
}));
|
1509 |
+
|
1510 |
+
return this;
|
1511 |
+
}*/
|
1512 |
+
|
1513 |
+
|
1514 |
+
// with side effect
|
1515 |
+
unfoldDurationMultipliers (): this {
|
1516 |
+
let timeDenominator = 4;
|
1517 |
+
|
1518 |
+
const unfoldMultipliers = (term): BaseTerm[] => {
|
1519 |
+
if (term instanceof TimeSignature)
|
1520 |
+
timeDenominator = term.value.denominator;
|
1521 |
+
|
1522 |
+
if (!(term instanceof MusicEvent) || !term.duration || !term.duration.multipliers || !term.duration.multipliers.length)
|
1523 |
+
return [term];
|
1524 |
+
|
1525 |
+
const factor = term.duration.multipliers.reduce((factor, multiplier) => factor * Number(multiplier), 1);
|
1526 |
+
if (!Number.isInteger(factor) || factor <= 0)
|
1527 |
+
return [term];
|
1528 |
+
|
1529 |
+
const denominator = Math.max(term.duration.denominator, timeDenominator);
|
1530 |
+
|
1531 |
+
const event = term.clone() as MusicEvent;
|
1532 |
+
event.duration.multipliers = [];
|
1533 |
+
|
1534 |
+
// break duration into multiple rest events
|
1535 |
+
const restCount = (event.duration.magnitude / WHOLE_DURATION_MAGNITUDE) * (factor - 1) * denominator;
|
1536 |
+
if (!Number.isInteger(restCount))
|
1537 |
+
console.warn("Rest count is not integear:", restCount, denominator, event.duration.magnitude, factor);
|
1538 |
+
|
1539 |
+
const rests = Array(Math.floor(restCount)).fill(null).map(() =>
|
1540 |
+
new Rest({name: "s", duration: new Duration({number: denominator, dots: 0})}));
|
1541 |
+
|
1542 |
+
return [event, ...rests];
|
1543 |
+
};
|
1544 |
+
|
1545 |
+
this.body = cc(this.body.map(unfoldMultipliers));
|
1546 |
+
|
1547 |
+
return this;
|
1548 |
+
}
|
1549 |
+
|
1550 |
+
|
1551 |
+
/*// pure
|
1552 |
+
flatten ({spreadRepeats = false} = {}): Relative {
|
1553 |
+
this.updateChordChains();
|
1554 |
+
|
1555 |
+
const chord = this.findFirst(Chord) as Chord;
|
1556 |
+
const anchor = this.anchorPitch || (chord && chord.anchorPitch);
|
1557 |
+
|
1558 |
+
const block = this.clone();
|
1559 |
+
if (spreadRepeats)
|
1560 |
+
block.spreadRepeatBlocks();
|
1561 |
+
block.spreadRelativeBlocks();
|
1562 |
+
block.unfoldDurationMultipliers();
|
1563 |
+
|
1564 |
+
return Relative.makeBlock(block, {anchor: anchor && anchor.clone()});
|
1565 |
+
}*/
|
1566 |
+
|
1567 |
+
|
1568 |
+
// with side effect
|
1569 |
+
expandVariables (dict: BaseTerm): this {
|
1570 |
+
this.body = this.body.map(term => {
|
1571 |
+
if (term instanceof Variable) {
|
1572 |
+
const value = term.queryValue(dict);
|
1573 |
+
const clonedValue = value instanceof BaseTerm ? value.clone() : value;
|
1574 |
+
|
1575 |
+
if (clonedValue instanceof BaseTerm) {
|
1576 |
+
clonedValue.forEachTerm(MusicBlock, block => block.expandVariables(dict));
|
1577 |
+
|
1578 |
+
if (clonedValue instanceof MusicBlock)
|
1579 |
+
clonedValue.expandVariables(dict);
|
1580 |
+
}
|
1581 |
+
|
1582 |
+
return clonedValue;
|
1583 |
+
}
|
1584 |
+
|
1585 |
+
return term;
|
1586 |
+
});
|
1587 |
+
|
1588 |
+
return this;
|
1589 |
+
}
|
1590 |
+
|
1591 |
+
|
1592 |
+
// with side effects
|
1593 |
+
redivide ({recursive = true, measureHeads = null}: {recursive?: boolean, measureHeads?: number[]} = {}) {
|
1594 |
+
if (recursive) {
|
1595 |
+
this.forEachTerm(MusicBlock, block => {
|
1596 |
+
if (!block._parent || block._parent.cmd !== "alternative")
|
1597 |
+
block.redivide({recursive, measureHeads});
|
1598 |
+
});
|
1599 |
+
}
|
1600 |
+
|
1601 |
+
// split rests
|
1602 |
+
if (measureHeads) {
|
1603 |
+
this.body = [].concat(...this.body.map(term => {
|
1604 |
+
if (!(term instanceof Rest) || term.name !== "s" || !Number.isInteger(term._measure))
|
1605 |
+
return [term];
|
1606 |
+
|
1607 |
+
const nextHead = measureHeads[term._measure];
|
1608 |
+
const endTick = term._tick + term.durationMagnitude;
|
1609 |
+
if (nextHead > 0 && endTick > nextHead) {
|
1610 |
+
const post_events = term.post_events;
|
1611 |
+
|
1612 |
+
let startTick = term._tick;
|
1613 |
+
const rests = [];
|
1614 |
+
let nextMeasure;
|
1615 |
+
for (nextMeasure = term._measure; nextMeasure < measureHeads.length && endTick > measureHeads[nextMeasure]; ++nextMeasure) {
|
1616 |
+
const duration = Duration.fromMagnitude(measureHeads[nextMeasure] - startTick);
|
1617 |
+
if (!duration) {
|
1618 |
+
console.warn("invalid middle rest duration, splitting gave up:", measureHeads[nextMeasure] - startTick, term);
|
1619 |
+
return [term];
|
1620 |
+
}
|
1621 |
+
|
1622 |
+
const rest = new Rest({name: "s", duration, post_events: []});
|
1623 |
+
rest._measure = nextMeasure;
|
1624 |
+
rest._lastMeasure = nextMeasure;
|
1625 |
+
rests.push(rest);
|
1626 |
+
|
1627 |
+
console.assert(!!rest.duration, "middle splitted rest duration invalid:", measureHeads[nextMeasure] - startTick);
|
1628 |
+
|
1629 |
+
startTick = measureHeads[nextMeasure];
|
1630 |
+
}
|
1631 |
+
|
1632 |
+
const duration = Duration.fromMagnitude(endTick - startTick);
|
1633 |
+
if (!duration) {
|
1634 |
+
console.warn("invalid tail rest duration, splitting gave up:", endTick - startTick, term);
|
1635 |
+
return [term];
|
1636 |
+
}
|
1637 |
+
|
1638 |
+
const rest = new Rest({name: "s", duration, post_events: post_events && [...post_events]});
|
1639 |
+
rest._measure = nextMeasure;
|
1640 |
+
rest._lastMeasure = nextMeasure;
|
1641 |
+
rests.push(rest);
|
1642 |
+
|
1643 |
+
console.assert(rests.reduce((sum, rest) => sum + rest.durationMagnitude, 0) === term.durationMagnitude, "duration splitting error:", rests, term);
|
1644 |
+
//if (rests.reduce((sum, rest) => sum + rest.durationMagnitude, 0) !== term.durationMagnitude)
|
1645 |
+
// debugger;
|
1646 |
+
|
1647 |
+
return rests;
|
1648 |
+
}
|
1649 |
+
|
1650 |
+
return [term];
|
1651 |
+
}));
|
1652 |
+
}
|
1653 |
+
|
1654 |
+
const isPostTerm = term => !term
|
1655 |
+
|| term instanceof PostEvent
|
1656 |
+
|| (term as Primitive).exp === "~"
|
1657 |
+
|| ["bar", "arpeggio", "glissando", "sustainOff", "sustainOn"].includes((term as Command).cmd)
|
1658 |
+
;
|
1659 |
+
|
1660 |
+
const list = this.body.filter(term => !(term instanceof Divide));
|
1661 |
+
let measure = null;
|
1662 |
+
for (const term of list) {
|
1663 |
+
if (Number.isInteger(measure) && isPostTerm(term))
|
1664 |
+
term._measure = measure;
|
1665 |
+
else
|
1666 |
+
measure = term._measure;
|
1667 |
+
}
|
1668 |
+
|
1669 |
+
const body: BaseTerm[] = [];
|
1670 |
+
const measures = new Set();
|
1671 |
+
|
1672 |
+
list.reverse().forEach(term => {
|
1673 |
+
if (term instanceof BaseTerm) {
|
1674 |
+
const newMeasures = term.measures.filter(m => !measures.has(m));
|
1675 |
+
if (newMeasures.length) {
|
1676 |
+
const comment = " " + newMeasures[0] + (newMeasures.length > 1 ? "-" + Math.max(...newMeasures) : "");
|
1677 |
+
|
1678 |
+
if (body.length)
|
1679 |
+
body.push(new Divide({_tailComment: Comment.createSingle(comment)}));
|
1680 |
+
|
1681 |
+
newMeasures.forEach(m => measures.add(m));
|
1682 |
+
}
|
1683 |
+
}
|
1684 |
+
|
1685 |
+
body.push(term);
|
1686 |
+
});
|
1687 |
+
|
1688 |
+
this.body = body.reverse();
|
1689 |
+
}
|
1690 |
+
|
1691 |
+
|
1692 |
+
clarifyHead () {
|
1693 |
+
const terms = this.body;
|
1694 |
+
|
1695 |
+
const head = terms.find(term => term.isMusic);
|
1696 |
+
if (head instanceof MusicEvent) {
|
1697 |
+
// clarify the first music event content
|
1698 |
+
const firstEventIndex = terms.indexOf(head);
|
1699 |
+
if (firstEventIndex >= 0) {
|
1700 |
+
const firstEvent = terms[firstEventIndex] as MusicEvent;
|
1701 |
+
//console.log("firstEvent:", firstEvent);
|
1702 |
+
if (firstEvent._previous) {
|
1703 |
+
const clarified = firstEvent.clarified;
|
1704 |
+
|
1705 |
+
terms.splice(firstEventIndex, 1, clarified);
|
1706 |
+
//console.log("terms:", firstEventIndex, terms, clarified);
|
1707 |
+
}
|
1708 |
+
}
|
1709 |
+
}
|
1710 |
+
else if (head) {
|
1711 |
+
const block = head.findFirst(MusicBlock) as MusicBlock;
|
1712 |
+
if (block)
|
1713 |
+
block.clarifyHead();
|
1714 |
+
else
|
1715 |
+
console.warn("[MusicBlock.clarifyHead] unexpected music head:", head);
|
1716 |
+
}
|
1717 |
+
}
|
1718 |
+
|
1719 |
+
|
1720 |
+
absoluteToRelative (): Relative {
|
1721 |
+
const anchor = this.findFirst(Chord) as Chord;
|
1722 |
+
if (!anchor)
|
1723 |
+
return null;
|
1724 |
+
|
1725 |
+
const anchorPitch = anchor.absolutePitch;
|
1726 |
+
let pitch = anchorPitch;
|
1727 |
+
|
1728 |
+
const newBody = this.clone();
|
1729 |
+
newBody.forEachTerm(Chord, chord => {
|
1730 |
+
const newPitch = chord.absolutePitch;
|
1731 |
+
chord.makeRelativeTo(pitch);
|
1732 |
+
|
1733 |
+
pitch = newPitch;
|
1734 |
+
});
|
1735 |
+
|
1736 |
+
return Relative.makeBlock(newBody, {anchor: anchorPitch});
|
1737 |
+
}
|
1738 |
+
};
|
1739 |
+
|
1740 |
+
|
1741 |
+
export class SimultaneousList extends BaseTerm {
|
1742 |
+
list: BaseTerm[];
|
1743 |
+
|
1744 |
+
|
1745 |
+
serialize () {
|
1746 |
+
return [
|
1747 |
+
"<<\n",
|
1748 |
+
...cc(this.list.map(item => [...BaseTerm.optionalSerialize(item), "\n"])),
|
1749 |
+
">>\n",
|
1750 |
+
];
|
1751 |
+
}
|
1752 |
+
|
1753 |
+
|
1754 |
+
removeStaffGroup () {
|
1755 |
+
for (let i = 0; i < this.list.length; ++i) {
|
1756 |
+
const item: any = this.list[i];
|
1757 |
+
if (item.head instanceof Command && item.head.args && item.head.args[0] === "StaffGroup")
|
1758 |
+
this.list[i] = item.body;
|
1759 |
+
}
|
1760 |
+
|
1761 |
+
this.list.forEach(item => {
|
1762 |
+
if (item instanceof SimultaneousList)
|
1763 |
+
item.removeStaffGroup();
|
1764 |
+
});
|
1765 |
+
}
|
1766 |
+
|
1767 |
+
|
1768 |
+
get isMusic (): boolean {
|
1769 |
+
return true;
|
1770 |
+
}
|
1771 |
+
|
1772 |
+
|
1773 |
+
get entries () {
|
1774 |
+
return this.list;
|
1775 |
+
}
|
1776 |
+
|
1777 |
+
|
1778 |
+
get durationMagnitude (): number {
|
1779 |
+
return Math.max(...this.list.filter(term => term instanceof BaseTerm).map(term => term.durationMagnitude));
|
1780 |
+
}
|
1781 |
+
|
1782 |
+
|
1783 |
+
get measureLayout (): measureLayout.MeasureLayout {
|
1784 |
+
const track = this.list.find(term => term instanceof BaseTerm && term.measureLayout);
|
1785 |
+
return track && track.measureLayout;
|
1786 |
+
}
|
1787 |
+
};
|
1788 |
+
|
1789 |
+
|
1790 |
+
export class ContextedMusic extends BaseTerm {
|
1791 |
+
head: Command;
|
1792 |
+
body: BaseTerm;
|
1793 |
+
lyrics?: BaseTerm;
|
1794 |
+
|
1795 |
+
|
1796 |
+
serialize () {
|
1797 |
+
return [
|
1798 |
+
...BaseTerm.optionalSerialize(this.head),
|
1799 |
+
...BaseTerm.optionalSerialize(this.body),
|
1800 |
+
...BaseTerm.optionalSerialize(this.lyrics),
|
1801 |
+
];
|
1802 |
+
}
|
1803 |
+
|
1804 |
+
|
1805 |
+
get isMusic (): boolean {
|
1806 |
+
return true;
|
1807 |
+
}
|
1808 |
+
|
1809 |
+
|
1810 |
+
get entries () {
|
1811 |
+
return [this.head, this.body];
|
1812 |
+
}
|
1813 |
+
|
1814 |
+
|
1815 |
+
get type (): string {
|
1816 |
+
return this.head.args[0];
|
1817 |
+
}
|
1818 |
+
|
1819 |
+
|
1820 |
+
get durationMagnitude (): number {
|
1821 |
+
return this.body.durationMagnitude;
|
1822 |
+
}
|
1823 |
+
|
1824 |
+
|
1825 |
+
get withClause (): Command {
|
1826 |
+
if (this.head.args[2] && this.head.args[2] instanceof Command && this.head.args[2].cmd === "with")
|
1827 |
+
return this.head.args[2];
|
1828 |
+
}
|
1829 |
+
|
1830 |
+
|
1831 |
+
get contextDict (): {[key: string]: string} {
|
1832 |
+
const withEntries = this.withClause ? Object.entries((this.withClause.args[0] as Block).assignmentDict) : [];
|
1833 |
+
const entries = withEntries.map(([key, value]) => [`${this.type}.${key}`, value]);
|
1834 |
+
|
1835 |
+
const pair = this.head.getAssignmentPair();
|
1836 |
+
if (pair)
|
1837 |
+
entries.push([pair.key.toString(), pair.value.toString()]);
|
1838 |
+
|
1839 |
+
return entries.reduce((dict, [key, value]) => ((dict[key] = value), dict), {});
|
1840 |
+
}
|
1841 |
+
|
1842 |
+
|
1843 |
+
get list (): BaseTerm[] {
|
1844 |
+
if (this.body instanceof SimultaneousList)
|
1845 |
+
return this.body.list;
|
1846 |
+
|
1847 |
+
return null;
|
1848 |
+
}
|
1849 |
+
|
1850 |
+
|
1851 |
+
set list (value: BaseTerm[]) {
|
1852 |
+
if (this.body instanceof SimultaneousList)
|
1853 |
+
this.body.list = value;
|
1854 |
+
}
|
1855 |
+
|
1856 |
+
|
1857 |
+
get measureLayout (): measureLayout.MeasureLayout {
|
1858 |
+
return this.body.measureLayout;
|
1859 |
+
}
|
1860 |
+
};
|
1861 |
+
|
1862 |
+
|
1863 |
+
export class Divide extends BaseTerm {
|
1864 |
+
serialize () {
|
1865 |
+
return ["|", "\n"];
|
1866 |
+
}
|
1867 |
+
}
|
1868 |
+
|
1869 |
+
|
1870 |
+
export class Scheme extends BaseTerm {
|
1871 |
+
exp: (boolean|BaseTerm);
|
1872 |
+
|
1873 |
+
|
1874 |
+
serialize () {
|
1875 |
+
if (BaseTerm.isTerm(this.exp))
|
1876 |
+
return ["#", "\b", ...(this.exp as BaseTerm).serialize()];
|
1877 |
+
else if (typeof this.exp === "boolean")
|
1878 |
+
return ["#", "\b", this.exp ? "#t" : "#f"];
|
1879 |
+
// TODO: enhance grammar to parse empty scheme list
|
1880 |
+
//else if (this.exp === null)
|
1881 |
+
// return ["#", "\b", "'()"];
|
1882 |
+
else
|
1883 |
+
return ["#", "\b", this.exp];
|
1884 |
+
}
|
1885 |
+
|
1886 |
+
|
1887 |
+
query (key: string): any {
|
1888 |
+
if (this.exp instanceof SchemeFunction)
|
1889 |
+
return this.exp.query(key);
|
1890 |
+
}
|
1891 |
+
|
1892 |
+
|
1893 |
+
get entries () {
|
1894 |
+
if (this.exp instanceof BaseTerm)
|
1895 |
+
return [this.exp];
|
1896 |
+
|
1897 |
+
return [];
|
1898 |
+
}
|
1899 |
+
};
|
1900 |
+
|
1901 |
+
|
1902 |
+
export class SchemeFunction extends BaseTerm {
|
1903 |
+
func: (string | BaseTerm);
|
1904 |
+
args: (boolean | string | BaseTerm)[];
|
1905 |
+
|
1906 |
+
|
1907 |
+
serialize () {
|
1908 |
+
return [
|
1909 |
+
"(", "\b",
|
1910 |
+
...BaseTerm.optionalSerialize(this.func),
|
1911 |
+
...cc(this.args.map(BaseTerm.serializeScheme)),
|
1912 |
+
"\b", ")",
|
1913 |
+
];
|
1914 |
+
}
|
1915 |
+
|
1916 |
+
|
1917 |
+
query (key: string): any {
|
1918 |
+
if (key === this.func) {
|
1919 |
+
const term = this;
|
1920 |
+
|
1921 |
+
return {
|
1922 |
+
get value () {
|
1923 |
+
return term.args.length === 1 ? term.args[0] : term.args;
|
1924 |
+
},
|
1925 |
+
|
1926 |
+
set value (value) {
|
1927 |
+
if (term.args.length === 1)
|
1928 |
+
term.args[0] = value as string|BaseTerm;
|
1929 |
+
else
|
1930 |
+
term.args = value as (string|BaseTerm)[];
|
1931 |
+
},
|
1932 |
+
};
|
1933 |
+
}
|
1934 |
+
}
|
1935 |
+
|
1936 |
+
|
1937 |
+
get asList (): (boolean | string | BaseTerm)[] {
|
1938 |
+
return [this.func, ...this.args];
|
1939 |
+
}
|
1940 |
+
|
1941 |
+
|
1942 |
+
get entries () {
|
1943 |
+
return this.asList.filter(term => term instanceof BaseTerm) as BaseTerm[];
|
1944 |
+
}
|
1945 |
+
};
|
1946 |
+
|
1947 |
+
|
1948 |
+
export class SchemePair extends BaseTerm {
|
1949 |
+
left: any;
|
1950 |
+
right: any;
|
1951 |
+
|
1952 |
+
|
1953 |
+
serialize () {
|
1954 |
+
return [
|
1955 |
+
"(", "\b",
|
1956 |
+
...BaseTerm.optionalSerialize(this.left), ".", ...BaseTerm.optionalSerialize(this.right),
|
1957 |
+
"\b", ")",
|
1958 |
+
];
|
1959 |
+
}
|
1960 |
+
};
|
1961 |
+
|
1962 |
+
|
1963 |
+
export class SchemePointer extends BaseTerm {
|
1964 |
+
value: any;
|
1965 |
+
|
1966 |
+
|
1967 |
+
serialize () {
|
1968 |
+
const content = this.value === null ? ["()"] : BaseTerm.optionalSerialize(this.value);
|
1969 |
+
|
1970 |
+
return [
|
1971 |
+
"'", "\b", ...content,
|
1972 |
+
];
|
1973 |
+
}
|
1974 |
+
|
1975 |
+
|
1976 |
+
get entries () {
|
1977 |
+
if (this.value instanceof BaseTerm)
|
1978 |
+
return [this.value];
|
1979 |
+
|
1980 |
+
return [];
|
1981 |
+
}
|
1982 |
+
};
|
1983 |
+
|
1984 |
+
|
1985 |
+
export class SchemeEmbed extends BaseTerm {
|
1986 |
+
value: Root;
|
1987 |
+
|
1988 |
+
|
1989 |
+
serialize () {
|
1990 |
+
return [
|
1991 |
+
"#{",
|
1992 |
+
...BaseTerm.optionalSerialize(this.value),
|
1993 |
+
"#}",
|
1994 |
+
];
|
1995 |
+
}
|
1996 |
+
};
|
1997 |
+
|
1998 |
+
|
1999 |
+
export class Assignment extends BaseTerm {
|
2000 |
+
key: (string|any[]);
|
2001 |
+
value: any;
|
2002 |
+
|
2003 |
+
|
2004 |
+
constructor (data) {
|
2005 |
+
super(data);
|
2006 |
+
|
2007 |
+
if (this.value instanceof BaseTerm)
|
2008 |
+
this.value._parent = this;
|
2009 |
+
}
|
2010 |
+
|
2011 |
+
|
2012 |
+
serialize () {
|
2013 |
+
const keys = (Array.isArray(this.key) ? this.key : [this.key]).map(BaseTerm.optionalSerialize);
|
2014 |
+
const values = (Array.isArray(this.value) ? this.value : [this.value]).map(BaseTerm.optionalSerialize);
|
2015 |
+
|
2016 |
+
return [
|
2017 |
+
...cc(keys),
|
2018 |
+
"=",
|
2019 |
+
...cc(values),
|
2020 |
+
];
|
2021 |
+
}
|
2022 |
+
|
2023 |
+
|
2024 |
+
get entries () {
|
2025 |
+
if (this.value instanceof BaseTerm)
|
2026 |
+
return [this.value];
|
2027 |
+
|
2028 |
+
return null;
|
2029 |
+
}
|
2030 |
+
|
2031 |
+
|
2032 |
+
query (key) {
|
2033 |
+
if (this.key === key) {
|
2034 |
+
const term = this;
|
2035 |
+
|
2036 |
+
return {
|
2037 |
+
get value () {
|
2038 |
+
return term.value;
|
2039 |
+
},
|
2040 |
+
|
2041 |
+
set value (value) {
|
2042 |
+
term.value = value;
|
2043 |
+
},
|
2044 |
+
};
|
2045 |
+
}
|
2046 |
+
}
|
2047 |
+
};
|
2048 |
+
|
2049 |
+
|
2050 |
+
export class MusicEvent extends BaseTerm {
|
2051 |
+
duration?: Duration;
|
2052 |
+
post_events?: (string | PostEvent)[];
|
2053 |
+
|
2054 |
+
declare _previous?: MusicEvent;
|
2055 |
+
//_anchorPitch?: ChordElement;
|
2056 |
+
_lastMeasure?: number;
|
2057 |
+
|
2058 |
+
|
2059 |
+
constructor (data: object) {
|
2060 |
+
super(data);
|
2061 |
+
|
2062 |
+
if (this.post_events)
|
2063 |
+
this.post_events = this.post_events.map(parseRaw);
|
2064 |
+
}
|
2065 |
+
|
2066 |
+
|
2067 |
+
getPreviousT (T) {
|
2068 |
+
if (this._previous instanceof T)
|
2069 |
+
return this._previous;
|
2070 |
+
|
2071 |
+
if (this._previous)
|
2072 |
+
return this._previous.getPreviousT(T);
|
2073 |
+
}
|
2074 |
+
|
2075 |
+
|
2076 |
+
get durationValue (): Duration {
|
2077 |
+
return this.duration || (this._previous ? this._previous.durationValue : Duration.default);
|
2078 |
+
}
|
2079 |
+
|
2080 |
+
|
2081 |
+
get durationMagnitude (): number {
|
2082 |
+
return this.durationValue.magnitude;
|
2083 |
+
}
|
2084 |
+
|
2085 |
+
|
2086 |
+
get division (): number {
|
2087 |
+
return this.durationValue.division;
|
2088 |
+
}
|
2089 |
+
|
2090 |
+
|
2091 |
+
get withMultiplier () {
|
2092 |
+
return this.duration && this.duration.withMultiplier;
|
2093 |
+
}
|
2094 |
+
|
2095 |
+
|
2096 |
+
get isMusic (): boolean {
|
2097 |
+
return true;
|
2098 |
+
}
|
2099 |
+
|
2100 |
+
|
2101 |
+
get isTying (): boolean {
|
2102 |
+
return this.post_events && this.post_events.some(event => event instanceof PostEvent && event.isTying);
|
2103 |
+
}
|
2104 |
+
|
2105 |
+
|
2106 |
+
get isStaccato (): boolean {
|
2107 |
+
return this.post_events && this.post_events.some(event => event instanceof PostEvent && event.isStaccato);
|
2108 |
+
}
|
2109 |
+
|
2110 |
+
|
2111 |
+
// to be implement in derived classes
|
2112 |
+
get isRest (): boolean {
|
2113 |
+
return null;
|
2114 |
+
}
|
2115 |
+
|
2116 |
+
|
2117 |
+
get beamOn (): boolean {
|
2118 |
+
return this.post_events && this.post_events.includes("[");
|
2119 |
+
}
|
2120 |
+
|
2121 |
+
|
2122 |
+
get beamOff (): boolean {
|
2123 |
+
return this.post_events && this.post_events.includes("]");
|
2124 |
+
}
|
2125 |
+
|
2126 |
+
|
2127 |
+
get measures (): number[] {
|
2128 |
+
if (!Number.isFinite(this._measure) || !Number.isFinite(this._lastMeasure))
|
2129 |
+
return [];
|
2130 |
+
|
2131 |
+
return Array(this._lastMeasure + 1 - this._measure).fill(null).map((_, i) => this._measure + i);
|
2132 |
+
}
|
2133 |
+
|
2134 |
+
|
2135 |
+
get measureLayout (): measureLayout.MeasureLayout {
|
2136 |
+
if (this.measures.length > 1)
|
2137 |
+
return measureLayout.BlockMLayout.fromSeq(this.measures.map(measure => measureLayout.SingleMLayout.from(measure)));
|
2138 |
+
|
2139 |
+
if (this.measures.length === 1)
|
2140 |
+
return measureLayout.SingleMLayout.from(this._measure);
|
2141 |
+
|
2142 |
+
return null;
|
2143 |
+
}
|
2144 |
+
|
2145 |
+
|
2146 |
+
get implicitType (): ImplicitType {
|
2147 |
+
if (this.post_events) {
|
2148 |
+
for (const event of this.post_events) {
|
2149 |
+
if (event instanceof PostEvent && event.arg instanceof Command) {
|
2150 |
+
switch (event.arg.cmd) {
|
2151 |
+
case "startTrillSpan":
|
2152 |
+
case "trill":
|
2153 |
+
return ImplicitType.Trill;
|
2154 |
+
|
2155 |
+
case "turn":
|
2156 |
+
return ImplicitType.Turn;
|
2157 |
+
|
2158 |
+
case "mordent":
|
2159 |
+
return ImplicitType.Mordent;
|
2160 |
+
|
2161 |
+
case "prall":
|
2162 |
+
return ImplicitType.Prall;
|
2163 |
+
|
2164 |
+
// Arpeggio is not implemented in 'articulate.ly' yet
|
2165 |
+
case "arpeggio":
|
2166 |
+
return ImplicitType.Arpeggio;
|
2167 |
+
}
|
2168 |
+
}
|
2169 |
+
}
|
2170 |
+
}
|
2171 |
+
|
2172 |
+
return ImplicitType.None;
|
2173 |
+
}
|
2174 |
+
|
2175 |
+
|
2176 |
+
get clarified (): MusicEvent {
|
2177 |
+
const clarified = this instanceof Chord ? this.clarifiedChord : this.clone();
|
2178 |
+
clarified.duration = this.durationValue && this.durationValue.clone();
|
2179 |
+
|
2180 |
+
return clarified;
|
2181 |
+
}
|
2182 |
+
};
|
2183 |
+
|
2184 |
+
|
2185 |
+
export class Chord extends MusicEvent {
|
2186 |
+
pitches: (ChordElement | Command)[];
|
2187 |
+
options: {
|
2188 |
+
exclamations?: string[],
|
2189 |
+
questions?: string[],
|
2190 |
+
rest?: string,
|
2191 |
+
withAngle?: boolean,
|
2192 |
+
};
|
2193 |
+
|
2194 |
+
|
2195 |
+
constructor (data) {
|
2196 |
+
super(data);
|
2197 |
+
|
2198 |
+
this.connectPitches();
|
2199 |
+
}
|
2200 |
+
|
2201 |
+
|
2202 |
+
connectPitches () {
|
2203 |
+
if (this.basePitch)
|
2204 |
+
this.basePitch._parent = this;
|
2205 |
+
for (let i = 1; i < this.pitchElements.length; ++i)
|
2206 |
+
this.pitchElements[i]._previous = this.pitchElements[i - 1];
|
2207 |
+
}
|
2208 |
+
|
2209 |
+
|
2210 |
+
get single (): boolean {
|
2211 |
+
return this.pitches.length === 1;
|
2212 |
+
}
|
2213 |
+
|
2214 |
+
|
2215 |
+
get entries () {
|
2216 |
+
const list: any[] = [...this.pitches];
|
2217 |
+
if (Array.isArray(this.post_events))
|
2218 |
+
list.push(...this.post_events);
|
2219 |
+
|
2220 |
+
return list;
|
2221 |
+
}
|
2222 |
+
|
2223 |
+
|
2224 |
+
serialize () {
|
2225 |
+
const innerPitches = this.pitches.map(BaseTerm.optionalSerialize);
|
2226 |
+
const pitches = (this.single && !this.options.withAngle) ? cc(innerPitches) : [
|
2227 |
+
"<", "\b", ...cc(innerPitches), "\b", ">",
|
2228 |
+
];
|
2229 |
+
|
2230 |
+
const {exclamations, questions, rest} = this.options;
|
2231 |
+
const postfix = cc([...(exclamations || []), ...(questions || []), ...BaseTerm.optionalSerialize(this.duration), rest]
|
2232 |
+
.filter(item => item)
|
2233 |
+
.map(item => ["\b", item]),
|
2234 |
+
).concat(...(this.post_events || []).map(BaseTerm.optionalSerialize));
|
2235 |
+
|
2236 |
+
return [
|
2237 |
+
new OpenLocator(this),
|
2238 |
+
...pitches,
|
2239 |
+
...postfix,
|
2240 |
+
new CloseLocator(this),
|
2241 |
+
];
|
2242 |
+
}
|
2243 |
+
|
2244 |
+
|
2245 |
+
get pitchElements (): ChordElement[] {
|
2246 |
+
return this.pitches.filter(pitch => pitch instanceof ChordElement) as ChordElement[];
|
2247 |
+
}
|
2248 |
+
|
2249 |
+
|
2250 |
+
get pitchNames (): string[] {
|
2251 |
+
return this.pitchElements.map((elem: ChordElement) => elem.pitch.replace(/'|,/g, ""));
|
2252 |
+
}
|
2253 |
+
|
2254 |
+
|
2255 |
+
get basePitch (): ChordElement {
|
2256 |
+
return this.pitchElements[0];
|
2257 |
+
}
|
2258 |
+
|
2259 |
+
|
2260 |
+
get absolutePitch (): ChordElement {
|
2261 |
+
console.assert(!!this.basePitch, "absolutePitch on non pitch:", this.join());
|
2262 |
+
return this.basePitch.absolutePitch;
|
2263 |
+
}
|
2264 |
+
|
2265 |
+
|
2266 |
+
get anchorPitch (): ChordElement {
|
2267 |
+
if (this._anchorPitch)
|
2268 |
+
return this._anchorPitch;
|
2269 |
+
|
2270 |
+
const previous = this.getPreviousT(Chord);
|
2271 |
+
if (previous)
|
2272 |
+
return previous.absolutePitch;
|
2273 |
+
|
2274 |
+
return this.basePitch;
|
2275 |
+
}
|
2276 |
+
|
2277 |
+
|
2278 |
+
get isRest (): boolean {
|
2279 |
+
return !!this.options.rest;
|
2280 |
+
}
|
2281 |
+
|
2282 |
+
|
2283 |
+
get completeTied (): boolean {
|
2284 |
+
return this.pitchElements.filter(pitch => !pitch._tied).length === 0;
|
2285 |
+
}
|
2286 |
+
|
2287 |
+
|
2288 |
+
get pitchesValue (): (ChordElement | Command)[] {
|
2289 |
+
if (this._previous instanceof Chord && this.basePitch.pitch === "q") {
|
2290 |
+
const pitches = this._previous.pitchesValue.map(pitch => {
|
2291 |
+
const newPitch = pitch.clone();
|
2292 |
+
if (newPitch instanceof ChordElement) {
|
2293 |
+
newPitch._location = this.basePitch._location;
|
2294 |
+
newPitch._tied = this.basePitch._tied;
|
2295 |
+
newPitch._parent = (pitch as ChordElement)._parent && this;
|
2296 |
+
newPitch._previous = (pitch as ChordElement)._previous;
|
2297 |
+
}
|
2298 |
+
|
2299 |
+
return newPitch;
|
2300 |
+
});
|
2301 |
+
const base = pitches.find(pitch => pitch instanceof ChordElement) as ChordElement;
|
2302 |
+
if (base)
|
2303 |
+
base.pitch = base.pitch.replace(/[,']/g, "");
|
2304 |
+
|
2305 |
+
return pitches;
|
2306 |
+
}
|
2307 |
+
|
2308 |
+
return this.pitches;
|
2309 |
+
}
|
2310 |
+
|
2311 |
+
|
2312 |
+
get clarifiedChord (): MusicEvent {
|
2313 |
+
const clarified = this.clone();
|
2314 |
+
clarified.pitches = this.pitchesValue.filter(pitch => !(pitch as ChordElement)._tied).map(pitch => pitch.clone());
|
2315 |
+
clarified.connectPitches();
|
2316 |
+
|
2317 |
+
// replace by rest if all pitches tied
|
2318 |
+
if (!clarified.pitches.length)
|
2319 |
+
return new Rest({name: "r", duration: this.duration});
|
2320 |
+
|
2321 |
+
return clarified;
|
2322 |
+
}
|
2323 |
+
|
2324 |
+
|
2325 |
+
shiftAnchor (newAnchor: ChordElement) {
|
2326 |
+
//console.warn("shiftAnchor:", this.join(), newAnchor.join(), this.absolutePitch.pitchValue, newAnchor.pitchValue, this.anchorPitch.pitchValue);
|
2327 |
+
const _location = this.basePitch._location;
|
2328 |
+
const shift = idioms.phonetDifferToShift(this.basePitch.phonetStep - newAnchor.phonetStep);
|
2329 |
+
const relativeOctave = this.basePitch.absoluteOctave(this.anchorPitch) - newAnchor.octave - shift;
|
2330 |
+
//console.log("_location:", _location);
|
2331 |
+
|
2332 |
+
this.pitches[0] = ChordElement.from({
|
2333 |
+
phonet: this.basePitch.phonet,
|
2334 |
+
alters: this.basePitch.alters,
|
2335 |
+
octave: relativeOctave,
|
2336 |
+
});
|
2337 |
+
|
2338 |
+
this.pitches[0]._location = _location;
|
2339 |
+
this.pitches[0]._parent = this;
|
2340 |
+
|
2341 |
+
this.connectPitches();
|
2342 |
+
|
2343 |
+
//console.log("shiftAnchor.1:", this.join(), this.absolutePitch.pitchValue, {relativeOctave, shift, "newAnchor.octave": newAnchor.octave});
|
2344 |
+
}
|
2345 |
+
|
2346 |
+
|
2347 |
+
makeRelativeTo (from: ChordElement) {
|
2348 |
+
const _location = this.basePitch._location;
|
2349 |
+
|
2350 |
+
const octave = this.basePitch.relativeOctave(from);
|
2351 |
+
this.pitches[0] = ChordElement.from({
|
2352 |
+
phonet: this.basePitch.phonet,
|
2353 |
+
alters: this.basePitch.alters,
|
2354 |
+
octave,
|
2355 |
+
});
|
2356 |
+
|
2357 |
+
this.pitches[0]._location = _location;
|
2358 |
+
this.pitches[0]._parent = this;
|
2359 |
+
}
|
2360 |
+
};
|
2361 |
+
|
2362 |
+
|
2363 |
+
export class Rest extends MusicEvent {
|
2364 |
+
name: string;
|
2365 |
+
|
2366 |
+
|
2367 |
+
serialize () {
|
2368 |
+
return [
|
2369 |
+
new OpenLocator(this),
|
2370 |
+
...compact([
|
2371 |
+
this.name,
|
2372 |
+
...BaseTerm.optionalSerialize(this.duration),
|
2373 |
+
]),
|
2374 |
+
...cc((this.post_events || []).map(BaseTerm.optionalSerialize)),
|
2375 |
+
new CloseLocator(this),
|
2376 |
+
];
|
2377 |
+
}
|
2378 |
+
|
2379 |
+
|
2380 |
+
get isSpacer () {
|
2381 |
+
return this.name === "s";
|
2382 |
+
}
|
2383 |
+
|
2384 |
+
|
2385 |
+
get isRest (): boolean {
|
2386 |
+
return true;
|
2387 |
+
}
|
2388 |
+
};
|
2389 |
+
|
2390 |
+
|
2391 |
+
export class ChordElement extends BaseTerm {
|
2392 |
+
pitch: string;
|
2393 |
+
options: {
|
2394 |
+
exclamations?: string[],
|
2395 |
+
questions?: string[],
|
2396 |
+
post_events?: PostEvent[],
|
2397 |
+
};
|
2398 |
+
|
2399 |
+
declare _parent?: Chord;
|
2400 |
+
declare _previous?: ChordElement;
|
2401 |
+
_tied?: MusicEvent;
|
2402 |
+
_transposition?: number;
|
2403 |
+
|
2404 |
+
// cache for property of absolutePitch
|
2405 |
+
_absolutePitch?: ChordElement;
|
2406 |
+
|
2407 |
+
|
2408 |
+
static from ({phonet, alters = "", octave, options = {}}): ChordElement {
|
2409 |
+
const octaveString = octave ? Array(Math.abs(octave)).fill(octave > 0 ? "'" : ",").join("") : "";
|
2410 |
+
const pitch = phonet + (alters || "") + octaveString;
|
2411 |
+
|
2412 |
+
return new ChordElement({pitch, options: {...options, proto: "_PLAIN"}});
|
2413 |
+
}
|
2414 |
+
|
2415 |
+
|
2416 |
+
static get default (): ChordElement {
|
2417 |
+
return ChordElement.from({phonet: "c", octave: 0});
|
2418 |
+
}
|
2419 |
+
|
2420 |
+
|
2421 |
+
constructor (data: object) {
|
2422 |
+
super(data);
|
2423 |
+
|
2424 |
+
if (this.options.post_events)
|
2425 |
+
this.options.post_events = this.options.post_events.map(parseRaw);
|
2426 |
+
|
2427 |
+
if (!this.pitch)
|
2428 |
+
console.log("null pitch:", this);
|
2429 |
+
}
|
2430 |
+
|
2431 |
+
|
2432 |
+
serialize () {
|
2433 |
+
const {exclamations, questions, post_events} = this.options;
|
2434 |
+
const postfix = [].concat(...[...(exclamations || []), ...(questions || [])]
|
2435 |
+
.filter(item => item)
|
2436 |
+
.map(item => ["\b", item]),
|
2437 |
+
).concat(...(post_events || []).map(item => ["\b", ...BaseTerm.optionalSerialize(item)]));
|
2438 |
+
|
2439 |
+
return [
|
2440 |
+
new OpenLocator(this),
|
2441 |
+
this.pitch,
|
2442 |
+
...postfix,
|
2443 |
+
new CloseLocator(this),
|
2444 |
+
];
|
2445 |
+
}
|
2446 |
+
|
2447 |
+
|
2448 |
+
get octave (): number {
|
2449 |
+
const positive = (this.pitch.match(/'/g) || []).length;
|
2450 |
+
const negative = (this.pitch.match(/,/g) || []).length;
|
2451 |
+
|
2452 |
+
return positive - negative;
|
2453 |
+
}
|
2454 |
+
|
2455 |
+
|
2456 |
+
get phonet (): string {
|
2457 |
+
const ph = this.pitch.substr(0, 1);
|
2458 |
+
|
2459 |
+
return idioms.PHONETS_ALIAS[ph] || ph;
|
2460 |
+
}
|
2461 |
+
|
2462 |
+
|
2463 |
+
get phonetStep (): number {
|
2464 |
+
return idioms.PHONETS.indexOf(this.phonet);
|
2465 |
+
}
|
2466 |
+
|
2467 |
+
|
2468 |
+
get alters (): string {
|
2469 |
+
const captures = this.pitch.substr(1).match(/^\w+/);
|
2470 |
+
return captures && captures[0];
|
2471 |
+
}
|
2472 |
+
|
2473 |
+
|
2474 |
+
get alteredPhonet (): string {
|
2475 |
+
const captures = this.pitch.match(/^\w+/);
|
2476 |
+
return captures && captures[0];
|
2477 |
+
}
|
2478 |
+
|
2479 |
+
|
2480 |
+
get anchorPitch (): ChordElement {
|
2481 |
+
if (this._previous)
|
2482 |
+
return this._previous.absolutePitch;
|
2483 |
+
|
2484 |
+
if (this._parent)
|
2485 |
+
return this._parent.anchorPitch;
|
2486 |
+
|
2487 |
+
return ChordElement.from({phonet: this.phonet, octave: 0});
|
2488 |
+
}
|
2489 |
+
|
2490 |
+
|
2491 |
+
getAbsolutePitch (): ChordElement {
|
2492 |
+
if (this.phonet === "q")
|
2493 |
+
return this.anchorPitch;
|
2494 |
+
|
2495 |
+
if (this.anchorPitch === this)
|
2496 |
+
return this;
|
2497 |
+
|
2498 |
+
const octave = this.absoluteOctave(this.anchorPitch);
|
2499 |
+
|
2500 |
+
return ChordElement.from({phonet: this.phonet, alters: this.alters, octave});
|
2501 |
+
}
|
2502 |
+
|
2503 |
+
|
2504 |
+
get absolutePitch (): ChordElement {
|
2505 |
+
if (!this._absolutePitch)
|
2506 |
+
this._absolutePitch = this.getAbsolutePitch();
|
2507 |
+
|
2508 |
+
return this._absolutePitch;
|
2509 |
+
}
|
2510 |
+
|
2511 |
+
|
2512 |
+
absoluteOctave (anchor: ChordElement): number {
|
2513 |
+
if (this.phonet === "q")
|
2514 |
+
return anchor.octave;
|
2515 |
+
|
2516 |
+
const phonetDiffer = this.phonetStep - anchor.phonetStep;
|
2517 |
+
const shift = idioms.phonetDifferToShift(phonetDiffer);
|
2518 |
+
|
2519 |
+
return anchor.octave + shift + this.octave;
|
2520 |
+
}
|
2521 |
+
|
2522 |
+
|
2523 |
+
relativeOctave (from: ChordElement): number {
|
2524 |
+
if (this.phonet === "q") {
|
2525 |
+
if (this.anchorPitch)
|
2526 |
+
return this.anchorPitch.relativeOctave(from);
|
2527 |
+
else
|
2528 |
+
return 0;
|
2529 |
+
}
|
2530 |
+
|
2531 |
+
const phonetDiffer = this.phonetStep - from.phonetStep;
|
2532 |
+
const shift = idioms.phonetDifferToShift(phonetDiffer);
|
2533 |
+
|
2534 |
+
return this.octave - shift - from.octave;
|
2535 |
+
}
|
2536 |
+
|
2537 |
+
|
2538 |
+
get alterValue (): number {
|
2539 |
+
return idioms.ALTER_VALUES[this.alters] || 0;
|
2540 |
+
}
|
2541 |
+
|
2542 |
+
|
2543 |
+
get pitchValue (): number {
|
2544 |
+
const phonetValue = idioms.PHONET_VALUES[this.phonet];
|
2545 |
+
console.assert(Number.isInteger(phonetValue), "invalid phonet:", this.phonet);
|
2546 |
+
console.assert(!this.alters || idioms.ALTER_VALUES[this.alters], "invalid alters:", this.alters);
|
2547 |
+
|
2548 |
+
return 48 + this.octave * 12 + phonetValue + this.alterValue;
|
2549 |
+
}
|
2550 |
+
|
2551 |
+
|
2552 |
+
get absolutePitchValue (): number {
|
2553 |
+
return this.absolutePitch.pitchValue;
|
2554 |
+
}
|
2555 |
+
|
2556 |
+
|
2557 |
+
// middle C is zero
|
2558 |
+
get notePosition (): number {
|
2559 |
+
const phonet = idioms.PHONETS.indexOf(this.phonet);
|
2560 |
+
|
2561 |
+
return (this.octave - 1) * 7 + phonet;
|
2562 |
+
}
|
2563 |
+
|
2564 |
+
|
2565 |
+
get absoluteNotePosition (): number {
|
2566 |
+
return this.absolutePitch.notePosition;
|
2567 |
+
}
|
2568 |
+
|
2569 |
+
|
2570 |
+
get tiedParent (): ChordElement {
|
2571 |
+
if (!this._tied || !(this._tied instanceof Chord))
|
2572 |
+
return null;
|
2573 |
+
|
2574 |
+
const pitch = this._tied.pitchElements.find(p => p.absolutePitchValue === this.absolutePitchValue);
|
2575 |
+
if (!pitch)
|
2576 |
+
return null;
|
2577 |
+
|
2578 |
+
if (pitch._tied)
|
2579 |
+
return pitch.tiedParent;
|
2580 |
+
|
2581 |
+
return pitch;
|
2582 |
+
}
|
2583 |
+
};
|
2584 |
+
|
2585 |
+
|
2586 |
+
export class Duration extends BaseTerm {
|
2587 |
+
number: string;
|
2588 |
+
dots: number;
|
2589 |
+
multipliers?: string[];
|
2590 |
+
|
2591 |
+
|
2592 |
+
static _default: Duration;
|
2593 |
+
static get default (): Duration {
|
2594 |
+
if (!Duration._default)
|
2595 |
+
Duration._default = new Duration({number: 4, dots: 0});
|
2596 |
+
|
2597 |
+
return Duration._default;
|
2598 |
+
}
|
2599 |
+
|
2600 |
+
|
2601 |
+
static fromMagnitude (magnitude: number): Duration {
|
2602 |
+
const MULTI = 1024;
|
2603 |
+
const MULTI_DURATION_MAGNITUDE = WHOLE_DURATION_MAGNITUDE * MULTI;
|
2604 |
+
const multiMag = magnitude * MULTI;
|
2605 |
+
if (!Number.isInteger(multiMag)) {
|
2606 |
+
console.warn("magnitude must be integer:", magnitude);
|
2607 |
+
return null;
|
2608 |
+
}
|
2609 |
+
|
2610 |
+
const di = gcd(multiMag, MULTI_DURATION_MAGNITUDE);
|
2611 |
+
const denominator = MULTI_DURATION_MAGNITUDE / di;
|
2612 |
+
const numerator = multiMag / di;
|
2613 |
+
if (!Number.isInteger(Math.log2(denominator)))
|
2614 |
+
return new Duration({number: 1, dots: 0, multipliers: [`${numerator}/${denominator}`]});
|
2615 |
+
|
2616 |
+
switch (numerator) {
|
2617 |
+
case 1:
|
2618 |
+
return new Duration({number: denominator, dots: 0});
|
2619 |
+
|
2620 |
+
case 3:
|
2621 |
+
return new Duration({number: denominator / 2, dots: 1});
|
2622 |
+
|
2623 |
+
case 7:
|
2624 |
+
return new Duration({number: denominator / 4, dots: 2});
|
2625 |
+
|
2626 |
+
default:
|
2627 |
+
return new Duration({number: denominator, dots: 0, multipliers: [numerator.toString()]});
|
2628 |
+
}
|
2629 |
+
}
|
2630 |
+
|
2631 |
+
|
2632 |
+
serialize () {
|
2633 |
+
const dots = Array(this.dots).fill(".").join("");
|
2634 |
+
const multipliers = this.multipliers && this.multipliers.map(multiplier => `*${multiplier}`).join("");
|
2635 |
+
|
2636 |
+
return compact([
|
2637 |
+
this.number, dots, multipliers,
|
2638 |
+
]);
|
2639 |
+
}
|
2640 |
+
|
2641 |
+
|
2642 |
+
get withMultiplier () {
|
2643 |
+
return this.multipliers && this.multipliers.length > 0;
|
2644 |
+
}
|
2645 |
+
|
2646 |
+
|
2647 |
+
get denominator (): number {
|
2648 |
+
switch (this.number) {
|
2649 |
+
case "\\breve":
|
2650 |
+
return 0.5;
|
2651 |
+
|
2652 |
+
case "\\longa":
|
2653 |
+
return 0.25;
|
2654 |
+
}
|
2655 |
+
|
2656 |
+
return Number(this.number);
|
2657 |
+
}
|
2658 |
+
|
2659 |
+
|
2660 |
+
get division (): number {
|
2661 |
+
return Math.log2(this.denominator);
|
2662 |
+
}
|
2663 |
+
|
2664 |
+
|
2665 |
+
// how many smallest rhythm unit in a whole note
|
2666 |
+
get subdivider (): number {
|
2667 |
+
return this.denominator * (2 ** this.dots);
|
2668 |
+
}
|
2669 |
+
|
2670 |
+
|
2671 |
+
get magnitude (): number {
|
2672 |
+
let value = WHOLE_DURATION_MAGNITUDE / this.denominator;
|
2673 |
+
|
2674 |
+
if (this.dots)
|
2675 |
+
value *= 2 - 0.5 ** this.dots;
|
2676 |
+
|
2677 |
+
if (this.multipliers)
|
2678 |
+
this.multipliers.forEach(multiplier => value *= eval(multiplier));
|
2679 |
+
|
2680 |
+
return value;
|
2681 |
+
}
|
2682 |
+
};
|
2683 |
+
|
2684 |
+
|
2685 |
+
interface BriefChordBody {
|
2686 |
+
pitch: string;
|
2687 |
+
duration: Duration;
|
2688 |
+
separator: string;
|
2689 |
+
items: string[];
|
2690 |
+
};
|
2691 |
+
|
2692 |
+
|
2693 |
+
export class BriefChord extends BaseTerm {
|
2694 |
+
body: BriefChordBody;
|
2695 |
+
post_events: any[];
|
2696 |
+
|
2697 |
+
|
2698 |
+
constructor (data: object) {
|
2699 |
+
super(data);
|
2700 |
+
|
2701 |
+
if (this.body)
|
2702 |
+
this.body.duration = parseRaw(this.body.duration);
|
2703 |
+
}
|
2704 |
+
|
2705 |
+
|
2706 |
+
serialize () {
|
2707 |
+
const {pitch, duration, separator, items} = this.body;
|
2708 |
+
|
2709 |
+
return [
|
2710 |
+
...compact(cc([pitch, duration, separator, ...(items || [])].map(BaseTerm.optionalSerialize))),
|
2711 |
+
...cc((this.post_events || []).map(BaseTerm.optionalSerialize)),
|
2712 |
+
];
|
2713 |
+
}
|
2714 |
+
|
2715 |
+
|
2716 |
+
get isMusic (): boolean {
|
2717 |
+
return true;
|
2718 |
+
}
|
2719 |
+
|
2720 |
+
|
2721 |
+
get durationMagnitude (): number {
|
2722 |
+
if (this.body.duration)
|
2723 |
+
return this.body.duration.magnitude;
|
2724 |
+
|
2725 |
+
return 0;
|
2726 |
+
}
|
2727 |
+
};
|
2728 |
+
|
2729 |
+
|
2730 |
+
export class NumberUnit extends BaseTerm {
|
2731 |
+
number: number;
|
2732 |
+
unit: string;
|
2733 |
+
|
2734 |
+
|
2735 |
+
serialize () {
|
2736 |
+
return [this.number, "\b", this.unit];
|
2737 |
+
}
|
2738 |
+
|
2739 |
+
|
2740 |
+
set ({number, unit}) {
|
2741 |
+
this.number = Number(number.toFixed(2));
|
2742 |
+
|
2743 |
+
if (unit !== undefined)
|
2744 |
+
this.unit = unit;
|
2745 |
+
}
|
2746 |
+
}
|
2747 |
+
|
2748 |
+
|
2749 |
+
export class Tempo extends BaseTerm {
|
2750 |
+
beatsPerMinute?: number;
|
2751 |
+
unit?: Duration;
|
2752 |
+
text?: string;
|
2753 |
+
|
2754 |
+
|
2755 |
+
static fromNoteBpm (note: number, bpm: number): Tempo {
|
2756 |
+
return new Tempo ({
|
2757 |
+
unit: new Duration({number: note.toString(), dots: 0}),
|
2758 |
+
beatsPerMinute: bpm,
|
2759 |
+
});
|
2760 |
+
}
|
2761 |
+
|
2762 |
+
|
2763 |
+
serialize () {
|
2764 |
+
const assignment = Number.isFinite(this.beatsPerMinute) ? [...BaseTerm.optionalSerialize(this.unit), "=", this.beatsPerMinute] : [];
|
2765 |
+
|
2766 |
+
return [
|
2767 |
+
"\\tempo",
|
2768 |
+
...BaseTerm.optionalSerialize(this.text),
|
2769 |
+
...assignment,
|
2770 |
+
];
|
2771 |
+
}
|
2772 |
+
}
|
2773 |
+
|
2774 |
+
|
2775 |
+
const DIRECTION_CHAR = {
|
2776 |
+
up: "^",
|
2777 |
+
down: "_",
|
2778 |
+
middle: "-",
|
2779 |
+
};
|
2780 |
+
|
2781 |
+
|
2782 |
+
export class PostEvent extends BaseTerm {
|
2783 |
+
direction: string;
|
2784 |
+
arg: string | BaseTerm;
|
2785 |
+
|
2786 |
+
|
2787 |
+
serialize () {
|
2788 |
+
const dir = DIRECTION_CHAR[this.direction];
|
2789 |
+
const prefix = dir ? [dir, "\b"] : [];
|
2790 |
+
|
2791 |
+
return prefix.concat(BaseTerm.optionalSerialize(this.arg));
|
2792 |
+
}
|
2793 |
+
|
2794 |
+
|
2795 |
+
get entries () {
|
2796 |
+
if (this.arg instanceof BaseTerm)
|
2797 |
+
return [this.arg];
|
2798 |
+
|
2799 |
+
return null;
|
2800 |
+
}
|
2801 |
+
|
2802 |
+
|
2803 |
+
get isTying (): boolean {
|
2804 |
+
return this.arg === "~";
|
2805 |
+
}
|
2806 |
+
|
2807 |
+
|
2808 |
+
get isStaccato (): boolean {
|
2809 |
+
if (this.arg instanceof Command)
|
2810 |
+
return ["staccato", "staccatissimo", "portato"].includes(this.arg.cmd);
|
2811 |
+
|
2812 |
+
if ([".", "!"].includes(this.arg as string))
|
2813 |
+
return true;
|
2814 |
+
|
2815 |
+
return false;
|
2816 |
+
}
|
2817 |
+
};
|
2818 |
+
|
2819 |
+
|
2820 |
+
export class Fingering extends BaseTerm {
|
2821 |
+
value: number;
|
2822 |
+
|
2823 |
+
|
2824 |
+
serialize () {
|
2825 |
+
return [this.value];
|
2826 |
+
}
|
2827 |
+
};
|
2828 |
+
|
2829 |
+
|
2830 |
+
export class Markup extends BaseTerm {
|
2831 |
+
head: any[];
|
2832 |
+
body: (string|BaseTerm);
|
2833 |
+
|
2834 |
+
|
2835 |
+
serialize () {
|
2836 |
+
return [
|
2837 |
+
...cc(this.head.map(BaseTerm.optionalSerialize)),
|
2838 |
+
...BaseTerm.optionalSerialize(this.body),
|
2839 |
+
];
|
2840 |
+
}
|
2841 |
+
};
|
2842 |
+
|
2843 |
+
|
2844 |
+
export class Lyric extends MusicEvent {
|
2845 |
+
content: string | LiteralString;
|
2846 |
+
|
2847 |
+
|
2848 |
+
serialize () {
|
2849 |
+
return [
|
2850 |
+
...BaseTerm.optionalSerialize(this.content),
|
2851 |
+
...BaseTerm.optionalSerialize(this.duration),
|
2852 |
+
...cc((this.post_events || []).map(BaseTerm.optionalSerialize)),
|
2853 |
+
];
|
2854 |
+
}
|
2855 |
+
};
|
2856 |
+
|
2857 |
+
|
2858 |
+
export class Comment extends BaseTerm {
|
2859 |
+
text: string;
|
2860 |
+
scoped: boolean;
|
2861 |
+
|
2862 |
+
|
2863 |
+
serialize () {
|
2864 |
+
return [
|
2865 |
+
this.text,
|
2866 |
+
"\n",
|
2867 |
+
];
|
2868 |
+
}
|
2869 |
+
|
2870 |
+
|
2871 |
+
static createSingle (text): Comment {
|
2872 |
+
return new Comment({text: "%" + text});
|
2873 |
+
}
|
2874 |
+
|
2875 |
+
|
2876 |
+
static createScoped (text): Comment {
|
2877 |
+
console.assert(!/%\}/.test(text), "invalid scoped comment text:", text);
|
2878 |
+
|
2879 |
+
return new Comment({text: `%{${text}%}`, scoped: true});
|
2880 |
+
}
|
2881 |
+
};
|
2882 |
+
|
2883 |
+
|
2884 |
+
export class Unexpect extends BaseTerm {
|
2885 |
+
constructor (data) {
|
2886 |
+
super(data);
|
2887 |
+
|
2888 |
+
console.warn("unexpected term", data);
|
2889 |
+
}
|
2890 |
+
};
|
2891 |
+
|
2892 |
+
|
2893 |
+
export const termDictionary = {
|
2894 |
+
Root,
|
2895 |
+
LiteralString,
|
2896 |
+
Command,
|
2897 |
+
Variable,
|
2898 |
+
MarkupCommand,
|
2899 |
+
Repeat,
|
2900 |
+
Relative,
|
2901 |
+
ParallelMusic,
|
2902 |
+
TimeSignature,
|
2903 |
+
Partial,
|
2904 |
+
Times,
|
2905 |
+
Tuplet,
|
2906 |
+
Grace,
|
2907 |
+
AfterGrace,
|
2908 |
+
Clef,
|
2909 |
+
KeySignature,
|
2910 |
+
OctaveShift,
|
2911 |
+
Include,
|
2912 |
+
Version,
|
2913 |
+
Language,
|
2914 |
+
LyricMode,
|
2915 |
+
ChordMode,
|
2916 |
+
Transposition,
|
2917 |
+
StemDirection,
|
2918 |
+
Change,
|
2919 |
+
Block,
|
2920 |
+
InlineBlock,
|
2921 |
+
Scheme,
|
2922 |
+
SchemeFunction,
|
2923 |
+
SchemePair,
|
2924 |
+
SchemePointer,
|
2925 |
+
SchemeEmbed,
|
2926 |
+
Assignment,
|
2927 |
+
Duration,
|
2928 |
+
ChordElement,
|
2929 |
+
Chord,
|
2930 |
+
Rest,
|
2931 |
+
BriefChord,
|
2932 |
+
NumberUnit,
|
2933 |
+
MusicBlock,
|
2934 |
+
SimultaneousList,
|
2935 |
+
ContextedMusic,
|
2936 |
+
Divide,
|
2937 |
+
Tempo,
|
2938 |
+
PostEvent,
|
2939 |
+
Fingering,
|
2940 |
+
Markup,
|
2941 |
+
Lyric,
|
2942 |
+
Primitive,
|
2943 |
+
Comment,
|
2944 |
+
};
|
2945 |
+
|
2946 |
+
|
2947 |
+
const termProtoMap: Map<object, string> = Object.entries(termDictionary)
|
2948 |
+
.reduce((map, [name, cls]: [string, {prototype: object}]) => (map.set(cls.prototype, name), map), new Map());
|
2949 |
+
|
2950 |
+
|
2951 |
+
const parseRawEnforce = data => {
|
2952 |
+
switch (typeof data) {
|
2953 |
+
case "string":
|
2954 |
+
case "number":
|
2955 |
+
return new Primitive({exp: data});
|
2956 |
+
|
2957 |
+
default:
|
2958 |
+
return parseRaw(data);
|
2959 |
+
}
|
2960 |
+
};
|
2961 |
+
|
2962 |
+
|
2963 |
+
export const parseRaw = data => {
|
2964 |
+
if (data instanceof BaseTerm)
|
2965 |
+
return data;
|
2966 |
+
|
2967 |
+
if (!data)
|
2968 |
+
return data;
|
2969 |
+
|
2970 |
+
switch (typeof data) {
|
2971 |
+
case "object":
|
2972 |
+
if (Array.isArray(data))
|
2973 |
+
return data.map(item => parseRaw(item));
|
2974 |
+
|
2975 |
+
const {proto, ...fields} = data;
|
2976 |
+
if (proto) {
|
2977 |
+
if (proto === "_PLAIN")
|
2978 |
+
return fields;
|
2979 |
+
|
2980 |
+
const termClass = termDictionary[proto];
|
2981 |
+
if (!termClass)
|
2982 |
+
throw new Error(`Unexpected term class: ${data.proto}`);
|
2983 |
+
|
2984 |
+
return new termClass(fields);
|
2985 |
+
}
|
2986 |
+
|
2987 |
+
return new Unexpect(data);
|
2988 |
+
}
|
2989 |
+
|
2990 |
+
return data;
|
2991 |
+
};
|