]> OzVa Git service - shopping-channel/commitdiff
Added sound pannel
authorMax Value <greenwoodw50@gmail.com>
Fri, 28 Mar 2025 23:37:25 +0000 (23:37 +0000)
committerMax Value <greenwoodw50@gmail.com>
Fri, 28 Mar 2025 23:37:25 +0000 (23:37 +0000)
+ added html sound page
+ added clock.wav
+ added impulse responce (impulse.wav)
~ page handling to different functions
~ changed gitignore

todo
+ timer sfx

.gitignore
static/sounds/clock.wav [new file with mode: 0644]
static/sounds/impulse.wav [new file with mode: 0644]
teleshopping.py
templates/sounds.html [new file with mode: 0644]

index 05e655d6beb196081923a5dfa6b88b3e7043aaae..5fd2490344d8be2089ebd5ce7989c34f3d84cddf 100644 (file)
@@ -1,3 +1,3 @@
 *.wsgi
-data.db
+*.db
 docs/*
diff --git a/static/sounds/clock.wav b/static/sounds/clock.wav
new file mode 100644 (file)
index 0000000..a760531
Binary files /dev/null and b/static/sounds/clock.wav differ
diff --git a/static/sounds/impulse.wav b/static/sounds/impulse.wav
new file mode 100644 (file)
index 0000000..2d66562
Binary files /dev/null and b/static/sounds/impulse.wav differ
index a860e35d85ca978435fe685dc0ba775dd677c919..bd9064f697b9c928b3bda23c2e06f3d616411af7 100755 (executable)
@@ -62,9 +62,13 @@ def gfx_main():
        return Response(render_template("gfx.html"), mimetype="text/html")
 
 @app.route("/autocue")
-def gfx_page(page):
+def gfx_page():
        return Response(render_template("autocue.html"), mimetype="text/html")
 
+@app.route("/sounds")
+def sounds_page():
+       return Response(render_template("sounds.html"), mimetype="text/html")
+
 
 
 
diff --git a/templates/sounds.html b/templates/sounds.html
new file mode 100644 (file)
index 0000000..756e16e
--- /dev/null
@@ -0,0 +1,231 @@
+<!DOCTYPE html>
+<html lang="en">
+       <head>
+               <meta charset="utf-8">
+               <title>XMDV</title>
+       </head>
+       <body>
+               <h2>Timers</h2>
+               <table>
+                       <tr>
+                               <td><audio id="timer1" src=/static/sounds/timer1.wav controls></audio></td>
+                               <td>Timer 1</td>
+                       </tr>
+                       <tr>
+                               <td><audio id="timer2" src=/static/sounds/timer2.wav controls></audio></td>
+                               <td>Timer 1</td>
+                       </tr>
+                       <tr>
+                               <td><audio id="timer3" src=/static/sounds/timer3.wav controls></audio></td>
+                               <td>Timer 1</td>
+                       </tr>
+                       <tr>
+                               <td><audio id="timer4" src=/static/sounds/timer4.wav controls></audio></td>
+                               <td>Timer 1</td>
+                       </tr>
+                       <tr>
+                               <td><audio id="timer5" src=/static/sounds/timer5.wav controls></audio></td>
+                               <td>Timer 1</td>
+                       </tr>
+                       <tr>
+                               <td><audio id="timer6" src=/static/sounds/timer6.wav controls></audio></td>
+                               <td>Timer 1</td>
+                       </tr>
+               </table>
+               <h2>Doomsday</h2>
+               <audio id="clock" src=/static/sounds/clock.wav controls></audio>
+               <br>
+               <input type="button" onclick="start();" value="Start listen" id="start" /> <em style="color:green" id="state"></em>
+
+               <p>
+                       Above are all the audio elements that will play out the SFX for the timers and the clock. Feel free to test them out. When you click the start button you'll still have full control over the audio elements but the system will also automatically fade them in when a timer is updated or when a clock tick is triggered. If you do step in and stop one of them, it wont play until its triggered again. For the timers it will automatically seek so that the sound file ends when the timer does. Note that the internal tracker for the clock will update on start, but the timers will not. This means that the system will try to "catch-up" and play sounds for any running timers.
+               </p>
+
+               <script>
+
+// todo
+// - add leveling on clock tick
+// - potentially rework leveling system
+
+const fadeLength = 4.;
+const levelLength = 4; // how long it takes for the level to converge on the target
+const effectLength = 6; // how long it takes for the effects to converge on the target
+
+let pastTime = [
+               0,
+               0,
+               0,
+               0,
+               0,
+               0
+       ];
+let pastClock = 0;
+let timersPlaying = 0;
+
+const context = new AudioContext();
+
+// helper function for creating the convolution reverb
+async function createReverb() {
+       let reverb = context.createConvolver();
+
+       // load impulse response from file
+       let response = await fetch("/static/sounds/impulse.wav");
+       let arrayBuffer = await response.arrayBuffer();
+       reverb.buffer = await context.decodeAudioData(arrayBuffer);
+
+       return reverb;
+}
+
+const amp = context.createGain();
+const filter = new BiquadFilterNode(context, options={type:"lowpass",frequency:1000});
+const reverb = createReverb();
+const wetGain = context.createGain();
+const dryGain = context.createGain();
+const compressor = new DynamicsCompressorNode(context, options={ratio:4});
+
+wetGain.gain.value = 0;
+
+const timers = [
+       document.getElementById("timer1"),
+       document.getElementById("timer2"),
+       document.getElementById("timer3"),
+       document.getElementById("timer4"),
+       document.getElementById("timer5"),
+       document.getElementById("timer6"),
+       document.getElementById("clock")
+       ];
+const clock = document.getElementById("clock");
+
+for (let t = 0; t < 6; t++) {
+       let timer = timers[t];
+       let track = context.createMediaElementSource(timer);
+       track.connect(amp);
+}
+
+
+reverb.then(reverb => {
+       track = context.createMediaElementSource(clock);
+       track.connect(compressor);
+       amp.connect(filter)
+       filter.connect(reverb)
+       reverb.connect(wetGain)
+       wetGain.connect(compressor);
+       amp.connect(dryGain).connect(compressor)
+
+       compressor.connect(context.destination);
+})
+
+function getTimes() {
+       fetch("./api", {cache: "no-store"})
+               .then(data => data.json())
+               .then(data => {
+
+                       if (pastClock != data["current_position"]) {
+                               pastClock = data["current_position"]
+                               if (context.state === "suspended") {
+                                       context.resume();
+                               }
+
+                               console.log(`Clock tick`);
+                               clock.currentTime = 0.0;
+                               clock.play();
+                       }
+
+                       timersPlaying = 0;
+                       for (let t = 0; t < 6; t++) {
+                               // if the timer is playing
+                               let playing = false;
+                               if (timers[t].currentTime > 0 && !timers[t].paused && !timers[t].ended) {
+                                       timersPlaying += 1;
+                                       playing = true;
+                               }
+
+                               let timeEnd = data[`end_timer_${t+1}`];
+                               let current = Math.round((Date.now() + data['timer_offset']) / 1000);
+                               let time = timeEnd - current;
+
+                               if (timeEnd != pastTime[t]) {
+                                       pastTime[t] = timeEnd;
+
+                                       //console.log(`Timer ${t} changed (${time}s)`);
+
+                                       if (time > 0) {
+                                               if (context.state === "suspended") {
+                                                       context.resume();
+                                               }
+
+                                               console.log(`Starting timer ${t}`);
+                                               timers[t].currentTime = timers[t].duration - time;
+                                               timers[t].volume = 1;
+                                               timers[t].play();
+
+                                               if (!playing) {
+                                                       timers[t].volume = 0;
+                                                       let fadeIn = setInterval(function () {
+                                                               if (timers[t].volume < 1.0) {
+                                                                       if (timers[t].volume + 0.2 / fadeLength > 1) {timers[t].volume = 1.0;}
+                                                                       else {timers[t].volume += 0.2 / fadeLength;} // 0.2 related to interval time
+                                                               } else {
+
+                                                                       clearInterval(fadeIn);
+                                                               }
+                                                       }, 200);
+                                               }
+                                       } else {
+                                               timers[t].pause();
+                                       }
+                               }
+                       }
+               });
+}
+
+function updateLeveler() {
+       let targetLevel = 0.5;
+       let targetWet = 0;
+       if (timersPlaying > 2) {
+               targetLevel = 1 / timersPlaying;
+       }
+       if (clock.currentTime > 0 && !clock.paused && !clock.ended) {
+               //targetLevel *= 0.75;
+               targetWet = 0.75;
+       }
+
+       if (Math.abs(targetLevel - amp.gain.value) < 0.1 / levelLength) {
+               amp.gain.value = targetLevel;
+       } else if (targetLevel > amp.gain.value) {
+               amp.gain.value += 0.1 / levelLength;
+       } else if (targetLevel < amp.gain.value) {
+               amp.gain.value -= 0.1 / levelLength;
+       }
+       //console.log(`${timersPlaying} Timers playing so target volume is ${targetLevel * 100}%. Current volume is ${amp.gain.value * 100}%.`);
+
+       if (Math.abs(targetWet - wetGain.gain.value) < 0.1 / effectLength) {
+               wetGain.gain.value = targetWet;
+       } else if (targetWet > wetGain.gain.value) {
+               wetGain.gain.value += 0.1 / effectLength;
+       } else if (targetWet < wetGain.gain.value) {
+               wetGain.gain.value -= 0.1 / effectLength;
+       }
+       dryGain.gain.value = 1 - wetGain.gain.value;
+       //console.log(`Target wet is ${targetWet * 100}%. Current wet is ${wetGain.gain.value * 100}%.`);
+}
+
+let repeat;
+let leveler;
+function start() {
+       document.getElementById("state").innerHTML = "Currently armed...";
+       document.getElementById("start").disabled = true;
+
+       clearInterval(repeat);
+       clearInterval(leveler);
+       repeat = setInterval(getTimes, 500);
+       leveler = setInterval(updateLeveler, 100);
+
+       fetch("./api", {cache: "no-store"})
+               .then(data => data.json())
+               .then(data => {pastClock = data["current_position"];})
+}
+
+               </script>
+       </body>
+</html>