秋叶如花
<style>#papa { left: -214px; width: 1024px; height: 683px; background: silver url('https://638183.freep.cn/638183/t22/webp/qyrh.webp') no-repeat center/cover; box-shadow: 3px 3px 20px #000; display: grid; place-items: center; user-select: none; position: relative; z-index: 1; }
#mplayer { position: absolute; width: 400px; height: 110px; left: 20px; bottom: 10px; font: normal 14px sans-serif; z-index: 9; }
#mplayer::before, #mplayer::after { position: absolute; width: 100%; height: 50%; color: tan; margin: 70px 0; }
#mplayer::before { content: attr(data-cur); }
#mplayer::after { content: attr(data-dur); text-align: right; }
.btnplay { position: absolute; width: 40px; height: 20px; left: 50%; top: 48px; background: linear-gradient(to left,hsla(120,50%,70%,.6),hsla(60,80%,50%,.7)); border-radius: 0 100%; transform-origin: 0 0; animation: rot linear 4s infinite; animation-play-state: paused; }
.btnplay::before, .btnplay::after { position: absolute; width: 100%; height: 100%; content: ''; background: inherit; border-radius: inherit; transform-origin: 0 0; transform: rotate(120deg); }
.btnplay::after { transform: rotate(240deg); }
#prog { --posX: 0px; position: absolute; bottom: 10px; width: 100%; height: 1px; background: tan; cursor: pointer; }
#prog::before, #prog::after { position: absolute; content: ''; left: 0; }
#prog::before { width: var(--posX); height: 1px; background: green; }
#prog::after { left: var(--posX); top: calc(50% - 5px); width: 3px; height: 10px; background: darkgreen; }
#lrc { --motion: cover1; --tt: 1s; --state: paused; position: absolute; right: 20px; bottom: 15px; font: bold 2.4em sans-serif; color: hsl(240,100%,90%); -webkit-background-clip: text; filter: drop-shadow(1px 1px 2px hsla(0,100%,0%,.85)); }
#lrc::before { position: absolute; content: attr(data-lrc); width: 20%; height: 100%; color: transparent; overflow: hidden; white-space: nowrap; background: linear-gradient(180deg,hsla(120,30%,50%,.75),hsla(60,100%,50%,.65)); 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%; } }
@keyframes rot { to { transform: rotate(1turn); } }
</style>
<div id="papa">
<div id="lrc" data-lrc="花潮论坛lrc在线">花潮论坛lrc在线</div>
<div id="mplayer" data-cur="00:00" data-dur="00:00">
<span id="prog"></span>
<span class="btnplay"></span>
</div>
</div>
<audio id="aud" src="https://music.163.com/song/media/outer/url?id=26222194.mp3" autoplay loop>audio api</audio>
<script>
(function() {
let mKey = 0, mFlag = true, btnplay = document.querySelector('.btnplay');
let lrcAr = [,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,];
mplayer.onmousemove = (e) => {mplayer.style.cursor = e.offsetY > 90 ? 'pointer' : e.offsetY < 80 ? 'pointer' : 'default'; }
mplayer.onclick = (e) => {if(e.offsetY > 90) {aud.currentTime = aud.duration * e.offsetX / prog.offsetWidth;} else if(e.offsetY < 80) {aud.paused ? aud.play() : aud.pause();}}
aud.addEventListener('pause', () => mState());
aud.addEventListener('play', () => mState());
aud.addEventListener('seeked', () => calcKey());
aud.addEventListener('timeupdate', () => {prog.style.setProperty('--posX', prog.offsetWidth * aud.currentTime / aud.duration + 'px');mplayer.dataset.cur = toMin(aud.currentTime);mplayer.dataset.dur = toMin(aud.duration);for (j = 0; j < lrcAr.length; j++) {if (aud.currentTime >= lrcAr) {if (mKey === j) showLrc(lrcAr);else continue;}}});
let mState = () => aud.paused ? (btnplay.style.animationPlayState = 'paused', lrc.style.setProperty('--state', 'paused')) : (btnplay.style.animationPlayState = 'running', lrc.style.setProperty('--state', 'running'));
let showLrc = (time) => {let name = mFlag ? 'cover1' : 'cover2';lrc.innerHTML = lrc.dataset.lrc = lrcAr;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 < lrcAr.length; j ++) {if(aud.currentTime <= lrcAr) {mKey = j - 1;break;}}if(mKey <0) mKey = 0;if(mKey > lrcAr.length - 1) mKey = lrcAr.length - 1;let time = lrcAr - (aud.currentTime - 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; };
papa.oncontextmenu = () => false;
(function() {for(j=0; j<50; j++) {let leaf = document.createElement('span');leaf.className = 'btnplay';leaf.style.cssText = `left: 80%;top: 45%;transform: rotate(${Math.random() * 360}deg) translate(${Math.random() * 150}px);`;papa.appendChild(leaf);}})();
})();
</script>
代码
<style>
#papa { margin: auto; width: 1024px; height: 683px; background: silver url('https://638183.freep.cn/638183/t22/webp/qyrh.webp') no-repeat center/cover; box-shadow: 3px 3px 20px #000; display: grid; place-items: center; user-select: none; position: relative; z-index: 1; }
#mplayer { position: absolute; width: 400px; height: 110px; left: 20px; bottom: 10px; font: normal 14px sans-serif; z-index: 9; }
#mplayer::before, #mplayer::after { position: absolute; width: 100%; height: 50%; color: tan; margin: 70px 0; }
#mplayer::before { content: attr(data-cur); }
#mplayer::after { content: attr(data-dur); text-align: right; }
.btnplay { position: absolute; width: 40px; height: 20px; left: 50%; top: 48px; background: linear-gradient(to left,hsla(120,50%,70%,.6),hsla(60,80%,50%,.7)); border-radius: 0 100%; transform-origin: 0 0; animation: rot linear 4s infinite; animation-play-state: paused; }
.btnplay::before, .btnplay::after { position: absolute; width: 100%; height: 100%; content: ''; background: inherit; border-radius: inherit; transform-origin: 0 0; transform: rotate(120deg); }
.btnplay::after { transform: rotate(240deg); }
#prog { --posX: 0px; position: absolute; bottom: 10px; width: 100%; height: 1px; background: tan; cursor: pointer; }
#prog::before, #prog::after { position: absolute; content: ''; left: 0; }
#prog::before { width: var(--posX); height: 1px; background: green; }
#prog::after { left: var(--posX); top: calc(50% - 5px); width: 3px; height: 10px; background: darkgreen; }
#lrc { --motion: cover1; --tt: 1s; --state: paused; position: absolute; right: 20px; bottom: 15px; font: bold 2.4em sans-serif; color: hsl(240,100%,90%); -webkit-background-clip: text; filter: drop-shadow(1px 1px 2px hsla(0,100%,0%,.85)); }
#lrc::before { position: absolute; content: attr(data-lrc); width: 20%; height: 100%; color: transparent; overflow: hidden; white-space: nowrap; background: linear-gradient(180deg,hsla(120,30%,50%,.75),hsla(60,100%,50%,.65)); 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%; } }
@keyframes rot { to { transform: rotate(1turn); } }
</style>
<div id="papa">
<div id="lrc" data-lrc="花潮论坛lrc在线">花潮论坛lrc在线</div>
<div id="mplayer" data-cur="00:00" data-dur="00:00">
<span id="prog"></span>
<span class="btnplay"></span>
</div>
</div>
<audio id="aud" src="https://music.163.com/song/media/outer/url?id=26222194.mp3" autoplay loop>audio api</audio>
<script>
(function() {
let mKey = 0, mFlag = true, btnplay = document.querySelector('.btnplay');
let lrcAr = [,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,];
mplayer.onmousemove = (e) => {mplayer.style.cursor = e.offsetY > 90 ? 'pointer' : e.offsetY < 80 ? 'pointer' : 'default'; }
mplayer.onclick = (e) => {if(e.offsetY > 90) {aud.currentTime = aud.duration * e.offsetX / prog.offsetWidth;} else if(e.offsetY < 80) {aud.paused ? aud.play() : aud.pause();}}
aud.addEventListener('pause', () => mState());
aud.addEventListener('play', () => mState());
aud.addEventListener('seeked', () => calcKey());
aud.addEventListener('timeupdate', () => {prog.style.setProperty('--posX', prog.offsetWidth * aud.currentTime / aud.duration + 'px');mplayer.dataset.cur = toMin(aud.currentTime);mplayer.dataset.dur = toMin(aud.duration);for (j = 0; j < lrcAr.length; j++) {if (aud.currentTime >= lrcAr) {if (mKey === j) showLrc(lrcAr);else continue;}}});
let mState = () => aud.paused ? (btnplay.style.animationPlayState = 'paused', lrc.style.setProperty('--state', 'paused')) : (btnplay.style.animationPlayState = 'running', lrc.style.setProperty('--state', 'running'));
let showLrc = (time) => {let name = mFlag ? 'cover1' : 'cover2';lrc.innerHTML = lrc.dataset.lrc = lrcAr;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 < lrcAr.length; j ++) {if(aud.currentTime <= lrcAr) {mKey = j - 1;break;}}if(mKey <0) mKey = 0;if(mKey > lrcAr.length - 1) mKey = lrcAr.length - 1;let time = lrcAr - (aud.currentTime - 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; };
papa.oncontextmenu = () => false;
(function() {for(j=0; j<50; j++) {let leaf = document.createElement('span');leaf.className = 'btnplay';leaf.style.cssText = `left: 80%;top: 45%;transform: rotate(${Math.random() * 360}deg) translate(${Math.random() * 150}px);`;papa.appendChild(leaf);}})();
})();
</script>
本帖的三叶草按钮,变 id 为 class,CSS选择器里 #btnplay 换成 .btnplay,便于后面继续调用——帖子右边绕圈圈聚拢的三叶草,也是调用这个 class 选择器的,它们用JS追加到 id="papa" 的 div 标签里:
//自执行匿名函数
(function() {
for(j=0; j<50; j++) {
let leaf = document.createElement('span');
leaf.className = 'btnplay';
leaf.style.cssText = `
left: 80%;
top: 45%;
transform: rotate(${Math.random() * 360}deg) translate(${Math.random() * 150}px);
`;
papa.appendChild(leaf);
}
})();
红色那句,就是声明所创建的 50 个 span 标签,class="btnplay"。蓝色那句,让这些 span 标签及HTML中的三叶草,以一定的随机角度旋转、以一定的随机距离移位,因为它们之前固定在一个点上(left 和 top值固定它们在同一个点),所以旋转加移位的结果是绕圈圈,又因为多,所以出现堆叠现象(即使少也会有重合或部分重合现象的)。
细心的小盆友可能会发现,JS里,没有 btnplay 的点击事件。是的,没有。
播放控制器实际上是由 id="mplayer" 的 div 实现:鼠标在 > 90 的 Y轴方向点击,是控制进度,在 < 70 的 Y轴 方向点击,是控制播放/暂停。
这么做原因有二:① 三叶草只有在叶片上点击才有效,旋转时不是很好点对它们;② 进度条极细,也不方便控制进度。 火红的秋叶显得更美。 又一种三叶草按钮的播放器形式,漂亮{:4_187:} 三叶草被整体拿出来做画面效果,和上次的单个叶片的装饰也不同{:4_187:} 我这里网受限制,歌没法听,只能先看个画面效果{:4_204:} 马黑黑 发表于 2022-11-3 07:40
细心的小盆友可能会发现,JS里,没有 btnplay 的点击事件。是的,没有。
播放控制器实际上是由 id="mpla ...
这>90 和 <70 没搞明白是什么意思。
控制似乎并不困难,进度条以上点击控制启停,并不需要点在叶片上;进度条以下点击控制进度,在当前位置左边后退,右边前进。 起个网名好难 发表于 2022-11-3 11:34
这>90 和
现在当然不困难,因为是用了 mplayer 父框的尺寸做点击载体。90 和 70 我前面解释清楚了 红影 发表于 2022-11-3 10:38
我这里网受限制,歌没法听,只能先看个画面效果
今天不是大大前天的大大后天吗 梦油 发表于 2022-11-3 09:39
火红的秋叶显得更美。
取景也漂亮 红影 发表于 2022-11-3 10:34
又一种三叶草按钮的播放器形式,漂亮
形状是一样的,颜色不同而已 红影 发表于 2022-11-3 10:37
三叶草被整体拿出来做画面效果,和上次的单个叶片的装饰也不同
你说得对,三跟一是有区别的 起个网名好难 发表于 2022-11-3 11:34
这>90 和
但如果不用播放器父框的话,而是点击三叶草、进度条,是有一点点不顺手的 马黑黑 发表于 2022-11-3 12:44
现在当然不困难,因为是用了 mplayer 父框的尺寸做点击载体。90 和 70 我前面解释清楚了
90, 70 是角度又或着是长度,如果是长度哪里是计量的起点。 马黑黑 发表于 2022-11-3 12:45
取景也漂亮
是的,黑黑朋友。从你的作品中可以看出,你的审美观是上乘的 马黑黑 发表于 2022-11-3 12:45
形状是一样的,颜色不同而已
播放器进度条也不同了啊,两种都漂亮{:4_204:} 马黑黑 发表于 2022-11-3 12:46
你说得对,三跟一是有区别的
也是每次刷新位置有所不同{:4_173:} 马黑黑 发表于 2022-11-3 12:44
今天不是大大前天的大大后天吗
不是不是,明天才是{:4_173:}一会回家路上必须找地方再做个核酸了,省得过去后被关。