implement global mic, muting, get chrome audio fully working again
This commit is contained in:
parent
e35e5ae2fe
commit
3c54a5aafd
|
@ -1,8 +1,9 @@
|
||||||
import { derived, type Readable, type Unsubscriber, writable } from "svelte/store";
|
import { derived, type Readable, type Unsubscriber, writable } from "svelte/store";
|
||||||
import { Player } from "./client";
|
import { Player } from "./client";
|
||||||
|
import { localStore, subscribe } from "./stores";
|
||||||
|
|
||||||
export class AudioManager {
|
export class AudioManager {
|
||||||
globalGain = writable(1.0);
|
micGain: Readable<number>;
|
||||||
useMusic: boolean;
|
useMusic: boolean;
|
||||||
ac: AudioContext;
|
ac: AudioContext;
|
||||||
dest: MediaStreamAudioDestinationNode;
|
dest: MediaStreamAudioDestinationNode;
|
||||||
|
@ -10,7 +11,7 @@ export class AudioManager {
|
||||||
source: AudioNode;
|
source: AudioNode;
|
||||||
public readonly output: MediaStream;
|
public readonly output: MediaStream;
|
||||||
|
|
||||||
constructor(media: HTMLAudioElement | MediaStream) {
|
constructor(media: HTMLAudioElement | MediaStream, micVolume: Readable<number>, muted: Readable<boolean>) {
|
||||||
this.ac = new AudioContext();
|
this.ac = new AudioContext();
|
||||||
this.dest = this.ac.createMediaStreamDestination();
|
this.dest = this.ac.createMediaStreamDestination();
|
||||||
this.gain = this.ac.createGain();
|
this.gain = this.ac.createGain();
|
||||||
|
@ -26,68 +27,82 @@ export class AudioManager {
|
||||||
this.source = this.ac.createMediaStreamSource(media);
|
this.source = this.ac.createMediaStreamSource(media);
|
||||||
this.useMusic = false;
|
this.useMusic = false;
|
||||||
}
|
}
|
||||||
this.gain.gain.value /= 32;
|
subscribe([micVolume, muted], ([micVolume, muted]) => {
|
||||||
|
this.gain.gain.value = micVolume * +!muted;
|
||||||
|
});
|
||||||
this.source.connect(this.gain).connect(this.dest)
|
this.source.connect(this.gain).connect(this.dest)
|
||||||
this.output = this.dest.stream;
|
this.output = this.dest.stream;
|
||||||
}
|
}
|
||||||
|
|
||||||
createPlayerAudio(outputDiv: HTMLDivElement, localPlayer: Readable<Player | undefined>, peerPlayer: Readable<Player | undefined>) {
|
createPlayerAudio(
|
||||||
const panner = this.ac.createPanner();
|
outputDiv: HTMLDivElement,
|
||||||
panner.distanceModel = "linear";
|
localPlayer: Readable<Player | undefined>,
|
||||||
panner.refDistance = 1500;
|
peerPlayer: Readable<Player | undefined>,
|
||||||
panner.maxDistance = 2500;
|
globalSpeak: Readable<boolean>,
|
||||||
|
) {
|
||||||
const gain = this.ac.createGain();
|
const gain = this.ac.createGain();
|
||||||
const dest = this.ac.createMediaStreamDestination();
|
const dest = this.ac.createMediaStreamDestination();
|
||||||
gain.connect(dest); //.connect(panner)
|
|
||||||
|
|
||||||
|
const isDev = ((globalThis as any).isDev = "hot" in import.meta);
|
||||||
let audio = document.createElement("audio");
|
let audio = document.createElement("audio");
|
||||||
audio.srcObject = dest.stream;
|
audio.srcObject = dest.stream;
|
||||||
audio.autoplay = true;
|
audio.autoplay = true;
|
||||||
audio.controls = false;
|
audio.controls = isDev;
|
||||||
|
audio.play();
|
||||||
outputDiv.appendChild(audio);
|
outputDiv.appendChild(audio);
|
||||||
|
|
||||||
return new PlayerAudio(this.ac, panner, gain, localPlayer, peerPlayer);
|
return new PlayerAudio(this.ac, dest, gain, localPlayer, peerPlayer, globalSpeak);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export class PlayerAudio {
|
export class PlayerAudio {
|
||||||
private source: MediaStreamAudioSourceNode | undefined;
|
private source: MediaStreamAudioSourceNode | undefined;
|
||||||
private playerWatcher: Readable<[number[], boolean]>;
|
// private playerWatcher: Readable<[number[], boolean]>;
|
||||||
constructor(
|
constructor(
|
||||||
private context: AudioContext,
|
private context: AudioContext,
|
||||||
private panner: PannerNode,
|
private dest: MediaStreamAudioDestinationNode,
|
||||||
private gain: GainNode,
|
private gain: GainNode,
|
||||||
localPlayer: Readable<Player | undefined>,
|
localPlayer: Readable<Player | undefined>,
|
||||||
peerPlayer: Readable<Player | undefined>
|
peerPlayer: Readable<Player | undefined>,
|
||||||
|
globalSpeak: Readable<boolean>,
|
||||||
) {
|
) {
|
||||||
this.playerWatcher = derived([localPlayer, peerPlayer], ([localPlayer, peerPlayer], set) => {
|
console.log("created player audio");
|
||||||
|
|
||||||
|
subscribe([localPlayer, peerPlayer], ([localPlayer, peerPlayer]) => {
|
||||||
if (!localPlayer || !peerPlayer) {
|
if (!localPlayer || !peerPlayer) {
|
||||||
console.log("got player binding")
|
this.gain.gain.value = 0;
|
||||||
set([[0, 0, 0], false]);
|
// console.warn("missing a player...");
|
||||||
return () => { };
|
return () => { };
|
||||||
}
|
}
|
||||||
|
|
||||||
return derived([localPlayer.stageInfo, peerPlayer.stageInfo], ([localInfo, peerInfo]): [number[], boolean] => {
|
console.log("got player binding for (local, peer)", localPlayer.name, peerPlayer.name);
|
||||||
return [
|
|
||||||
localInfo.position.map((localPosition, i) => peerInfo.position[i] - localPosition),
|
|
||||||
localInfo.stage === peerInfo.stage
|
|
||||||
];
|
|
||||||
}).subscribe(set);
|
|
||||||
});
|
|
||||||
|
|
||||||
this.playerWatcher.subscribe(([[x, y, z], sameStage]) => {
|
return subscribe([localPlayer.proximityInfo, peerPlayer.proximityInfo, globalSpeak], ([localInfo, peerInfo, globalSpeak]) => {
|
||||||
// this.panner.positionX.value = x;
|
const [x, y, z] = localInfo.position.map((localPosition, i) => peerInfo.position[i] - localPosition);
|
||||||
// this.panner.positionY.value = y;
|
const sameStage = localInfo.stage === peerInfo.stage;
|
||||||
// this.panner.positionZ.value = z;
|
|
||||||
let magnitude = Math.sqrt(Math.pow(x, 2) + Math.pow(y, 2) + Math.pow(z, 2));
|
let magnitude = Math.sqrt(Math.pow(x, 2) + Math.pow(y, 2) + Math.pow(z, 2));
|
||||||
const max = 2500, min = 1500;
|
const max = 3500, min = 750;
|
||||||
let scaledGain = 1 - (magnitude - min) / max;
|
let scaledGain = 1 - (magnitude - min) / max;
|
||||||
this.gain.gain.value = scaledGain * +sameStage;
|
let t = Math.max(Math.min(scaledGain * +sameStage, 1), 0);
|
||||||
|
function easeInExpo(x: number): number {
|
||||||
|
return x === 0 ? 0 : Math.pow(2, 10 * x - 10);
|
||||||
|
}
|
||||||
|
this.gain.gain.value = globalSpeak ? 1 : easeInExpo(t);
|
||||||
|
});
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
gotTrack(stream: MediaStream) {
|
gotTrack(stream: MediaStream) {
|
||||||
|
let a: HTMLAudioElement | null = new Audio();
|
||||||
|
a.muted = true;
|
||||||
|
a.srcObject = stream;
|
||||||
|
a.addEventListener('canplaythrough', () => {
|
||||||
|
a = null;
|
||||||
|
});
|
||||||
|
a.play();
|
||||||
this.source = this.context.createMediaStreamSource(stream);
|
this.source = this.context.createMediaStreamSource(stream);
|
||||||
this.source.connect(this.gain);
|
let gainOut = this.source.connect(this.gain);
|
||||||
|
gainOut.connect(this.dest);
|
||||||
|
// gainOut.connect(this.context.destination);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -6,14 +6,14 @@ import { ReadKind, readEvent } from "./protocol/serverEvent";
|
||||||
import { AudioManager, PlayerAudio } from "./audio";
|
import { AudioManager, PlayerAudio } from "./audio";
|
||||||
import { type WriteEvent, writeEvent, WriteKind } from "./protocol/clientEvent";
|
import { type WriteEvent, writeEvent, WriteKind } from "./protocol/clientEvent";
|
||||||
import { BinaryWriter } from "./binary/writer";
|
import { BinaryWriter } from "./binary/writer";
|
||||||
import { cached, type CachedWritable, type Cached, cachedWritable } from "./stores";
|
import { cached, type CachedWritable, type Cached, cachedWritable, subscribe } from "./stores";
|
||||||
import { BinaryReader } from "./binary/reader";
|
import { BinaryReader } from "./binary/reader";
|
||||||
import { base } from "$app/paths";
|
// @ts-ignore
|
||||||
|
import { PUBLIC_WT_URL } from "$env/static/public";
|
||||||
|
|
||||||
export class Client {
|
export class Client {
|
||||||
static async connect(name: Readable<string>, audioManager: AudioManager, outputAudio: HTMLDivElement) {
|
static async connect(name: Readable<string>, globalSpeak: Readable<boolean>, audioManager: AudioManager, outputAudio: HTMLDivElement) {
|
||||||
const settings = await (await fetch("settings.json")).json();
|
const transport = new WebTransport(PUBLIC_WT_URL, { allowPooling: false });
|
||||||
const transport = new WebTransport(`https://${settings.host}:${settings.port}`, { allowPooling: false });
|
|
||||||
await transport.ready;
|
await transport.ready;
|
||||||
console.log("ready!");
|
console.log("ready!");
|
||||||
const stream = await transport.createBidirectionalStream();
|
const stream = await transport.createBidirectionalStream();
|
||||||
|
@ -24,17 +24,20 @@ export class Client {
|
||||||
const array = new Uint8Array(0x20);
|
const array = new Uint8Array(0x20);
|
||||||
new TextEncoder().encodeInto(get(name), array);
|
new TextEncoder().encodeInto(get(name), array);
|
||||||
await writer.write(array);
|
await writer.write(array);
|
||||||
|
writer.write(new Uint8Array([+get(globalSpeak)]))
|
||||||
|
|
||||||
const reader = BinaryStreamReader.fromReader(stream.readable.getReader());
|
const reader = BinaryStreamReader.fromReader(stream.readable.getReader());
|
||||||
const { players, peers } = await readHello(reader);
|
const { players, peers } = await readHello(reader);
|
||||||
console.log("done reading hello");
|
console.log("done reading hello");
|
||||||
|
|
||||||
const client = new Client(audioManager, outputAudio, name, cachedWritable(players), uuid, transport, stream, writer);
|
const client = new Client(audioManager, outputAudio, name, globalSpeak, cachedWritable(players), uuid, transport, stream, writer);
|
||||||
globalThis.client = client;
|
globalThis.client = client;
|
||||||
|
globalThis.get = get;
|
||||||
|
|
||||||
for (const [id, name] of peers) {
|
for (const [id, name, globalSpeak] of peers) {
|
||||||
const peer = client.getPeer(id);
|
const peer = client.getPeer(id);
|
||||||
peer.targetPlayerName.set(name);
|
peer.targetPlayerName.set(name);
|
||||||
|
peer.globalSpeak.set(globalSpeak)
|
||||||
peer.offer();
|
peer.offer();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -49,6 +52,7 @@ export class Client {
|
||||||
private async closeHandler() {
|
private async closeHandler() {
|
||||||
await this.transport.closed;
|
await this.transport.closed;
|
||||||
console.error("closed");
|
console.error("closed");
|
||||||
|
this.connected.set(false);
|
||||||
}
|
}
|
||||||
|
|
||||||
private getPeer(id: string) {
|
private getPeer(id: string) {
|
||||||
|
@ -93,25 +97,27 @@ export class Client {
|
||||||
return x;
|
return x;
|
||||||
})
|
})
|
||||||
break;
|
break;
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
case ReadKind.Moved: {
|
case ReadKind.Moved: {
|
||||||
const player = this.players.get()[event.id];
|
const player = this.players.get()[event.id];
|
||||||
if (player)
|
if (player)
|
||||||
player.stageInfo.update(info => Object.assign(info, { position: event.position }));
|
player.proximityInfo.update(info => Object.assign(info, { position: event.position }));
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case ReadKind.StageChanged: {
|
case ReadKind.StageChanged: {
|
||||||
console.log("goodbye", event.id);
|
console.log("goodbye", event.id);
|
||||||
const player = this.players.get()[event.id];
|
const player = this.players.get()[event.id];
|
||||||
if (player)
|
if (player)
|
||||||
player.stageInfo.update(info => Object.assign(info, { stage: event.stage }));
|
player.proximityInfo.update(info => Object.assign(info, { stage: event.stage }));
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case ReadKind.PeerConnectionChanged: {
|
case ReadKind.PeerConnectionChanged: {
|
||||||
console.log("peer changed", event.id, event.connected, event.target);
|
console.log("peer changed", event.id, event.connected, event.target);
|
||||||
if (event.connected)
|
if (event.connected) {
|
||||||
this.getPeer(event.id).targetPlayerName.set(event.target);
|
const peer = this.getPeer(event.id);
|
||||||
|
peer.targetPlayerName.set(event.target);
|
||||||
|
peer.globalSpeak.set(event.globalSpeak);
|
||||||
|
}
|
||||||
else
|
else
|
||||||
delete this.peers[event.id];
|
delete this.peers[event.id];
|
||||||
break;
|
break;
|
||||||
|
@ -135,11 +141,13 @@ export class Client {
|
||||||
private peers: Record<string, Peer> = {};
|
private peers: Record<string, Peer> = {};
|
||||||
public boundPlayer: Readable<Player | undefined>;
|
public boundPlayer: Readable<Player | undefined>;
|
||||||
public playersByName: Cached<Record<string, Player>>;
|
public playersByName: Cached<Record<string, Player>>;
|
||||||
|
public connected: Writable<boolean> = writable(true);
|
||||||
|
|
||||||
private constructor(
|
private constructor(
|
||||||
public audioManager: AudioManager,
|
public audioManager: AudioManager,
|
||||||
public outputAudio: HTMLDivElement,
|
public outputAudio: HTMLDivElement,
|
||||||
boundPlayerName: Readable<string>,
|
boundPlayerName: Readable<string>,
|
||||||
|
globalSpeak: Readable<boolean>,
|
||||||
public players: CachedWritable<Record<number, Player>>,
|
public players: CachedWritable<Record<number, Player>>,
|
||||||
public uuid: string,
|
public uuid: string,
|
||||||
public transport: WebTransport,
|
public transport: WebTransport,
|
||||||
|
@ -149,26 +157,30 @@ export class Client {
|
||||||
this.playersByName = cached(this.players, (players) => {
|
this.playersByName = cached(this.players, (players) => {
|
||||||
console.log("players by name derivation", Object.values(players), Object.fromEntries(Object.values(players).map(x => [x.name, x])))
|
console.log("players by name derivation", Object.values(players), Object.fromEntries(Object.values(players).map(x => [x.name, x])))
|
||||||
return Object.fromEntries(Object.values(players).map(x => [x.name, x]))
|
return Object.fromEntries(Object.values(players).map(x => [x.name, x]))
|
||||||
})
|
});
|
||||||
|
subscribe([boundPlayerName, globalSpeak], ([name, globalSpeak]) => {
|
||||||
|
console.log("writing target change");
|
||||||
|
this.writeEvent({
|
||||||
|
kind: WriteKind.TargetChanged,
|
||||||
|
name,
|
||||||
|
globalSpeak
|
||||||
|
});
|
||||||
|
});
|
||||||
this.boundPlayer = derived([boundPlayerName, this.playersByName], ([playerName, players]) => players[playerName]);
|
this.boundPlayer = derived([boundPlayerName, this.playersByName], ([playerName, players]) => players[playerName]);
|
||||||
}
|
}
|
||||||
|
|
||||||
closed() {
|
|
||||||
return this.transport.closed;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export class Player {
|
export class Player {
|
||||||
public stageInfo: Writable<{ position: number[], stage: string }>;
|
public proximityInfo: CachedWritable<{ position: number[], stage: string }>;
|
||||||
constructor(public name: string, public id: number, stageInfo?: { position: number[], stage: string }) {
|
constructor(public name: string, public id: number, proximityInfo?: { position: number[], stage: string }) {
|
||||||
|
this.proximityInfo = cachedWritable(Object.assign({ position: [0, 0, 0], stage: "nostage" + Math.random() }, proximityInfo ?? {}));
|
||||||
this.stageInfo = writable(Object.assign({ position: [0, 0, 0], stage: "nostage" + Math.random() }, stageInfo ?? {}));
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export class Peer {
|
export class Peer {
|
||||||
private connection: RTCPeerConnection;
|
private connection: RTCPeerConnection;
|
||||||
private playerAudio: PlayerAudio;
|
private playerAudio: PlayerAudio;
|
||||||
|
public globalSpeak: Writable<boolean> = writable(false);
|
||||||
public targetPlayerName: Writable<string>;
|
public targetPlayerName: Writable<string>;
|
||||||
private targetPlayer: Readable<Player | undefined>;
|
private targetPlayer: Readable<Player | undefined>;
|
||||||
private state: Writable<string> = writable("not connected");
|
private state: Writable<string> = writable("not connected");
|
||||||
|
@ -191,7 +203,7 @@ export class Peer {
|
||||||
});
|
});
|
||||||
this.targetPlayerName = writable("");
|
this.targetPlayerName = writable("");
|
||||||
this.targetPlayer = derived([this.targetPlayerName, this.client.playersByName], ([playerName, players]) => players[playerName]);
|
this.targetPlayer = derived([this.targetPlayerName, this.client.playersByName], ([playerName, players]) => players[playerName]);
|
||||||
this.playerAudio = this.client.audioManager.createPlayerAudio(this.client.outputAudio, this.client.boundPlayer, this.targetPlayer);
|
this.playerAudio = this.client.audioManager.createPlayerAudio(this.client.outputAudio, this.client.boundPlayer, this.targetPlayer, this.globalSpeak);
|
||||||
for (const track of this.client.audioManager.output.getAudioTracks()) {
|
for (const track of this.client.audioManager.output.getAudioTracks()) {
|
||||||
this.connection.addTrack(track, this.client.audioManager.output);
|
this.connection.addTrack(track, this.client.audioManager.output);
|
||||||
}
|
}
|
||||||
|
@ -205,7 +217,12 @@ export class Peer {
|
||||||
else
|
else
|
||||||
console.log(this.peerId, "finished candidate gathering")
|
console.log(this.peerId, "finished candidate gathering")
|
||||||
});
|
});
|
||||||
|
// this.connection.addEventListener("negotiationneeded", () => {
|
||||||
|
// console.warn("Needed renegotiation")
|
||||||
|
// this.offer();
|
||||||
|
// });
|
||||||
this.connection.addEventListener("track", (ev) => {
|
this.connection.addEventListener("track", (ev) => {
|
||||||
|
console.log("got track with it", ev.streams.length, ev.streams);
|
||||||
this.playerAudio.gotTrack(ev.streams[0]);
|
this.playerAudio.gotTrack(ev.streams[0]);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
@ -228,7 +245,7 @@ export class Peer {
|
||||||
this.client.writeEvent({
|
this.client.writeEvent({
|
||||||
kind: WriteKind.Answer,
|
kind: WriteKind.Answer,
|
||||||
to: this.peerId,
|
to: this.peerId,
|
||||||
answerSdp: answer.sdp ?? throwExpr("it didn't create an offer???"),
|
answerSdp: answer.sdp ?? throwExpr("it didn't create an answer???"),
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -20,12 +20,14 @@ export type WriteEvent = {
|
||||||
answerSdp: string
|
answerSdp: string
|
||||||
} | {
|
} | {
|
||||||
kind: WriteKind.TargetChanged,
|
kind: WriteKind.TargetChanged,
|
||||||
name: string
|
name: string,
|
||||||
|
globalSpeak: boolean,
|
||||||
}
|
}
|
||||||
export function writeEvent(writer: BinaryWriter, event: WriteEvent) {
|
export function writeEvent(writer: BinaryWriter, event: WriteEvent) {
|
||||||
writer.writeUInt8(event.kind);
|
writer.writeUInt8(event.kind);
|
||||||
if (event.kind === WriteKind.TargetChanged) {
|
if (event.kind === WriteKind.TargetChanged) {
|
||||||
writer.writeFixedString(0x20, event.name);
|
writer.writeFixedString(0x20, event.name);
|
||||||
|
writer.writeBoolean(event.globalSpeak);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -16,8 +16,8 @@ export async function readHello(reader: BinaryStreamReader) {
|
||||||
{ position, stage },
|
{ position, stage },
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
const peers = await reader.readArray(await reader.readUInt32(), async (reader): Promise<[string, string]> => {
|
const peers = await reader.readArray(await reader.readUInt32(), async (reader): Promise<[string, string, boolean]> => {
|
||||||
return [await reader.readFixedString(36), await reader.readFixedString(0x20)];
|
return [await reader.readFixedString(36), await reader.readFixedString(0x20), await reader.readBoolean()];
|
||||||
});
|
});
|
||||||
|
|
||||||
return { players, peers };
|
return { players, peers };
|
||||||
|
|
|
@ -38,7 +38,8 @@ export async function readEvent(reader: BinaryReader | BinaryStreamReader) {
|
||||||
const id = await reader.readFixedString(36);
|
const id = await reader.readFixedString(36);
|
||||||
const connected = await reader.readBoolean();
|
const connected = await reader.readBoolean();
|
||||||
const target = await reader.readFixedString(0x20);
|
const target = await reader.readFixedString(0x20);
|
||||||
return { kind, id, connected, target };
|
const globalSpeak = await reader.readBoolean();
|
||||||
|
return { kind, id, connected, target, globalSpeak };
|
||||||
}
|
}
|
||||||
case ReadKind.Offer: {
|
case ReadKind.Offer: {
|
||||||
const id = await reader.readFixedString(36);
|
const id = await reader.readFixedString(36);
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
import { writable, type Writable, type Readable, derived } from "svelte/store";
|
import { writable, type Writable, type Readable, derived, type Unsubscriber } from "svelte/store";
|
||||||
|
|
||||||
export function localStore<T>(key: string, defaultValue: T, initialConverter: (value: string) => T, stringifier: (value: T) => string): Writable<T> {
|
export function localStore<T>(key: string, defaultValue: T, initialConverter: (value: string) => T, stringifier: (value: T) => string): Writable<T> {
|
||||||
let baseWritable = writable(defaultValue);
|
let baseWritable = writable(defaultValue);
|
||||||
|
@ -67,6 +67,13 @@ export function cachedWritable<T>(initial: T): CachedWritable<T> {
|
||||||
return newValue;
|
return newValue;
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
subscribe: store.subscribe
|
subscribe(run, invalidate) {
|
||||||
|
console.log('fucyk');
|
||||||
|
return store.subscribe(run, invalidate)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function subscribe<S extends Stores>(stores: S, subscriber: (values: StoresValues<S>) => void): Unsubscriber {
|
||||||
|
return derived(stores, identity).subscribe(subscriber);
|
||||||
|
}
|
||||||
|
|
|
@ -3,20 +3,41 @@
|
||||||
import { Client, Player } from "$lib/client";
|
import { Client, Player } from "$lib/client";
|
||||||
import { AudioManager } from "$lib/audio";
|
import { AudioManager } from "$lib/audio";
|
||||||
import { identity, localStore } from "$lib/stores";
|
import { identity, localStore } from "$lib/stores";
|
||||||
import { readable, readonly, type Writable } from "svelte/store";
|
import {
|
||||||
|
readable,
|
||||||
|
readonly,
|
||||||
|
type Readable,
|
||||||
|
type Writable,
|
||||||
|
} from "svelte/store";
|
||||||
|
|
||||||
const isDev = ((globalThis as any).isDev = location.pathname.includes("dev"));
|
const setupName = new URLSearchParams(location.search).get("name");
|
||||||
|
const isDev = ((globalThis as any).isDev = "hot" in import.meta);
|
||||||
let name = localStore("name", "", identity, identity);
|
let name = localStore("name", "", identity, identity);
|
||||||
|
if (setupName) name.set(setupName);
|
||||||
let useMusic = isDev
|
let useMusic = isDev
|
||||||
? localStore(
|
? localStore(
|
||||||
"useMusic",
|
"useMusic",
|
||||||
false,
|
false,
|
||||||
(v) => v == "true",
|
(v) => v == "true",
|
||||||
(v) => v.toString()
|
(v) => v.toString(),
|
||||||
)
|
)
|
||||||
: readable(false);
|
: readable(false);
|
||||||
|
let micVolume = localStore("micVolume", 0.4, parseFloat, (v) => v.toString());
|
||||||
|
let muted = localStore(
|
||||||
|
"muted",
|
||||||
|
false,
|
||||||
|
(v) => v == "true",
|
||||||
|
(v) => v.toString(),
|
||||||
|
);
|
||||||
|
let globalSpeak = localStore(
|
||||||
|
"globalSpeak",
|
||||||
|
true,
|
||||||
|
(v) => v == "true",
|
||||||
|
(v) => v.toString(),
|
||||||
|
);
|
||||||
let client: Client = $state(undefined as any);
|
let client: Client = $state(undefined as any);
|
||||||
let players: Writable<Record<number, Player>> = undefined as any;
|
let players: Writable<Record<number, Player>> = undefined as any;
|
||||||
|
let connected: Readable<boolean> = undefined as any;
|
||||||
let ac: AudioManager;
|
let ac: AudioManager;
|
||||||
let wyrm: HTMLAudioElement;
|
let wyrm: HTMLAudioElement;
|
||||||
let outputAudio: HTMLDivElement;
|
let outputAudio: HTMLDivElement;
|
||||||
|
@ -27,35 +48,75 @@
|
||||||
audio: true,
|
audio: true,
|
||||||
video: false,
|
video: false,
|
||||||
});
|
});
|
||||||
ac = new AudioManager(media);
|
ac = new AudioManager(media, micVolume, muted);
|
||||||
}
|
}
|
||||||
async function connect() {
|
async function connect() {
|
||||||
await audioPrep();
|
await audioPrep();
|
||||||
client = await Client.connect(readonly(name), ac, outputAudio);
|
client = await Client.connect(
|
||||||
|
readonly(name),
|
||||||
|
readonly(globalSpeak),
|
||||||
|
ac,
|
||||||
|
outputAudio,
|
||||||
|
);
|
||||||
players = client.players;
|
players = client.players;
|
||||||
|
players.subscribe(() => {
|
||||||
|
console.log("players", $players);
|
||||||
|
players = client.players;
|
||||||
|
});
|
||||||
|
connected = readonly(client.connected);
|
||||||
console.log(client);
|
console.log(client);
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<div bind:this={outputAudio}>audio outputs</div>
|
{#if "WebTransport" in globalThis}
|
||||||
|
<div bind:this={outputAudio}>{void 0}</div>
|
||||||
|
|
||||||
<div>
|
<div class="controls">
|
||||||
|
<div>
|
||||||
I am <input
|
I am <input
|
||||||
type="text"
|
type="text"
|
||||||
maxlength="32"
|
maxlength="32"
|
||||||
list="connectedPlayers"
|
list="connectedPlayers"
|
||||||
bind:value={$name}
|
bind:value={$name}
|
||||||
/>
|
/>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<label for="micVolume">Mic Volume: </label>
|
||||||
|
<input
|
||||||
|
id="micVolume"
|
||||||
|
type="range"
|
||||||
|
min="0"
|
||||||
|
max="1"
|
||||||
|
step="0.1"
|
||||||
|
bind:value={$micVolume}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<label for="muted">Muted: </label>
|
||||||
|
<input id="muted" type="checkbox" bind:checked={$muted} />
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<label for="globalSpeak">Global Mic: </label>
|
||||||
|
<input id="globalSpeak" type="checkbox" bind:checked={$globalSpeak} />
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
<button onclick={() => connect()}>Connect</button>
|
<button onclick={() => connect()}>Connect</button>
|
||||||
|
</div>
|
||||||
{#if isDev}
|
{#if isDev}
|
||||||
<label for="useMusic">Use Music</label>
|
<div>
|
||||||
|
<label for="useMusic">Use Music (dev only!)</label>
|
||||||
<input id="useMusic" type="checkbox" bind:checked={$useMusic} />
|
<input id="useMusic" type="checkbox" bind:checked={$useMusic} />
|
||||||
<audio bind:this={wyrm} src="/wyrmjewelbox.wav" autoplay={false} controls
|
<audio
|
||||||
>audio of the wyrm's houseki box</audio
|
bind:this={wyrm}
|
||||||
|
src="/wyrmjewelbox.wav"
|
||||||
|
autoplay={false}
|
||||||
|
controls>audio of the wyrm's houseki box</audio
|
||||||
>
|
>
|
||||||
|
</div>
|
||||||
{/if}
|
{/if}
|
||||||
</div>
|
</div>
|
||||||
{#if client !== undefined}
|
{#if client !== undefined}
|
||||||
|
<input type="checkbox" disabled bind:checked={$connected} />
|
||||||
<datalist>
|
<datalist>
|
||||||
{#each Object.values($players) as player}
|
{#each Object.values($players) as player}
|
||||||
<option>{player.name}</option>
|
<option>{player.name}</option>
|
||||||
|
@ -67,10 +128,22 @@
|
||||||
<pre class="name">{player.name}</pre>
|
<pre class="name">{player.name}</pre>
|
||||||
{/each}
|
{/each}
|
||||||
</div>
|
</div>
|
||||||
{/if}
|
{/if}
|
||||||
|
|
||||||
<style>
|
<style>
|
||||||
|
.controls {
|
||||||
|
display: flex;
|
||||||
|
flex-basis: 1 1 auto;
|
||||||
|
flex-direction: column;
|
||||||
|
}
|
||||||
.name {
|
.name {
|
||||||
font-style: italic;
|
font-style: italic;
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
|
{:else}
|
||||||
|
!secureContext... was the correct link shared? <a
|
||||||
|
href="chrome://flags/#unsafely-treat-insecure-origin-as-secure"
|
||||||
|
>make sure you have the insecure flag set in case you need it</a
|
||||||
|
>
|
||||||
|
to <b>{window.location}</b>
|
||||||
|
{/if}
|
||||||
|
|
|
@ -1,4 +0,0 @@
|
||||||
{
|
|
||||||
"host": "example.com",
|
|
||||||
"port": 4433
|
|
||||||
}
|
|
|
@ -21,7 +21,10 @@ const config = {
|
||||||
fallback: undefined,
|
fallback: undefined,
|
||||||
precompress: false,
|
precompress: false,
|
||||||
strict: true
|
strict: true
|
||||||
})
|
}),
|
||||||
|
// paths: {
|
||||||
|
// base: "/dev"
|
||||||
|
// }
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -8,6 +8,11 @@ export default defineConfig({
|
||||||
// key: readFileSync('key.pem'),
|
// key: readFileSync('key.pem'),
|
||||||
// cert: readFileSync('cert.pem'),
|
// cert: readFileSync('cert.pem'),
|
||||||
// },
|
// },
|
||||||
|
// hmr: {
|
||||||
|
// path: "ws",
|
||||||
|
// host: "smosh.eepy.engineering"
|
||||||
|
// },
|
||||||
|
// origin: "https://smosh.eepy.engineering"
|
||||||
},
|
},
|
||||||
plugins: [
|
plugins: [
|
||||||
sveltekit(),
|
sveltekit(),
|
||||||
|
|
Loading…
Reference in a new issue