红影 发表于 2022-12-12 23:31
只是名词上的差异,意思都懂
心里领会
马黑黑 发表于 2022-12-12 23:41
心里领会
无论是什么词,都很感谢黑黑带来的馈赠{:4_187:}
红影 发表于 2022-12-13 16:13
无论是什么词,都很感谢黑黑带来的馈赠
言重了
马黑黑 发表于 2022-12-13 17:23
言重了
这份感谢是发自内心的{:4_181:}
红影 发表于 2022-12-13 21:24
这份感谢是发自内心的
啊?那那那……{:4_172:}
马黑黑 发表于 2022-12-13 21:24
啊?那那那……
那就喝酒吧,敬你{:4_191:}
红影 发表于 2022-12-13 22:28
那就喝酒吧,敬你
谢谢
马黑黑 发表于 2022-12-13 23:40
谢谢
不客气啊,中午好{:4_204:}
三十二、模拟示波播放器(支持lrc同步)
(一)插件代码
(function(mkPlayer) {let defaults = {lrcAr: [],ypData: new Array(600).fill(0).map((v, k) => Math.random() * 150 + Math.random() * (k < 300 ? k : (500 - k)) * 0.15),player_css: 'bottom: 15px; left: 50%; transform: translate(-50%); ',playerCode: `<style>#mplayer { --color1: red; --color2: pink; --ww: 200; --hh: 80; position: absolute; cursor: pointer; }#lrc { --motion: cover2;--tt: 2s;--state: paused; --bg: linear-gradient(180deg, hsla(60, 50%, 50%, .45), hsla(0, 100%, 50%, .75)); position: absolute; top: 20px; font: bold 2.4em sans-serif; color: hsl(0, 10%, 90%); white-space: pre; -webkit-background-clip: text; filter: drop-shadow(1px 1px 2px hsla(0, 0%, 0%, .95)); } #lrc::before { position: absolute; content: attr(data-lrc); width: 20%;height: 100%; color: transparent; overflow: hidden; white-space: pre; background: var(--bg); filter: inherit; -webkit-background-clip: text; animation: var(--motion) var(--tt) linear forwards; animation-play-state: var(--state); }@keyframes cover1 {from {width: 0;} to {width: 100%;}}@keyframes cover2 {from {width: 0;} to {width: 100%;}}</style><div id="lrc" data-lrc="HCPlayer">HCPlayer</div><canvas id="mplayer"></canvas>`,};let playCode = (user_config) => {let data = Object.assign({}, defaults, user_config);papa.innerHTML += data.playerCode;mplayer.style.cssText += data.player_css;lrc.style.cssText += data.lrc_css;aud.loop = false;let mKey = 0, mFlag = true;let getCssVal = (e,v) => getComputedStyle(e).getPropertyValue(v);let ctx = mplayer.getContext('2d');let w = mplayer.width = getCssVal(mplayer,'--ww').replace(/[^0-9]/ig,''), h = mplayer.height = getCssVal(mplayer,'--hh').replace(/[^0-9]/ig,''), idx = 0;let slice = w / data.ypData.length, vmax = Math.max.apply(null, data.ypData);mplayer.onmousemove = (e) => { mplayer.title = e.offsetY < h * 0.7 ? toMin(aud.duration * e.offsetX / w) : '暂停/播放'; };mplayer.onclick = (e) => {if(e.offsetY < h * 0.7) {let ac = aud.duration * e.offsetX / w;idx = Math.round(data.ypData.length * ac / aud.duration);aud.currentTime = ac;} else {aud.paused ? aud.play() : aud.pause();}};aud.addEventListener('pause', () => mState());aud.addEventListener('playing', () => mState());aud.addEventListener('seeked', () => calcKey());aud.addEventListener('ended', () => { mKey = 0; aud.play(); });aud.addEventListener('timeupdate', () => {idx = Math.round(data.ypData.length * aud.currentTime / aud.duration);if(idx > data.ypData.length - 1) idx = data.ypData.length - 1;draw();for (j = 0; j < data.lrcAr.length; j++) {if (aud.currentTime >= data.lrcAr) {cKey = j;if (mKey === j) showLrc(data.lrcAr);else continue;}}});let draw = () => {ctx.clearRect(0,0,w,h);ctx.strokeStyle = getCssVal(mplayer,'--color1');ctx.fillStyle = getCssVal(mplayer,'--color2');ctx.font = '16px sans-serif';ctx.shadowOffsetX = ctx.shadowOffsetY = 1;ctx.shadowBlur = 2;ctx.shadowColor = '#555';ctx.textAlign = 'center';ctx.beginPath();for(j = 0; j <= idx; j ++) {ctx.lineTo(slice*j, 0.7 * (h - data.ypData * h / vmax));}ctx.stroke();ctx.beginPath();ctx.strokeStyle = getCssVal(mplayer,'--color2');for(j = idx; j < data.ypData.length; j ++) {ctx.lineTo(slice*j, 0.7 * (h - data.ypData * h / vmax));}ctx.stroke();ctx.fillText((aud.paused ? '播放 ' : '暂停 ') + toMin(aud.currentTime) + ' / ' + toMin(aud.duration), w/2, h*0.95);};let calcKey = () => {for (j = 0; j < data.lrcAr.length; j++) {if (aud.currentTime <= data.lrcAr) {mKey = j - 1;break;}}if (mKey < 0) mKey = 0;if (mKey > data.lrcAr.length - 1) mKey = data.lrcAr.length - 1;let time = data.lrcAr - (aud.currentTime - data.lrcAr);showLrc(time);};let showLrc = (time) => {let name = mFlag ? 'cover1' : 'cover2';lrc.innerHTML = data.lrcAr;lrc.dataset.lrc = data.lrcAr.replace(/<br>/, '\n');lrc.style.setProperty('--motion', name);lrc.style.setProperty('--tt', time + 's');lrc.style.setProperty('--state', 'running');mKey += 1;mFlag = !mFlag;};let mState = () => aud.paused ? lrc.style.setProperty('--state', 'paused') : lrc.style.setProperty('--state', 'running');let toMin = (val) => {if (!val) return '00:00';val = Math.floor(val);let min = parseInt(val / 60), sec = parseFloat(val % 60);if (min < 10) min = '0' + min;if (sec < 10) sec = '0' + sec;return min + ':' + sec;};draw();};mkPlayer.HCPlayer = playCode;})(this);
(二)插件参数配置说明
1、lrcAr 参数 : 配置歌词数组。例:
let ar = [,,...,];
HCPlayer({
lrcAr: ar,
});
【注】发纯音乐理论上可无需配置 lrcAr 参数,建议设定成帖名,ar 变量可声明为:let ar = [];
2、ypData 参数 : 示波图谱数组,即通过本版提供的工具制作的音频声波数据数组。可以省略,若省略,插件会提供一个随机的示波数据,但不能体现歌曲的声音状态。若提供自己的声波数组,配置如下:
let ar = [,,...,];
let ypSj = ;
HCPlayer({
lrcAr: ar,
ypData: ypSj,
});
3、lrc_css 参数 : 设置lrc歌词颜色与位置。同步颜色使用 CSS变量 --bg 设置,底色用 CSS属性 color 设置。歌词位置请使用 CSS 属性 left/right 和 top/bottom 两两配对的方式设置。例:
let ar = [,,...,];
let ypSj = ;
HCPlayer({
lrcAr: ar,
ypData: ypSj,
lrc_css: '--bg: green; color: #ccc; left: 240px; top: 20px;',
});
4、player_css 参数 : 配置播放控制器。着色示波为 CSS变量 --color1,底色示波为 --color2,播放时间信息也用 --color2。播放器宽高分别用 --ww 和 --hh 设置。位置的设置同 lrc_css 的操作。例:
let ar = [,,...,];
let ypSj = ;
HCPlayer({
lrcAr: ar,
ypData: ypSj,
lrc_css: '--bg: green; color: #ccc; left: 240px; top: 20px;',
player_css: '--color1: blue; --color2: lightblue; --ww: 400px; --hh: 100px; left: 200px; bottom: 10px;',
});
(三)帖子模板
<style>
#papa { margin: 0 0 0 calc(50% - 593px); width: 1024px; height: 600px; display: grid; place-items: center; background: gray url('./pic/18u.webp') no-repeat center/cover; box-shadow: 3px 3px 20px #000; position: relative; }
</style>
<div id="papa">
<audio id="aud" src="音乐地址" autoplay></audio>
</div>
<script>
//这里是插件代码
//这里是歌词数组 let lrcAr = [];
//这里是音频数据 let ypData = ;
HCPlayer({
lrcAr: lrcAr,
ypData: ypData,
lrc_css: '--bg: green; color:pink;',
player_css: '--color1: green; --color2: lightgreen;--ww: 300px; --hh: 80px; bottom: 20px; ',
});
</script>
(四)效果演示:
https://www.huachaowang.com/forum.php?mod=viewthread&tid=65333
三十三、模拟动态示波进度条播放器
(播放控制器基于canvas画布,示波随机动态抖动。插件支持歌词同步)
(一)插件代码(function(mkPlayer) {let defaults = {lrcAr: [],lrc_css: 'top: 15px; left: 50%; transform: translate(-50%)',player_css: 'bottom: 10px; left: 50%; transform: translate(-50%)',playerCode: `<style>#mplayer { --ww: 300px; --color1: hsla(0,80%,50%,.95); --color2: hsla(0,0%,95%,.45); position: absolute; z-index: 901; }#lrc { --motion: cover2; --tt: 1s; --state: running; --bg: linear-gradient(180deg,hsla(100,10%,50%,.75),hsla(100,100%,20%,.65)); position: absolute; font: bold 2.4em sans-serif; color: hsl(100, 100%, 90%); white-space: pre; -webkit-background-clip: text; filter: drop-shadow(1px 1px 2px hsla(0, 100%, 0%, .85)); z-index: 900; }#lrc::before { position: absolute; content: attr(data-lrc); width: 20%; height: 100%; color: transparent; overflow: hidden; white-space: pre; background: var(--bg); filter: inherit; -webkit-background-clip: text; animation: var(--motion) var(--tt) linear forwards; animation-play-state: var(--state); }@keyframes cover1 { from { width: 0; }to { width: 100%; } }@keyframes cover2 { from { width: 0; }to { width: 100%; } }</style><canvas id="mplayer" width="300" height="75"></canvas><div id="lrc" data-lrc="HCPlayer">HCPlayer</div>`,};let playCode = (user_config) => {let data = Object.assign({}, defaults, user_config);papa.innerHTML += data.playerCode;mplayer.style.cssText += data.player_css;lrc.style.cssText += data.lrc_css;let mKey = 0, mFlag = true, btfFlag = false;let getCssVal = (e,v) => getComputedStyle(e).getPropertyValue(v);let ctx = mplayer.getContext('2d');let w = mplayer.width = getCssVal(mplayer,'--ww').replace(/[^0-9]/ig,''), h = mplayer.height = 75;let btnFlag = false;let player = {prog: 0,track: w,color: ,cur: '00:00',dur: '00:00'};let drawCircle = (x,y,r,color) => {ctx.beginPath();ctx.strokeStyle = color;ctx.lineWidth = 2;ctx.arc(x,y,r,0,2*Math.PI);ctx.stroke();};let drawTriangle = (x,y,len,color) => {ctx.beginPath();ctx.fillStyle = color;ctx.moveTo(x,y);ctx.lineTo(x, y+len);ctx.lineTo(x+len, y + len/2);ctx.lineTo(x,y);ctx.fill();};let drawRect = (x,y,ww,hh,color) => {ctx.beginPath();ctx.fillStyle = color;ctx.fillRect(x,y,ww,hh);};let drawOsc = (w1,w2) => {ctx.beginPath();ctx.lineWidth = 1;ctx.strokeStyle = player.color;let slice = 1, tt1 = Math.round(w1/slice), tt2 = Math.round(w2/slice);for(j=0; j<=tt2; j++) {ctx.lineTo(j*slice, Math.random() *30);}ctx.stroke();ctx.beginPath();ctx.strokeStyle = player.color;for(j=tt2; j<tt1; j++) {ctx.lineTo(j*slice, Math.random() *30);}ctx.stroke();};let drawTxt = (text,x,y,align,color) => {ctx.beginPath();ctx.font = '16px sans-serif';ctx.textAlign = align;ctx.textBaseline = 'middle';ctx.fillStyle = color;ctx.fillText(text,x,y);};let btnRender = () => {let btnColor = btnFlag ? player.color : player.color;ctx.clearRect(w/2-18, h-38, 36, 36);drawCircle(w/2,h-20,16,btnColor);aud.paused ? drawTriangle(w/2-6,h-28,16,btnColor) : (drawRect(w/2-6, h-28,4,16,btnColor), drawRect(w/2+2, h-28,4,16,btnColor));};let drawAll = () => {ctx.clearRect(0,0,w,h);drawOsc(w,player.prog);drawTxt(player.cur,w/2-26,h-20,'right',player.color);drawTxt(player.dur,w/2 + 26,h-20,'left',player.color);btnRender();};let overBtn = (e) => Math.sqrt((e.offsetX-w/2) ** 2 + (e.offsetY-(h-20)) ** 2) < 16;let overProg = (e) => e.offsetY > 0 && e.offsetY < 30;mplayer.onmousemove = (e) => {mplayer.style.cursor = overBtn(e) || overProg(e) ? 'pointer' : 'default';mplayer.title = overProg(e) ? toMin(e.offsetX*aud.duration/w) : '';overBtn(e) ? (btnFlag = true,btnRender()) : (btnFlag = false,btnRender());};mplayer.onclick = (e) => {if(overBtn(e)) { aud.paused ? aud.play() : aud.pause(); drawAll(); }if(overProg(e)) aud.currentTime = aud.duration * e.offsetX / w;};mplayer.onmouseout = () => { btnFlag = false; btnRender(); };aud.addEventListener('timeupdate', () => {player.prog = aud.currentTime * w /aud.duration;player.cur = toMin(aud.currentTime);player.dur = toMin(aud.duration);drawAll();for (j = 0; j < data.lrcAr.length; j++) {if (aud.currentTime >= data.lrcAr) {if (mKey === j) showLrc(data.lrcAr);else continue;}}});aud.addEventListener('pause', () => mState());aud.addEventListener('play', () => mState());aud.addEventListener('seeked', () => calcKey());let mState = () => aud.paused ? (lrc.style.setProperty('--state', 'paused')) : (lrc.style.setProperty('--state', 'running'));let showLrc = (time) => {let name = mFlag ? 'cover1' : 'cover2';lrc.innerHTML = data.lrcAr;lrc.dataset.lrc = data.lrcAr.replace(/<br>/, '\n');lrc.style.setProperty('--motion', name);lrc.style.setProperty('--tt', time + 's');lrc.style.setProperty('--state', 'running');mKey += 1;mFlag = !mFlag;};let calcKey = () => {for (j = 0; j < data.lrcAr.length; j++) {if (aud.currentTime <= data.lrcAr) {mKey = j - 1;break;}}if (mKey < 0) mKey = 0;if (mKey > data.lrcAr.length - 1) mKey = data.lrcAr.length - 1;let time = data.lrcAr - (aud.currentTime - data.lrcAr);showLrc(time);};let toMin = (val) => {if (!val) return '00:00';val = Math.floor(val);let min = parseInt(val / 60), sec = parseFloat(val % 60);if (min < 10) min = '0' + min;if (sec < 10) sec = '0' + sec;return min + ':' + sec;};drawAll();};mkPlayer.HCPlayer = playCode;})(this);(二)参数即配置
1、lrcAr : 接收帖子歌词同步数组,例如:let arr = [ ]; 配置举例如下:
<div id="papa"></div>
<audio id="aud" src="MP3地址" autoplay loop></audio>
<script>
//这里是插件代码
let arr = [ , , /* ... */ ];
HCPlayer({
lrcAr: arr,
});
2、lrc_css : 歌词显示样式设定,① 接收一个CSS变量 --bg ,支持颜色和渐变颜色,例如:--bg: red; 或 --bg: linear(180deg, red, pink); ② 接收CSS属性 color 设置歌词底色,仅支持颜色表达法,例如: color: #ccc; ③ 定位使用CSS属性 left/right 和 top/bottom 配对进行。 配置举例:
<div id="papa"></div>
<audio id="aud" src="MP3地址" autoplay loop></audio>
<script>
//这里是插件代码
let arr = [ , , /* ... */ ];
HCPlayer({
lrcAr: arr,
lrc_css: '--bg: linear-gradient(180deg,hsla(0,100%,50%,.35),hsla(200,100%,50%,.65)); color: lightblue; top: 10px;',
});
3、player_css : 播放控制器设定。① --ww 变量设置播放器长度;② --color1 设置示波表示播放进度部分的颜色,--color2 设置示波底轨颜色;③ ③ 定位使用CSS属性 left/right 和 top/bottom 配对进行。配置举例:
<div id="papa"></div>
<audio id="aud" src="MP3地址" autoplay loop></audio>
<script>
//这里是插件代码
let arr = [ , , /* ... */ ];
HCPlayer({
lrcAr: arr,
lrc_css: '--bg: linear-gradient(180deg,hsla(0,100%,50%,.35),hsla(200,100%,50%,.65)); color: lightblue; top: 10px;',
player_css: '--color1: red; --color2: lightblue; --ww: 400px; bottom: 5px;',
});
【注意】每一个参数配置结束,均以小角逗号 ,收尾!
(三)应用实例:少女净妖师 - 欢乐水吧 - 花潮论坛 - Powered by Discuz! (huachaowang.com)
黑黑的速度真的神的,已经封装到三十三了
这些分享的资源是花潮论坛的财富,很值钱呢{:4_170:}
小辣椒 发表于 2022-12-21 12:41
这些分享的资源是花潮论坛的财富,很值钱呢
嗯嗯{:4_181:},五六毛有的
马黑黑 发表于 2022-12-21 12:44
嗯嗯,五六毛有的
{:4_173:},土豪居然说只有五 六毛
小辣椒 发表于 2022-12-21 12:46
,土豪居然说只有五 六毛
够用就好
马黑黑 发表于 2022-12-21 12:54
够用就好
钱太多黄梅天会很累,要晒太阳麻烦{:4_203:}
小辣椒 发表于 2022-12-21 13:03
钱太多黄梅天会很累,要晒太阳麻烦
就是,银行家很累的
马黑黑 发表于 2022-12-21 15:15
就是,银行家很累的
不过黑黑不会累,家有女佣{:4_189:}
小辣椒 发表于 2022-12-21 20:42
不过黑黑不会累,家有女佣
还行
马黑黑 发表于 2022-12-21 20:56
还行
那是肯定不会太累的{:4_189:}