// ==UserScript==
// @name 听歌小助手
// @namespace https://github.com/fan0530
// @version 1.1.4
// @author fanxq
// @description 这个脚本主要为Hifini(音乐磁场)网站提供了一个歌曲管理页面(可添加、删除、列表播放歌曲等)以及歌曲下载功能
// @icon https://cdn.jsdelivr.net/gh/fan0530/music-player/favicon.ico
// @match https://hifini.com/*
// @match https://www.hifini.com/*
// @require https://unpkg.com/vue@3.4.19/dist/vue.global.prod.js
// @require data:application/javascript,window.Vue%3DVue%3B
// @require https://unpkg.com/element-plus@2.5.6/dist/index.full.min.js
// @require https://unpkg.com/idb-keyval@6.2.1/dist/umd.js
// @resource element-plus/dist/index.css https://unpkg.com/element-plus@2.5.6/dist/index.css
// @resource player.html https://cdn.jsdelivr.net/gh/fan0530/music-player@main/index.v2024122201.html
// @connect hifini.com
// @connect gitee.com
// @connect kuwo.cn
// @connect 126.net
// @connect qq.com
// @grant GM_addStyle
// @grant GM_getResourceText
// @grant GM_xmlhttpRequest
// @grant unsafeWindow
// @grant window.focus
// @run-at document-end
// ==/UserScript==
(e=>{if(typeof GM_addStyle=="function"){GM_addStyle(e);return}const t=document.createElement("style");t.textContent=e,document.head.append(t)})(" .my-help-dialog{--el-dialog-width: 92% !important;max-width:none;min-width:none}.my-help-dialog .img-wrapper{margin:10px 0;padding:20px;background:#f7f7f7}@media screen and (min-width: 500px){.my-help-dialog{--el-dialog-width: 55% !important;max-width:800px;min-width:500px}}.loading-toast{--el-dialog-width: 140px !important;--el-dialog-border-radius: 10px !important}.loading-toast .el-dialog__header{display:none}.btn-container[data-v-0a2e6f7e]{position:fixed;bottom:100px;right:20px}.menu-popper .menu-list[data-v-0a2e6f7e]{list-style:none;margin:0;padding:0}.menu-popper .menu-list li[data-v-0a2e6f7e]{display:flex;flex-direction:column}.menu-popper .menu-list li .menu-item[data-v-0a2e6f7e]{display:flex;align-items:center;justify-content:center;color:#666}.menu-popper .menu-list li .menu-item[data-v-0a2e6f7e]:active{background-color:#80808033} ");
(function (vue, elementPlus, idbKeyval) {
'use strict';
var _GM_getResourceText = /* @__PURE__ */ (() => typeof GM_getResourceText != "undefined" ? GM_getResourceText : void 0)();
var _GM_xmlhttpRequest = /* @__PURE__ */ (() => typeof GM_xmlhttpRequest != "undefined" ? GM_xmlhttpRequest : void 0)();
var _unsafeWindow = /* @__PURE__ */ (() => typeof unsafeWindow != "undefined" ? unsafeWindow : void 0)();
var DEFAULT_ICON_CONFIGS = {
size: "1em",
strokeWidth: 4,
strokeLinecap: "round",
strokeLinejoin: "round",
rtl: false,
theme: "outline",
colors: {
outline: {
fill: "#333",
background: "transparent"
},
filled: {
fill: "#333",
background: "#FFF"
},
twoTone: {
fill: "#333",
twoTone: "#2F88FF"
},
multiColor: {
outStrokeColor: "#333",
outFillColor: "#2F88FF",
innerStrokeColor: "#FFF",
innerFillColor: "#43CCF8"
}
},
prefix: "i"
};
function guid() {
return "icon-" + ((1 + Math.random()) * 4294967296 | 0).toString(16).substring(1);
}
function IconConverter(id, icon, config) {
var fill = typeof icon.fill === "string" ? [icon.fill] : icon.fill || [];
var colors = [];
var theme = icon.theme || config.theme;
switch (theme) {
case "outline":
colors.push(typeof fill[0] === "string" ? fill[0] : "currentColor");
colors.push("none");
colors.push(typeof fill[0] === "string" ? fill[0] : "currentColor");
colors.push("none");
break;
case "filled":
colors.push(typeof fill[0] === "string" ? fill[0] : "currentColor");
colors.push(typeof fill[0] === "string" ? fill[0] : "currentColor");
colors.push("#FFF");
colors.push("#FFF");
break;
case "two-tone":
colors.push(typeof fill[0] === "string" ? fill[0] : "currentColor");
colors.push(typeof fill[1] === "string" ? fill[1] : config.colors.twoTone.twoTone);
colors.push(typeof fill[0] === "string" ? fill[0] : "currentColor");
colors.push(typeof fill[1] === "string" ? fill[1] : config.colors.twoTone.twoTone);
break;
case "multi-color":
colors.push(typeof fill[0] === "string" ? fill[0] : "currentColor");
colors.push(typeof fill[1] === "string" ? fill[1] : config.colors.multiColor.outFillColor);
colors.push(typeof fill[2] === "string" ? fill[2] : config.colors.multiColor.innerStrokeColor);
colors.push(typeof fill[3] === "string" ? fill[3] : config.colors.multiColor.innerFillColor);
break;
}
return {
size: icon.size || config.size,
strokeWidth: icon.strokeWidth || config.strokeWidth,
strokeLinecap: icon.strokeLinecap || config.strokeLinecap,
strokeLinejoin: icon.strokeLinejoin || config.strokeLinejoin,
colors,
id
};
}
var IconContext = Symbol("icon-context");
function IconWrapper(name, rtl, render) {
var options = {
name: "icon-" + name,
props: ["size", "strokeWidth", "strokeLinecap", "strokeLinejoin", "theme", "fill", "spin"],
setup: function setup(props) {
var id = guid();
var ICON_CONFIGS = vue.inject(IconContext, DEFAULT_ICON_CONFIGS);
return function() {
var size = props.size, strokeWidth = props.strokeWidth, strokeLinecap = props.strokeLinecap, strokeLinejoin = props.strokeLinejoin, theme = props.theme, fill = props.fill, spin = props.spin;
var svgProps = IconConverter(id, {
size,
strokeWidth,
strokeLinecap,
strokeLinejoin,
theme,
fill
}, ICON_CONFIGS);
var cls = [ICON_CONFIGS.prefix + "-icon"];
cls.push(ICON_CONFIGS.prefix + "-icon-" + name);
if (rtl && ICON_CONFIGS.rtl) {
cls.push(ICON_CONFIGS.prefix + "-icon-rtl");
}
if (spin) {
cls.push(ICON_CONFIGS.prefix + "-icon-spin");
}
return vue.createVNode("span", {
"class": cls.join(" ")
}, [render(svgProps)]);
};
}
};
return options;
}
const AddMusic = IconWrapper("add-music", true, function(props) {
return vue.createVNode("svg", {
"width": props.size,
"height": props.size,
"viewBox": "0 0 48 48",
"fill": "none"
}, [vue.createVNode("path", {
"d": "M24 44C12.9543 44 4 35.0457 4 24C4 12.9543 12.9543 4 24 4C35.0457 4 44 12.9543 44 24",
"stroke": props.colors[0],
"stroke-width": props.strokeWidth,
"stroke-linecap": props.strokeLinecap,
"stroke-linejoin": props.strokeLinejoin
}, null), vue.createVNode("path", {
"d": "M20 24V17.0718L26 20.5359L32 24L26 27.4641L20 30.9282V24Z",
"fill": props.colors[1],
"stroke": props.colors[0],
"stroke-width": props.strokeWidth,
"stroke-linejoin": props.strokeLinejoin
}, null), vue.createVNode("path", {
"d": "M37.0508 32L37.0508 42",
"stroke": props.colors[0],
"stroke-width": props.strokeWidth,
"stroke-linecap": props.strokeLinecap,
"stroke-linejoin": props.strokeLinejoin
}, null), vue.createVNode("path", {
"d": "M42 36.9497L32 36.9497",
"stroke": props.colors[0],
"stroke-width": props.strokeWidth,
"stroke-linecap": props.strokeLinecap,
"stroke-linejoin": props.strokeLinejoin
}, null)]);
});
const Down = IconWrapper("down", false, function(props) {
return vue.createVNode("svg", {
"width": props.size,
"height": props.size,
"viewBox": "0 0 48 48",
"fill": "none"
}, [vue.createVNode("path", {
"d": "M36 18L24 30L12 18",
"stroke": props.colors[0],
"stroke-width": props.strokeWidth,
"stroke-linecap": props.strokeLinecap,
"stroke-linejoin": props.strokeLinejoin
}, null)]);
});
const DownloadOne = IconWrapper("download-one", true, function(props) {
return vue.createVNode("svg", {
"width": props.size,
"height": props.size,
"viewBox": "0 0 48 48",
"fill": "none"
}, [vue.createVNode("path", {
"d": "M11.6777 20.271C7.27476 21.3181 4 25.2766 4 30C4 35.5228 8.47715 40 14 40C14.9474 40 15.864 39.8683 16.7325 39.6221",
"stroke": props.colors[0],
"stroke-width": props.strokeWidth,
"stroke-linecap": props.strokeLinecap,
"stroke-linejoin": props.strokeLinejoin
}, null), vue.createVNode("path", {
"d": "M36.0547 20.271C40.4577 21.3181 43.7324 25.2766 43.7324 30C43.7324 35.5228 39.2553 40 33.7324 40C32.785 40 31.8684 39.8683 30.9999 39.6221",
"stroke": props.colors[0],
"stroke-width": props.strokeWidth,
"stroke-linecap": props.strokeLinecap,
"stroke-linejoin": props.strokeLinejoin
}, null), vue.createVNode("path", {
"d": "M36 20C36 13.3726 30.6274 8 24 8C17.3726 8 12 13.3726 12 20",
"stroke": props.colors[0],
"stroke-width": props.strokeWidth,
"stroke-linecap": props.strokeLinecap,
"stroke-linejoin": props.strokeLinejoin
}, null), vue.createVNode("path", {
"d": "M17.0654 30.119L23.9999 37.0764L31.1318 30",
"stroke": props.colors[0],
"stroke-width": props.strokeWidth,
"stroke-linecap": props.strokeLinecap,
"stroke-linejoin": props.strokeLinejoin
}, null), vue.createVNode("path", {
"d": "M24 20V33.5382",
"stroke": props.colors[0],
"stroke-width": props.strokeWidth,
"stroke-linecap": props.strokeLinecap,
"stroke-linejoin": props.strokeLinejoin
}, null)]);
});
const Help = IconWrapper("help", true, function(props) {
return vue.createVNode("svg", {
"width": props.size,
"height": props.size,
"viewBox": "0 0 48 48",
"fill": "none"
}, [vue.createVNode("path", {
"d": "M24 44C29.5228 44 34.5228 41.7614 38.1421 38.1421C41.7614 34.5228 44 29.5228 44 24C44 18.4772 41.7614 13.4772 38.1421 9.85786C34.5228 6.23858 29.5228 4 24 4C18.4772 4 13.4772 6.23858 9.85786 9.85786C6.23858 13.4772 4 18.4772 4 24C4 29.5228 6.23858 34.5228 9.85786 38.1421C13.4772 41.7614 18.4772 44 24 44Z",
"fill": props.colors[1],
"stroke": props.colors[0],
"stroke-width": props.strokeWidth,
"stroke-linejoin": props.strokeLinejoin
}, null), vue.createVNode("path", {
"d": "M24 28.6248V24.6248C27.3137 24.6248 30 21.9385 30 18.6248C30 15.3111 27.3137 12.6248 24 12.6248C20.6863 12.6248 18 15.3111 18 18.6248",
"stroke": props.colors[2],
"stroke-width": props.strokeWidth,
"stroke-linecap": props.strokeLinecap,
"stroke-linejoin": props.strokeLinejoin
}, null), vue.createVNode("path", {
"fill-rule": "evenodd",
"clip-rule": "evenodd",
"d": "M24 37.6248C25.3807 37.6248 26.5 36.5055 26.5 35.1248C26.5 33.7441 25.3807 32.6248 24 32.6248C22.6193 32.6248 21.5 33.7441 21.5 35.1248C21.5 36.5055 22.6193 37.6248 24 37.6248Z",
"fill": props.colors[2]
}, null)]);
});
const Record = IconWrapper("record", true, function(props) {
return vue.createVNode("svg", {
"width": props.size,
"height": props.size,
"viewBox": "0 0 48 48",
"fill": "none"
}, [vue.createVNode("rect", {
"x": "5",
"y": "18",
"width": "38",
"height": "24",
"rx": "2",
"fill": props.colors[1],
"stroke": props.colors[0],
"stroke-width": props.strokeWidth,
"stroke-linecap": props.strokeLinecap,
"stroke-linejoin": props.strokeLinejoin
}, null), vue.createVNode("path", {
"d": "M8 12H40",
"stroke": props.colors[0],
"stroke-width": props.strokeWidth,
"stroke-linecap": props.strokeLinecap,
"stroke-linejoin": props.strokeLinejoin
}, null), vue.createVNode("path", {
"d": "M15 6L33 6",
"stroke": props.colors[0],
"stroke-width": props.strokeWidth,
"stroke-linecap": props.strokeLinecap,
"stroke-linejoin": props.strokeLinejoin
}, null), vue.createVNode("path", {
"d": "M26 24V30",
"stroke": props.colors[2],
"stroke-width": props.strokeWidth,
"stroke-linecap": props.strokeLinecap,
"stroke-linejoin": props.strokeLinejoin
}, null), vue.createVNode("path", {
"d": "M18 32.7491C18 31.2308 19.2894 30 20.88 30H26V33.2509C26 34.7692 24.7106 36 23.12 36H20.88C19.2894 36 18 34.7692 18 33.2509V32.7491Z",
"stroke": props.colors[2],
"stroke-width": props.strokeWidth,
"stroke-linejoin": props.strokeLinejoin
}, null), vue.createVNode("path", {
"d": "M31 25L26 24",
"stroke": props.colors[2],
"stroke-width": props.strokeWidth,
"stroke-linecap": props.strokeLinecap,
"stroke-linejoin": props.strokeLinejoin
}, null)]);
});
const RecordPlayer = IconWrapper("record-player", true, function(props) {
return vue.createVNode("svg", {
"width": props.size,
"height": props.size,
"viewBox": "0 0 48 48",
"fill": "none"
}, [vue.createVNode("rect", {
"x": "5",
"y": "8",
"width": "38",
"height": "32",
"rx": "2",
"stroke": props.colors[0],
"stroke-width": props.strokeWidth
}, null), vue.createVNode("path", {
"d": "M13 8V40",
"stroke": props.colors[0],
"stroke-width": props.strokeWidth,
"stroke-linecap": props.strokeLinecap,
"stroke-linejoin": props.strokeLinejoin
}, null), vue.createVNode("circle", {
"cx": "28",
"cy": "24",
"r": "9",
"fill": props.colors[1],
"stroke": props.colors[0],
"stroke-width": props.strokeWidth
}, null), vue.createVNode("circle", {
"cx": "28",
"cy": "24",
"r": "3",
"fill": props.colors[2]
}, null), vue.createVNode("path", {
"d": "M5 16H13",
"stroke": props.colors[0],
"stroke-width": props.strokeWidth,
"stroke-linecap": props.strokeLinecap,
"stroke-linejoin": props.strokeLinejoin
}, null), vue.createVNode("path", {
"d": "M5 24H13",
"stroke": props.colors[0],
"stroke-width": props.strokeWidth,
"stroke-linecap": props.strokeLinecap,
"stroke-linejoin": props.strokeLinejoin
}, null), vue.createVNode("path", {
"d": "M5 32H13",
"stroke": props.colors[0],
"stroke-width": props.strokeWidth,
"stroke-linecap": props.strokeLinecap,
"stroke-linejoin": props.strokeLinejoin
}, null)]);
});
const _hoisted_1$2 = ["id"];
const _hoisted_2$1 = /* @__PURE__ */ vue.createElementVNode("p", null, [
/* @__PURE__ */ vue.createTextVNode("具体使用说明请查看这篇文章: "),
/* @__PURE__ */ vue.createElementVNode("a", {
target: "_blank",
href: "https://mp.weixin.qq.com/s?__biz=Mzg2NDgzMjU1OA==&mid=2247483945&idx=1&sn=cdf6194c875eeeb143c51a7a9cf8f520&chksm=ce621f68f915967e712d03de99b81e6f64f3e28ca4b970815e78d388b4ad76d04d6fd40e69e2#rd"
}, " 有用这颗“黑凤梨”听歌的朋友吗?我写了个油猴脚本送给你 ")
], -1);
const _hoisted_3$1 = /* @__PURE__ */ vue.createElementVNode("p", null, [
/* @__PURE__ */ vue.createTextVNode("如果在使用该脚本的过程遇到了问题,或者觉得有需要改善的地方,可以在 "),
/* @__PURE__ */ vue.createElementVNode("a", {
target: "_blank",
href: "https://greasyfork.org/zh-CN/scripts/497671-%E5%90%AC%E6%AD%8C%E5%B0%8F%E5%8A%A9%E6%89%8B/feedback"
}, " 此处(https://greasyfork.org/zh-CN/scripts/497671-%E5%90%AC%E6%AD%8C%E5%B0%8F%E5%8A%A9%E6%89%8B/feedback) "),
/* @__PURE__ */ vue.createTextVNode(" 反馈。 ")
], -1);
const _hoisted_4$1 = /* @__PURE__ */ vue.createElementVNode("p", null, "最后,如果这个脚本对你有点帮助的话,可以考虑请我喝瓶快乐水,你的支持将给我动力持续去维护好这个脚本。", -1);
const _hoisted_5$1 = /* @__PURE__ */ vue.createElementVNode("p", null, [
/* @__PURE__ */ vue.createElementVNode("img", {
src: "https://gitee.com/fanxiqian/music-player/raw/master/code.jpg",
alt: "赞赏码",
style: { "display": "block", "margin": "0 auto", "width": "100%", "max-width": "320px", "height": "auto" }
})
], -1);
const _sfc_main$2 = {
__name: "HelpDialog",
props: {
show: {
type: Boolean,
default: false
}
},
emits: ["update:show"],
setup(__props, { emit: __emit }) {
const emits = __emit;
const updateShow = (visible) => {
emits("update:show", visible);
};
return (_ctx, _cache) => {
return vue.openBlock(), vue.createBlock(vue.unref(elementPlus.ElDialog), {
"model-value": __props.show,
onClose: _cache[0] || (_cache[0] = ($event) => updateShow(false)),
class: "my-help-dialog",
"align-center": ""
}, {
header: vue.withCtx(({ titleId, titleClass }) => [
vue.createElementVNode("h4", {
id: titleId,
class: vue.normalizeClass(titleClass)
}, "使用说明", 10, _hoisted_1$2)
]),
default: vue.withCtx(() => [
vue.createVNode(vue.unref(elementPlus.ElScrollbar), { height: "70vh" }, {
default: vue.withCtx(() => [
_hoisted_2$1,
_hoisted_3$1,
_hoisted_4$1,
_hoisted_5$1
]),
_: 1
})
]),
_: 1
}, 8, ["model-value"]);
};
}
};
const _hoisted_1$1 = {
"element-loading-text": "正在打开...",
style: { "height": "108px" }
};
const _sfc_main$1 = {
__name: "LoadingToast",
props: {
show: {
type: Boolean,
default: false
}
},
emits: ["update:show"],
setup(__props, { emit: __emit }) {
const emits = __emit;
const updateShow = (visible) => {
emits("update:show", visible);
};
return (_ctx, _cache) => {
const _directive_loading = vue.resolveDirective("loading");
return vue.openBlock(), vue.createBlock(vue.unref(elementPlus.ElDialog), {
"model-value": __props.show,
onClose: _cache[0] || (_cache[0] = ($event) => updateShow(false)),
"align-center": "",
"show-close": false,
modal: false,
"close-on-click-modal": false,
"close-on-press-escape": false,
class: "loading-toast"
}, {
default: vue.withCtx(() => [
vue.withDirectives(vue.createElementVNode("div", _hoisted_1$1, null, 512), [
[_directive_loading, true]
])
]),
_: 1
}, 8, ["model-value"]);
};
}
};
const _export_sfc = (sfc, props) => {
const target = sfc.__vccOpts || sfc;
for (const [key, val] of props) {
target[key] = val;
}
return target;
};
const _withScopeId = (n) => (vue.pushScopeId("data-v-0a2e6f7e"), n = n(), vue.popScopeId(), n);
const _hoisted_1 = { style: { "position": "absolute", "top": "5px", "right": "5px" } };
const _hoisted_2 = { style: { "display": "flex", "align-items": "center", "justify-content": "center" } };
const _hoisted_3 = /* @__PURE__ */ _withScopeId(() => /* @__PURE__ */ vue.createElementVNode("span", { style: { "margin-left": "8px" } }, "添加至", -1));
const _hoisted_4 = { style: { "display": "flex", "align-items": "center" } };
const _hoisted_5 = { class: "btn-container" };
const _hoisted_6 = { class: "menu-list" };
const _hoisted_7 = ["onClick"];
const _hoisted_8 = { style: { "margin-left": "12px" } };
const _sfc_main = {
__name: "App",
setup(__props) {
const isPlayerExisted = vue.ref(false);
const showHelpDialog = vue.ref(false);
const isAdding = vue.ref(false);
const showLoading = vue.ref(false);
let targetWindow = null;
const msgHub = [];
const menus = vue.ref([
{
name: "usage",
title: "使用说明",
icon: vue.markRaw(Help),
handler: () => {
showHelpDialog.value = true;
}
},
{
name: "player",
title: "打开歌单",
icon: vue.markRaw(RecordPlayer),
handler: () => {
if (!showLoading.value) {
showLoading.value = true;
}
openPlayer();
}
}
]);
const requestAudioBlobData = (url) => {
return new Promise((resolve, reject) => {
_GM_xmlhttpRequest({
method: "GET",
url,
headers: {
"referer": "https://hifini.com/"
},
responseType: "blob",
onload: function(res) {
if (!(res == null ? void 0 : res.response)) {
resolve(null);
}
resolve(res.response);
},
onerror: function(err) {
resolve(null);
}
});
});
};
const getAudioData = async () => {
var _a;
let audioItem = null;
const targetScript = Array.from(_unsafeWindow.document.querySelectorAll("script")).filter((x) => x.innerHTML).find((x) => x.innerHTML.indexOf("APlayer") !== -1);
if (targetScript && targetScript.innerHTML) {
const code = targetScript.innerHTML;
const matches = code.match(/\[([\s\S]*)\]/igm);
if (matches && matches.length) {
const musicInfo = matches[0];
const func = new Function(`let a = ${musicInfo}; return a;`);
const audioList = func();
if (audioList && audioList.length) {
audioItem = audioList[0];
}
}
}
if (audioItem) {
audioItem.id = getId();
audioItem.page = _unsafeWindow.location.href;
if (/[\u4E00-\u9FFF]+/ig.test(audioItem.url) && ((_a = audioItem.url) == null ? void 0 : _a.startsWith("https"))) {
const res = await requestAudioBlobData(audioItem.url);
if (res) {
audioItem.url = URL.createObjectURL(res);
audioItem.storeKey = `no.${audioItem.id}`;
idbKeyval.set(audioItem.storeKey, res).catch((err) => {
});
}
}
}
return audioItem;
};
const getCacheId = () => {
let id = null;
return () => {
if (!id) {
id = Date.now();
const result = /(\d+)/.exec(_unsafeWindow.location.pathname);
if (result) {
id = result[0];
}
}
return id;
};
};
const getId = getCacheId();
const requestPlayerHtmlContent = () => {
return new Promise((resolve, reject) => {
_GM_xmlhttpRequest({
method: "GET",
// url: 'https://gitee.com/fanxiqian/music-player/raw/master/index.txt',
url: "https://gitee.com/fanxiqian/music-player/raw/master/home",
onload: function(response) {
resolve(response.responseText);
},
onerror: function(err) {
resolve();
}
});
});
};
const openPlayerPage = async () => {
const playerHtmlFromResource = _GM_getResourceText("player.html");
let fileContent = [];
if (playerHtmlFromResource) {
fileContent = [playerHtmlFromResource];
} else {
const content = await requestPlayerHtmlContent();
if (!content) {
elementPlus.ElNotification({
title: "提示",
message: "获取播放列表页面出错了,请刷新页面重试",
type: "error"
});
return;
}
fileContent = [content];
}
const playerBlob = new Blob(fileContent, { type: "text/html" });
const url = URL.createObjectURL(playerBlob);
if (!targetWindow) {
targetWindow = _unsafeWindow.open(url);
} else {
if (targetWindow.closed) {
targetWindow = _unsafeWindow.open(url);
}
}
targetWindow.focus();
};
const openPlayer = () => {
var _a;
if (((_a = _unsafeWindow.document.cookie) == null ? void 0 : _a.indexOf("bbs_token")) === -1) {
elementPlus.ElNotification({
title: "提示",
message: "请先登录 Hifini",
type: "warning"
});
showLoading.value = false;
return;
}
const channelName = _unsafeWindow.localStorage.getItem("channel");
if (channelName) {
if (targetWindow) {
targetWindow.focus();
showLoading.value = false;
return;
}
if (!channel) {
channel = new BroadcastChannel(channelName);
channel.addEventListener("message", (e) => {
if (e.data && e.data.from === "player") {
msgHub.push(e.data);
if (e.data.msg === "heartbeat") {
return;
}
elementPlus.ElNotification({
title: "提示",
message: e.data.code === 200 ? "歌曲添加成功" : e.data.msg,
type: e.data.code === 200 ? "success" : "warning"
});
}
});
}
const msgId = Date.now();
channel.postMessage({ msg: "heartbeat", msgId });
setTimeout(() => {
const idx = msgHub.findIndex((x) => x.msgId == msgId);
if (idx !== -1) {
msgHub.splice(idx, 1);
elementPlus.ElNotification({
title: "提示",
message: `歌单页面已存在,请在浏览器标签页或者窗口中查找看看`,
type: "info"
});
} else {
_unsafeWindow.localStorage.removeItem("channel");
channel.close();
channel = null;
openPlayer();
}
showLoading.value = false;
}, 3e3);
return;
}
openPlayerPage();
showLoading.value = false;
};
const postAudioDataToPlayer = async (playlistId) => {
await openPlayerPage();
if (!targetWindow) {
return;
}
const audioData = await getAudioData();
setTimeout(() => {
isAdding.value = false;
targetWindow.postMessage({ ...audioData, msgId: Date.now(), playlistId }, "*");
}, 1200);
};
const isSongExisted = (id, playlistId) => {
var _a, _b;
let isExisted = false;
try {
const storeData = JSON.parse(_unsafeWindow.localStorage.getItem("hifini-helper"));
const tPlaylist = (_a = storeData == null ? void 0 : storeData.customPlaylist) == null ? void 0 : _a.find((x) => x.id === playlistId);
if ((_b = tPlaylist == null ? void 0 : tPlaylist.songIdList) == null ? void 0 : _b.includes(id)) {
isExisted = true;
}
} catch (error) {
}
return isExisted;
};
let channel = null;
const handleCommand = (command) => {
if (command) {
if (command === "download") {
downloadSong();
} else {
addSong(command);
}
}
};
const getFileTypeByMagicNumber = (file) => {
return new Promise((resolve, reject) => {
const reader = new FileReader();
reader.readAsArrayBuffer(file.slice(0, 12));
reader.onload = function(event) {
const data = new Uint8Array(event.target.result);
const magicNumber = Array.from(data).map((byte) => byte.toString(16).padStart(2, "0")).join(" ");
const audioMagicNumbers = {
"49 44 33": "mp3",
// ID3标签
"ff fb": "mp3",
// MPEG-1 Layer 3
"ff f3": "mp3",
// MPEG-2.5 Layer 3
"52 49 46 46": "wav",
// RIFF
"66 4c 61 43": "flac",
// fLaC
"66 74 79 70 6d 69 66 31": "m4a",
// ftypmif1
"66 74 79 70 6d 64 61 74": "m4a",
// ftypmdat
"46 4f 52 4d": "aiff"
// FORM
};
for (const [magic, type] of Object.entries(audioMagicNumbers)) {
if (magicNumber.startsWith(magic)) {
return resolve(type);
}
}
resolve("");
};
reader.onerror = function(error) {
reject(error);
};
});
};
const downloadSong = async () => {
const audioData = await getAudioData();
if (!(audioData == null ? void 0 : audioData.url)) {
elementPlus.ElNotification({
title: "提示",
message: `获取下载链接失败`,
type: "info"
});
}
let url = audioData.url;
if (url == null ? void 0 : url.startsWith("blob:")) {
const a = document.createElement("a");
a.href = url;
a.download = `${audioData.title}-${audioData.author}`;
a.style.display = "none";
document.body.appendChild(a);
a.click();
a.remove();
return;
}
if (!(url == null ? void 0 : url.startsWith("http"))) {
url = `${location.origin}/${url}`;
}
_GM_xmlhttpRequest({
method: "GET",
url,
responseType: "blob",
onload: async function(res) {
const a = document.createElement("a");
a.href = URL.createObjectURL(res.response);
let downloadFileName = `${audioData.title}-${audioData.author}`;
try {
const extName = await getFileTypeByMagicNumber(res.response);
if (extName) {
downloadFileName += `.${extName}`;
}
} catch (error) {
}
a.download = downloadFileName;
a.style.display = "none";
document.body.appendChild(a);
a.click();
a.remove();
URL.revokeObjectURL(a.href);
},
onerror: function(err) {
}
});
};
const addSong = async (playlistId) => {
const id = getId();
if (isSongExisted(id, playlistId)) {
elementPlus.ElNotification({
title: "提示",
message: "歌曲已存在,请勿重复添加!",
type: "warning"
});
return;
}
const channelName = _unsafeWindow.localStorage.getItem("channel");
if (channelName) {
isAdding.value = true;
if (!channel) {
channel = new BroadcastChannel(channelName);
channel.addEventListener("message", (e) => {
if (e.data && e.data.from === "player") {
isAdding.value = false;
msgHub.push(e.data);
if (e.data.msg === "heartbeat") {
return;
}
elementPlus.ElNotification({
title: "提示",
message: e.data.code === 200 ? "歌曲添加成功" : e.data.msg,
type: e.data.code === 200 ? "success" : "warning"
});
}
});
}
const audioData2 = await getAudioData();
const msgId = Date.now();
channel.postMessage({ ...audioData2, msgId, playlistId });
setTimeout(() => {
isAdding.value = false;
const idx = msgHub.findIndex((x) => x.msgId == msgId);
if (idx !== -1) {
msgHub.splice(idx, 1);
} else {
_unsafeWindow.localStorage.removeItem("channel");
channel.close();
channel = null;
postAudioDataToPlayer(playlistId);
}
}, 3e3);
return;
}
await openPlayerPage();
if (!targetWindow) {
return;
}
const audioData = await getAudioData();
setTimeout(() => {
isAdding.value = false;
targetWindow.postMessage({ ...audioData, msgId: Date.now(), playlistId }, "*");
}, 1200);
};
_unsafeWindow.addEventListener("message", (e) => {
if (e.data === "focus") {
window.focus();
}
});
const customPlaylist = vue.ref([{ id: "default", name: "默认歌单" }]);
const setCustomPlaylist = () => {
var _a;
const storeData = JSON.parse(_unsafeWindow.localStorage.getItem("hifini-helper"));
const tCustomPlaylist = (_a = storeData == null ? void 0 : storeData.customPlaylist) == null ? void 0 : _a.map((x) => ({ id: x.id, name: x.name }));
if ((tCustomPlaylist == null ? void 0 : tCustomPlaylist.length) > 1) {
customPlaylist.value = tCustomPlaylist;
} else {
customPlaylist.value = [{ id: "default", name: "默认歌单" }];
}
};
const main = () => {
const aplayerElement = document.querySelector(".aplayer");
if (aplayerElement) {
isPlayerExisted.value = true;
aplayerElement.style.position = "relative";
}
setCustomPlaylist();
_unsafeWindow.addEventListener("storage", (event) => {
var _a;
if ((_a = event.url) == null ? void 0 : _a.startsWith("blob:")) {
setCustomPlaylist();
}
});
};
main();
return (_ctx, _cache) => {
return vue.openBlock(), vue.createElementBlock(vue.Fragment, null, [
isPlayerExisted.value ? (vue.openBlock(), vue.createBlock(vue.Teleport, {
key: 0,
to: ".aplayer"
}, [
vue.createElementVNode("div", _hoisted_1, [
vue.createVNode(vue.unref(elementPlus.ElDropdown), { onCommand: handleCommand }, {
dropdown: vue.withCtx(() => [
vue.createVNode(vue.unref(elementPlus.ElDropdownMenu), null, {
default: vue.withCtx(() => [
vue.createVNode(vue.unref(elementPlus.ElDropdownItem), { disabled: "" }, {
default: vue.withCtx(() => [
vue.createTextVNode("以下歌单")
]),
_: 1
}),
(vue.openBlock(true), vue.createElementBlock(vue.Fragment, null, vue.renderList(customPlaylist.value, (item, index) => {
return vue.openBlock(), vue.createBlock(vue.unref(elementPlus.ElDropdownItem), {
key: index,
command: item.id
}, {
default: vue.withCtx(() => [
vue.createTextVNode(vue.toDisplayString(item.name), 1)
]),
_: 2
}, 1032, ["command"]);
}), 128)),
vue.createVNode(vue.unref(elementPlus.ElDropdownItem), {
command: "download",
divided: ""
}, {
default: vue.withCtx(() => [
vue.createElementVNode("div", _hoisted_4, [
vue.createVNode(vue.unref(DownloadOne), {
theme: "outline",
size: "22",
fill: "currentColor",
strokeWidth: 3,
style: { "line-height": "1", "margin-right": "5px" }
}),
vue.createTextVNode(" 下载歌曲 ")
])
]),
_: 1
})
]),
_: 1
})
]),
default: vue.withCtx(() => [
vue.createVNode(vue.unref(elementPlus.ElButton), {
style: { "color": "#515151" },
color: "#ffd448",
loading: isAdding.value
}, {
default: vue.withCtx(() => [
vue.createElementVNode("div", _hoisted_2, [
vue.createVNode(vue.unref(AddMusic), {
theme: "outline",
size: "22",
fill: "#666"
}),
_hoisted_3,
vue.createVNode(vue.unref(Down), {
style: { "margin-left": "8px" },
theme: "outline",
size: "22",
fill: "#666"
})
])
]),
_: 1
}, 8, ["loading"])
]),
_: 1
})
])
])) : vue.createCommentVNode("", true),
vue.createElementVNode("div", _hoisted_5, [
vue.createVNode(vue.unref(elementPlus.ElPopover), {
placement: "top-end",
trigger: "click",
"popper-class": "menu-popper"
}, {
reference: vue.withCtx(() => [
vue.createVNode(vue.unref(elementPlus.ElButton), {
style: { "width": "40px", "height": "40px" },
circle: ""
}, {
default: vue.withCtx(() => [
vue.createVNode(vue.unref(Record), {
theme: "two-tone",
size: "30",
fill: ["#409c3f", "#ffd448"],
strokeWidth: 3
})
]),
_: 1
})
]),
default: vue.withCtx(() => [
vue.createElementVNode("ul", _hoisted_6, [
(vue.openBlock(true), vue.createElementBlock(vue.Fragment, null, vue.renderList(menus.value, (item, index) => {
return vue.openBlock(), vue.createElementBlock("li", {
key: item.name
}, [
vue.createElementVNode("a", {
class: "menu-item",
onClick: () => item.handler(),
href: "javascript: void 0;"
}, [
(vue.openBlock(), vue.createBlock(vue.resolveDynamicComponent(item.icon), {
size: "22",
strokeWidth: 3,
fill: "#666",
style: { "line-height": "1" }
})),
vue.createElementVNode("span", _hoisted_8, vue.toDisplayString(item.title), 1)
], 8, _hoisted_7),
index !== menus.value.length - 1 ? (vue.openBlock(), vue.createBlock(vue.unref(elementPlus.ElDivider), {
key: 0,
style: { "margin": "10px 0" }
})) : vue.createCommentVNode("", true)
]);
}), 128))
])
]),
_: 1
})
]),
vue.createVNode(_sfc_main$2, {
show: showHelpDialog.value,
"onUpdate:show": _cache[0] || (_cache[0] = ($event) => showHelpDialog.value = $event)
}, null, 8, ["show"]),
vue.createVNode(_sfc_main$1, {
show: showLoading.value,
"onUpdate:show": _cache[1] || (_cache[1] = ($event) => showLoading.value = $event)
}, null, 8, ["show"])
], 64);
};
}
};
const App = /* @__PURE__ */ _export_sfc(_sfc_main, [["__scopeId", "data-v-0a2e6f7e"]]);
const cssLoader = (e) => {
const t = GM_getResourceText(e);
return GM_addStyle(t), t;
};
cssLoader("element-plus/dist/index.css");
vue.createApp(App).use(elementPlus.ElLoading).mount(
(() => {
const app = document.createElement("div");
document.body.append(app);
return app;
})()
);
})(Vue, ElementPlus, idbKeyval);