├── README.md
├── LICENSE
├── Plugin.php
├── SQPlayer.css
└── SQPlayer.js
/README.md:
--------------------------------------------------------------------------------
1 | # Square Player
2 |
3 | 一个简洁到极致的单曲播放器,基于 ES6 标准开发的试水之作。
4 |
5 | ## 使用方法
6 |
7 | 1. `Star` 本项目
8 | 2. 从这里 [下载](https://github.com/Dreamer-Paul/Square-Player/archive/master.zip) 源码
9 | 3. 可使用本地、网易云音乐两种方式食用,参照 [文档站](https://docs.paul.ren/square) 即可快速部署完成
10 | 4. 本项目提供 Typecho 插件版本,如需使用请先将下载后得到的文件夹从 `Square-Player-master` 重命名为 `SQP`
11 |
12 | ## 开源协议
13 |
14 | 本项目采用 MIT 开源协议进行授权,请保留原作者的版权注释(CSS、JS 文件)
15 |
16 | 原创不易!如果喜欢本项目,请 `Star` 它以示对我的支持~
17 |
18 | 同时欢迎前往 [保罗的小窝](https://paul.ren/donate) 为我提供赞助,谢谢您!
19 |
20 | ## 感谢
21 |
22 | 感谢来自开源社区提供的解决方案,如果没有它们,Square Player 可能还无法如此完善~
23 |
24 | - [Meting](https://github.com/metowolf/Meting)
25 |
--------------------------------------------------------------------------------
/LICENSE:
--------------------------------------------------------------------------------
1 | MIT License
2 |
3 | Copyright (C) 2019 Dreamer-Paul
4 |
5 | Permission is hereby granted, free of charge, to any person obtaining a copy
6 | of this software and associated documentation files (the "Software"), to deal
7 | in the Software without restriction, including without limitation the rights
8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9 | copies of the Software, and to permit persons to whom the Software is
10 | furnished to do so, subject to the following conditions:
11 |
12 | The above copyright notice and this permission notice shall be included in all
13 | copies or substantial portions of the Software.
14 |
15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21 | SOFTWARE.
--------------------------------------------------------------------------------
/Plugin.php:
--------------------------------------------------------------------------------
1 | header = array('SQP_Plugin', 'header');
16 | Typecho_Plugin::factory('Widget_Archive') -> footer = array('SQP_Plugin', 'footer');
17 | }
18 |
19 | /* 禁用插件方法 */
20 | public static function deactivate(){}
21 |
22 | /* 插件配置方法 */
23 | public static function config(Typecho_Widget_Helper_Form $form){}
24 |
25 | /* 个人用户的配置方法 */
26 | public static function personalConfig(Typecho_Widget_Helper_Form $form){}
27 |
28 | /* 插件实现方法 */
29 | public static function header(){
30 | if(Typecho_Widget::widget('Widget_Archive') -> is("post")){
31 | echo '';
32 | }
33 | }
34 |
35 | public static function footer(){
36 | if(Typecho_Widget::widget('Widget_Archive') -> is("post")){
37 | echo '';
38 | }
39 | }
40 | }
--------------------------------------------------------------------------------
/SQPlayer.css:
--------------------------------------------------------------------------------
1 | /* ----
2 |
3 | # Square Player
4 | # By: Dreamer-Paul
5 | # Last Update: 2024.4.20
6 |
7 | 一个简洁到极致的单曲播放器。
8 |
9 | 本代码为奇趣保罗原创,并遵守 MIT 开源协议。欢迎访问我的博客:https://paugram.com
10 |
11 | ---- */
12 |
13 | sqp {
14 | width: 8em;
15 | height: 8em;
16 | color: #fff;
17 | float: right;
18 | display: block;
19 | overflow: hidden;
20 | user-select: none;
21 | position: relative;
22 | border-radius: 1em;
23 | margin: 0 0 1em 1em;
24 | background: #ccc center/cover no-repeat;
25 | }
26 | sqp[left] {
27 | float: left;
28 | margin: 0 1em 1em 0;
29 | }
30 |
31 | sqp .sqp-info {
32 | left: 0;
33 | right: 0;
34 | bottom: 0;
35 | padding: .75em 0;
36 | font-size: .75em;
37 | position: absolute;
38 | backdrop-filter: blur(2px);
39 | background-color: rgba(0, 0, 0, .4);
40 | }
41 |
42 | sqp .sqp-title {
43 | padding: 0 1em;
44 | white-space: nowrap;
45 | display: inline-block;
46 | }
47 |
48 | sqp .sqp-toggle {
49 | width: 2em;
50 | height: 2em;
51 | opacity: .8;
52 | cursor: pointer;
53 | position: relative;
54 | border-radius: 66%;
55 | display: inline-block;
56 | backdrop-filter: blur(2px);
57 | box-shadow: 0 0 0 .25em rgba(0, 0, 0, .4);
58 | transform: translate(calc(4em - 50%), 100%);
59 | transition: opacity .3s, width .3s, height .3s, transform .3s;
60 | background: #fff center/1em no-repeat;
61 | background-image: url()
62 | }
63 |
64 | sqp .sqp-toggle:hover {
65 | opacity: 1;
66 | }
67 |
68 | sqp .sqp-toggle.playing {
69 | width: 1.5em;
70 | height: 1.5em;
71 | transform: translate(.5em, .5em);
72 | background-image: url();
73 | }
74 |
--------------------------------------------------------------------------------
/SQPlayer.js:
--------------------------------------------------------------------------------
1 | 'use strict';
2 |
3 | /* ----
4 |
5 | # Square Player
6 | # By: Dreamer-Paul
7 | # Last Update: 2024.4.20
8 |
9 | 一个简洁到极致的单曲播放器。
10 |
11 | 本代码为奇趣保罗原创,并遵守 MIT 开源协议。欢迎访问我的博客:https://paugram.com
12 |
13 | ---- */
14 |
15 | window._SQPPlayers = [];
16 |
17 | class SQPlayer {
18 | key = "";
19 | elements = undefined;
20 |
21 | constructor(wrapper, key, set) {
22 | this.key = key;
23 | this.elements = {
24 | wrap: wrapper,
25 | player: new Audio(),
26 | info: this.creator("div", { className: "info" }),
27 | title: this.creator("span", { className: "title", content: "加载中..." }),
28 | toggle: this.creator("div", { className: "toggle" })
29 | };
30 |
31 | this.elements.wrap.setAttribute("loaded", "");
32 | this.elements.player.setAttribute("preload", "none");
33 |
34 | if (wrapper.dataset.cid) {
35 | this.setupByCloudMusic(wrapper.dataset.cid, set.server);
36 | }
37 | else {
38 | this.setup(wrapper.dataset);
39 | }
40 |
41 | this.elements.info.appendChild(this.elements.title);
42 | this.elements.wrap.appendChild(this.elements.info);
43 | this.elements.wrap.appendChild(this.elements.toggle);
44 | }
45 |
46 | // 切换
47 | toggle = () => this.events.onToggle();
48 |
49 | // 播放
50 | play = () => {
51 | this.elements.player.play();
52 |
53 | _SQPPlayers.forEach(item => {
54 | if (item.key !== this.key) item.pause();
55 | });
56 | }
57 |
58 | // 暂停
59 | pause = () => {
60 | this.elements.player.pause();
61 | }
62 |
63 | // 元素创建器
64 | creator(tag, attr) {
65 | const el = document.createElement(tag);
66 |
67 | if (attr?.className) {
68 | el.className = `sqp-${attr.className}`;
69 | }
70 |
71 | if (attr?.content) {
72 | el.innerHTML = attr.content;
73 | }
74 |
75 | return el;
76 | }
77 |
78 | // 事件
79 | events = {
80 | onToggle: () => {
81 | this.elements.player.paused ? this.play() : this.pause();
82 | },
83 | onPlay: () => {
84 | this.elements.toggle.classList.add("playing");
85 | },
86 | onPause: () => {
87 | this.elements.toggle.classList.remove("playing");
88 | },
89 | }
90 |
91 | // 修改元素的操作
92 | modify = {
93 | updateTitleText: (nextTitle) => {
94 | const fontSize = Number(window.getComputedStyle(document.querySelector("html")).fontSize.replace("px", ""));
95 |
96 | const el = this.elements.title;
97 | el.innerText = nextTitle;
98 |
99 | const offset = el.offsetWidth - (fontSize * 8);
100 | const duration = parseInt(el.offsetWidth / 30) * 1000;
101 |
102 | if (offset > 0) {
103 | el.animate([
104 | { transform: "translateX(0)" },
105 | { transform: `translateX(${-offset}px)` },
106 | { transform: "translateX(0)" },
107 | ], {
108 | duration,
109 | iterations: Infinity,
110 | });
111 | }
112 | },
113 | }
114 |
115 | // 设置播放器
116 | setup = (item) => {
117 | // 播放器主体初始化
118 | let titleText = "未知标题";
119 |
120 | if (item.artist && item.title) {
121 | titleText = `${item.title} - ${item.artist}`;
122 | }
123 | else if (item.title) {
124 | titleText = item.title;
125 | }
126 |
127 | if (item.link) {
128 | this.elements.player.src = item.link;
129 | }
130 | else {
131 | titleText = "无效文件路径";
132 | console.error("SQP: Error, No files to play!");
133 | }
134 |
135 | this.modify.updateTitleText(titleText);
136 |
137 | if (item.cover) {
138 | this.elements.wrap.style.backgroundImage = `url(${item.cover})`;
139 | }
140 |
141 | this.elements.toggle.addEventListener("click", this.events.onToggle);
142 | this.elements.player.addEventListener("play", this.events.onPlay);
143 | this.elements.player.addEventListener("pause", this.events.onPause);
144 | }
145 |
146 | // 销毁
147 | destroy = () => {
148 | this.elements.player.pause();
149 | this.elements.toggle.removeEventListener("click", this.events.onToggle);
150 | this.elements.player.removeEventListener("play", this.events.onPlay);
151 | this.elements.player.removeEventListener("pause", this.events.onPause);
152 |
153 | this.elements.wrap.remove();
154 | this.elements.wrap = undefined;
155 | this.elements = undefined;
156 | }
157 |
158 | setupByCloudMusic = (cid, server) => {
159 | const getData = {
160 | "meto": () => (
161 | fetch(`https://api.i-meto.com/meting/api?server=netease&id=${cid}`).then(
162 | (res) => res.json()
163 | ).then((items) => {
164 | const item = items[0];
165 |
166 | if (!item) {
167 | throw new Error("返回数据为空");
168 | }
169 |
170 | this.setup({
171 | title: item.title,
172 | artist: item.author,
173 | cover: item.pic,
174 | link: item.url
175 | });
176 | })
177 | ),
178 | "paul": () => (
179 | fetch(`https://api.paugram.com/netease/?id=${cid}`).then(
180 | (res) => res.json()
181 | ).then((item) => {
182 | this.setup(item);
183 | })
184 | ),
185 | };
186 |
187 | if (server in getData) {
188 | getData[server]().catch((err) => {
189 | this.modify.updateTitleText(`获取数据异常:${err.message}`);
190 | });
191 | }
192 | }
193 | }
194 |
195 | console.log("%c Square Player %c https://paugram.com ", "color: #fff; margin: 1em 0; padding: 5px 0; background: #1875b3;", "margin: 1em 0; padding: 5px 0; background: #efefef;");
196 |
197 | class SQP_Extend {
198 | constructor(settings) {
199 | this.settings = settings;
200 | this.init();
201 | }
202 |
203 | init = () => {
204 | this.wrapper = document.querySelectorAll("sqp");
205 |
206 | this.wrapper.forEach((item, key) => {
207 | if (!item.hasAttribute("loaded")) {
208 | _SQPPlayers.push(new SQPlayer(item, key + new Date().getTime(), this.settings));
209 | }
210 | });
211 | }
212 |
213 | destroy = () => {
214 | while (_SQPPlayers.length > 0) {
215 | _SQPPlayers[0].destroy();
216 | _SQPPlayers.splice(_SQPPlayers[0], 1);
217 | }
218 | }
219 | }
220 |
221 | window._SQP_Extend = new SQP_Extend({
222 | server: "meto"
223 | });
224 |
--------------------------------------------------------------------------------