More to pulsing music player
Some checks failed
On Push Deploy / deploy (push) Has been cancelled

This commit is contained in:
Johnny322
2026-02-08 15:41:36 +01:00
parent 9e95f847b1
commit 74fd4fde32
2 changed files with 134 additions and 43 deletions

View File

@@ -155,20 +155,24 @@
</div> </div>
<div class="player-body"> <div class="player-body">
<div <div
v-if="currentClipUrl"
class="custom-player" class="custom-player"
:style="{ '--pulse': pulseLevel.toFixed(3) }" :class="{ idle: !currentClipUrl }"
:style="{
'--pulse': pulseLevel.toFixed(3),
'--ray': rayLevel.toFixed(3),
'--ray-hue': rayHue.toFixed(0)
}"
> >
<div class="pulse-orb" :class="{ active: isPlaying }"></div> <div
<div class="player-controls"> class="pulse-orb"
<button class="ghost" @click="togglePlayback"> :class="{
{{ isPlaying ? 'Pause' : 'Play' }} active: isPlaying,
</button> playing: isPlaying && !isAnswerClip,
<div class="player-meta"> answer: isPlaying && isAnswerClip,
<span class="label">Status</span> idle: !currentClipUrl
<span class="value">{{ isPlaying ? 'Playing' : 'Paused' }}</span> }"
</div> ></div>
</div> <div class="pulse-rays" :class="{ active: isPlaying }"></div>
</div> </div>
<audio <audio
ref="player" ref="player"
@@ -180,7 +184,7 @@
@pause="handlePlayerPause" @pause="handlePlayerPause"
@ended="handlePlayerPause" @ended="handlePlayerPause"
></audio> ></audio>
<div v-else class="player-empty"> <div v-if="!currentClipUrl" class="player-empty">
Pick a tile to start the round. Pick a tile to start the round.
</div> </div>
</div> </div>
@@ -260,9 +264,13 @@ export default {
audioContext: null as AudioContext | null, audioContext: null as AudioContext | null,
analyser: null as AnalyserNode | null, analyser: null as AnalyserNode | null,
analyserData: null as Uint8Array | null, analyserData: null as Uint8Array | null,
frequencyData: null as Uint8Array | null,
rafId: 0, rafId: 0,
mediaSource: null as MediaElementAudioSourceNode | null, mediaSource: null as MediaElementAudioSourceNode | null,
mediaElement: null as HTMLAudioElement | null mediaElement: null as HTMLAudioElement | null,
rayLevel: 0,
rayHue: 200,
isAnswerClip: false
} }
}, },
computed: { computed: {
@@ -335,7 +343,9 @@ export default {
this.analyser = analyser this.analyser = analyser
this.analyserData = new Uint8Array(analyser.fftSize) this.analyserData = new Uint8Array(analyser.fftSize)
this.frequencyData = new Uint8Array(analyser.frequencyBinCount)
this.pulseLevel = 0 this.pulseLevel = 0
this.rayLevel = 0
}, },
teardownAudio() { teardownAudio() {
if (this.rafId) { if (this.rafId) {
@@ -352,28 +362,47 @@ export default {
} }
this.mediaElement = null this.mediaElement = null
this.analyserData = null this.analyserData = null
this.frequencyData = null
this.isPlaying = false this.isPlaying = false
this.pulseLevel = 0 this.pulseLevel = 0
this.rayLevel = 0
}, },
startPulse() { startPulse() {
if (!this.analyser || !this.analyserData) return if (!this.analyser || !this.analyserData || !this.frequencyData) return
const analyser = this.analyser const analyser = this.analyser
const data = this.analyserData const data = this.analyserData
const freq = this.frequencyData
const tick = () => { const tick = () => {
if (!this.isPlaying || !this.analyser || !this.analyserData) { if (!this.isPlaying || !this.analyser || !this.analyserData || !this.frequencyData) {
this.rafId = 0 this.rafId = 0
return return
} }
analyser.getByteTimeDomainData(data) analyser.getByteTimeDomainData(data)
analyser.getByteFrequencyData(freq)
let sumSquares = 0 let sumSquares = 0
for (let i = 0; i < data.length; i += 1) { for (let i = 0; i < data.length; i += 1) {
const centered = (data[i] - 128) / 128 const centered = (data[i] - 128) / 128
sumSquares += centered * centered sumSquares += centered * centered
} }
const rms = Math.sqrt(sumSquares / data.length) const rms = Math.sqrt(sumSquares / data.length)
const target = Math.min(1, rms * 2.8) const target = Math.min(1, rms * 3.6)
this.pulseLevel = this.pulseLevel * 0.7 + target * 0.3 this.pulseLevel = this.pulseLevel * 0.65 + target * 0.35
const lowBandEnd = Math.floor(freq.length * 0.2)
const highBandStart = Math.floor(freq.length * 0.55)
let lowSum = 0
for (let i = 0; i < lowBandEnd; i += 1) lowSum += freq[i]
const lowAvg = lowSum / Math.max(1, lowBandEnd) / 255
let highSum = 0
for (let i = highBandStart; i < freq.length; i += 1) highSum += freq[i]
const highAvg = highSum / Math.max(1, freq.length - highBandStart) / 255
const rayTarget = Math.min(1, (lowAvg * 0.6 + highAvg * 0.9) * 1.4)
this.rayLevel = this.rayLevel * 0.7 + rayTarget * 0.3
const hueTarget = 180 + highAvg * 120
this.rayHue = this.rayHue * 0.8 + hueTarget * 0.2
this.rafId = requestAnimationFrame(tick) this.rafId = requestAnimationFrame(tick)
} }
@@ -435,6 +464,7 @@ export default {
this.currentTileKey = null this.currentTileKey = null
this.currentClipUrl = '' this.currentClipUrl = ''
this.lastAwardedTeamId = null this.lastAwardedTeamId = null
this.isAnswerClip = false
this.teardownAudio() this.teardownAudio()
}, },
tileKey(cIndex: number, qIndex: number) { tileKey(cIndex: number, qIndex: number) {
@@ -479,6 +509,7 @@ export default {
this.tiles[key] = { status: 'playing', lastTeamId: null } this.tiles[key] = { status: 'playing', lastTeamId: null }
this.currentTileKey = key this.currentTileKey = key
this.currentClipUrl = encodeURI(clue.song) this.currentClipUrl = encodeURI(clue.song)
this.isAnswerClip = false
await nextTick() await nextTick()
const player = this.getPlayer() const player = this.getPlayer()
if (player) { if (player) {
@@ -502,6 +533,7 @@ export default {
this.lastAwardedTeamId = null this.lastAwardedTeamId = null
if (clue.answer) { if (clue.answer) {
this.currentClipUrl = encodeURI(clue.answer) this.currentClipUrl = encodeURI(clue.answer)
this.isAnswerClip = true
await nextTick() await nextTick()
const player = this.getPlayer() const player = this.getPlayer()
if (player) { if (player) {
@@ -541,6 +573,7 @@ export default {
this.tiles[key].status = 'void' this.tiles[key].status = 'void'
this.currentTileKey = null this.currentTileKey = null
this.currentClipUrl = '' this.currentClipUrl = ''
this.isAnswerClip = false
const player = this.getPlayer() const player = this.getPlayer()
player?.pause() player?.pause()
this.teardownAudio() this.teardownAudio()

View File

@@ -386,6 +386,10 @@ audio.hidden-audio {
overflow: hidden; overflow: hidden;
} }
.custom-player.idle {
background: radial-gradient(circle at top, rgba(20, 20, 20, 0.5), rgba(5, 6, 12, 0.95) 70%);
}
.custom-player::after { .custom-player::after {
content: ''; content: '';
position: absolute; position: absolute;
@@ -400,39 +404,72 @@ audio.hidden-audio {
height: 96px; height: 96px;
border-radius: 50%; border-radius: 50%;
background: radial-gradient(circle at 30% 30%, #fbd72b, #f58b2b 60%, #1d3b8b); background: radial-gradient(circle at 30% 30%, #fbd72b, #f58b2b 60%, #1d3b8b);
transform: scale(calc(1 + var(--pulse, 0) * 0.22)); transform: scale(calc(1 + var(--pulse, 0) * 0.45));
box-shadow: 0 0 calc(20px + var(--pulse, 0) * 60px) rgba(251, 215, 43, 0.35); box-shadow:
transition: transform 0.08s ease-out; 0 0 calc(18px + var(--pulse, 0) * 90px) rgba(251, 215, 43, 0.5),
0 0 calc(40px + var(--pulse, 0) * 160px) rgba(61, 214, 255, 0.35);
transition: transform 0.06s ease-out;
position: relative;
z-index: 2;
}
.pulse-orb.idle {
background: radial-gradient(circle at 40% 40%, #1a1a1a, #000 65%);
box-shadow: 0 0 12px rgba(0, 0, 0, 0.6);
transform: scale(1);
}
.pulse-orb.playing {
background: linear-gradient(
135deg,
#7df9ff,
#6c63ff,
#ff7ad9,
#ffd36a,
#7df9ff
);
background-size: 240% 240%;
animation: gradientShift 6s ease-in-out infinite;
}
.pulse-orb.answer {
background: radial-gradient(circle at 30% 30%, #6bff9f, #1ecb5a 65%, #0b5c2b);
box-shadow:
0 0 calc(18px + var(--pulse, 0) * 90px) rgba(43, 220, 123, 0.6),
0 0 calc(40px + var(--pulse, 0) * 160px) rgba(43, 220, 123, 0.3);
} }
.pulse-orb.active { .pulse-orb.active {
animation: pulseGlow 1.6s ease-in-out infinite; animation: pulseGlow 1.6s ease-in-out infinite;
} }
.player-controls { .pulse-rays {
display: grid; position: absolute;
gap: 12px; inset: -20%;
place-items: center; border-radius: 50%;
background:
conic-gradient(
from 0deg,
hsla(var(--ray-hue, 200), 90%, 70%, 0.3) 0deg,
transparent 30deg,
hsla(var(--ray-hue, 200), 90%, 70%, 0.45) 60deg,
transparent 95deg,
hsla(var(--ray-hue, 200), 90%, 70%, 0.25) 130deg,
transparent 170deg,
hsla(var(--ray-hue, 200), 90%, 70%, 0.35) 210deg,
transparent 250deg,
hsla(var(--ray-hue, 200), 90%, 70%, 0.4) 300deg,
transparent 340deg
);
opacity: calc(0.1 + var(--ray, 0) * 0.9);
filter: blur(6px);
transform: scale(calc(0.9 + var(--ray, 0) * 0.6));
transition: opacity 0.1s ease-out, transform 0.1s ease-out;
z-index: 1;
} }
.player-meta { .pulse-rays.active {
text-align: center; animation: raySpin 6s linear infinite;
font-size: 0.85rem;
color: rgba(255, 255, 255, 0.7);
}
.player-meta .label {
text-transform: uppercase;
letter-spacing: 0.2em;
font-size: 0.65rem;
display: block;
}
.player-meta .value {
display: block;
color: #f6f4ef;
font-weight: 600;
margin-top: 4px;
} }
@keyframes pulseGlow { @keyframes pulseGlow {
@@ -445,6 +482,27 @@ audio.hidden-audio {
} }
} }
@keyframes gradientShift {
0% {
background-position: 0% 50%;
}
50% {
background-position: 100% 50%;
}
100% {
background-position: 0% 50%;
}
}
@keyframes raySpin {
from {
transform: scale(calc(0.9 + var(--ray, 0) * 0.6)) rotate(0deg);
}
to {
transform: scale(calc(0.9 + var(--ray, 0) * 0.6)) rotate(360deg);
}
}
.player-empty { .player-empty {
color: rgba(255, 255, 255, 0.6); color: rgba(255, 255, 255, 0.6);
} }