@moq/watch 0.1.0 → 0.1.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/element.js CHANGED
@@ -1,9 +1,9 @@
1
- import * as r from "@moq/lite";
2
- import { Signal as a, Effect as n } from "@moq/signals";
3
- import { h as l, B as c, g as h } from "./broadcast-CSXy8QiX.js";
4
- const u = ["url", "name", "path", "paused", "volume", "muted", "reload", "jitter", "latency"], d = new FinalizationRegistry((o) => o.close());
1
+ import * as o from "@moq/lite";
2
+ import { Signal as n, Effect as a } from "@moq/signals";
3
+ import { h as l, B as h, g as u } from "./broadcast-BFfLDq15.js";
4
+ const c = ["url", "name", "path", "paused", "volume", "muted", "reload", "jitter", "latency"], d = new FinalizationRegistry((r) => r.close());
5
5
  class b extends HTMLElement {
6
- static observedAttributes = u;
6
+ static observedAttributes = c;
7
7
  // The connection to the moq-relay server.
8
8
  connection;
9
9
  // The broadcast being watched.
@@ -13,16 +13,16 @@ class b extends HTMLElement {
13
13
  // The backend that powers this element.
14
14
  #t;
15
15
  // Set when the element is connected to the DOM.
16
- #e = new a(!1);
16
+ #e = new n(!1);
17
17
  // Expose the Effect class, so users can easily create effects scoped to this element.
18
- signals = new n();
18
+ signals = new a();
19
19
  constructor() {
20
- super(), d.register(this, this.signals), this.connection = new r.Connection.Reload({
20
+ super(), d.register(this, this.signals), this.connection = new o.Connection.Reload({
21
21
  enabled: this.#e
22
- }), this.signals.cleanup(() => this.connection.close()), this.broadcast = new c({
22
+ }), this.signals.cleanup(() => this.connection.close()), this.broadcast = new h({
23
23
  connection: this.connection.established,
24
24
  enabled: this.#e
25
- }), this.signals.cleanup(() => this.broadcast.close()), this.#t = new h({
25
+ }), this.signals.cleanup(() => this.broadcast.close()), this.#t = new u({
26
26
  broadcast: this.broadcast
27
27
  }), this.signals.cleanup(() => this.#t.close());
28
28
  const s = () => {
@@ -31,20 +31,20 @@ class b extends HTMLElement {
31
31
  throw new Error("Cannot have both canvas and video elements");
32
32
  this.#t.element.set(t ?? e);
33
33
  }, i = new MutationObserver(s);
34
- i.observe(this, { childList: !0, subtree: !0 }), this.signals.cleanup(() => i.disconnect()), s(), this.signals.effect((t) => {
34
+ i.observe(this, { childList: !0, subtree: !0 }), this.signals.cleanup(() => i.disconnect()), s(), this.signals.run((t) => {
35
35
  const e = t.get(this.url);
36
36
  e ? this.setAttribute("url", e.toString()) : this.removeAttribute("url");
37
- }), this.signals.effect((t) => {
37
+ }), this.signals.run((t) => {
38
38
  const e = t.get(this.path);
39
39
  e ? this.setAttribute("path", e.toString()) : this.removeAttribute("path");
40
- }), this.signals.effect((t) => {
40
+ }), this.signals.run((t) => {
41
41
  t.get(this.audio.muted) ? this.setAttribute("muted", "") : this.removeAttribute("muted");
42
- }), this.signals.effect((t) => {
42
+ }), this.signals.run((t) => {
43
43
  t.get(this.paused) ? this.setAttribute("paused", "true") : this.removeAttribute("paused");
44
- }), this.signals.effect((t) => {
44
+ }), this.signals.run((t) => {
45
45
  const e = t.get(this.audio.volume);
46
46
  this.setAttribute("volume", e.toString());
47
- }), this.signals.effect((t) => {
47
+ }), this.signals.run((t) => {
48
48
  const e = Math.floor(t.get(this.jitter));
49
49
  this.setAttribute("jitter", e.toString());
50
50
  });
@@ -62,7 +62,7 @@ class b extends HTMLElement {
62
62
  if (s === "url")
63
63
  this.url.set(t ? new URL(t) : void 0);
64
64
  else if (s === "name" || s === "path")
65
- this.path.set(t ? r.Path.from(t) : void 0);
65
+ this.path.set(t ? o.Path.from(t) : void 0);
66
66
  else if (s === "paused")
67
67
  this.paused.set(t !== null);
68
68
  else if (s === "volume") {
package/element.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"file":"element.js","sources":["../src/element.ts"],"sourcesContent":["import type { Time } from \"@moq/lite\";\nimport * as Moq from \"@moq/lite\";\nimport { Effect, Signal } from \"@moq/signals\";\nimport type * as Audio from \"./audio\";\nimport { type Backend, MultiBackend } from \"./backend\";\nimport { Broadcast } from \"./broadcast\";\nimport { Sync } from \"./sync\";\nimport type * as Video from \"./video\";\n\n// TODO remove name; replaced with path\n// TODO remove latency; replaced with buffer\nconst OBSERVED = [\"url\", \"name\", \"path\", \"paused\", \"volume\", \"muted\", \"reload\", \"jitter\", \"latency\"] as const;\ntype Observed = (typeof OBSERVED)[number];\n\n// Close everything when this element is garbage collected.\n// This is primarily to avoid a console.warn that we didn't close() before GC.\n// There's no destructor for web components so this is the best we can do.\nconst cleanup = new FinalizationRegistry<Effect>((signals) => signals.close());\n\n// An optional web component that wraps a <canvas>\nexport default class MoqWatch extends HTMLElement implements Backend {\n\tstatic observedAttributes = OBSERVED;\n\n\t// The connection to the moq-relay server.\n\tconnection: Moq.Connection.Reload;\n\n\t// The broadcast being watched.\n\tbroadcast: Broadcast;\n\n\t// Used to sync audio and video playback at a target latency.\n\tsync = new Sync();\n\n\t// The backend that powers this element.\n\t#backend: MultiBackend;\n\n\t// Set when the element is connected to the DOM.\n\t#enabled = new Signal(false);\n\n\t// Expose the Effect class, so users can easily create effects scoped to this element.\n\tsignals = new Effect();\n\n\tconstructor() {\n\t\tsuper();\n\n\t\tcleanup.register(this, this.signals);\n\n\t\tthis.connection = new Moq.Connection.Reload({\n\t\t\tenabled: this.#enabled,\n\t\t});\n\t\tthis.signals.cleanup(() => this.connection.close());\n\n\t\tthis.broadcast = new Broadcast({\n\t\t\tconnection: this.connection.established,\n\t\t\tenabled: this.#enabled,\n\t\t});\n\t\tthis.signals.cleanup(() => this.broadcast.close());\n\n\t\tthis.#backend = new MultiBackend({\n\t\t\tbroadcast: this.broadcast,\n\t\t});\n\t\tthis.signals.cleanup(() => this.#backend.close());\n\n\t\t// Watch to see if the canvas element is added or removed.\n\t\tconst setElement = () => {\n\t\t\tconst canvas = this.querySelector(\"canvas\") as HTMLCanvasElement | undefined;\n\t\t\tconst video = this.querySelector(\"video\") as HTMLVideoElement | undefined;\n\t\t\tif (canvas && video) {\n\t\t\t\tthrow new Error(\"Cannot have both canvas and video elements\");\n\t\t\t}\n\t\t\tthis.#backend.element.set(canvas ?? video);\n\t\t};\n\n\t\tconst observer = new MutationObserver(setElement);\n\t\tobserver.observe(this, { childList: true, subtree: true });\n\t\tthis.signals.cleanup(() => observer.disconnect());\n\t\tsetElement();\n\n\t\t// Optionally update attributes to match the library state.\n\t\t// This is kind of dangerous because it can create loops.\n\t\t// NOTE: This only runs when the element is connected to the DOM, which is not obvious.\n\t\t// This is because there's no destructor for web components to clean up our effects.\n\t\tthis.signals.effect((effect) => {\n\t\t\tconst url = effect.get(this.url);\n\t\t\tif (url) {\n\t\t\t\tthis.setAttribute(\"url\", url.toString());\n\t\t\t} else {\n\t\t\t\tthis.removeAttribute(\"url\");\n\t\t\t}\n\t\t});\n\n\t\tthis.signals.effect((effect) => {\n\t\t\tconst broadcast = effect.get(this.path);\n\t\t\tif (broadcast) {\n\t\t\t\tthis.setAttribute(\"path\", broadcast.toString());\n\t\t\t} else {\n\t\t\t\tthis.removeAttribute(\"path\");\n\t\t\t}\n\t\t});\n\n\t\tthis.signals.effect((effect) => {\n\t\t\tconst muted = effect.get(this.audio.muted);\n\t\t\tif (muted) {\n\t\t\t\tthis.setAttribute(\"muted\", \"\");\n\t\t\t} else {\n\t\t\t\tthis.removeAttribute(\"muted\");\n\t\t\t}\n\t\t});\n\n\t\tthis.signals.effect((effect) => {\n\t\t\tconst paused = effect.get(this.paused);\n\t\t\tif (paused) {\n\t\t\t\tthis.setAttribute(\"paused\", \"true\");\n\t\t\t} else {\n\t\t\t\tthis.removeAttribute(\"paused\");\n\t\t\t}\n\t\t});\n\n\t\tthis.signals.effect((effect) => {\n\t\t\tconst volume = effect.get(this.audio.volume);\n\t\t\tthis.setAttribute(\"volume\", volume.toString());\n\t\t});\n\n\t\tthis.signals.effect((effect) => {\n\t\t\tconst jitter = Math.floor(effect.get(this.jitter));\n\t\t\tthis.setAttribute(\"jitter\", jitter.toString());\n\t\t});\n\t}\n\n\t// Annoyingly, we have to use these callbacks to figure out when the element is connected to the DOM.\n\t// This wouldn't be so bad if there was a destructor for web components to clean up our effects.\n\tconnectedCallback() {\n\t\tthis.#enabled.set(true);\n\t\tthis.style.display = \"block\";\n\t\tthis.style.position = \"relative\";\n\t}\n\n\tdisconnectedCallback() {\n\t\t// Stop everything but don't actually cleanup just in case we get added back to the DOM.\n\t\tthis.#enabled.set(false);\n\t}\n\n\tattributeChangedCallback(name: Observed, oldValue: string | null, newValue: string | null) {\n\t\tif (oldValue === newValue) {\n\t\t\treturn;\n\t\t}\n\n\t\tif (name === \"url\") {\n\t\t\tthis.url.set(newValue ? new URL(newValue) : undefined);\n\t\t} else if (name === \"name\" || name === \"path\") {\n\t\t\tthis.path.set(newValue ? Moq.Path.from(newValue) : undefined);\n\t\t} else if (name === \"paused\") {\n\t\t\tthis.paused.set(newValue !== null);\n\t\t} else if (name === \"volume\") {\n\t\t\tconst volume = newValue ? Number.parseFloat(newValue) : 0.5;\n\t\t\tthis.audio.volume.set(volume);\n\t\t} else if (name === \"muted\") {\n\t\t\tthis.audio.muted.set(newValue !== null);\n\t\t} else if (name === \"reload\") {\n\t\t\tthis.broadcast.reload.set(newValue !== null);\n\t\t} else if (name === \"jitter\" || name === \"latency\") {\n\t\t\t// \"latency\" is a legacy alias for \"jitter\"\n\t\t\tthis.jitter.set((newValue ? Number.parseFloat(newValue) : 100) as Time.Milli);\n\t\t} else {\n\t\t\tconst exhaustive: never = name;\n\t\t\tthrow new Error(`Invalid attribute: ${exhaustive}`);\n\t\t}\n\t}\n\n\tget url(): Signal<URL | undefined> {\n\t\treturn this.connection.url;\n\t}\n\n\tget path(): Signal<Moq.Path.Valid | undefined> {\n\t\treturn this.broadcast.path;\n\t}\n\n\tget jitter(): Signal<Time.Milli> {\n\t\treturn this.#backend.jitter;\n\t}\n\n\tget paused(): Signal<boolean> {\n\t\treturn this.#backend.paused;\n\t}\n\n\tget audio(): Audio.Backend {\n\t\treturn this.#backend.audio;\n\t}\n\n\tget video(): Video.Backend {\n\t\treturn this.#backend.video;\n\t}\n}\n\ncustomElements.define(\"moq-watch\", MoqWatch);\n\ndeclare global {\n\tinterface HTMLElementTagNameMap {\n\t\t\"moq-watch\": MoqWatch;\n\t}\n}\n"],"names":["OBSERVED","cleanup","signals","MoqWatch","Sync","#backend","#enabled","Signal","Effect","Moq","Broadcast","MultiBackend","setElement","canvas","video","observer","effect","url","broadcast","volume","jitter","name","oldValue","newValue","exhaustive"],"mappings":";;;AAWA,MAAMA,IAAW,CAAC,OAAO,QAAQ,QAAQ,UAAU,UAAU,SAAS,UAAU,UAAU,SAAS,GAM7FC,IAAU,IAAI,qBAA6B,CAACC,MAAYA,EAAQ,OAAO;AAG7E,MAAqBC,UAAiB,YAA+B;AAAA,EACpE,OAAO,qBAAqBH;AAAA;AAAA,EAG5B;AAAA;AAAA,EAGA;AAAA;AAAA,EAGA,OAAO,IAAII,EAAA;AAAA;AAAA,EAGXC;AAAA;AAAA,EAGAC,KAAW,IAAIC,EAAO,EAAK;AAAA;AAAA,EAG3B,UAAU,IAAIC,EAAA;AAAA,EAEd,cAAc;AACb,UAAA,GAEAP,EAAQ,SAAS,MAAM,KAAK,OAAO,GAEnC,KAAK,aAAa,IAAIQ,EAAI,WAAW,OAAO;AAAA,MAC3C,SAAS,KAAKH;AAAA,IAAA,CACd,GACD,KAAK,QAAQ,QAAQ,MAAM,KAAK,WAAW,OAAO,GAElD,KAAK,YAAY,IAAII,EAAU;AAAA,MAC9B,YAAY,KAAK,WAAW;AAAA,MAC5B,SAAS,KAAKJ;AAAA,IAAA,CACd,GACD,KAAK,QAAQ,QAAQ,MAAM,KAAK,UAAU,OAAO,GAEjD,KAAKD,KAAW,IAAIM,EAAa;AAAA,MAChC,WAAW,KAAK;AAAA,IAAA,CAChB,GACD,KAAK,QAAQ,QAAQ,MAAM,KAAKN,GAAS,OAAO;AAGhD,UAAMO,IAAa,MAAM;AACxB,YAAMC,IAAS,KAAK,cAAc,QAAQ,GACpCC,IAAQ,KAAK,cAAc,OAAO;AACxC,UAAID,KAAUC;AACb,cAAM,IAAI,MAAM,4CAA4C;AAE7D,WAAKT,GAAS,QAAQ,IAAIQ,KAAUC,CAAK;AAAA,IAC1C,GAEMC,IAAW,IAAI,iBAAiBH,CAAU;AAChD,IAAAG,EAAS,QAAQ,MAAM,EAAE,WAAW,IAAM,SAAS,IAAM,GACzD,KAAK,QAAQ,QAAQ,MAAMA,EAAS,YAAY,GAChDH,EAAA,GAMA,KAAK,QAAQ,OAAO,CAACI,MAAW;AAC/B,YAAMC,IAAMD,EAAO,IAAI,KAAK,GAAG;AAC/B,MAAIC,IACH,KAAK,aAAa,OAAOA,EAAI,SAAA,CAAU,IAEvC,KAAK,gBAAgB,KAAK;AAAA,IAE5B,CAAC,GAED,KAAK,QAAQ,OAAO,CAACD,MAAW;AAC/B,YAAME,IAAYF,EAAO,IAAI,KAAK,IAAI;AACtC,MAAIE,IACH,KAAK,aAAa,QAAQA,EAAU,SAAA,CAAU,IAE9C,KAAK,gBAAgB,MAAM;AAAA,IAE7B,CAAC,GAED,KAAK,QAAQ,OAAO,CAACF,MAAW;AAE/B,MADcA,EAAO,IAAI,KAAK,MAAM,KAAK,IAExC,KAAK,aAAa,SAAS,EAAE,IAE7B,KAAK,gBAAgB,OAAO;AAAA,IAE9B,CAAC,GAED,KAAK,QAAQ,OAAO,CAACA,MAAW;AAE/B,MADeA,EAAO,IAAI,KAAK,MAAM,IAEpC,KAAK,aAAa,UAAU,MAAM,IAElC,KAAK,gBAAgB,QAAQ;AAAA,IAE/B,CAAC,GAED,KAAK,QAAQ,OAAO,CAACA,MAAW;AAC/B,YAAMG,IAASH,EAAO,IAAI,KAAK,MAAM,MAAM;AAC3C,WAAK,aAAa,UAAUG,EAAO,SAAA,CAAU;AAAA,IAC9C,CAAC,GAED,KAAK,QAAQ,OAAO,CAACH,MAAW;AAC/B,YAAMI,IAAS,KAAK,MAAMJ,EAAO,IAAI,KAAK,MAAM,CAAC;AACjD,WAAK,aAAa,UAAUI,EAAO,SAAA,CAAU;AAAA,IAC9C,CAAC;AAAA,EACF;AAAA;AAAA;AAAA,EAIA,oBAAoB;AACnB,SAAKd,GAAS,IAAI,EAAI,GACtB,KAAK,MAAM,UAAU,SACrB,KAAK,MAAM,WAAW;AAAA,EACvB;AAAA,EAEA,uBAAuB;AAEtB,SAAKA,GAAS,IAAI,EAAK;AAAA,EACxB;AAAA,EAEA,yBAAyBe,GAAgBC,GAAyBC,GAAyB;AAC1F,QAAID,MAAaC;AAIjB,UAAIF,MAAS;AACZ,aAAK,IAAI,IAAIE,IAAW,IAAI,IAAIA,CAAQ,IAAI,MAAS;AAAA,eAC3CF,MAAS,UAAUA,MAAS;AACtC,aAAK,KAAK,IAAIE,IAAWd,EAAI,KAAK,KAAKc,CAAQ,IAAI,MAAS;AAAA,eAClDF,MAAS;AACnB,aAAK,OAAO,IAAIE,MAAa,IAAI;AAAA,eACvBF,MAAS,UAAU;AAC7B,cAAMF,IAASI,IAAW,OAAO,WAAWA,CAAQ,IAAI;AACxD,aAAK,MAAM,OAAO,IAAIJ,CAAM;AAAA,MAC7B,WAAWE,MAAS;AACnB,aAAK,MAAM,MAAM,IAAIE,MAAa,IAAI;AAAA,eAC5BF,MAAS;AACnB,aAAK,UAAU,OAAO,IAAIE,MAAa,IAAI;AAAA,eACjCF,MAAS,YAAYA,MAAS;AAExC,aAAK,OAAO,IAAKE,IAAW,OAAO,WAAWA,CAAQ,IAAI,GAAkB;AAAA,WACtE;AACN,cAAMC,IAAoBH;AAC1B,cAAM,IAAI,MAAM,sBAAsBG,CAAU,EAAE;AAAA,MACnD;AAAA,EACD;AAAA,EAEA,IAAI,MAA+B;AAClC,WAAO,KAAK,WAAW;AAAA,EACxB;AAAA,EAEA,IAAI,OAA2C;AAC9C,WAAO,KAAK,UAAU;AAAA,EACvB;AAAA,EAEA,IAAI,SAA6B;AAChC,WAAO,KAAKnB,GAAS;AAAA,EACtB;AAAA,EAEA,IAAI,SAA0B;AAC7B,WAAO,KAAKA,GAAS;AAAA,EACtB;AAAA,EAEA,IAAI,QAAuB;AAC1B,WAAO,KAAKA,GAAS;AAAA,EACtB;AAAA,EAEA,IAAI,QAAuB;AAC1B,WAAO,KAAKA,GAAS;AAAA,EACtB;AACD;AAEA,eAAe,OAAO,aAAaF,CAAQ;"}
1
+ {"version":3,"file":"element.js","sources":["../src/element.ts"],"sourcesContent":["import type { Time } from \"@moq/lite\";\nimport * as Moq from \"@moq/lite\";\nimport { Effect, Signal } from \"@moq/signals\";\nimport type * as Audio from \"./audio\";\nimport { type Backend, MultiBackend } from \"./backend\";\nimport { Broadcast } from \"./broadcast\";\nimport { Sync } from \"./sync\";\nimport type * as Video from \"./video\";\n\n// TODO remove name; replaced with path\n// TODO remove latency; replaced with buffer\nconst OBSERVED = [\"url\", \"name\", \"path\", \"paused\", \"volume\", \"muted\", \"reload\", \"jitter\", \"latency\"] as const;\ntype Observed = (typeof OBSERVED)[number];\n\n// Close everything when this element is garbage collected.\n// This is primarily to avoid a console.warn that we didn't close() before GC.\n// There's no destructor for web components so this is the best we can do.\nconst cleanup = new FinalizationRegistry<Effect>((signals) => signals.close());\n\n// An optional web component that wraps a <canvas>\nexport default class MoqWatch extends HTMLElement implements Backend {\n\tstatic observedAttributes = OBSERVED;\n\n\t// The connection to the moq-relay server.\n\tconnection: Moq.Connection.Reload;\n\n\t// The broadcast being watched.\n\tbroadcast: Broadcast;\n\n\t// Used to sync audio and video playback at a target latency.\n\tsync = new Sync();\n\n\t// The backend that powers this element.\n\t#backend: MultiBackend;\n\n\t// Set when the element is connected to the DOM.\n\t#enabled = new Signal(false);\n\n\t// Expose the Effect class, so users can easily create effects scoped to this element.\n\tsignals = new Effect();\n\n\tconstructor() {\n\t\tsuper();\n\n\t\tcleanup.register(this, this.signals);\n\n\t\tthis.connection = new Moq.Connection.Reload({\n\t\t\tenabled: this.#enabled,\n\t\t});\n\t\tthis.signals.cleanup(() => this.connection.close());\n\n\t\tthis.broadcast = new Broadcast({\n\t\t\tconnection: this.connection.established,\n\t\t\tenabled: this.#enabled,\n\t\t});\n\t\tthis.signals.cleanup(() => this.broadcast.close());\n\n\t\tthis.#backend = new MultiBackend({\n\t\t\tbroadcast: this.broadcast,\n\t\t});\n\t\tthis.signals.cleanup(() => this.#backend.close());\n\n\t\t// Watch to see if the canvas element is added or removed.\n\t\tconst setElement = () => {\n\t\t\tconst canvas = this.querySelector(\"canvas\") as HTMLCanvasElement | undefined;\n\t\t\tconst video = this.querySelector(\"video\") as HTMLVideoElement | undefined;\n\t\t\tif (canvas && video) {\n\t\t\t\tthrow new Error(\"Cannot have both canvas and video elements\");\n\t\t\t}\n\t\t\tthis.#backend.element.set(canvas ?? video);\n\t\t};\n\n\t\tconst observer = new MutationObserver(setElement);\n\t\tobserver.observe(this, { childList: true, subtree: true });\n\t\tthis.signals.cleanup(() => observer.disconnect());\n\t\tsetElement();\n\n\t\t// Optionally update attributes to match the library state.\n\t\t// This is kind of dangerous because it can create loops.\n\t\t// NOTE: This only runs when the element is connected to the DOM, which is not obvious.\n\t\t// This is because there's no destructor for web components to clean up our effects.\n\t\tthis.signals.run((effect) => {\n\t\t\tconst url = effect.get(this.url);\n\t\t\tif (url) {\n\t\t\t\tthis.setAttribute(\"url\", url.toString());\n\t\t\t} else {\n\t\t\t\tthis.removeAttribute(\"url\");\n\t\t\t}\n\t\t});\n\n\t\tthis.signals.run((effect) => {\n\t\t\tconst broadcast = effect.get(this.path);\n\t\t\tif (broadcast) {\n\t\t\t\tthis.setAttribute(\"path\", broadcast.toString());\n\t\t\t} else {\n\t\t\t\tthis.removeAttribute(\"path\");\n\t\t\t}\n\t\t});\n\n\t\tthis.signals.run((effect) => {\n\t\t\tconst muted = effect.get(this.audio.muted);\n\t\t\tif (muted) {\n\t\t\t\tthis.setAttribute(\"muted\", \"\");\n\t\t\t} else {\n\t\t\t\tthis.removeAttribute(\"muted\");\n\t\t\t}\n\t\t});\n\n\t\tthis.signals.run((effect) => {\n\t\t\tconst paused = effect.get(this.paused);\n\t\t\tif (paused) {\n\t\t\t\tthis.setAttribute(\"paused\", \"true\");\n\t\t\t} else {\n\t\t\t\tthis.removeAttribute(\"paused\");\n\t\t\t}\n\t\t});\n\n\t\tthis.signals.run((effect) => {\n\t\t\tconst volume = effect.get(this.audio.volume);\n\t\t\tthis.setAttribute(\"volume\", volume.toString());\n\t\t});\n\n\t\tthis.signals.run((effect) => {\n\t\t\tconst jitter = Math.floor(effect.get(this.jitter));\n\t\t\tthis.setAttribute(\"jitter\", jitter.toString());\n\t\t});\n\t}\n\n\t// Annoyingly, we have to use these callbacks to figure out when the element is connected to the DOM.\n\t// This wouldn't be so bad if there was a destructor for web components to clean up our effects.\n\tconnectedCallback() {\n\t\tthis.#enabled.set(true);\n\t\tthis.style.display = \"block\";\n\t\tthis.style.position = \"relative\";\n\t}\n\n\tdisconnectedCallback() {\n\t\t// Stop everything but don't actually cleanup just in case we get added back to the DOM.\n\t\tthis.#enabled.set(false);\n\t}\n\n\tattributeChangedCallback(name: Observed, oldValue: string | null, newValue: string | null) {\n\t\tif (oldValue === newValue) {\n\t\t\treturn;\n\t\t}\n\n\t\tif (name === \"url\") {\n\t\t\tthis.url.set(newValue ? new URL(newValue) : undefined);\n\t\t} else if (name === \"name\" || name === \"path\") {\n\t\t\tthis.path.set(newValue ? Moq.Path.from(newValue) : undefined);\n\t\t} else if (name === \"paused\") {\n\t\t\tthis.paused.set(newValue !== null);\n\t\t} else if (name === \"volume\") {\n\t\t\tconst volume = newValue ? Number.parseFloat(newValue) : 0.5;\n\t\t\tthis.audio.volume.set(volume);\n\t\t} else if (name === \"muted\") {\n\t\t\tthis.audio.muted.set(newValue !== null);\n\t\t} else if (name === \"reload\") {\n\t\t\tthis.broadcast.reload.set(newValue !== null);\n\t\t} else if (name === \"jitter\" || name === \"latency\") {\n\t\t\t// \"latency\" is a legacy alias for \"jitter\"\n\t\t\tthis.jitter.set((newValue ? Number.parseFloat(newValue) : 100) as Time.Milli);\n\t\t} else {\n\t\t\tconst exhaustive: never = name;\n\t\t\tthrow new Error(`Invalid attribute: ${exhaustive}`);\n\t\t}\n\t}\n\n\tget url(): Signal<URL | undefined> {\n\t\treturn this.connection.url;\n\t}\n\n\tget path(): Signal<Moq.Path.Valid | undefined> {\n\t\treturn this.broadcast.path;\n\t}\n\n\tget jitter(): Signal<Time.Milli> {\n\t\treturn this.#backend.jitter;\n\t}\n\n\tget paused(): Signal<boolean> {\n\t\treturn this.#backend.paused;\n\t}\n\n\tget audio(): Audio.Backend {\n\t\treturn this.#backend.audio;\n\t}\n\n\tget video(): Video.Backend {\n\t\treturn this.#backend.video;\n\t}\n}\n\ncustomElements.define(\"moq-watch\", MoqWatch);\n\ndeclare global {\n\tinterface HTMLElementTagNameMap {\n\t\t\"moq-watch\": MoqWatch;\n\t}\n}\n"],"names":["OBSERVED","cleanup","signals","MoqWatch","Sync","#backend","#enabled","Signal","Effect","Moq","Broadcast","MultiBackend","setElement","canvas","video","observer","effect","url","broadcast","volume","jitter","name","oldValue","newValue","exhaustive"],"mappings":"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"}
package/index.js CHANGED
@@ -1,70 +1,70 @@
1
- import { D as w, M as p, R as v, S as f, a as m, E as y, b as _, c as S, P as h, d as k, e as M, f as P } from "./broadcast-CSXy8QiX.js";
2
- import { B as L, m as N, g as V, h as W, t as Y } from "./broadcast-CSXy8QiX.js";
3
- import { Signal as i, Effect as l } from "@moq/signals";
1
+ import { D as w, M as p, R as v, S as m, a as y, E as _, b as S, c as k, P as h, d as M, e as P, f } from "./broadcast-BFfLDq15.js";
2
+ import { B as L, m as N, g as V, h as W, t as Y } from "./broadcast-BFfLDq15.js";
3
+ import { Signal as i, Effect as c } from "@moq/signals";
4
4
  const B = /* @__PURE__ */ Object.freeze(/* @__PURE__ */ Object.defineProperty({
5
5
  __proto__: null,
6
6
  Decoder: w,
7
7
  Mse: p,
8
8
  Renderer: v,
9
- Source: f
9
+ Source: m
10
10
  }, Symbol.toStringTag, { value: "Module" })), D = /* @__PURE__ */ Object.freeze(/* @__PURE__ */ Object.defineProperty({
11
11
  __proto__: null,
12
- Decoder: m,
13
- Emitter: y,
14
- Mse: _,
15
- Source: S
12
+ Decoder: y,
13
+ Emitter: _,
14
+ Mse: S,
15
+ Source: k
16
16
  }, Symbol.toStringTag, { value: "Module" }));
17
17
  class O {
18
18
  broadcast;
19
19
  enabled;
20
20
  // Empty string is a valid message.
21
- #e = new i(void 0);
22
- latest = this.#e;
21
+ #t = new i(void 0);
22
+ latest = this.#t;
23
23
  #s = new i(void 0);
24
24
  catalog = this.#s;
25
- #t = new l();
26
- constructor(s, t, a) {
27
- this.broadcast = s, this.enabled = i.from(a?.enabled ?? !1), this.#t.effect((e) => {
28
- e.get(this.enabled) && this.#s.set(e.get(t)?.chat?.message);
29
- }), this.#t.effect(this.#a.bind(this));
25
+ #e = new c();
26
+ constructor(s, e, a) {
27
+ this.broadcast = s, this.enabled = i.from(a?.enabled ?? !1), this.#e.run((t) => {
28
+ t.get(this.enabled) && this.#s.set(t.get(e)?.chat?.message);
29
+ }), this.#e.run(this.#a.bind(this));
30
30
  }
31
31
  #a(s) {
32
- const t = s.getAll([this.enabled, this.#s, this.broadcast]);
33
- if (!t) return;
34
- const [a, e, o] = t, n = o.subscribe(e.name, h.chat);
35
- s.cleanup(() => n.close()), s.set(this.#e, ""), s.cleanup(() => this.#e.set(void 0)), s.spawn(async () => {
32
+ const e = s.getAll([this.enabled, this.#s, this.broadcast]);
33
+ if (!e) return;
34
+ const [a, t, o] = e, n = o.subscribe(t.name, h.chat);
35
+ s.cleanup(() => n.close()), s.set(this.#t, ""), s.cleanup(() => this.#t.set(void 0)), s.spawn(async () => {
36
36
  for (; ; ) {
37
- const c = await n.readString();
38
- if (c === void 0) break;
39
- this.#e.set(c);
37
+ const r = await n.readString();
38
+ if (r === void 0) break;
39
+ this.#t.set(r);
40
40
  }
41
41
  });
42
42
  }
43
43
  close() {
44
- this.#t.close();
44
+ this.#e.close();
45
45
  }
46
46
  }
47
47
  class T {
48
48
  broadcast;
49
49
  enabled;
50
50
  active;
51
- #e = new i(void 0);
52
- catalog = this.#e;
53
- #s = new l();
54
- constructor(s, t, a) {
55
- this.broadcast = s, this.active = new i(void 0), this.enabled = i.from(a?.enabled ?? !1), this.#s.effect((e) => {
56
- e.get(this.enabled) && this.#e.set(e.get(t)?.chat?.typing);
57
- }), this.#s.effect(this.#t.bind(this));
58
- }
59
- #t(s) {
60
- const t = s.getAll([this.enabled, this.#e, this.broadcast]);
61
- if (!t) return;
62
- const [a, e, o] = t, n = o.subscribe(e.name, h.typing);
51
+ #t = new i(void 0);
52
+ catalog = this.#t;
53
+ #s = new c();
54
+ constructor(s, e, a) {
55
+ this.broadcast = s, this.active = new i(void 0), this.enabled = i.from(a?.enabled ?? !1), this.#s.run((t) => {
56
+ t.get(this.enabled) && this.#t.set(t.get(e)?.chat?.typing);
57
+ }), this.#s.run(this.#e.bind(this));
58
+ }
59
+ #e(s) {
60
+ const e = s.getAll([this.enabled, this.#t, this.broadcast]);
61
+ if (!e) return;
62
+ const [a, t, o] = e, n = o.subscribe(t.name, h.typing);
63
63
  s.cleanup(() => n.close()), s.spawn(async () => {
64
64
  for (; ; ) {
65
- const c = await n.readBool();
66
- if (c === void 0) break;
67
- this.active.set(c);
65
+ const r = await n.readBool();
66
+ if (r === void 0) break;
67
+ this.active.set(r);
68
68
  }
69
69
  }), s.cleanup(() => this.active.set(void 0));
70
70
  }
@@ -75,12 +75,12 @@ class T {
75
75
  class j {
76
76
  message;
77
77
  typing;
78
- #e = new i(void 0);
79
- #s = new l();
80
- constructor(s, t, a) {
81
- this.message = new O(s, t, a?.message), this.typing = new T(s, t, a?.typing), this.#s.effect((e) => {
82
- const o = e.get(this.message.catalog), n = e.get(this.typing.catalog);
83
- !o && !n || e.set(this.#e, {
78
+ #t = new i(void 0);
79
+ #s = new c();
80
+ constructor(s, e, a) {
81
+ this.message = new O(s, e, a?.message), this.typing = new T(s, e, a?.typing), this.#s.run((t) => {
82
+ const o = t.get(this.message.catalog), n = t.get(this.typing.catalog);
83
+ !o && !n || t.set(this.#t, {
84
84
  message: o,
85
85
  typing: n
86
86
  });
@@ -94,34 +94,34 @@ const E = /* @__PURE__ */ Object.freeze(/* @__PURE__ */ Object.defineProperty({
94
94
  __proto__: null,
95
95
  Chat: j
96
96
  }, Symbol.toStringTag, { value: "Module" }));
97
- async function g(r, s) {
98
- const t = await r.readJson();
99
- if (t !== void 0)
100
- return s.parse(t);
97
+ async function g(l, s) {
98
+ const e = await l.readJson();
99
+ if (e !== void 0)
100
+ return s.parse(e);
101
101
  }
102
102
  class x {
103
103
  enabled;
104
104
  broadcast;
105
- #e = new i(void 0);
105
+ #t = new i(void 0);
106
106
  #s = new i(void 0);
107
- signals = new l();
108
- constructor(s, t, a) {
109
- this.broadcast = s, this.enabled = i.from(a?.enabled ?? !1), this.signals.effect((e) => {
110
- this.#e.set(e.get(t)?.location?.peers);
111
- }), this.signals.effect(this.#t.bind(this));
112
- }
113
- #t(s) {
114
- const t = s.getAll([this.enabled, this.#e, this.broadcast]);
115
- if (!t) return;
116
- const [a, e, o] = t, n = o.subscribe(e.name, h.location);
107
+ signals = new c();
108
+ constructor(s, e, a) {
109
+ this.broadcast = s, this.enabled = i.from(a?.enabled ?? !1), this.signals.run((t) => {
110
+ this.#t.set(t.get(e)?.location?.peers);
111
+ }), this.signals.run(this.#e.bind(this));
112
+ }
113
+ #e(s) {
114
+ const e = s.getAll([this.enabled, this.#t, this.broadcast]);
115
+ if (!e) return;
116
+ const [a, t, o] = e, n = o.subscribe(t.name, h.location);
117
117
  s.cleanup(() => n.close()), s.spawn(this.#a.bind(this, n));
118
118
  }
119
119
  async #a(s) {
120
120
  try {
121
121
  for (; ; ) {
122
- const t = await g(s, k);
123
- if (!t) break;
124
- this.#s.set(t);
122
+ const e = await g(s, M);
123
+ if (!e) break;
124
+ this.#s.set(e);
125
125
  }
126
126
  } finally {
127
127
  this.#s.set(void 0), s.close();
@@ -137,37 +137,37 @@ class x {
137
137
  class A {
138
138
  broadcast;
139
139
  enabled;
140
- #e = new i(void 0);
141
- handle = this.#e;
142
- #s = new i(void 0);
143
140
  #t = new i(void 0);
144
- position = this.#t;
145
- signals = new l();
146
- constructor(s, t, a) {
147
- this.broadcast = s, this.enabled = i.from(a?.enabled ?? !1), this.signals.effect((e) => {
148
- this.#s.set(e.get(t)?.location);
149
- }), this.signals.effect((e) => {
150
- e.get(this.enabled) && this.#t.set(e.get(this.#s)?.initial);
151
- }), this.signals.effect((e) => {
152
- this.#e.set(e.get(this.#s)?.handle);
153
- }), this.signals.effect((e) => {
154
- const o = e.get(this.broadcast);
141
+ handle = this.#t;
142
+ #s = new i(void 0);
143
+ #e = new i(void 0);
144
+ position = this.#e;
145
+ signals = new c();
146
+ constructor(s, e, a) {
147
+ this.broadcast = s, this.enabled = i.from(a?.enabled ?? !1), this.signals.run((t) => {
148
+ this.#s.set(t.get(e)?.location);
149
+ }), this.signals.run((t) => {
150
+ t.get(this.enabled) && this.#e.set(t.get(this.#s)?.initial);
151
+ }), this.signals.run((t) => {
152
+ this.#t.set(t.get(this.#s)?.handle);
153
+ }), this.signals.run((t) => {
154
+ const o = t.get(this.broadcast);
155
155
  if (!o) return;
156
- const n = e.get(this.#s)?.track;
156
+ const n = t.get(this.#s)?.track;
157
157
  if (!n) return;
158
- const c = o.subscribe(n.name, h.location);
159
- e.cleanup(() => c.close()), e.spawn(this.#a.bind(this, c));
158
+ const r = o.subscribe(n.name, h.location);
159
+ t.cleanup(() => r.close()), t.spawn(this.#a.bind(this, r));
160
160
  });
161
161
  }
162
162
  async #a(s) {
163
163
  try {
164
164
  for (; ; ) {
165
- const t = await g(s, M);
166
- if (!t) break;
167
- this.#t.set(t);
165
+ const e = await g(s, P);
166
+ if (!e) break;
167
+ this.#e.set(e);
168
168
  }
169
169
  } finally {
170
- this.#t.set(void 0), s.close();
170
+ this.#e.set(void 0), s.close();
171
171
  }
172
172
  }
173
173
  close() {
@@ -177,9 +177,9 @@ class A {
177
177
  class R {
178
178
  window;
179
179
  peers;
180
- signals = new l();
181
- constructor(s, t, a) {
182
- this.window = new A(s, t, a?.window), this.peers = new x(s, t, a?.peers);
180
+ signals = new c();
181
+ constructor(s, e, a) {
182
+ this.window = new A(s, e, a?.window), this.peers = new x(s, e, a?.peers);
183
183
  }
184
184
  close() {
185
185
  this.signals.close(), this.window.close(), this.peers.close();
@@ -193,24 +193,24 @@ class I {
193
193
  broadcast;
194
194
  enabled;
195
195
  preview = new i(void 0);
196
- #e = new i(void 0);
197
- #s = new l();
198
- constructor(s, t, a) {
199
- this.broadcast = s, this.enabled = i.from(a?.enabled ?? !1), this.#s.effect((e) => {
200
- this.#e.set(e.get(t)?.preview);
201
- }), this.#s.effect((e) => {
202
- const o = e.getAll([this.enabled, this.broadcast, this.#e]);
196
+ #t = new i(void 0);
197
+ #s = new c();
198
+ constructor(s, e, a) {
199
+ this.broadcast = s, this.enabled = i.from(a?.enabled ?? !1), this.#s.run((t) => {
200
+ this.#t.set(t.get(e)?.preview);
201
+ }), this.#s.run((t) => {
202
+ const o = t.getAll([this.enabled, this.broadcast, this.#t]);
203
203
  if (!o) return;
204
- const [n, c, u] = o, b = c.subscribe(u.name, h.preview);
205
- e.cleanup(() => b.close()), e.spawn(async () => {
204
+ const [n, r, b] = o, u = r.subscribe(b.name, h.preview);
205
+ t.cleanup(() => u.close()), t.spawn(async () => {
206
206
  try {
207
- const d = await g(b, P);
207
+ const d = await g(u, f);
208
208
  if (!d) return;
209
209
  this.preview.set(d);
210
210
  } catch (d) {
211
211
  console.warn("Failed to parse preview JSON:", d);
212
212
  }
213
- }), e.cleanup(() => this.preview.set(void 0));
213
+ }), t.cleanup(() => this.preview.set(void 0));
214
214
  });
215
215
  }
216
216
  close() {
package/index.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"file":"index.js","sources":["../src/chat/message.ts","../src/chat/typing.ts","../src/chat/index.ts","../../lite/src/zod.ts","../src/location/peers.ts","../src/location/window.ts","../src/location/index.ts","../src/preview.ts"],"sourcesContent":["import * as Catalog from \"@moq/hang/catalog\";\nimport type * as Moq from \"@moq/lite\";\nimport { Effect, type Getter, Signal } from \"@moq/signals\";\n\nexport interface MessageProps {\n\t// Whether to start downloading the chat.\n\t// Defaults to false so you can make sure everything is ready before starting.\n\tenabled?: boolean | Signal<boolean>;\n}\n\nexport class Message {\n\tbroadcast: Signal<Moq.Broadcast | undefined>;\n\tenabled: Signal<boolean>;\n\n\t// Empty string is a valid message.\n\t#latest = new Signal<string | undefined>(undefined);\n\treadonly latest: Getter<string | undefined> = this.#latest;\n\n\t#catalog = new Signal<Catalog.Track | undefined>(undefined);\n\treadonly catalog: Getter<Catalog.Track | undefined> = this.#catalog;\n\n\t#signals = new Effect();\n\n\tconstructor(\n\t\tbroadcast: Signal<Moq.Broadcast | undefined>,\n\t\tcatalog: Signal<Catalog.Root | undefined>,\n\t\tprops?: MessageProps,\n\t) {\n\t\tthis.broadcast = broadcast;\n\t\tthis.enabled = Signal.from(props?.enabled ?? false);\n\n\t\t// Grab the chat section from the catalog (if it's changed).\n\t\tthis.#signals.effect((effect) => {\n\t\t\tif (!effect.get(this.enabled)) return;\n\t\t\tthis.#catalog.set(effect.get(catalog)?.chat?.message);\n\t\t});\n\n\t\tthis.#signals.effect(this.#run.bind(this));\n\t}\n\n\t#run(effect: Effect) {\n\t\tconst values = effect.getAll([this.enabled, this.#catalog, this.broadcast]);\n\t\tif (!values) return;\n\t\tconst [_, catalog, broadcast] = values;\n\n\t\tconst track = broadcast.subscribe(catalog.name, Catalog.PRIORITY.chat);\n\t\teffect.cleanup(() => track.close());\n\n\t\t// Undefined is only when we're not subscribed to the track.\n\t\teffect.set(this.#latest, \"\");\n\t\teffect.cleanup(() => this.#latest.set(undefined));\n\n\t\teffect.spawn(async () => {\n\t\t\tfor (;;) {\n\t\t\t\tconst frame = await track.readString();\n\t\t\t\tif (frame === undefined) break;\n\n\t\t\t\t// Use a function to avoid the dequal check.\n\t\t\t\tthis.#latest.set(frame);\n\t\t\t}\n\t\t});\n\t}\n\n\tclose() {\n\t\tthis.#signals.close();\n\t}\n}\n","import * as Catalog from \"@moq/hang/catalog\";\nimport type * as Moq from \"@moq/lite\";\nimport { Effect, type Getter, Signal } from \"@moq/signals\";\n\nexport interface TypingProps {\n\t// Whether to start downloading the chat.\n\t// Defaults to false so you can make sure everything is ready before starting.\n\tenabled?: boolean | Signal<boolean>;\n}\n\nexport class Typing {\n\tbroadcast: Signal<Moq.Broadcast | undefined>;\n\tenabled: Signal<boolean>;\n\tactive: Signal<boolean | undefined>;\n\n\t#catalog = new Signal<Catalog.Track | undefined>(undefined);\n\treadonly catalog: Getter<Catalog.Track | undefined> = this.#catalog;\n\n\t#signals = new Effect();\n\n\tconstructor(\n\t\tbroadcast: Signal<Moq.Broadcast | undefined>,\n\t\tcatalog: Signal<Catalog.Root | undefined>,\n\t\tprops?: TypingProps,\n\t) {\n\t\tthis.broadcast = broadcast;\n\t\tthis.active = new Signal<boolean | undefined>(undefined);\n\t\tthis.enabled = Signal.from(props?.enabled ?? false);\n\n\t\t// Grab the chat section from the catalog (if it's changed).\n\t\tthis.#signals.effect((effect) => {\n\t\t\tif (!effect.get(this.enabled)) return;\n\t\t\tthis.#catalog.set(effect.get(catalog)?.chat?.typing);\n\t\t});\n\n\t\tthis.#signals.effect(this.#run.bind(this));\n\t}\n\n\t#run(effect: Effect) {\n\t\tconst values = effect.getAll([this.enabled, this.#catalog, this.broadcast]);\n\t\tif (!values) return;\n\t\tconst [_, catalog, broadcast] = values;\n\n\t\tconst track = broadcast.subscribe(catalog.name, Catalog.PRIORITY.typing);\n\t\teffect.cleanup(() => track.close());\n\n\t\teffect.spawn(async () => {\n\t\t\tfor (;;) {\n\t\t\t\tconst value = await track.readBool();\n\t\t\t\tif (value === undefined) break;\n\n\t\t\t\tthis.active.set(value);\n\t\t\t}\n\t\t});\n\n\t\teffect.cleanup(() => this.active.set(undefined));\n\t}\n\n\tclose() {\n\t\tthis.#signals.close();\n\t}\n}\n","import type * as Catalog from \"@moq/hang/catalog\";\nimport type * as Moq from \"@moq/lite\";\nimport { Effect, Signal } from \"@moq/signals\";\nimport { Message, type MessageProps } from \"./message\";\nimport { Typing, type TypingProps } from \"./typing\";\n\nexport interface ChatProps {\n\tmessage?: MessageProps;\n\ttyping?: TypingProps;\n}\n\nexport class Chat {\n\tmessage: Message;\n\ttyping: Typing;\n\n\t#catalog = new Signal<Catalog.Chat | undefined>(undefined);\n\t#signals = new Effect();\n\n\tconstructor(\n\t\tbroadcast: Signal<Moq.Broadcast | undefined>,\n\t\tcatalog: Signal<Catalog.Root | undefined>,\n\t\tprops?: ChatProps,\n\t) {\n\t\tthis.message = new Message(broadcast, catalog, props?.message);\n\t\tthis.typing = new Typing(broadcast, catalog, props?.typing);\n\n\t\t// Grab the chat section from the catalog (if it's changed).\n\t\tthis.#signals.effect((effect) => {\n\t\t\tconst message = effect.get(this.message.catalog);\n\t\t\tconst typing = effect.get(this.typing.catalog);\n\t\t\tif (!message && !typing) return;\n\n\t\t\teffect.set(this.#catalog, {\n\t\t\t\tmessage,\n\t\t\t\ttyping,\n\t\t\t});\n\t\t});\n\t}\n\n\tclose() {\n\t\tthis.#signals.close();\n\t\tthis.message.close();\n\t\tthis.typing.close();\n\t}\n}\n","// Helper containers for Zod-validated track encoding/decoding.\n\nimport type * as z from \"zod\";\nimport type { Group } from \"./group.ts\";\nimport type { Track } from \"./track.ts\";\n\nexport async function read<T = unknown>(source: Track | Group, schema: z.ZodSchema<T>): Promise<T | undefined> {\n\tconst next = await source.readJson();\n\tif (next === undefined) return undefined; // only treat undefined as EOF, not other falsy values\n\treturn schema.parse(next);\n}\n\nexport function write<T = unknown>(source: Track | Group, value: T, schema: z.ZodSchema<T>) {\n\tconst valid = schema.parse(value);\n\tsource.writeJson(valid);\n}\n","import * as Catalog from \"@moq/hang/catalog\";\nimport type * as Moq from \"@moq/lite\";\nimport * as Zod from \"@moq/lite/zod\";\nimport { Effect, type Getter, Signal } from \"@moq/signals\";\n\nexport interface PeersProps {\n\tenabled?: boolean | Signal<boolean>;\n}\n\nexport class Peers {\n\tenabled: Signal<boolean>;\n\tbroadcast: Signal<Moq.Broadcast | undefined>;\n\n\t#catalog = new Signal<Catalog.Track | undefined>(undefined);\n\t#positions = new Signal<Record<string, Catalog.Position> | undefined>(undefined);\n\n\tsignals = new Effect();\n\n\tconstructor(\n\t\tbroadcast: Signal<Moq.Broadcast | undefined>,\n\t\tcatalog: Signal<Catalog.Root | undefined>,\n\t\tprops?: PeersProps,\n\t) {\n\t\tthis.broadcast = broadcast;\n\t\tthis.enabled = Signal.from(props?.enabled ?? false);\n\n\t\tthis.signals.effect((effect) => {\n\t\t\tthis.#catalog.set(effect.get(catalog)?.location?.peers);\n\t\t});\n\n\t\tthis.signals.effect(this.#run.bind(this));\n\t}\n\n\t#run(effect: Effect) {\n\t\tconst values = effect.getAll([this.enabled, this.#catalog, this.broadcast]);\n\t\tif (!values) return;\n\t\tconst [_, catalog, broadcast] = values;\n\n\t\tconst track = broadcast.subscribe(catalog.name, Catalog.PRIORITY.location);\n\t\teffect.cleanup(() => track.close());\n\n\t\teffect.spawn(this.#runTrack.bind(this, track));\n\t}\n\n\tasync #runTrack(track: Moq.Track) {\n\t\ttry {\n\t\t\tfor (;;) {\n\t\t\t\tconst frame = await Zod.read(track, Catalog.PeersSchema);\n\t\t\t\tif (!frame) break;\n\n\t\t\t\tthis.#positions.set(frame);\n\t\t\t}\n\t\t} finally {\n\t\t\tthis.#positions.set(undefined);\n\t\t\ttrack.close();\n\t\t}\n\t}\n\n\tget positions(): Getter<Record<string, Catalog.Position> | undefined> {\n\t\treturn this.#positions;\n\t}\n\n\tclose() {\n\t\tthis.signals.close();\n\t}\n}\n","import * as Catalog from \"@moq/hang/catalog\";\nimport type * as Moq from \"@moq/lite\";\nimport * as Zod from \"@moq/lite/zod\";\nimport { Effect, type Getter, Signal } from \"@moq/signals\";\n\nexport interface WindowProps {\n\tenabled?: boolean | Signal<boolean>;\n}\n\nexport class Window {\n\tbroadcast: Signal<Moq.Broadcast | undefined>;\n\n\tenabled: Signal<boolean>;\n\n\t#handle = new Signal<string | undefined>(undefined);\n\treadonly handle: Getter<string | undefined> = this.#handle;\n\n\t#catalog = new Signal<Catalog.Location | undefined>(undefined);\n\n\t#position = new Signal<Catalog.Position | undefined>(undefined);\n\treadonly position: Getter<Catalog.Position | undefined> = this.#position;\n\n\tsignals = new Effect();\n\n\tconstructor(\n\t\tbroadcast: Signal<Moq.Broadcast | undefined>,\n\t\tcatalog: Signal<Catalog.Root | undefined>,\n\t\tprops?: WindowProps,\n\t) {\n\t\tthis.broadcast = broadcast;\n\t\tthis.enabled = Signal.from(props?.enabled ?? false);\n\n\t\tthis.signals.effect((effect) => {\n\t\t\tthis.#catalog.set(effect.get(catalog)?.location);\n\t\t});\n\n\t\tthis.signals.effect((effect) => {\n\t\t\tif (!effect.get(this.enabled)) return;\n\t\t\tthis.#position.set(effect.get(this.#catalog)?.initial);\n\t\t});\n\n\t\tthis.signals.effect((effect) => {\n\t\t\tthis.#handle.set(effect.get(this.#catalog)?.handle);\n\t\t});\n\n\t\tthis.signals.effect((effect) => {\n\t\t\tconst broadcast = effect.get(this.broadcast);\n\t\t\tif (!broadcast) return;\n\n\t\t\tconst updates = effect.get(this.#catalog)?.track;\n\t\t\tif (!updates) return;\n\n\t\t\tconst track = broadcast.subscribe(updates.name, Catalog.PRIORITY.location);\n\t\t\teffect.cleanup(() => track.close());\n\n\t\t\teffect.spawn(this.#runTrack.bind(this, track));\n\t\t});\n\t}\n\n\tasync #runTrack(track: Moq.Track) {\n\t\ttry {\n\t\t\tfor (;;) {\n\t\t\t\tconst position = await Zod.read(track, Catalog.PositionSchema);\n\t\t\t\tif (!position) break;\n\n\t\t\t\tthis.#position.set(position);\n\t\t\t}\n\t\t} finally {\n\t\t\tthis.#position.set(undefined);\n\t\t\ttrack.close();\n\t\t}\n\t}\n\n\tclose() {\n\t\tthis.signals.close();\n\t}\n}\n","import type * as Catalog from \"@moq/hang/catalog\";\nimport type * as Moq from \"@moq/lite\";\nimport { Effect, type Signal } from \"@moq/signals\";\nimport { Peers, type PeersProps } from \"./peers\";\nimport { Window, type WindowProps } from \"./window\";\n\nexport interface Props {\n\twindow?: WindowProps;\n\tpeers?: PeersProps;\n}\n\nexport class Root {\n\twindow: Window;\n\tpeers: Peers;\n\n\tsignals = new Effect();\n\n\tconstructor(\n\t\tbroadcast: Signal<Moq.Broadcast | undefined>,\n\t\tcatalog: Signal<Catalog.Root | undefined>,\n\t\tprops?: Props,\n\t) {\n\t\tthis.window = new Window(broadcast, catalog, props?.window);\n\t\tthis.peers = new Peers(broadcast, catalog, props?.peers);\n\t}\n\n\tclose() {\n\t\tthis.signals.close();\n\t\tthis.window.close();\n\t\tthis.peers.close();\n\t}\n}\n","import * as Catalog from \"@moq/hang/catalog\";\nimport type * as Moq from \"@moq/lite\";\nimport * as Zod from \"@moq/lite/zod\";\nimport { Effect, Signal } from \"@moq/signals\";\n\nexport interface PreviewProps {\n\tenabled?: boolean | Signal<boolean>;\n}\n\nexport class Preview {\n\tbroadcast: Signal<Moq.Broadcast | undefined>;\n\tenabled: Signal<boolean>;\n\tpreview = new Signal<Catalog.Preview | undefined>(undefined);\n\t#catalog = new Signal<Catalog.Track | undefined>(undefined);\n\n\t#signals = new Effect();\n\n\tconstructor(\n\t\tbroadcast: Signal<Moq.Broadcast | undefined>,\n\t\tcatalog: Signal<Catalog.Root | undefined>,\n\t\tprops?: PreviewProps,\n\t) {\n\t\tthis.broadcast = broadcast;\n\t\tthis.enabled = Signal.from(props?.enabled ?? false);\n\n\t\tthis.#signals.effect((effect) => {\n\t\t\tthis.#catalog.set(effect.get(catalog)?.preview);\n\t\t});\n\n\t\tthis.#signals.effect((effect) => {\n\t\t\tconst values = effect.getAll([this.enabled, this.broadcast, this.#catalog]);\n\t\t\tif (!values) return;\n\t\t\tconst [_, broadcast, catalog] = values;\n\n\t\t\t// Subscribe to the preview.json track directly\n\t\t\tconst track = broadcast.subscribe(catalog.name, Catalog.PRIORITY.preview);\n\t\t\teffect.cleanup(() => track.close());\n\n\t\t\teffect.spawn(async () => {\n\t\t\t\ttry {\n\t\t\t\t\tconst info = await Zod.read(track, Catalog.PreviewSchema);\n\t\t\t\t\tif (!info) return;\n\n\t\t\t\t\tthis.preview.set(info);\n\t\t\t\t} catch (error) {\n\t\t\t\t\tconsole.warn(\"Failed to parse preview JSON:\", error);\n\t\t\t\t}\n\t\t\t});\n\n\t\t\teffect.cleanup(() => this.preview.set(undefined));\n\t\t});\n\t}\n\n\tclose() {\n\t\tthis.#signals.close();\n\t}\n}\n"],"names":["Message","#latest","Signal","#catalog","#signals","Effect","broadcast","catalog","props","effect","#run","values","_","track","Catalog.PRIORITY","frame","Typing","value","Chat","message","typing","read","source","schema","next","Peers","#positions","#runTrack","Zod.read","Catalog.PeersSchema","Window","#handle","#position","updates","position","Catalog.PositionSchema","Root","Preview","info","Catalog.PreviewSchema","error"],"mappings":"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"}
1
+ {"version":3,"file":"index.js","sources":["../src/chat/message.ts","../src/chat/typing.ts","../src/chat/index.ts","../../lite/src/zod.ts","../src/location/peers.ts","../src/location/window.ts","../src/location/index.ts","../src/preview.ts"],"sourcesContent":["import * as Catalog from \"@moq/hang/catalog\";\nimport type * as Moq from \"@moq/lite\";\nimport { Effect, type Getter, Signal } from \"@moq/signals\";\n\nexport interface MessageProps {\n\t// Whether to start downloading the chat.\n\t// Defaults to false so you can make sure everything is ready before starting.\n\tenabled?: boolean | Signal<boolean>;\n}\n\nexport class Message {\n\tbroadcast: Signal<Moq.Broadcast | undefined>;\n\tenabled: Signal<boolean>;\n\n\t// Empty string is a valid message.\n\t#latest = new Signal<string | undefined>(undefined);\n\treadonly latest: Getter<string | undefined> = this.#latest;\n\n\t#catalog = new Signal<Catalog.Track | undefined>(undefined);\n\treadonly catalog: Getter<Catalog.Track | undefined> = this.#catalog;\n\n\t#signals = new Effect();\n\n\tconstructor(\n\t\tbroadcast: Signal<Moq.Broadcast | undefined>,\n\t\tcatalog: Signal<Catalog.Root | undefined>,\n\t\tprops?: MessageProps,\n\t) {\n\t\tthis.broadcast = broadcast;\n\t\tthis.enabled = Signal.from(props?.enabled ?? false);\n\n\t\t// Grab the chat section from the catalog (if it's changed).\n\t\tthis.#signals.run((effect) => {\n\t\t\tif (!effect.get(this.enabled)) return;\n\t\t\tthis.#catalog.set(effect.get(catalog)?.chat?.message);\n\t\t});\n\n\t\tthis.#signals.run(this.#run.bind(this));\n\t}\n\n\t#run(effect: Effect) {\n\t\tconst values = effect.getAll([this.enabled, this.#catalog, this.broadcast]);\n\t\tif (!values) return;\n\t\tconst [_, catalog, broadcast] = values;\n\n\t\tconst track = broadcast.subscribe(catalog.name, Catalog.PRIORITY.chat);\n\t\teffect.cleanup(() => track.close());\n\n\t\t// Undefined is only when we're not subscribed to the track.\n\t\teffect.set(this.#latest, \"\");\n\t\teffect.cleanup(() => this.#latest.set(undefined));\n\n\t\teffect.spawn(async () => {\n\t\t\tfor (;;) {\n\t\t\t\tconst frame = await track.readString();\n\t\t\t\tif (frame === undefined) break;\n\n\t\t\t\t// Use a function to avoid the dequal check.\n\t\t\t\tthis.#latest.set(frame);\n\t\t\t}\n\t\t});\n\t}\n\n\tclose() {\n\t\tthis.#signals.close();\n\t}\n}\n","import * as Catalog from \"@moq/hang/catalog\";\nimport type * as Moq from \"@moq/lite\";\nimport { Effect, type Getter, Signal } from \"@moq/signals\";\n\nexport interface TypingProps {\n\t// Whether to start downloading the chat.\n\t// Defaults to false so you can make sure everything is ready before starting.\n\tenabled?: boolean | Signal<boolean>;\n}\n\nexport class Typing {\n\tbroadcast: Signal<Moq.Broadcast | undefined>;\n\tenabled: Signal<boolean>;\n\tactive: Signal<boolean | undefined>;\n\n\t#catalog = new Signal<Catalog.Track | undefined>(undefined);\n\treadonly catalog: Getter<Catalog.Track | undefined> = this.#catalog;\n\n\t#signals = new Effect();\n\n\tconstructor(\n\t\tbroadcast: Signal<Moq.Broadcast | undefined>,\n\t\tcatalog: Signal<Catalog.Root | undefined>,\n\t\tprops?: TypingProps,\n\t) {\n\t\tthis.broadcast = broadcast;\n\t\tthis.active = new Signal<boolean | undefined>(undefined);\n\t\tthis.enabled = Signal.from(props?.enabled ?? false);\n\n\t\t// Grab the chat section from the catalog (if it's changed).\n\t\tthis.#signals.run((effect) => {\n\t\t\tif (!effect.get(this.enabled)) return;\n\t\t\tthis.#catalog.set(effect.get(catalog)?.chat?.typing);\n\t\t});\n\n\t\tthis.#signals.run(this.#run.bind(this));\n\t}\n\n\t#run(effect: Effect) {\n\t\tconst values = effect.getAll([this.enabled, this.#catalog, this.broadcast]);\n\t\tif (!values) return;\n\t\tconst [_, catalog, broadcast] = values;\n\n\t\tconst track = broadcast.subscribe(catalog.name, Catalog.PRIORITY.typing);\n\t\teffect.cleanup(() => track.close());\n\n\t\teffect.spawn(async () => {\n\t\t\tfor (;;) {\n\t\t\t\tconst value = await track.readBool();\n\t\t\t\tif (value === undefined) break;\n\n\t\t\t\tthis.active.set(value);\n\t\t\t}\n\t\t});\n\n\t\teffect.cleanup(() => this.active.set(undefined));\n\t}\n\n\tclose() {\n\t\tthis.#signals.close();\n\t}\n}\n","import type * as Catalog from \"@moq/hang/catalog\";\nimport type * as Moq from \"@moq/lite\";\nimport { Effect, Signal } from \"@moq/signals\";\nimport { Message, type MessageProps } from \"./message\";\nimport { Typing, type TypingProps } from \"./typing\";\n\nexport interface ChatProps {\n\tmessage?: MessageProps;\n\ttyping?: TypingProps;\n}\n\nexport class Chat {\n\tmessage: Message;\n\ttyping: Typing;\n\n\t#catalog = new Signal<Catalog.Chat | undefined>(undefined);\n\t#signals = new Effect();\n\n\tconstructor(\n\t\tbroadcast: Signal<Moq.Broadcast | undefined>,\n\t\tcatalog: Signal<Catalog.Root | undefined>,\n\t\tprops?: ChatProps,\n\t) {\n\t\tthis.message = new Message(broadcast, catalog, props?.message);\n\t\tthis.typing = new Typing(broadcast, catalog, props?.typing);\n\n\t\t// Grab the chat section from the catalog (if it's changed).\n\t\tthis.#signals.run((effect) => {\n\t\t\tconst message = effect.get(this.message.catalog);\n\t\t\tconst typing = effect.get(this.typing.catalog);\n\t\t\tif (!message && !typing) return;\n\n\t\t\teffect.set(this.#catalog, {\n\t\t\t\tmessage,\n\t\t\t\ttyping,\n\t\t\t});\n\t\t});\n\t}\n\n\tclose() {\n\t\tthis.#signals.close();\n\t\tthis.message.close();\n\t\tthis.typing.close();\n\t}\n}\n","// Helper containers for Zod-validated track encoding/decoding.\n\nimport type * as z from \"zod\";\nimport type { Group } from \"./group.ts\";\nimport type { Track } from \"./track.ts\";\n\nexport async function read<T = unknown>(source: Track | Group, schema: z.ZodSchema<T>): Promise<T | undefined> {\n\tconst next = await source.readJson();\n\tif (next === undefined) return undefined; // only treat undefined as EOF, not other falsy values\n\treturn schema.parse(next);\n}\n\nexport function write<T = unknown>(source: Track | Group, value: T, schema: z.ZodSchema<T>) {\n\tconst valid = schema.parse(value);\n\tsource.writeJson(valid);\n}\n","import * as Catalog from \"@moq/hang/catalog\";\nimport type * as Moq from \"@moq/lite\";\nimport * as Zod from \"@moq/lite/zod\";\nimport { Effect, type Getter, Signal } from \"@moq/signals\";\n\nexport interface PeersProps {\n\tenabled?: boolean | Signal<boolean>;\n}\n\nexport class Peers {\n\tenabled: Signal<boolean>;\n\tbroadcast: Signal<Moq.Broadcast | undefined>;\n\n\t#catalog = new Signal<Catalog.Track | undefined>(undefined);\n\t#positions = new Signal<Record<string, Catalog.Position> | undefined>(undefined);\n\n\tsignals = new Effect();\n\n\tconstructor(\n\t\tbroadcast: Signal<Moq.Broadcast | undefined>,\n\t\tcatalog: Signal<Catalog.Root | undefined>,\n\t\tprops?: PeersProps,\n\t) {\n\t\tthis.broadcast = broadcast;\n\t\tthis.enabled = Signal.from(props?.enabled ?? false);\n\n\t\tthis.signals.run((effect) => {\n\t\t\tthis.#catalog.set(effect.get(catalog)?.location?.peers);\n\t\t});\n\n\t\tthis.signals.run(this.#run.bind(this));\n\t}\n\n\t#run(effect: Effect) {\n\t\tconst values = effect.getAll([this.enabled, this.#catalog, this.broadcast]);\n\t\tif (!values) return;\n\t\tconst [_, catalog, broadcast] = values;\n\n\t\tconst track = broadcast.subscribe(catalog.name, Catalog.PRIORITY.location);\n\t\teffect.cleanup(() => track.close());\n\n\t\teffect.spawn(this.#runTrack.bind(this, track));\n\t}\n\n\tasync #runTrack(track: Moq.Track) {\n\t\ttry {\n\t\t\tfor (;;) {\n\t\t\t\tconst frame = await Zod.read(track, Catalog.PeersSchema);\n\t\t\t\tif (!frame) break;\n\n\t\t\t\tthis.#positions.set(frame);\n\t\t\t}\n\t\t} finally {\n\t\t\tthis.#positions.set(undefined);\n\t\t\ttrack.close();\n\t\t}\n\t}\n\n\tget positions(): Getter<Record<string, Catalog.Position> | undefined> {\n\t\treturn this.#positions;\n\t}\n\n\tclose() {\n\t\tthis.signals.close();\n\t}\n}\n","import * as Catalog from \"@moq/hang/catalog\";\nimport type * as Moq from \"@moq/lite\";\nimport * as Zod from \"@moq/lite/zod\";\nimport { Effect, type Getter, Signal } from \"@moq/signals\";\n\nexport interface WindowProps {\n\tenabled?: boolean | Signal<boolean>;\n}\n\nexport class Window {\n\tbroadcast: Signal<Moq.Broadcast | undefined>;\n\n\tenabled: Signal<boolean>;\n\n\t#handle = new Signal<string | undefined>(undefined);\n\treadonly handle: Getter<string | undefined> = this.#handle;\n\n\t#catalog = new Signal<Catalog.Location | undefined>(undefined);\n\n\t#position = new Signal<Catalog.Position | undefined>(undefined);\n\treadonly position: Getter<Catalog.Position | undefined> = this.#position;\n\n\tsignals = new Effect();\n\n\tconstructor(\n\t\tbroadcast: Signal<Moq.Broadcast | undefined>,\n\t\tcatalog: Signal<Catalog.Root | undefined>,\n\t\tprops?: WindowProps,\n\t) {\n\t\tthis.broadcast = broadcast;\n\t\tthis.enabled = Signal.from(props?.enabled ?? false);\n\n\t\tthis.signals.run((effect) => {\n\t\t\tthis.#catalog.set(effect.get(catalog)?.location);\n\t\t});\n\n\t\tthis.signals.run((effect) => {\n\t\t\tif (!effect.get(this.enabled)) return;\n\t\t\tthis.#position.set(effect.get(this.#catalog)?.initial);\n\t\t});\n\n\t\tthis.signals.run((effect) => {\n\t\t\tthis.#handle.set(effect.get(this.#catalog)?.handle);\n\t\t});\n\n\t\tthis.signals.run((effect) => {\n\t\t\tconst broadcast = effect.get(this.broadcast);\n\t\t\tif (!broadcast) return;\n\n\t\t\tconst updates = effect.get(this.#catalog)?.track;\n\t\t\tif (!updates) return;\n\n\t\t\tconst track = broadcast.subscribe(updates.name, Catalog.PRIORITY.location);\n\t\t\teffect.cleanup(() => track.close());\n\n\t\t\teffect.spawn(this.#runTrack.bind(this, track));\n\t\t});\n\t}\n\n\tasync #runTrack(track: Moq.Track) {\n\t\ttry {\n\t\t\tfor (;;) {\n\t\t\t\tconst position = await Zod.read(track, Catalog.PositionSchema);\n\t\t\t\tif (!position) break;\n\n\t\t\t\tthis.#position.set(position);\n\t\t\t}\n\t\t} finally {\n\t\t\tthis.#position.set(undefined);\n\t\t\ttrack.close();\n\t\t}\n\t}\n\n\tclose() {\n\t\tthis.signals.close();\n\t}\n}\n","import type * as Catalog from \"@moq/hang/catalog\";\nimport type * as Moq from \"@moq/lite\";\nimport { Effect, type Signal } from \"@moq/signals\";\nimport { Peers, type PeersProps } from \"./peers\";\nimport { Window, type WindowProps } from \"./window\";\n\nexport interface Props {\n\twindow?: WindowProps;\n\tpeers?: PeersProps;\n}\n\nexport class Root {\n\twindow: Window;\n\tpeers: Peers;\n\n\tsignals = new Effect();\n\n\tconstructor(\n\t\tbroadcast: Signal<Moq.Broadcast | undefined>,\n\t\tcatalog: Signal<Catalog.Root | undefined>,\n\t\tprops?: Props,\n\t) {\n\t\tthis.window = new Window(broadcast, catalog, props?.window);\n\t\tthis.peers = new Peers(broadcast, catalog, props?.peers);\n\t}\n\n\tclose() {\n\t\tthis.signals.close();\n\t\tthis.window.close();\n\t\tthis.peers.close();\n\t}\n}\n","import * as Catalog from \"@moq/hang/catalog\";\nimport type * as Moq from \"@moq/lite\";\nimport * as Zod from \"@moq/lite/zod\";\nimport { Effect, Signal } from \"@moq/signals\";\n\nexport interface PreviewProps {\n\tenabled?: boolean | Signal<boolean>;\n}\n\nexport class Preview {\n\tbroadcast: Signal<Moq.Broadcast | undefined>;\n\tenabled: Signal<boolean>;\n\tpreview = new Signal<Catalog.Preview | undefined>(undefined);\n\t#catalog = new Signal<Catalog.Track | undefined>(undefined);\n\n\t#signals = new Effect();\n\n\tconstructor(\n\t\tbroadcast: Signal<Moq.Broadcast | undefined>,\n\t\tcatalog: Signal<Catalog.Root | undefined>,\n\t\tprops?: PreviewProps,\n\t) {\n\t\tthis.broadcast = broadcast;\n\t\tthis.enabled = Signal.from(props?.enabled ?? false);\n\n\t\tthis.#signals.run((effect) => {\n\t\t\tthis.#catalog.set(effect.get(catalog)?.preview);\n\t\t});\n\n\t\tthis.#signals.run((effect) => {\n\t\t\tconst values = effect.getAll([this.enabled, this.broadcast, this.#catalog]);\n\t\t\tif (!values) return;\n\t\t\tconst [_, broadcast, catalog] = values;\n\n\t\t\t// Subscribe to the preview.json track directly\n\t\t\tconst track = broadcast.subscribe(catalog.name, Catalog.PRIORITY.preview);\n\t\t\teffect.cleanup(() => track.close());\n\n\t\t\teffect.spawn(async () => {\n\t\t\t\ttry {\n\t\t\t\t\tconst info = await Zod.read(track, Catalog.PreviewSchema);\n\t\t\t\t\tif (!info) return;\n\n\t\t\t\t\tthis.preview.set(info);\n\t\t\t\t} catch (error) {\n\t\t\t\t\tconsole.warn(\"Failed to parse preview JSON:\", error);\n\t\t\t\t}\n\t\t\t});\n\n\t\t\teffect.cleanup(() => this.preview.set(undefined));\n\t\t});\n\t}\n\n\tclose() {\n\t\tthis.#signals.close();\n\t}\n}\n"],"names":["Message","#latest","Signal","#catalog","#signals","Effect","broadcast","catalog","props","effect","#run","values","_","track","Catalog.PRIORITY","frame","Typing","value","Chat","message","typing","read","source","schema","next","Peers","#positions","#runTrack","Zod.read","Catalog.PeersSchema","Window","#handle","#position","updates","position","Catalog.PositionSchema","Root","Preview","info","Catalog.PreviewSchema","error"],"mappings":"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"}
package/package.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "name": "@moq/watch",
3
3
  "type": "module",
4
- "version": "0.1.0",
4
+ "version": "0.1.1",
5
5
  "description": "Watch/subscribe to Media over QUIC streams",
6
6
  "license": "(MIT OR Apache-2.0)",
7
7
  "repository": "github:moq-dev/moq",
@@ -33,9 +33,9 @@
33
33
  "./support/element.js"
34
34
  ],
35
35
  "dependencies": {
36
- "@moq/hang": "^0.1.3",
36
+ "@moq/hang": "^0.2.0",
37
37
  "@moq/lite": "^0.1.3",
38
- "@moq/signals": "^0.1.2",
38
+ "@moq/signals": "^0.1.3",
39
39
  "@moq/ui-core": "^0.1.0"
40
40
  }
41
41
  }
@@ -12,7 +12,7 @@ function d(h, e, ...t) {
12
12
  typeof r == "string" ? n.appendChild(document.createTextNode(r)) : n.appendChild(r);
13
13
  }), Object.assign(n, s), n;
14
14
  }
15
- const f = navigator.userAgent.toLowerCase().includes("firefox"), C = ["show", "details"];
15
+ const g = navigator.userAgent.toLowerCase().includes("firefox"), C = ["show", "details"];
16
16
  class y extends HTMLElement {
17
17
  #i = new u("warning");
18
18
  #e = new u(!1);
@@ -24,7 +24,7 @@ class y extends HTMLElement {
24
24
  super(), m().then((e) => this.#n.set(e)).catch((e) => console.error("Failed to detect watch support:", e));
25
25
  }
26
26
  connectedCallback() {
27
- this.#t = new b(), this.#t.effect(this.#s.bind(this));
27
+ this.#t = new b(), this.#t.run(this.#s.bind(this));
28
28
  }
29
29
  disconnectedCallback() {
30
30
  this.#t?.close(), this.#t = void 0;
@@ -94,7 +94,7 @@ class y extends HTMLElement {
94
94
  });
95
95
  n.event(o, "click", () => {
96
96
  this.#e.update((s) => !s);
97
- }), n.effect((s) => {
97
+ }), n.run((s) => {
98
98
  o.textContent = s.get(this.#e) ? "Details ➖" : "Details ➕";
99
99
  });
100
100
  const l = d(
@@ -121,7 +121,7 @@ class y extends HTMLElement {
121
121
  padding: "1rem",
122
122
  fontSize: "0.875rem"
123
123
  }
124
- }), a = (r) => r ? "🟢 Yes" : "🔴 No", o = (r) => r?.hardware ? "🟢 Hardware" : r?.software ? `🟡 Software${f ? "*" : ""}` : "🔴 No", l = (r) => r === "full" ? "🟢 Full" : r === "partial" ? "🟡 Polyfill" : "🔴 None", s = (r, c, g) => {
124
+ }), a = (r) => r ? "🟢 Yes" : "🔴 No", o = (r) => r?.hardware ? "🟢 Hardware" : r?.software ? `🟡 Software${g ? "*" : ""}` : "🔴 No", l = (r) => r === "full" ? "🟢 Full" : r === "partial" ? "🟡 Polyfill" : "🔴 None", s = (r, c, f) => {
125
125
  const p = d(
126
126
  "div",
127
127
  {
@@ -146,11 +146,11 @@ class y extends HTMLElement {
146
146
  {
147
147
  style: { gridColumnStart: "3" }
148
148
  },
149
- g
149
+ f
150
150
  );
151
151
  i.appendChild(p), i.appendChild(w), i.appendChild(v);
152
152
  };
153
- if (s("WebTransport", "", l(t.webtransport)), s("Rendering", "Audio", a(t.audio.render)), s("", "Video", a(t.video.render)), s("Decoding", "Opus", l(t.audio.decoding.opus)), s("", "AAC", a(t.audio.decoding.aac)), s("", "AV1", o(t.video.decoding?.av1)), s("", "H.265", o(t.video.decoding?.h265)), s("", "H.264", o(t.video.decoding?.h264)), s("", "VP9", o(t.video.decoding?.vp9)), s("", "VP8", o(t.video.decoding?.vp8)), f) {
153
+ if (s("WebTransport", "", l(t.webtransport)), s("Rendering", "Audio", a(t.audio.render)), s("", "Video", a(t.video.render)), s("Decoding", "Opus", l(t.audio.decoding.opus)), s("", "AAC", a(t.audio.decoding.aac)), s("", "AV1", o(t.video.decoding?.av1)), s("", "H.265", o(t.video.decoding?.h265)), s("", "H.264", o(t.video.decoding?.h264)), s("", "VP9", o(t.video.decoding?.vp9)), s("", "VP8", o(t.video.decoding?.vp8)), g) {
154
154
  const r = d(
155
155
  "div",
156
156
  {