import FileSaver from 'file-saver/dist/FileSaver.js' export const saveAs: typeof import('file-saver').saveAs = FileSaver.saveAs export const getIndexPath = (id: number): string => { const idStr = String(id) // 获取最后三位,倒序排列 // x, y, z are the reversed last digits of the score id. Example: id 123456789, x = 9, y = 8, z = 7 // https://developers.musescore.com/#/file-urls // "5449062" -> ["2", "6", "0"] const indexN = idStr.split('').reverse().slice(0, 3) return indexN.join('/') } export const fetchData = async (url: string, init?: RequestInit): Promise => { const r = await fetch(url, init) const data = await r.arrayBuffer() return new Uint8Array(data) } export const useTimeout = async (promise: T | Promise, ms: number): Promise => { if (!(promise instanceof Promise)) { return promise } return new Promise((resolve, reject) => { const i = setTimeout(() => { reject(new Error('timeout')) }, ms) promise.then(resolve, reject).finally(() => clearTimeout(i)) }) } export const getSandboxWindow = (): Window => { const iframe = document.createElement('iframe') iframe.style.display = 'none' document.body.append(iframe) const w = iframe.contentWindow return w as Window } export const windowOpen: Window['open'] = (...args): Window | null => { return getSandboxWindow().open(...args) } export const waitForDocumentLoaded = (): Promise => { if (document.readyState !== 'complete') { return new Promise(resolve => { const cb = () => { if (document.readyState === 'complete') { resolve() document.removeEventListener('readystatechange', cb) } } document.addEventListener('readystatechange', cb) }) } else { return Promise.resolve() } }