v0.16.0
This commit is contained in:
parent
3b9dd171c7
commit
adb7f5075d
3 changed files with 265 additions and 388 deletions
641
dist/main.js
vendored
641
dist/main.js
vendored
|
@ -5,7 +5,7 @@
|
||||||
// @supportURL https://github.com/Xmader/musescore-downloader/issues
|
// @supportURL https://github.com/Xmader/musescore-downloader/issues
|
||||||
// @updateURL https://msdl.librescore.org/install.user.js
|
// @updateURL https://msdl.librescore.org/install.user.js
|
||||||
// @downloadURL https://msdl.librescore.org/install.user.js
|
// @downloadURL https://msdl.librescore.org/install.user.js
|
||||||
// @version 0.15.20
|
// @version 0.16.0
|
||||||
// @description download sheet music from musescore.com for free, no login or Musescore Pro required | 免登录、免 Musescore Pro,免费下载 musescore.com 上的曲谱
|
// @description download sheet music from musescore.com for free, no login or Musescore Pro required | 免登录、免 Musescore Pro,免费下载 musescore.com 上的曲谱
|
||||||
// @author Xmader
|
// @author Xmader
|
||||||
// @match https://musescore.com/*/*
|
// @match https://musescore.com/*/*
|
||||||
|
@ -22,10 +22,11 @@
|
||||||
window.eval('(' + function () {
|
window.eval('(' + function () {
|
||||||
|
|
||||||
function __awaiter(thisArg, _arguments, P, generator) {
|
function __awaiter(thisArg, _arguments, P, generator) {
|
||||||
|
function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
|
||||||
return new (P || (P = Promise))(function (resolve, reject) {
|
return new (P || (P = Promise))(function (resolve, reject) {
|
||||||
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
|
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
|
||||||
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
|
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
|
||||||
function step(result) { result.done ? resolve(result.value) : new P(function (resolve) { resolve(result.value); }).then(fulfilled, rejected); }
|
function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
|
||||||
step((generator = generator.apply(thisArg, _arguments || [])).next());
|
step((generator = generator.apply(thisArg, _arguments || [])).next());
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
@ -36,185 +37,27 @@
|
||||||
return module = { exports: {} }, fn(module, module.exports), module.exports;
|
return module = { exports: {} }, fn(module, module.exports), module.exports;
|
||||||
}
|
}
|
||||||
|
|
||||||
var FileSaver = createCommonjsModule(function (module, exports) {
|
var FileSaver_min = createCommonjsModule(function (module, exports) {
|
||||||
(function (global, factory) {
|
(function(a,b){b();})(commonjsGlobal,function(){function b(a,b){return "undefined"==typeof b?b={autoBom:!1}:"object"!=typeof b&&(console.warn("Deprecated: Expected third argument to be a object"),b={autoBom:!b}),b.autoBom&&/^\s*(?:text\/\S*|application\/xml|\S*\/\S*\+xml)\s*;.*charset\s*=\s*utf-8/i.test(a.type)?new Blob(["\uFEFF",a],{type:a.type}):a}function c(a,b,c){var d=new XMLHttpRequest;d.open("GET",a),d.responseType="blob",d.onload=function(){g(d.response,b,c);},d.onerror=function(){console.error("could not download file");},d.send();}function d(a){var b=new XMLHttpRequest;b.open("HEAD",a,!1);try{b.send();}catch(a){}return 200<=b.status&&299>=b.status}function e(a){try{a.dispatchEvent(new MouseEvent("click"));}catch(c){var b=document.createEvent("MouseEvents");b.initMouseEvent("click",!0,!0,window,0,0,0,80,20,!1,!1,!1,!1,0,null),a.dispatchEvent(b);}}var f="object"==typeof window&&window.window===window?window:"object"==typeof self&&self.self===self?self:"object"==typeof commonjsGlobal&&commonjsGlobal.global===commonjsGlobal?commonjsGlobal:void 0,a=f.navigator&&/Macintosh/.test(navigator.userAgent)&&/AppleWebKit/.test(navigator.userAgent)&&!/Safari/.test(navigator.userAgent),g=f.saveAs||("object"!=typeof window||window!==f?function(){}:"download"in HTMLAnchorElement.prototype&&!a?function(b,g,h){var i=f.URL||f.webkitURL,j=document.createElement("a");g=g||b.name||"download",j.download=g,j.rel="noopener","string"==typeof b?(j.href=b,j.origin===location.origin?e(j):d(j.href)?c(b,g,h):e(j,j.target="_blank")):(j.href=i.createObjectURL(b),setTimeout(function(){i.revokeObjectURL(j.href);},4E4),setTimeout(function(){e(j);},0));}:"msSaveOrOpenBlob"in navigator?function(f,g,h){if(g=g||f.name||"download","string"!=typeof f)navigator.msSaveOrOpenBlob(b(f,h),g);else if(d(f))c(f,g,h);else {var i=document.createElement("a");i.href=f,i.target="_blank",setTimeout(function(){e(i);});}}:function(b,d,e,g){if(g=g||open("","_blank"),g&&(g.document.title=g.document.body.innerText="downloading..."),"string"==typeof b)return c(b,d,e);var h="application/octet-stream"===b.type,i=/constructor/i.test(f.HTMLElement)||f.safari,j=/CriOS\/[\d]+/.test(navigator.userAgent);if((j||h&&i||a)&&"undefined"!=typeof FileReader){var k=new FileReader;k.onloadend=function(){var a=k.result;a=j?a:a.replace(/^data:[^;]*;/,"data:attachment/file;"),g?g.location.href=a:location=a,g=null;},k.readAsDataURL(b);}else {var l=f.URL||f.webkitURL,m=l.createObjectURL(b);g?g.location=m:location.href=m,g=null,setTimeout(function(){l.revokeObjectURL(m);},4E4);}});f.saveAs=g.saveAs=g,(module.exports=g);});
|
||||||
{
|
|
||||||
factory();
|
|
||||||
}
|
|
||||||
})(commonjsGlobal, function () {
|
|
||||||
|
|
||||||
/*
|
|
||||||
* FileSaver.js
|
|
||||||
* A saveAs() FileSaver implementation.
|
|
||||||
*
|
|
||||||
* By Eli Grey, http://eligrey.com
|
|
||||||
*
|
|
||||||
* License : https://github.com/eligrey/FileSaver.js/blob/master/LICENSE.md (MIT)
|
|
||||||
* source : http://purl.eligrey.com/github/FileSaver.js
|
|
||||||
*/
|
|
||||||
// The one and only way of getting global scope in all environments
|
|
||||||
// https://stackoverflow.com/q/3277182/1008999
|
|
||||||
var _global = typeof window === 'object' && window.window === window ? window : typeof self === 'object' && self.self === self ? self : typeof commonjsGlobal === 'object' && commonjsGlobal.global === commonjsGlobal ? commonjsGlobal : void 0;
|
|
||||||
|
|
||||||
function bom(blob, opts) {
|
});
|
||||||
if (typeof opts === 'undefined') opts = {
|
|
||||||
autoBom: false
|
// Only Node.JS has a process variable that is of [[Class]] process
|
||||||
};else if (typeof opts !== 'object') {
|
var detectNode = Object.prototype.toString.call(typeof process !== 'undefined' ? process : 0) === '[object process]';
|
||||||
console.warn('Deprecated: Expected third argument to be a object');
|
|
||||||
opts = {
|
const escapeFilename = (s) => {
|
||||||
autoBom: !opts
|
return s.replace(/[\s<>:{}"/\\|?*~.\0\cA-\cZ]+/g, '_');
|
||||||
};
|
};
|
||||||
} // prepend BOM for UTF-8 XML and text/* types (including HTML)
|
const getFetch = () => {
|
||||||
// note: your browser will automatically convert UTF-16 U+FEFF to EF BB BF
|
if (!detectNode) {
|
||||||
|
return fetch;
|
||||||
if (opts.autoBom && /^\s*(?:text\/\S*|application\/xml|\S*\/\S*\+xml)\s*;.*charset\s*=\s*utf-8/i.test(blob.type)) {
|
|
||||||
return new Blob([String.fromCharCode(0xFEFF), blob], {
|
|
||||||
type: blob.type
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
else {
|
||||||
return blob;
|
// eslint-disable-next-line @typescript-eslint/no-unsafe-return
|
||||||
|
return require('node-fetch');
|
||||||
}
|
}
|
||||||
|
|
||||||
function download(url, name, opts) {
|
|
||||||
var xhr = new XMLHttpRequest();
|
|
||||||
xhr.open('GET', url);
|
|
||||||
xhr.responseType = 'blob';
|
|
||||||
|
|
||||||
xhr.onload = function () {
|
|
||||||
saveAs(xhr.response, name, opts);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
xhr.onerror = function () {
|
|
||||||
console.error('could not download file');
|
|
||||||
};
|
|
||||||
|
|
||||||
xhr.send();
|
|
||||||
}
|
|
||||||
|
|
||||||
function corsEnabled(url) {
|
|
||||||
var xhr = new XMLHttpRequest(); // use sync to avoid popup blocker
|
|
||||||
|
|
||||||
xhr.open('HEAD', url, false);
|
|
||||||
|
|
||||||
try {
|
|
||||||
xhr.send();
|
|
||||||
} catch (e) {}
|
|
||||||
|
|
||||||
return xhr.status >= 200 && xhr.status <= 299;
|
|
||||||
} // `a.click()` doesn't work for all browsers (#465)
|
|
||||||
|
|
||||||
|
|
||||||
function click(node) {
|
|
||||||
try {
|
|
||||||
node.dispatchEvent(new MouseEvent('click'));
|
|
||||||
} catch (e) {
|
|
||||||
var evt = document.createEvent('MouseEvents');
|
|
||||||
evt.initMouseEvent('click', true, true, window, 0, 0, 0, 80, 20, false, false, false, false, 0, null);
|
|
||||||
node.dispatchEvent(evt);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
var saveAs = _global.saveAs || ( // probably in some web worker
|
|
||||||
typeof window !== 'object' || window !== _global ? function saveAs() {}
|
|
||||||
/* noop */
|
|
||||||
// Use download attribute first if possible (#193 Lumia mobile)
|
|
||||||
: 'download' in HTMLAnchorElement.prototype ? function saveAs(blob, name, opts) {
|
|
||||||
var URL = _global.URL || _global.webkitURL;
|
|
||||||
var a = document.createElement('a');
|
|
||||||
name = name || blob.name || 'download';
|
|
||||||
a.download = name;
|
|
||||||
a.rel = 'noopener'; // tabnabbing
|
|
||||||
// TODO: detect chrome extensions & packaged apps
|
|
||||||
// a.target = '_blank'
|
|
||||||
|
|
||||||
if (typeof blob === 'string') {
|
|
||||||
// Support regular links
|
|
||||||
a.href = blob;
|
|
||||||
|
|
||||||
if (a.origin !== location.origin) {
|
|
||||||
corsEnabled(a.href) ? download(blob, name, opts) : click(a, a.target = '_blank');
|
|
||||||
} else {
|
|
||||||
click(a);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
// Support blobs
|
|
||||||
a.href = URL.createObjectURL(blob);
|
|
||||||
setTimeout(function () {
|
|
||||||
URL.revokeObjectURL(a.href);
|
|
||||||
}, 4E4); // 40s
|
|
||||||
|
|
||||||
setTimeout(function () {
|
|
||||||
click(a);
|
|
||||||
}, 0);
|
|
||||||
}
|
|
||||||
} // Use msSaveOrOpenBlob as a second approach
|
|
||||||
: 'msSaveOrOpenBlob' in navigator ? function saveAs(blob, name, opts) {
|
|
||||||
name = name || blob.name || 'download';
|
|
||||||
|
|
||||||
if (typeof blob === 'string') {
|
|
||||||
if (corsEnabled(blob)) {
|
|
||||||
download(blob, name, opts);
|
|
||||||
} else {
|
|
||||||
var a = document.createElement('a');
|
|
||||||
a.href = blob;
|
|
||||||
a.target = '_blank';
|
|
||||||
setTimeout(function () {
|
|
||||||
click(a);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
navigator.msSaveOrOpenBlob(bom(blob, opts), name);
|
|
||||||
}
|
|
||||||
} // Fallback to using FileReader and a popup
|
|
||||||
: function saveAs(blob, name, opts, popup) {
|
|
||||||
// Open a popup immediately do go around popup blocker
|
|
||||||
// Mostly only available on user interaction and the fileReader is async so...
|
|
||||||
popup = popup || open('', '_blank');
|
|
||||||
|
|
||||||
if (popup) {
|
|
||||||
popup.document.title = popup.document.body.innerText = 'downloading...';
|
|
||||||
}
|
|
||||||
|
|
||||||
if (typeof blob === 'string') return download(blob, name, opts);
|
|
||||||
var force = blob.type === 'application/octet-stream';
|
|
||||||
|
|
||||||
var isSafari = /constructor/i.test(_global.HTMLElement) || _global.safari;
|
|
||||||
|
|
||||||
var isChromeIOS = /CriOS\/[\d]+/.test(navigator.userAgent);
|
|
||||||
|
|
||||||
if ((isChromeIOS || force && isSafari) && typeof FileReader === 'object') {
|
|
||||||
// Safari doesn't allow downloading of blob URLs
|
|
||||||
var reader = new FileReader();
|
|
||||||
|
|
||||||
reader.onloadend = function () {
|
|
||||||
var url = reader.result;
|
|
||||||
url = isChromeIOS ? url : url.replace(/^data:[^;]*;/, 'data:attachment/file;');
|
|
||||||
if (popup) popup.location.href = url;else location = url;
|
|
||||||
popup = null; // reverse-tabnabbing #460
|
|
||||||
};
|
|
||||||
|
|
||||||
reader.readAsDataURL(blob);
|
|
||||||
} else {
|
|
||||||
var URL = _global.URL || _global.webkitURL;
|
|
||||||
var url = URL.createObjectURL(blob);
|
|
||||||
if (popup) popup.location = url;else location.href = url;
|
|
||||||
popup = null; // reverse-tabnabbing #460
|
|
||||||
|
|
||||||
setTimeout(function () {
|
|
||||||
URL.revokeObjectURL(url);
|
|
||||||
}, 4E4); // 40s
|
|
||||||
}
|
|
||||||
});
|
|
||||||
_global.saveAs = saveAs.saveAs = saveAs;
|
|
||||||
|
|
||||||
{
|
|
||||||
module.exports = saveAs;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
const saveAs = FileSaver.saveAs;
|
|
||||||
const fetchData = (url, init) => __awaiter(void 0, void 0, void 0, function* () {
|
const fetchData = (url, init) => __awaiter(void 0, void 0, void 0, function* () {
|
||||||
const r = yield fetch(url, init);
|
const r = yield fetch(url, init);
|
||||||
const data = yield r.arrayBuffer();
|
const data = yield r.arrayBuffer();
|
||||||
|
@ -236,6 +79,8 @@
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
const getSandboxWindow = () => {
|
const getSandboxWindow = () => {
|
||||||
|
if (typeof document === 'undefined')
|
||||||
|
return {};
|
||||||
const iframe = document.createElement('iframe');
|
const iframe = document.createElement('iframe');
|
||||||
iframe.style.display = 'none';
|
iframe.style.display = 'none';
|
||||||
document.body.append(iframe);
|
document.body.append(iframe);
|
||||||
|
@ -273,10 +118,11 @@
|
||||||
(function () {
|
(function () {
|
||||||
|
|
||||||
function __awaiter(thisArg, _arguments, P, generator) {
|
function __awaiter(thisArg, _arguments, P, generator) {
|
||||||
|
function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
|
||||||
return new (P || (P = Promise))(function (resolve, reject) {
|
return new (P || (P = Promise))(function (resolve, reject) {
|
||||||
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
|
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
|
||||||
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
|
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
|
||||||
function step(result) { result.done ? resolve(result.value) : new P(function (resolve) { resolve(result.value); }).then(fulfilled, rejected); }
|
function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
|
||||||
step((generator = generator.apply(thisArg, _arguments || [])).next());
|
step((generator = generator.apply(thisArg, _arguments || [])).next());
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
@ -10869,7 +10715,7 @@
|
||||||
this.write_in_progress = true;
|
this.write_in_progress = true;
|
||||||
|
|
||||||
var self = this;
|
var self = this;
|
||||||
nextTick(function() {
|
process.nextTick(function() {
|
||||||
self.write_in_progress = false;
|
self.write_in_progress = false;
|
||||||
var res = self._write(flush, input, in_off, in_len, out, out_off, out_len);
|
var res = self._write(flush, input, in_off, in_len, out, out_off, out_len);
|
||||||
self.callback(res[0], res[1]);
|
self.callback(res[0], res[1]);
|
||||||
|
@ -11413,7 +11259,7 @@
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
} else {
|
} else {
|
||||||
nextTick(callback);
|
process.nextTick(callback);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -11437,7 +11283,7 @@
|
||||||
|
|
||||||
if (ws.ended) {
|
if (ws.ended) {
|
||||||
if (callback)
|
if (callback)
|
||||||
nextTick(callback);
|
process.nextTick(callback);
|
||||||
} else if (ws.ending) {
|
} else if (ws.ending) {
|
||||||
if (callback)
|
if (callback)
|
||||||
this.once('end', callback);
|
this.once('end', callback);
|
||||||
|
@ -11454,7 +11300,7 @@
|
||||||
|
|
||||||
Zlib$1.prototype.close = function(callback) {
|
Zlib$1.prototype.close = function(callback) {
|
||||||
if (callback)
|
if (callback)
|
||||||
nextTick(callback);
|
process.nextTick(callback);
|
||||||
|
|
||||||
if (this._closed)
|
if (this._closed)
|
||||||
return;
|
return;
|
||||||
|
@ -11464,7 +11310,7 @@
|
||||||
this._binding.close();
|
this._binding.close();
|
||||||
|
|
||||||
var self = this;
|
var self = this;
|
||||||
nextTick(function() {
|
process.nextTick(function() {
|
||||||
self.emit('close');
|
self.emit('close');
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
@ -23198,13 +23044,17 @@
|
||||||
evt.initMouseEvent('click', true, true, window, 0, 0, 0, 80, 20, false, false, false, false, 0, null);
|
evt.initMouseEvent('click', true, true, window, 0, 0, 0, 80, 20, false, false, false, false, 0, null);
|
||||||
node.dispatchEvent(evt);
|
node.dispatchEvent(evt);
|
||||||
}
|
}
|
||||||
}
|
} // Detect WebView inside a native macOS app by ruling out all browsers
|
||||||
|
// We just need to check for 'Safari' because all other browsers (besides Firefox) include that too
|
||||||
|
// https://www.whatismybrowser.com/guides/the-latest-user-agent/macos
|
||||||
|
|
||||||
|
|
||||||
|
var isMacOSWebView = _global.navigator && /Macintosh/.test(navigator.userAgent) && /AppleWebKit/.test(navigator.userAgent) && !/Safari/.test(navigator.userAgent);
|
||||||
var saveAs = _global.saveAs || ( // probably in some web worker
|
var saveAs = _global.saveAs || ( // probably in some web worker
|
||||||
typeof window !== 'object' || window !== _global ? function saveAs() {}
|
typeof window !== 'object' || window !== _global ? function saveAs() {}
|
||||||
/* noop */
|
/* noop */
|
||||||
// Use download attribute first if possible (#193 Lumia mobile)
|
// Use download attribute first if possible (#193 Lumia mobile) unless this is a macOS WebView
|
||||||
: 'download' in HTMLAnchorElement.prototype ? function saveAs(blob, name, opts) {
|
: 'download' in HTMLAnchorElement.prototype && !isMacOSWebView ? function saveAs(blob, name, opts) {
|
||||||
var URL = _global.URL || _global.webkitURL;
|
var URL = _global.URL || _global.webkitURL;
|
||||||
var a = document.createElement('a');
|
var a = document.createElement('a');
|
||||||
name = name || blob.name || 'download';
|
name = name || blob.name || 'download';
|
||||||
|
@ -23268,7 +23118,7 @@
|
||||||
|
|
||||||
var isChromeIOS = /CriOS\/[\d]+/.test(navigator.userAgent);
|
var isChromeIOS = /CriOS\/[\d]+/.test(navigator.userAgent);
|
||||||
|
|
||||||
if ((isChromeIOS || force && isSafari) && typeof FileReader === 'object') {
|
if ((isChromeIOS || force && isSafari || isMacOSWebView) && typeof FileReader !== 'undefined') {
|
||||||
// Safari doesn't allow downloading of blob URLs
|
// Safari doesn't allow downloading of blob URLs
|
||||||
var reader = new FileReader();
|
var reader = new FileReader();
|
||||||
|
|
||||||
|
@ -26346,94 +26196,6 @@ Please pipe the document into a Node stream.\
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* eslint-disable @typescript-eslint/no-unsafe-return */
|
|
||||||
// run at document-start
|
|
||||||
const ugappJsStore = (() => {
|
|
||||||
try {
|
|
||||||
const l = document.body.children;
|
|
||||||
const el = [...l].find(e => Object.keys(e.dataset).length > 0);
|
|
||||||
const json = Object.values(el.dataset)[0];
|
|
||||||
return JSON.parse(json);
|
|
||||||
}
|
|
||||||
catch (err) {
|
|
||||||
console$1.error(err);
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
})();
|
|
||||||
const IPNS_KEY = 'QmSdXtvzC8v8iTTZuj5cVmiugnzbR1QATYRcGix4bBsioP';
|
|
||||||
const RADIX = 20;
|
|
||||||
const scoreinfo = {
|
|
||||||
get playerdata() {
|
|
||||||
// @ts-ignore
|
|
||||||
return ugappJsStore.store.page.data.score;
|
|
||||||
},
|
|
||||||
get id() {
|
|
||||||
try {
|
|
||||||
return this.playerdata.id;
|
|
||||||
}
|
|
||||||
catch (_a) {
|
|
||||||
const el = document.querySelector("meta[property='al:ios:url']");
|
|
||||||
const m = el.content.match(/(\d+)$/);
|
|
||||||
return +m[1];
|
|
||||||
}
|
|
||||||
},
|
|
||||||
get idLastDigit() {
|
|
||||||
return (+this.id) % RADIX;
|
|
||||||
},
|
|
||||||
get title() {
|
|
||||||
try {
|
|
||||||
return this.playerdata.title;
|
|
||||||
}
|
|
||||||
catch (_a) {
|
|
||||||
const el = document.querySelector("meta[property='og:title']");
|
|
||||||
return el.content;
|
|
||||||
}
|
|
||||||
},
|
|
||||||
get fileName() {
|
|
||||||
return this.title.replace(/[\s<>:{}"/\\|?*~.\0\cA-\cZ]+/g, '_');
|
|
||||||
},
|
|
||||||
get pageCount() {
|
|
||||||
try {
|
|
||||||
return this.playerdata.pages_count;
|
|
||||||
}
|
|
||||||
catch (_a) {
|
|
||||||
return document.querySelectorAll('.gXB83').length;
|
|
||||||
}
|
|
||||||
},
|
|
||||||
get baseUrl() {
|
|
||||||
let thumbnailUrl;
|
|
||||||
try {
|
|
||||||
thumbnailUrl = this.playerdata.thumbnails.original;
|
|
||||||
}
|
|
||||||
catch (_a) {
|
|
||||||
const el = document.querySelector("meta[property='og:image']");
|
|
||||||
thumbnailUrl = el.content;
|
|
||||||
}
|
|
||||||
const { origin, pathname } = new URL(thumbnailUrl);
|
|
||||||
// remove the last part
|
|
||||||
return origin + pathname.split('/').slice(0, -1).join('/') + '/';
|
|
||||||
},
|
|
||||||
get msczIpfsRef() {
|
|
||||||
return `/ipns/${IPNS_KEY}/${this.idLastDigit}/${this.id}.mscz`;
|
|
||||||
},
|
|
||||||
get msczCidUrl() {
|
|
||||||
return `https://ipfs.infura.io:5001/api/v0/block/stat?arg=${this.msczIpfsRef}`;
|
|
||||||
},
|
|
||||||
get sheetImgType() {
|
|
||||||
try {
|
|
||||||
const imgE = document.querySelector('img[src*=score_]');
|
|
||||||
const { pathname } = new URL(imgE.src);
|
|
||||||
// eslint-disable-next-line @typescript-eslint/no-non-null-assertion
|
|
||||||
const imgtype = pathname.match(/\.(\w+)$/)[1];
|
|
||||||
return imgtype;
|
|
||||||
}
|
|
||||||
catch (_) {
|
|
||||||
// return null
|
|
||||||
return 'svg';
|
|
||||||
}
|
|
||||||
},
|
|
||||||
};
|
|
||||||
|
|
||||||
/* eslint-disable no-extend-native */
|
/* eslint-disable no-extend-native */
|
||||||
/* eslint-disable @typescript-eslint/ban-types */
|
/* eslint-disable @typescript-eslint/ban-types */
|
||||||
/**
|
/**
|
||||||
|
@ -26587,8 +26349,8 @@ Please pipe the document into a Node stream.\
|
||||||
midi: magicHookConstr('midi'),
|
midi: magicHookConstr('midi'),
|
||||||
mp3: magicHookConstr('mp3'),
|
mp3: magicHookConstr('mp3'),
|
||||||
};
|
};
|
||||||
const getApiUrl = (type, index) => {
|
const getApiUrl = (id, type, index) => {
|
||||||
return `/api/jmuse?id=${scoreinfo.id}&type=${type}&index=${index}&v2=1`;
|
return `/api/jmuse?id=${id}&type=${type}&index=${index}&v2=1`;
|
||||||
};
|
};
|
||||||
const getApiAuth = (type, index) => __awaiter(void 0, void 0, void 0, function* () {
|
const getApiAuth = (type, index) => __awaiter(void 0, void 0, void 0, function* () {
|
||||||
var _a;
|
var _a;
|
||||||
|
@ -26617,8 +26379,8 @@ Please pipe the document into a Node stream.\
|
||||||
}
|
}
|
||||||
return magic;
|
return magic;
|
||||||
});
|
});
|
||||||
const getFileUrl = (type, index = 0) => __awaiter(void 0, void 0, void 0, function* () {
|
const getFileUrl = (id, type, index = 0) => __awaiter(void 0, void 0, void 0, function* () {
|
||||||
const url = getApiUrl(type, index);
|
const url = getApiUrl(id, type, index);
|
||||||
const auth = yield getApiAuth(type);
|
const auth = yield getApiAuth(type);
|
||||||
const r = yield fetch(url, {
|
const r = yield fetch(url, {
|
||||||
headers: {
|
headers: {
|
||||||
|
@ -26632,7 +26394,7 @@ Please pipe the document into a Node stream.\
|
||||||
let pdfBlob;
|
let pdfBlob;
|
||||||
const _downloadPDF = (imgURLs, imgType, name = '') => __awaiter(void 0, void 0, void 0, function* () {
|
const _downloadPDF = (imgURLs, imgType, name = '') => __awaiter(void 0, void 0, void 0, function* () {
|
||||||
if (pdfBlob) {
|
if (pdfBlob) {
|
||||||
return saveAs(pdfBlob, `${name}.pdf`);
|
return FileSaver_min.saveAs(pdfBlob, `${name}.pdf`);
|
||||||
}
|
}
|
||||||
const cachedImg = document.querySelector('img[src*=score_]');
|
const cachedImg = document.querySelector('img[src*=score_]');
|
||||||
const { naturalWidth: width, naturalHeight: height } = cachedImg;
|
const { naturalWidth: width, naturalHeight: height } = cachedImg;
|
||||||
|
@ -26640,87 +26402,57 @@ Please pipe the document into a Node stream.\
|
||||||
const pdfArrayBuffer = yield worker.generatePDF(imgURLs, imgType, width, height);
|
const pdfArrayBuffer = yield worker.generatePDF(imgURLs, imgType, width, height);
|
||||||
worker.terminate();
|
worker.terminate();
|
||||||
pdfBlob = new Blob([pdfArrayBuffer]);
|
pdfBlob = new Blob([pdfArrayBuffer]);
|
||||||
saveAs(pdfBlob, `${name}.pdf`);
|
FileSaver_min.saveAs(pdfBlob, `${name}.pdf`);
|
||||||
});
|
});
|
||||||
const downloadPDF = () => __awaiter(void 0, void 0, void 0, function* () {
|
const downloadPDF = (scoreinfo, sheet) => __awaiter(void 0, void 0, void 0, function* () {
|
||||||
const imgType = scoreinfo.sheetImgType;
|
const imgType = sheet.imgType;
|
||||||
const pageCount = scoreinfo.pageCount;
|
const pageCount = sheet.pageCount;
|
||||||
const rs = Array.from({ length: pageCount }).map((_, i) => {
|
const rs = Array.from({ length: pageCount }).map((_, i) => {
|
||||||
if (i === 0) { // The url to the first page is static. We don't need to use API to obtain it.
|
if (i === 0) { // The url to the first page is static. We don't need to use API to obtain it.
|
||||||
return scoreinfo.baseUrl + `score_${i}.${imgType}`;
|
return sheet.thumbnailUrl;
|
||||||
}
|
}
|
||||||
else { // obtain image urls using the API
|
else { // obtain image urls using the API
|
||||||
return getFileUrl('img', i);
|
return getFileUrl(scoreinfo.id, 'img', i);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
const sheetImgURLs = yield Promise.all(rs);
|
const sheetImgURLs = yield Promise.all(rs);
|
||||||
return _downloadPDF(sheetImgURLs, imgType, scoreinfo.fileName);
|
return _downloadPDF(sheetImgURLs, imgType, scoreinfo.fileName);
|
||||||
});
|
});
|
||||||
|
|
||||||
let msczBufferP;
|
const MSCZ_BUF_SYM = Symbol('msczBufferP');
|
||||||
const fetchMscz = () => __awaiter(void 0, void 0, void 0, function* () {
|
const fetchMscz = (scoreinfo, _fetch = getFetch()) => __awaiter(void 0, void 0, void 0, function* () {
|
||||||
|
let msczBufferP = scoreinfo.store.get(MSCZ_BUF_SYM);
|
||||||
if (!msczBufferP) {
|
if (!msczBufferP) {
|
||||||
const url = scoreinfo.msczCidUrl;
|
const url = scoreinfo.msczCidUrl;
|
||||||
msczBufferP = (() => __awaiter(void 0, void 0, void 0, function* () {
|
msczBufferP = (() => __awaiter(void 0, void 0, void 0, function* () {
|
||||||
const r0 = yield fetch(url);
|
const r0 = yield _fetch(url);
|
||||||
|
// ipfs-http-gateway specific error
|
||||||
|
// may read further error msg as json
|
||||||
|
if (r0.status !== 500) {
|
||||||
assertRes(r0);
|
assertRes(r0);
|
||||||
const { Key } = yield r0.json();
|
}
|
||||||
const r = yield fetch(`https://ipfs.infura.io/ipfs/${Key}`);
|
const cidRes = yield r0.json();
|
||||||
|
const r = yield _fetch(scoreinfo.loadMsczUrl(cidRes));
|
||||||
assertRes(r);
|
assertRes(r);
|
||||||
const data = yield r.arrayBuffer();
|
const data = yield r.arrayBuffer();
|
||||||
return data;
|
return data;
|
||||||
}))();
|
}))();
|
||||||
|
scoreinfo.store.set(MSCZ_BUF_SYM, msczBufferP);
|
||||||
}
|
}
|
||||||
return msczBufferP;
|
return msczBufferP;
|
||||||
});
|
});
|
||||||
const downloadMscz = () => __awaiter(void 0, void 0, void 0, function* () {
|
const downloadMscz = (scoreinfo, saveAs) => __awaiter(void 0, void 0, void 0, function* () {
|
||||||
const data = new Blob([yield fetchMscz()]);
|
const data = new Blob([yield fetchMscz(scoreinfo)]);
|
||||||
const filename = scoreinfo.fileName;
|
const filename = scoreinfo.fileName;
|
||||||
saveAs(data, `${filename}.mscz`);
|
saveAs(data, `${filename}.mscz`);
|
||||||
});
|
});
|
||||||
|
|
||||||
const WEBMSCORE_URL = 'https://cdn.jsdelivr.net/npm/webmscore@0.10/webmscore.js';
|
/**
|
||||||
// fonts for Chinese characters (CN) and Korean hangul (KR)
|
* type checking only so no missing keys
|
||||||
// JP characters are included in the CN font
|
*/
|
||||||
const FONT_URLS = ['CN', 'KR'].map(l => `https://cdn.jsdelivr.net/npm/@librescore/fonts/SourceHanSans${l}-Regular.woff2`);
|
function createLocale(obj) {
|
||||||
const SF3_URL = 'https://cdn.jsdelivr.net/npm/@librescore/sf3/FluidR3Mono_GM.sf3';
|
return Object.freeze(obj);
|
||||||
const SOUND_FONT_LOADED = Symbol('SoundFont loaded');
|
|
||||||
const initMscore = (w) => __awaiter(void 0, void 0, void 0, function* () {
|
|
||||||
if (!w['WebMscore']) {
|
|
||||||
// init webmscore (https://github.com/LibreScore/webmscore)
|
|
||||||
const script = w.document.createElement('script');
|
|
||||||
script.src = WEBMSCORE_URL;
|
|
||||||
w.document.body.append(script);
|
|
||||||
yield new Promise(resolve => { script.onload = resolve; });
|
|
||||||
}
|
}
|
||||||
});
|
|
||||||
let fonts;
|
|
||||||
const initFonts = () => {
|
|
||||||
// load CJK fonts
|
|
||||||
// CJK (East Asian) characters will be rendered as "tofu" if there is no font
|
|
||||||
if (!fonts) {
|
|
||||||
fonts = Promise.all(FONT_URLS.map(url => fetchData(url)));
|
|
||||||
}
|
|
||||||
};
|
|
||||||
const loadSoundFont = (score) => {
|
|
||||||
if (!score[SOUND_FONT_LOADED]) {
|
|
||||||
const loadPromise = (() => __awaiter(void 0, void 0, void 0, function* () {
|
|
||||||
yield score.setSoundFont(yield fetchData(SF3_URL));
|
|
||||||
}))();
|
|
||||||
score[SOUND_FONT_LOADED] = loadPromise;
|
|
||||||
}
|
|
||||||
return score[SOUND_FONT_LOADED];
|
|
||||||
};
|
|
||||||
const loadMscore = (w) => __awaiter(void 0, void 0, void 0, function* () {
|
|
||||||
initFonts();
|
|
||||||
yield initMscore(w);
|
|
||||||
const WebMscore = w['WebMscore'];
|
|
||||||
// parse mscz data
|
|
||||||
const data = new Uint8Array(new Uint8Array(yield fetchMscz()));
|
|
||||||
const score = yield WebMscore.load('mscz', data, yield fonts);
|
|
||||||
yield score.generateExcerpts();
|
|
||||||
return score;
|
|
||||||
});
|
|
||||||
|
|
||||||
var en = createLocale({
|
var en = createLocale({
|
||||||
'PROCESSING'() {
|
'PROCESSING'() {
|
||||||
|
@ -26776,20 +26508,23 @@ Please pipe the document into a Node stream.\
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
/**
|
|
||||||
* type checking only so no missing keys
|
|
||||||
*/
|
|
||||||
function createLocale(obj) {
|
|
||||||
return Object.freeze(obj);
|
|
||||||
}
|
|
||||||
const locales = ((l) => Object.freeze(l))({
|
const locales = ((l) => Object.freeze(l))({
|
||||||
en,
|
en,
|
||||||
es,
|
es,
|
||||||
});
|
});
|
||||||
// detect browser language
|
// detect browser language
|
||||||
const lang = (() => {
|
const lang = (() => {
|
||||||
|
let userLangs;
|
||||||
|
if (!detectNode) {
|
||||||
|
userLangs = navigator.languages;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
const env = process.env;
|
||||||
|
const l = env.LC_ALL || env.LC_MESSAGES || env.LANG || env.LANGUAGE || '';
|
||||||
|
userLangs = [l.slice(0, 2)];
|
||||||
|
}
|
||||||
const names = Object.keys(locales);
|
const names = Object.keys(locales);
|
||||||
const _lang = navigator.languages.find(l => {
|
const _lang = userLangs.find(l => {
|
||||||
// find the first occurrence of valid languages
|
// find the first occurrence of valid languages
|
||||||
return names.includes(l);
|
return names.includes(l);
|
||||||
});
|
});
|
||||||
|
@ -26801,6 +26536,105 @@ Please pipe the document into a Node stream.\
|
||||||
return locale[key];
|
return locale[key];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* eslint-disable @typescript-eslint/no-var-requires */
|
||||||
|
const WEBMSCORE_URL = 'https://cdn.jsdelivr.net/npm/webmscore@0.10/webmscore.js';
|
||||||
|
// fonts for Chinese characters (CN) and Korean hangul (KR)
|
||||||
|
// JP characters are included in the CN font
|
||||||
|
const FONT_URLS = ['CN', 'KR'].map(l => `https://cdn.jsdelivr.net/npm/@librescore/fonts/SourceHanSans${l}-Regular.woff2`);
|
||||||
|
const SF3_URL = 'https://cdn.jsdelivr.net/npm/@librescore/sf3/FluidR3Mono_GM.sf3';
|
||||||
|
const SOUND_FONT_LOADED = Symbol('SoundFont loaded');
|
||||||
|
const initMscore = (w) => __awaiter(void 0, void 0, void 0, function* () {
|
||||||
|
if (!detectNode) { // attached to a page
|
||||||
|
if (!w['WebMscore']) {
|
||||||
|
// init webmscore (https://github.com/LibreScore/webmscore)
|
||||||
|
const script = w.document.createElement('script');
|
||||||
|
script.src = WEBMSCORE_URL;
|
||||||
|
w.document.body.append(script);
|
||||||
|
yield new Promise(resolve => { script.onload = resolve; });
|
||||||
|
}
|
||||||
|
return w['WebMscore'];
|
||||||
|
}
|
||||||
|
else { // nodejs
|
||||||
|
return require('webmscore').default;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
let fonts;
|
||||||
|
const initFonts = () => {
|
||||||
|
// load CJK fonts
|
||||||
|
// CJK (East Asian) characters will be rendered as "tofu" if there is no font
|
||||||
|
if (!fonts) {
|
||||||
|
if (detectNode) {
|
||||||
|
// module.exports.CN = ..., module.exports.KR = ...
|
||||||
|
const FONTS = Object.values(require('@librescore/fonts'));
|
||||||
|
const fs = require('fs');
|
||||||
|
fonts = Promise.all(FONTS.map((path) => fs.promises.readFile(path)));
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
fonts = Promise.all(FONT_URLS.map(url => fetchData(url)));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
const loadSoundFont = (score) => {
|
||||||
|
if (!score[SOUND_FONT_LOADED]) {
|
||||||
|
const loadPromise = (() => __awaiter(void 0, void 0, void 0, function* () {
|
||||||
|
let data;
|
||||||
|
if (detectNode) {
|
||||||
|
// module.exports.FluidR3Mono = ...
|
||||||
|
const SF3 = Object.values(require('@librescore/sf3'))[0];
|
||||||
|
const fs = require('fs');
|
||||||
|
data = yield fs.promises.readFile(SF3);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
data = yield fetchData(SF3_URL);
|
||||||
|
}
|
||||||
|
yield score.setSoundFont(data);
|
||||||
|
}))();
|
||||||
|
score[SOUND_FONT_LOADED] = loadPromise;
|
||||||
|
}
|
||||||
|
return score[SOUND_FONT_LOADED];
|
||||||
|
};
|
||||||
|
const loadMscore = (scoreinfo, w) => __awaiter(void 0, void 0, void 0, function* () {
|
||||||
|
initFonts();
|
||||||
|
const WebMscore = yield initMscore(w);
|
||||||
|
// parse mscz data
|
||||||
|
const data = new Uint8Array(new Uint8Array(yield fetchMscz(scoreinfo)));
|
||||||
|
const score = yield WebMscore.load('mscz', data, yield fonts);
|
||||||
|
yield score.generateExcerpts();
|
||||||
|
return score;
|
||||||
|
});
|
||||||
|
const INDV_DOWNLOADS = [
|
||||||
|
{
|
||||||
|
name: i18n('DOWNLOAD')('PDF'),
|
||||||
|
fileExt: 'pdf',
|
||||||
|
action: (score) => score.savePdf(),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: i18n('DOWNLOAD')('MSCZ'),
|
||||||
|
fileExt: 'mscz',
|
||||||
|
action: (score) => score.saveMsc('mscz'),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: i18n('DOWNLOAD')('MusicXML'),
|
||||||
|
fileExt: 'mxl',
|
||||||
|
action: (score) => score.saveMxl(),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: i18n('DOWNLOAD')('MIDI'),
|
||||||
|
fileExt: 'mid',
|
||||||
|
action: (score) => score.saveMidi(true, true),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: i18n('DOWNLOAD_AUDIO')('FLAC'),
|
||||||
|
fileExt: 'flac',
|
||||||
|
action: (score) => loadSoundFont(score).then(() => score.saveAudio('flac')),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: i18n('DOWNLOAD_AUDIO')('OGG'),
|
||||||
|
fileExt: 'ogg',
|
||||||
|
action: (score) => loadSoundFont(score).then(() => score.saveAudio('ogg')),
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
var btnListCss = "div {\n flex-wrap: wrap;\n display: flex;\n align-items: center;\n font-family: 'Open Sans', 'Roboto', 'Helvetica neue', Helvetica, sans-serif;\n position: absolute;\n z-index: 999;\n background: #f6f6f6;\n}\n\nbutton {\n width: 205px !important;\n height: 38px;\n\n color: #fff;\n background: #1f74bd;\n\n cursor: pointer;\n\n margin-bottom: 4px;\n margin-right: 4px;\n padding: 4px 12px;\n\n justify-content: start;\n align-self: center;\n\n font-size: 16px;\n border-radius: 2px;\n border: 0;\n\n display: inline-flex;\n position: relative;\n\n font-family: inherit;\n}\n\nsvg {\n display: inline-block;\n margin-right: 5px;\n width: 20px;\n height: 20px;\n margin-top: auto;\n margin-bottom: auto;\n}\n\nspan {\n margin-top: auto;\n margin-bottom: auto;\n}";
|
var btnListCss = "div {\n flex-wrap: wrap;\n display: flex;\n align-items: center;\n font-family: 'Open Sans', 'Roboto', 'Helvetica neue', Helvetica, sans-serif;\n position: absolute;\n z-index: 999;\n background: #f6f6f6;\n}\n\nbutton {\n width: 205px !important;\n height: 38px;\n\n color: #fff;\n background: #1f74bd;\n\n cursor: pointer;\n\n margin-bottom: 4px;\n margin-right: 4px;\n padding: 4px 12px;\n\n justify-content: start;\n align-self: center;\n\n font-size: 16px;\n border-radius: 2px;\n border: 0;\n\n display: inline-flex;\n position: relative;\n\n font-family: inherit;\n}\n\nsvg {\n display: inline-block;\n margin-right: 5px;\n width: 20px;\n height: 20px;\n margin-top: auto;\n margin-bottom: auto;\n}\n\nspan {\n margin-top: auto;\n margin-bottom: auto;\n}";
|
||||||
|
|
||||||
const getBtnContainer = () => {
|
const getBtnContainer = () => {
|
||||||
|
@ -26957,7 +26791,7 @@ Please pipe the document into a Node stream.\
|
||||||
a.dispatchEvent(new MouseEvent('click'));
|
a.dispatchEvent(new MouseEvent('click'));
|
||||||
}), fallback, timeout);
|
}), fallback, timeout);
|
||||||
};
|
};
|
||||||
BtnAction.mscoreWindow = (fn) => {
|
BtnAction.mscoreWindow = (scoreinfo, fn) => {
|
||||||
return (btnName, btn, setText) => __awaiter(this, void 0, void 0, function* () {
|
return (btnName, btn, setText) => __awaiter(this, void 0, void 0, function* () {
|
||||||
const _onclick = btn.onclick;
|
const _onclick = btn.onclick;
|
||||||
btn.onclick = null;
|
btn.onclick = null;
|
||||||
|
@ -26979,7 +26813,7 @@ Please pipe the document into a Node stream.\
|
||||||
setText(btnName);
|
setText(btnName);
|
||||||
btn.onclick = _onclick;
|
btn.onclick = _onclick;
|
||||||
});
|
});
|
||||||
score = yield loadMscore(w);
|
score = yield loadMscore(scoreinfo, w);
|
||||||
fn(w, score, txt);
|
fn(w, score, txt);
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
@ -27015,9 +26849,83 @@ Please pipe the document into a Node stream.\
|
||||||
};
|
};
|
||||||
})(BtnAction || (BtnAction = {}));
|
})(BtnAction || (BtnAction = {}));
|
||||||
|
|
||||||
|
class ScoreInfo {
|
||||||
|
constructor() {
|
||||||
|
this.IPNS_KEY = 'QmSdXtvzC8v8iTTZuj5cVmiugnzbR1QATYRcGix4bBsioP';
|
||||||
|
this.RADIX = 20;
|
||||||
|
this.store = new Map();
|
||||||
|
}
|
||||||
|
get idLastDigit() {
|
||||||
|
return (+this.id) % this.RADIX;
|
||||||
|
}
|
||||||
|
get fileName() {
|
||||||
|
return escapeFilename(this.title);
|
||||||
|
}
|
||||||
|
get msczIpfsRef() {
|
||||||
|
return `/ipns/${this.IPNS_KEY}/${this.idLastDigit}/${this.id}.mscz`;
|
||||||
|
}
|
||||||
|
get msczCidUrl() {
|
||||||
|
return `https://ipfs.infura.io:5001/api/v0/block/stat?arg=${this.msczIpfsRef}`;
|
||||||
|
}
|
||||||
|
loadMsczUrl(cidRes) {
|
||||||
|
const cid = cidRes.Key;
|
||||||
|
if (!cid) {
|
||||||
|
// read further error msg
|
||||||
|
const err = cidRes.Message;
|
||||||
|
if (err.includes('no link named')) { // file not found
|
||||||
|
throw new Error('score not in dataset');
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
throw new Error(err);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
this.msczUrl = `https://ipfs.infura.io/ipfs/${cid}`;
|
||||||
|
return this.msczUrl;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
class ScoreInfoInPage extends ScoreInfo {
|
||||||
|
constructor(document) {
|
||||||
|
super();
|
||||||
|
this.document = document;
|
||||||
|
}
|
||||||
|
get id() {
|
||||||
|
const el = this.document.querySelector("meta[property='al:ios:url']");
|
||||||
|
const m = el.content.match(/(\d+)$/);
|
||||||
|
return +m[1];
|
||||||
|
}
|
||||||
|
get title() {
|
||||||
|
const el = this.document.querySelector("meta[property='og:title']");
|
||||||
|
return el.content;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
class SheetInfo {
|
||||||
|
get imgType() {
|
||||||
|
const thumbnail = this.thumbnailUrl;
|
||||||
|
const imgtype = thumbnail.match(/\.(\w+)$/)[1];
|
||||||
|
return imgtype;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
class SheetInfoInPage extends SheetInfo {
|
||||||
|
constructor(document) {
|
||||||
|
super();
|
||||||
|
this.document = document;
|
||||||
|
}
|
||||||
|
get pageCount() {
|
||||||
|
return this.document.querySelectorAll('.gXB83').length;
|
||||||
|
}
|
||||||
|
get thumbnailUrl() {
|
||||||
|
// url to the image of the first page
|
||||||
|
const el = this.document.querySelector('link[as=image]');
|
||||||
|
const url = el.href;
|
||||||
|
return url.split('@')[0];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const { saveAs } = FileSaver_min;
|
||||||
const main = () => {
|
const main = () => {
|
||||||
const btnList = new BtnList();
|
const btnList = new BtnList();
|
||||||
const filename = scoreinfo.fileName;
|
const scoreinfo = new ScoreInfoInPage(document);
|
||||||
|
const { fileName, id } = scoreinfo;
|
||||||
let indvPartBtn = null;
|
let indvPartBtn = null;
|
||||||
const fallback = () => {
|
const fallback = () => {
|
||||||
// btns fallback to load from MSCZ file (`Individual Parts`)
|
// btns fallback to load from MSCZ file (`Individual Parts`)
|
||||||
|
@ -27025,33 +26933,33 @@ Please pipe the document into a Node stream.\
|
||||||
};
|
};
|
||||||
btnList.add({
|
btnList.add({
|
||||||
name: i18n('DOWNLOAD')('MSCZ'),
|
name: i18n('DOWNLOAD')('MSCZ'),
|
||||||
action: BtnAction.process(downloadMscz),
|
action: BtnAction.process(() => downloadMscz(scoreinfo, saveAs)),
|
||||||
});
|
});
|
||||||
btnList.add({
|
btnList.add({
|
||||||
name: i18n('DOWNLOAD')('PDF'),
|
name: i18n('DOWNLOAD')('PDF'),
|
||||||
action: BtnAction.process(downloadPDF, fallback, 3 * 60 * 1000 /* 3min */),
|
action: BtnAction.process(() => downloadPDF(scoreinfo, new SheetInfoInPage(document)), fallback, 3 * 60 * 1000 /* 3min */),
|
||||||
});
|
});
|
||||||
btnList.add({
|
btnList.add({
|
||||||
name: i18n('DOWNLOAD')('MusicXML'),
|
name: i18n('DOWNLOAD')('MusicXML'),
|
||||||
action: BtnAction.mscoreWindow((w, score) => __awaiter(void 0, void 0, void 0, function* () {
|
action: BtnAction.mscoreWindow(scoreinfo, (w, score) => __awaiter(void 0, void 0, void 0, function* () {
|
||||||
const mxl = yield score.saveMxl();
|
const mxl = yield score.saveMxl();
|
||||||
const data = new Blob([mxl]);
|
const data = new Blob([mxl]);
|
||||||
saveAs(data, `${filename}.mxl`);
|
saveAs(data, `${fileName}.mxl`);
|
||||||
w.close();
|
w.close();
|
||||||
})),
|
})),
|
||||||
});
|
});
|
||||||
btnList.add({
|
btnList.add({
|
||||||
name: i18n('DOWNLOAD')('MIDI'),
|
name: i18n('DOWNLOAD')('MIDI'),
|
||||||
action: BtnAction.download(() => getFileUrl('midi'), fallback, 30 * 1000 /* 30s */),
|
action: BtnAction.download(() => getFileUrl(id, 'midi'), fallback, 30 * 1000 /* 30s */),
|
||||||
});
|
});
|
||||||
btnList.add({
|
btnList.add({
|
||||||
name: i18n('DOWNLOAD')('MP3'),
|
name: i18n('DOWNLOAD')('MP3'),
|
||||||
action: BtnAction.download(() => getFileUrl('mp3'), fallback, 30 * 1000 /* 30s */),
|
action: BtnAction.download(() => getFileUrl(id, 'mp3'), fallback, 30 * 1000 /* 30s */),
|
||||||
});
|
});
|
||||||
indvPartBtn = btnList.add({
|
indvPartBtn = btnList.add({
|
||||||
name: i18n('IND_PARTS')(),
|
name: i18n('IND_PARTS')(),
|
||||||
tooltip: i18n('IND_PARTS_TOOLTIP')(),
|
tooltip: i18n('IND_PARTS_TOOLTIP')(),
|
||||||
action: BtnAction.mscoreWindow((w, score, txt) => __awaiter(void 0, void 0, void 0, function* () {
|
action: BtnAction.mscoreWindow(scoreinfo, (w, score, txt) => __awaiter(void 0, void 0, void 0, function* () {
|
||||||
const metadata = yield score.metadata();
|
const metadata = yield score.metadata();
|
||||||
console$1.log('score metadata loaded by webmscore', metadata);
|
console$1.log('score metadata loaded by webmscore', metadata);
|
||||||
// add the "full score" option as a "part"
|
// add the "full score" option as a "part"
|
||||||
|
@ -27060,38 +26968,7 @@ Please pipe the document into a Node stream.\
|
||||||
txt.remove();
|
txt.remove();
|
||||||
const fieldset = w.document.createElement('fieldset');
|
const fieldset = w.document.createElement('fieldset');
|
||||||
w.document.body.append(fieldset);
|
w.document.body.append(fieldset);
|
||||||
const downloads = [
|
const downloads = INDV_DOWNLOADS;
|
||||||
{
|
|
||||||
name: i18n('DOWNLOAD')('PDF'),
|
|
||||||
fileExt: 'pdf',
|
|
||||||
action: (score) => score.savePdf(),
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: i18n('DOWNLOAD')('MSCZ'),
|
|
||||||
fileExt: 'mscz',
|
|
||||||
action: (score) => score.saveMsc('mscz'),
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: i18n('DOWNLOAD')('MusicXML'),
|
|
||||||
fileExt: 'mxl',
|
|
||||||
action: (score) => score.saveMxl(),
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: i18n('DOWNLOAD')('MIDI'),
|
|
||||||
fileExt: 'mid',
|
|
||||||
action: (score) => score.saveMidi(true, true),
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: i18n('DOWNLOAD_AUDIO')('FLAC'),
|
|
||||||
fileExt: 'flac',
|
|
||||||
action: (score) => loadSoundFont(score).then(() => score.saveAudio('flac')),
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: i18n('DOWNLOAD_AUDIO')('OGG'),
|
|
||||||
fileExt: 'ogg',
|
|
||||||
action: (score) => loadSoundFont(score).then(() => score.saveAudio('ogg')),
|
|
||||||
},
|
|
||||||
];
|
|
||||||
// part selection
|
// part selection
|
||||||
const DEFAULT_PART = -1; // initially select "full score"
|
const DEFAULT_PART = -1; // initially select "full score"
|
||||||
for (const excerpt of metadata.excerpts) {
|
for (const excerpt of metadata.excerpts) {
|
||||||
|
@ -27130,7 +27007,7 @@ Please pipe the document into a Node stream.\
|
||||||
const checked = fieldset.querySelector('input:checked');
|
const checked = fieldset.querySelector('input:checked');
|
||||||
const partName = checked.alt;
|
const partName = checked.alt;
|
||||||
const data = new Blob([yield d.action(score)]);
|
const data = new Blob([yield d.action(score)]);
|
||||||
saveAs(data, `${filename} - ${partName}.${d.fileExt}`);
|
saveAs(data, `${fileName} - ${partName}.${d.fileExt}`);
|
||||||
// unlock button
|
// unlock button
|
||||||
initBtn();
|
initBtn();
|
||||||
});
|
});
|
||||||
|
|
2
package-lock.json
generated
2
package-lock.json
generated
|
@ -1,6 +1,6 @@
|
||||||
{
|
{
|
||||||
"name": "musescore-downloader",
|
"name": "musescore-downloader",
|
||||||
"version": "0.15.20",
|
"version": "0.16.0",
|
||||||
"lockfileVersion": 1,
|
"lockfileVersion": 1,
|
||||||
"requires": true,
|
"requires": true,
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
{
|
{
|
||||||
"name": "musescore-downloader",
|
"name": "musescore-downloader",
|
||||||
"version": "0.15.20",
|
"version": "0.16.0",
|
||||||
"description": "download sheet music from musescore.com for free, no login or Musescore Pro required | 免登录、免 Musescore Pro,免费下载 musescore.com 上的曲谱",
|
"description": "download sheet music from musescore.com for free, no login or Musescore Pro required | 免登录、免 Musescore Pro,免费下载 musescore.com 上的曲谱",
|
||||||
"main": "dist/main.js",
|
"main": "dist/main.js",
|
||||||
"bin": "dist/cli.js",
|
"bin": "dist/cli.js",
|
||||||
|
|
Loading…
Reference in a new issue