@moq/watch 0.1.0

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.
Files changed (110) hide show
  1. package/README.md +102 -0
  2. package/audio/backend.d.ts +15 -0
  3. package/audio/backend.d.ts.map +1 -0
  4. package/audio/decoder.d.ts +25 -0
  5. package/audio/decoder.d.ts.map +1 -0
  6. package/audio/emitter.d.ts +17 -0
  7. package/audio/emitter.d.ts.map +1 -0
  8. package/audio/index.d.ts +6 -0
  9. package/audio/index.d.ts.map +1 -0
  10. package/audio/mse.d.ts +21 -0
  11. package/audio/mse.d.ts.map +1 -0
  12. package/audio/render-worklet.d.ts +2 -0
  13. package/audio/render-worklet.d.ts.map +1 -0
  14. package/audio/render.d.ts +24 -0
  15. package/audio/render.d.ts.map +1 -0
  16. package/audio/ring-buffer.d.ts +19 -0
  17. package/audio/ring-buffer.d.ts.map +1 -0
  18. package/audio/ring-buffer.test.d.ts +2 -0
  19. package/audio/ring-buffer.test.d.ts.map +1 -0
  20. package/audio/source.d.ts +34 -0
  21. package/audio/source.d.ts.map +1 -0
  22. package/backend.d.ts +53 -0
  23. package/backend.d.ts.map +1 -0
  24. package/broadcast-CSXy8QiX.js +8192 -0
  25. package/broadcast-CSXy8QiX.js.map +1 -0
  26. package/broadcast.d.ts +23 -0
  27. package/broadcast.d.ts.map +1 -0
  28. package/chat/index.d.ts +17 -0
  29. package/chat/index.d.ts.map +1 -0
  30. package/chat/message.d.ts +16 -0
  31. package/chat/message.d.ts.map +1 -0
  32. package/chat/typing.d.ts +16 -0
  33. package/chat/typing.d.ts.map +1 -0
  34. package/element.d.ts +35 -0
  35. package/element.d.ts.map +1 -0
  36. package/element.js +105 -0
  37. package/element.js.map +1 -0
  38. package/index.d.ts +10 -0
  39. package/index.d.ts.map +1 -0
  40. package/index.js +232 -0
  41. package/index.js.map +1 -0
  42. package/libav-opus-af-BlMWboA7.js +368 -0
  43. package/libav-opus-af-BlMWboA7.js.map +1 -0
  44. package/location/index.d.ts +17 -0
  45. package/location/index.d.ts.map +1 -0
  46. package/location/peers.d.ts +16 -0
  47. package/location/peers.d.ts.map +1 -0
  48. package/location/window.d.ts +17 -0
  49. package/location/window.d.ts.map +1 -0
  50. package/main-DGBFe0O7.js +2301 -0
  51. package/main-DGBFe0O7.js.map +1 -0
  52. package/mse.d.ts +19 -0
  53. package/mse.d.ts.map +1 -0
  54. package/package.json +41 -0
  55. package/preview.d.ts +23 -0
  56. package/preview.d.ts.map +1 -0
  57. package/support/element.d.ts +22 -0
  58. package/support/element.d.ts.map +1 -0
  59. package/support/element.js +184 -0
  60. package/support/element.js.map +1 -0
  61. package/support/index.d.ts +29 -0
  62. package/support/index.d.ts.map +1 -0
  63. package/support/index.js +55 -0
  64. package/support/index.js.map +1 -0
  65. package/sync.d.ts +21 -0
  66. package/sync.d.ts.map +1 -0
  67. package/ui/components/BufferControl.d.ts +8 -0
  68. package/ui/components/BufferControl.d.ts.map +1 -0
  69. package/ui/components/BufferingIndicator.d.ts +2 -0
  70. package/ui/components/BufferingIndicator.d.ts.map +1 -0
  71. package/ui/components/FullscreenButton.d.ts +2 -0
  72. package/ui/components/FullscreenButton.d.ts.map +1 -0
  73. package/ui/components/LatencySlider.d.ts +2 -0
  74. package/ui/components/LatencySlider.d.ts.map +1 -0
  75. package/ui/components/PlayPauseButton.d.ts +2 -0
  76. package/ui/components/PlayPauseButton.d.ts.map +1 -0
  77. package/ui/components/QualitySelector.d.ts +3 -0
  78. package/ui/components/QualitySelector.d.ts.map +1 -0
  79. package/ui/components/StatsButton.d.ts +5 -0
  80. package/ui/components/StatsButton.d.ts.map +1 -0
  81. package/ui/components/VolumeSlider.d.ts +2 -0
  82. package/ui/components/VolumeSlider.d.ts.map +1 -0
  83. package/ui/components/WatchControls.d.ts +2 -0
  84. package/ui/components/WatchControls.d.ts.map +1 -0
  85. package/ui/components/WatchStatusIndicator.d.ts +2 -0
  86. package/ui/components/WatchStatusIndicator.d.ts.map +1 -0
  87. package/ui/context.d.ts +41 -0
  88. package/ui/context.d.ts.map +1 -0
  89. package/ui/element.d.ts +5 -0
  90. package/ui/element.d.ts.map +1 -0
  91. package/ui/hooks/use-watch-ui.d.ts +3 -0
  92. package/ui/hooks/use-watch-ui.d.ts.map +1 -0
  93. package/ui/index.d.ts +7 -0
  94. package/ui/index.d.ts.map +1 -0
  95. package/ui/index.js +1121 -0
  96. package/ui/index.js.map +1 -0
  97. package/user.d.ts +17 -0
  98. package/user.d.ts.map +1 -0
  99. package/video/backend.d.ts +16 -0
  100. package/video/backend.d.ts.map +1 -0
  101. package/video/decoder.d.ts +25 -0
  102. package/video/decoder.d.ts.map +1 -0
  103. package/video/index.d.ts +6 -0
  104. package/video/index.d.ts.map +1 -0
  105. package/video/mse.d.ts +23 -0
  106. package/video/mse.d.ts.map +1 -0
  107. package/video/renderer.d.ts +15 -0
  108. package/video/renderer.d.ts.map +1 -0
  109. package/video/source.d.ts +35 -0
  110. package/video/source.d.ts.map +1 -0
package/mse.d.ts ADDED
@@ -0,0 +1,19 @@
1
+ import { type Getter, Signal } from "@moq/signals";
2
+ import type { Sync } from "./sync";
3
+ export type MuxerProps = {
4
+ element?: HTMLMediaElement | Signal<HTMLMediaElement | undefined>;
5
+ paused?: boolean | Signal<boolean>;
6
+ };
7
+ /**
8
+ * MSE-based video source for CMAF/fMP4 fragments.
9
+ * Uses Media Source Extensions to handle complete moof+mdat fragments.
10
+ */
11
+ export declare class Muxer {
12
+ #private;
13
+ element: Signal<HTMLMediaElement | undefined>;
14
+ paused: Signal<boolean>;
15
+ readonly mediaSource: Getter<MediaSource | undefined>;
16
+ constructor(sync: Sync, props?: MuxerProps);
17
+ close(): void;
18
+ }
19
+ //# sourceMappingURL=mse.d.ts.map
package/mse.d.ts.map ADDED
@@ -0,0 +1 @@
1
+ {"version":3,"file":"mse.d.ts","sourceRoot":"","sources":["../src/mse.ts"],"names":[],"mappings":"AACA,OAAO,EAAU,KAAK,MAAM,EAAE,MAAM,EAAE,MAAM,cAAc,CAAC;AAC3D,OAAO,KAAK,EAAE,IAAI,EAAE,MAAM,QAAQ,CAAC;AAEnC,MAAM,MAAM,UAAU,GAAG;IACxB,OAAO,CAAC,EAAE,gBAAgB,GAAG,MAAM,CAAC,gBAAgB,GAAG,SAAS,CAAC,CAAC;IAClE,MAAM,CAAC,EAAE,OAAO,GAAG,MAAM,CAAC,OAAO,CAAC,CAAC;CACnC,CAAC;AAEF;;;GAGG;AACH,qBAAa,KAAK;;IACjB,OAAO,EAAE,MAAM,CAAC,gBAAgB,GAAG,SAAS,CAAC,CAAC;IAE9C,MAAM,EAAE,MAAM,CAAC,OAAO,CAAC,CAAC;IAKxB,QAAQ,CAAC,WAAW,EAAE,MAAM,CAAC,WAAW,GAAG,SAAS,CAAC,CAAqB;gBAI9D,IAAI,EAAE,IAAI,EAAE,KAAK,CAAC,EAAE,UAAU;IA2H1C,KAAK,IAAI,IAAI;CAGb"}
package/package.json ADDED
@@ -0,0 +1,41 @@
1
+ {
2
+ "name": "@moq/watch",
3
+ "type": "module",
4
+ "version": "0.1.0",
5
+ "description": "Watch/subscribe to Media over QUIC streams",
6
+ "license": "(MIT OR Apache-2.0)",
7
+ "repository": "github:moq-dev/moq",
8
+ "exports": {
9
+ ".": {
10
+ "types": "./index.d.ts",
11
+ "default": "./index.js"
12
+ },
13
+ "./element": {
14
+ "types": "./element.d.ts",
15
+ "default": "./element.js"
16
+ },
17
+ "./ui": {
18
+ "types": "./ui/index.d.ts",
19
+ "default": "./ui/index.js"
20
+ },
21
+ "./support": {
22
+ "types": "./support/index.d.ts",
23
+ "default": "./support/index.js"
24
+ },
25
+ "./support/element": {
26
+ "types": "./support/element.d.ts",
27
+ "default": "./support/element.js"
28
+ }
29
+ },
30
+ "sideEffects": [
31
+ "./element.js",
32
+ "./ui/index.js",
33
+ "./support/element.js"
34
+ ],
35
+ "dependencies": {
36
+ "@moq/hang": "^0.1.3",
37
+ "@moq/lite": "^0.1.3",
38
+ "@moq/signals": "^0.1.2",
39
+ "@moq/ui-core": "^0.1.0"
40
+ }
41
+ }
package/preview.d.ts ADDED
@@ -0,0 +1,23 @@
1
+ import * as Catalog from "@moq/hang/catalog";
2
+ import type * as Moq from "@moq/lite";
3
+ import { Signal } from "@moq/signals";
4
+ export interface PreviewProps {
5
+ enabled?: boolean | Signal<boolean>;
6
+ }
7
+ export declare class Preview {
8
+ #private;
9
+ broadcast: Signal<Moq.Broadcast | undefined>;
10
+ enabled: Signal<boolean>;
11
+ preview: Moq.Signals.Signal<{
12
+ name?: string | undefined;
13
+ avatar?: string | undefined;
14
+ audio?: boolean | undefined;
15
+ video?: boolean | undefined;
16
+ typing?: boolean | undefined;
17
+ chat?: boolean | undefined;
18
+ screen?: boolean | undefined;
19
+ } | undefined>;
20
+ constructor(broadcast: Signal<Moq.Broadcast | undefined>, catalog: Signal<Catalog.Root | undefined>, props?: PreviewProps);
21
+ close(): void;
22
+ }
23
+ //# sourceMappingURL=preview.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"preview.d.ts","sourceRoot":"","sources":["../src/preview.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,OAAO,MAAM,mBAAmB,CAAC;AAC7C,OAAO,KAAK,KAAK,GAAG,MAAM,WAAW,CAAC;AAEtC,OAAO,EAAU,MAAM,EAAE,MAAM,cAAc,CAAC;AAE9C,MAAM,WAAW,YAAY;IAC5B,OAAO,CAAC,EAAE,OAAO,GAAG,MAAM,CAAC,OAAO,CAAC,CAAC;CACpC;AAED,qBAAa,OAAO;;IACnB,SAAS,EAAE,MAAM,CAAC,GAAG,CAAC,SAAS,GAAG,SAAS,CAAC,CAAC;IAC7C,OAAO,EAAE,MAAM,CAAC,OAAO,CAAC,CAAC;IACzB,OAAO;;;;;;;;mBAAsD;gBAM5D,SAAS,EAAE,MAAM,CAAC,GAAG,CAAC,SAAS,GAAG,SAAS,CAAC,EAC5C,OAAO,EAAE,MAAM,CAAC,OAAO,CAAC,IAAI,GAAG,SAAS,CAAC,EACzC,KAAK,CAAC,EAAE,YAAY;IAiCrB,KAAK;CAGL"}
@@ -0,0 +1,22 @@
1
+ declare const OBSERVED: readonly ["show", "details"];
2
+ type Observed = (typeof OBSERVED)[number];
3
+ export type Show = "always" | "warning" | "error" | "never";
4
+ export default class MoqWatchSupport extends HTMLElement {
5
+ #private;
6
+ static observedAttributes: readonly ["show", "details"];
7
+ constructor();
8
+ connectedCallback(): void;
9
+ disconnectedCallback(): void;
10
+ attributeChangedCallback(name: Observed, _oldValue: string | null, newValue: string | null): void;
11
+ get show(): Show;
12
+ set show(show: Show);
13
+ get details(): boolean;
14
+ set details(details: boolean);
15
+ }
16
+ declare global {
17
+ interface HTMLElementTagNameMap {
18
+ "moq-watch-support": MoqWatchSupport;
19
+ }
20
+ }
21
+ export {};
22
+ //# sourceMappingURL=element.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"element.d.ts","sourceRoot":"","sources":["../../src/support/element.ts"],"names":[],"mappings":"AAOA,QAAA,MAAM,QAAQ,8BAA+B,CAAC;AAC9C,KAAK,QAAQ,GAAG,CAAC,OAAO,QAAQ,CAAC,CAAC,MAAM,CAAC,CAAC;AAO1C,MAAM,MAAM,IAAI,GAAG,QAAQ,GAAG,SAAS,GAAG,OAAO,GAAG,OAAO,CAAC;AAE5D,MAAM,CAAC,OAAO,OAAO,eAAgB,SAAQ,WAAW;;IAQvD,MAAM,CAAC,kBAAkB,+BAAY;;IAUrC,iBAAiB;IAKjB,oBAAoB;IAKpB,wBAAwB,CAAC,IAAI,EAAE,QAAQ,EAAE,SAAS,EAAE,MAAM,GAAG,IAAI,EAAE,QAAQ,EAAE,MAAM,GAAG,IAAI;IAgB1F,IAAI,IAAI,IAAI,IAAI,CAEf;IAED,IAAI,IAAI,CAAC,IAAI,EAAE,IAAI,EAElB;IAED,IAAI,OAAO,IAAI,OAAO,CAErB;IAED,IAAI,OAAO,CAAC,OAAO,EAAE,OAAO,EAE3B;CAgND;AAID,OAAO,CAAC,MAAM,CAAC;IACd,UAAU,qBAAqB;QAC9B,mBAAmB,EAAE,eAAe,CAAC;KACrC;CACD"}
@@ -0,0 +1,184 @@
1
+ import { Signal as u, Effect as b } from "@moq/signals";
2
+ import { isSupported as m } from "./index.js";
3
+ function d(h, e, ...t) {
4
+ const n = document.createElement(h);
5
+ if (!e) return n;
6
+ const { style: i, classList: a, dataset: o, attributes: l, ...s } = e;
7
+ return i && Object.assign(n.style, i), a && n.classList.add(...a), o && Object.entries(o).forEach(([r, c]) => {
8
+ n.dataset[r] = c;
9
+ }), l && Object.entries(l).forEach(([r, c]) => {
10
+ n.setAttribute(r, c);
11
+ }), t && t.forEach((r) => {
12
+ typeof r == "string" ? n.appendChild(document.createTextNode(r)) : n.appendChild(r);
13
+ }), Object.assign(n, s), n;
14
+ }
15
+ const f = navigator.userAgent.toLowerCase().includes("firefox"), C = ["show", "details"];
16
+ class y extends HTMLElement {
17
+ #i = new u("warning");
18
+ #e = new u(!1);
19
+ #n = new u(void 0);
20
+ #r = new u(!1);
21
+ #t;
22
+ static observedAttributes = C;
23
+ constructor() {
24
+ super(), m().then((e) => this.#n.set(e)).catch((e) => console.error("Failed to detect watch support:", e));
25
+ }
26
+ connectedCallback() {
27
+ this.#t = new b(), this.#t.effect(this.#s.bind(this));
28
+ }
29
+ disconnectedCallback() {
30
+ this.#t?.close(), this.#t = void 0;
31
+ }
32
+ attributeChangedCallback(e, t, n) {
33
+ if (e === "show") {
34
+ const i = n ?? "warning";
35
+ if (i === "always" || i === "warning" || i === "error" || i === "never")
36
+ this.show = i;
37
+ else
38
+ throw new Error(`Invalid show: ${i}`);
39
+ } else if (e === "details")
40
+ this.details = n !== null;
41
+ else {
42
+ const i = e;
43
+ throw new Error(`Invalid attribute: ${i}`);
44
+ }
45
+ }
46
+ get show() {
47
+ return this.#i.peek();
48
+ }
49
+ set show(e) {
50
+ this.#i.set(e);
51
+ }
52
+ get details() {
53
+ return this.#e.peek();
54
+ }
55
+ set details(e) {
56
+ this.#e.set(e);
57
+ }
58
+ #o(e) {
59
+ return e.webtransport === "none" || !e.audio.decoding || !e.video.decoding || !e.audio.render || !e.video.render || !Object.values(e.audio.decoding).some((t) => t === !0 || t === "full" || t === "partial") || !Object.values(e.video.decoding).some((t) => t.software || t.hardware) ? "none" : !Object.values(e.audio.decoding).every((t) => t === !0 || t === "full") || !Object.values(e.video.decoding).every((t) => t.software || t.hardware) ? "partial" : "full";
60
+ }
61
+ #s(e) {
62
+ const t = e.get(this.#n);
63
+ if (!t || e.get(this.#r)) return;
64
+ const i = e.get(this.#i);
65
+ if (i === "never") return;
66
+ const a = this.#o(t);
67
+ if (i === "warning" && a === "full" || i === "error" && a !== "none") return;
68
+ const o = d("div", {
69
+ style: {
70
+ margin: "0 auto",
71
+ maxWidth: "28rem",
72
+ padding: "1rem"
73
+ }
74
+ });
75
+ this.appendChild(o), e.cleanup(() => this.removeChild(o)), this.#a(o, a, e), e.get(this.#e) && this.#d(o, t, e);
76
+ }
77
+ #a(e, t, n) {
78
+ const i = d("div", {
79
+ style: {
80
+ display: "flex",
81
+ flexDirection: "row",
82
+ gap: "1rem",
83
+ flexWrap: "wrap",
84
+ justifyContent: "space-between",
85
+ alignItems: "center"
86
+ }
87
+ }), a = d("div", {
88
+ style: { fontWeight: "bold" }
89
+ });
90
+ t === "full" ? a.textContent = "🟢 Full Browser Support" : t === "partial" ? a.textContent = "🟡 Partial Browser Support" : t === "none" && (a.textContent = "🔴 No Browser Support");
91
+ const o = d("button", {
92
+ type: "button",
93
+ style: { fontSize: "14px" }
94
+ });
95
+ n.event(o, "click", () => {
96
+ this.#e.update((s) => !s);
97
+ }), n.effect((s) => {
98
+ o.textContent = s.get(this.#e) ? "Details ➖" : "Details ➕";
99
+ });
100
+ const l = d(
101
+ "button",
102
+ {
103
+ type: "button",
104
+ style: { fontSize: "14px" }
105
+ },
106
+ "Close ❌"
107
+ );
108
+ n.event(l, "click", () => {
109
+ this.#r.set(!0);
110
+ }), i.appendChild(a), i.appendChild(o), i.appendChild(l), e.appendChild(i), n.cleanup(() => e.removeChild(i));
111
+ }
112
+ #d(e, t, n) {
113
+ const i = d("div", {
114
+ style: {
115
+ display: "grid",
116
+ gridTemplateColumns: "1fr 1fr 1fr",
117
+ columnGap: "0.5rem",
118
+ rowGap: "0.2rem",
119
+ backgroundColor: "rgba(0, 0, 0, 0.6)",
120
+ borderRadius: "0.5rem",
121
+ padding: "1rem",
122
+ fontSize: "0.875rem"
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) => {
125
+ const p = d(
126
+ "div",
127
+ {
128
+ style: {
129
+ gridColumnStart: "1",
130
+ fontWeight: "bold",
131
+ textAlign: "right"
132
+ }
133
+ },
134
+ r
135
+ ), w = d(
136
+ "div",
137
+ {
138
+ style: {
139
+ gridColumnStart: "2",
140
+ textAlign: "center"
141
+ }
142
+ },
143
+ c
144
+ ), v = d(
145
+ "div",
146
+ {
147
+ style: { gridColumnStart: "3" }
148
+ },
149
+ g
150
+ );
151
+ i.appendChild(p), i.appendChild(w), i.appendChild(v);
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) {
154
+ const r = d(
155
+ "div",
156
+ {
157
+ style: {
158
+ gridColumnStart: "1",
159
+ gridColumnEnd: "4",
160
+ textAlign: "center",
161
+ fontSize: "0.875rem",
162
+ fontStyle: "italic"
163
+ }
164
+ },
165
+ "Hardware acceleration is ",
166
+ d(
167
+ "a",
168
+ {
169
+ href: "https://github.com/w3c/webcodecs/issues/896"
170
+ },
171
+ "undetectable"
172
+ ),
173
+ " on Firefox."
174
+ );
175
+ i.appendChild(r);
176
+ }
177
+ e.appendChild(i), n.cleanup(() => e.removeChild(i));
178
+ }
179
+ }
180
+ customElements.define("moq-watch-support", y);
181
+ export {
182
+ y as default
183
+ };
184
+ //# sourceMappingURL=element.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"element.js","sources":["../../../signals/src/dom.ts","../../src/support/element.ts"],"sourcesContent":["import type { Effect } from \".\";\n\nexport type CreateOptions<T extends HTMLElement> = {\n\tstyle?: Partial<CSSStyleDeclaration>;\n\tclassName?: string;\n\tclassList?: string[];\n\tid?: string;\n\tdataset?: Record<string, string>;\n\tattributes?: Record<string, string>;\n} & Partial<Omit<T, \"style\" | \"dataset\">>;\n\nexport function create<K extends keyof HTMLElementTagNameMap>(\n\ttagName: K,\n\toptions?: CreateOptions<HTMLElementTagNameMap[K] & HTMLElement>,\n\t...children: (HTMLElement | string)[]\n): HTMLElementTagNameMap[K] {\n\tconst element = document.createElement(tagName);\n\n\tif (!options) return element;\n\n\tconst { style, classList, dataset, attributes, ...props } = options;\n\n\t// Apply styles\n\tif (style) {\n\t\tObject.assign(element.style, style);\n\t}\n\n\t// Apply class list\n\tif (classList) {\n\t\telement.classList.add(...classList);\n\t}\n\n\t// Apply dataset\n\tif (dataset) {\n\t\tObject.entries(dataset).forEach(([key, value]) => {\n\t\t\telement.dataset[key] = value;\n\t\t});\n\t}\n\n\t// Apply attributes\n\tif (attributes) {\n\t\tObject.entries(attributes).forEach(([key, value]) => {\n\t\t\telement.setAttribute(key, value);\n\t\t});\n\t}\n\n\t// Append children\n\tif (children) {\n\t\tchildren.forEach((child) => {\n\t\t\tif (typeof child === \"string\") {\n\t\t\t\telement.appendChild(document.createTextNode(child));\n\t\t\t} else {\n\t\t\t\telement.appendChild(child);\n\t\t\t}\n\t\t});\n\t}\n\n\t// Apply other properties\n\tObject.assign(element, props);\n\n\treturn element;\n}\n\n// Matches solid.js's JSX.Element type.\nexport type Element = Node | ArrayElement | (string & {}) | number | boolean | null | undefined;\ninterface ArrayElement extends Array<Element> {}\n\nexport function render(effect: Effect, parent: Node, element: Element | ((effect: Effect) => Element)) {\n\tconst e = typeof element === \"function\" ? element(effect) : element;\n\tif (e === undefined || e === null) return;\n\n\tlet node: Node;\n\tif (e instanceof Node) {\n\t\tnode = e;\n\t} else if (Array.isArray(e)) {\n\t\tnode = document.createDocumentFragment();\n\t\tfor (const child of e) {\n\t\t\trender(effect, node, child);\n\t\t}\n\t} else if (typeof e === \"number\" || typeof e === \"boolean\" || typeof e === \"string\") {\n\t\tnode = document.createTextNode(e.toString());\n\t} else {\n\t\tconst exhaustive: never = e;\n\t\tthrow new Error(`Invalid element type: ${exhaustive}`);\n\t}\n\n\tparent.appendChild(node);\n\teffect.cleanup(() => parent.removeChild(node));\n}\n\nexport function setClass(effect: Effect, element: HTMLElement, ...classNames: string[]) {\n\tfor (const className of classNames) {\n\t\telement.classList.add(className);\n\t}\n\n\teffect.cleanup(() => {\n\t\tfor (const className of classNames) {\n\t\t\telement.classList.remove(className);\n\t\t}\n\t});\n}\n","import { Effect, Signal } from \"@moq/signals\";\nimport * as DOM from \"@moq/signals/dom\";\nimport { type Codec, type Full, isSupported, type Partial } from \"./\";\n\n// https://bugzilla.mozilla.org/show_bug.cgi?id=1967793\nconst isFirefox = navigator.userAgent.toLowerCase().includes(\"firefox\");\n\nconst OBSERVED = [\"show\", \"details\"] as const;\ntype Observed = (typeof OBSERVED)[number];\n\n// Whether to display the support banner.\n// - \"always\": Always display the banner.\n// - \"warning\": Display the banner if a required feature needs a polyfill/fallback.\n// - \"error\": Display the banner if a required feature is unsupported.\n// - \"never\": Never display the banner.\nexport type Show = \"always\" | \"warning\" | \"error\" | \"never\";\n\nexport default class MoqWatchSupport extends HTMLElement {\n\t#show = new Signal<Show>(\"warning\");\n\t#details = new Signal<boolean>(false);\n\t#support = new Signal<Full | undefined>(undefined);\n\t#close = new Signal<boolean>(false);\n\n\t#signals?: Effect;\n\n\tstatic observedAttributes = OBSERVED;\n\n\tconstructor() {\n\t\tsuper();\n\n\t\tisSupported()\n\t\t\t.then((s) => this.#support.set(s))\n\t\t\t.catch((err) => console.error(\"Failed to detect watch support:\", err));\n\t}\n\n\tconnectedCallback() {\n\t\tthis.#signals = new Effect();\n\t\tthis.#signals.effect(this.#render.bind(this));\n\t}\n\n\tdisconnectedCallback() {\n\t\tthis.#signals?.close();\n\t\tthis.#signals = undefined;\n\t}\n\n\tattributeChangedCallback(name: Observed, _oldValue: string | null, newValue: string | null) {\n\t\tif (name === \"show\") {\n\t\t\tconst show = newValue ?? \"warning\";\n\t\t\tif (show === \"always\" || show === \"warning\" || show === \"error\" || show === \"never\") {\n\t\t\t\tthis.show = show;\n\t\t\t} else {\n\t\t\t\tthrow new Error(`Invalid show: ${show}`);\n\t\t\t}\n\t\t} else if (name === \"details\") {\n\t\t\tthis.details = newValue !== null;\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 show(): Show {\n\t\treturn this.#show.peek();\n\t}\n\n\tset show(show: Show) {\n\t\tthis.#show.set(show);\n\t}\n\n\tget details(): boolean {\n\t\treturn this.#details.peek();\n\t}\n\n\tset details(details: boolean) {\n\t\tthis.#details.set(details);\n\t}\n\n\t#getSummary(support: Full): Partial {\n\t\tif (support.webtransport === \"none\") return \"none\";\n\n\t\tif (!support.audio.decoding || !support.video.decoding) return \"none\";\n\t\tif (!support.audio.render || !support.video.render) return \"none\";\n\n\t\tif (!Object.values(support.audio.decoding).some((v) => v === true || v === \"full\" || v === \"partial\"))\n\t\t\treturn \"none\";\n\t\tif (!Object.values(support.video.decoding).some((v) => v.software || v.hardware)) return \"none\";\n\n\t\tif (!Object.values(support.audio.decoding).every((v) => v === true || v === \"full\")) return \"partial\";\n\t\tif (!Object.values(support.video.decoding).every((v) => v.software || v.hardware)) return \"partial\";\n\n\t\treturn \"full\";\n\t}\n\n\t#render(effect: Effect) {\n\t\tconst support = effect.get(this.#support);\n\t\tif (!support) return;\n\n\t\tconst close = effect.get(this.#close);\n\t\tif (close) return;\n\n\t\tconst show = effect.get(this.#show);\n\t\tif (show === \"never\") return;\n\n\t\tconst summary = this.#getSummary(support);\n\n\t\t// Don't render the banner if we have full support and they only asked for warnings.\n\t\tif (show === \"warning\" && summary === \"full\") return;\n\n\t\t// Don't render the banner if we have at least partial support and they only asked for errors.\n\t\tif (show === \"error\" && summary !== \"none\") return;\n\n\t\tconst container = DOM.create(\"div\", {\n\t\t\tstyle: {\n\t\t\t\tmargin: \"0 auto\",\n\t\t\t\tmaxWidth: \"28rem\",\n\t\t\t\tpadding: \"1rem\",\n\t\t\t},\n\t\t});\n\n\t\tthis.appendChild(container);\n\t\teffect.cleanup(() => this.removeChild(container));\n\n\t\tthis.#renderHeader(container, summary, effect);\n\n\t\tif (effect.get(this.#details)) {\n\t\t\tthis.#renderDetails(container, support, effect);\n\t\t}\n\t}\n\n\t#renderHeader(parent: HTMLDivElement, summary: Partial, effect: Effect) {\n\t\tconst headerDiv = DOM.create(\"div\", {\n\t\t\tstyle: {\n\t\t\t\tdisplay: \"flex\",\n\t\t\t\tflexDirection: \"row\",\n\t\t\t\tgap: \"1rem\",\n\t\t\t\tflexWrap: \"wrap\",\n\t\t\t\tjustifyContent: \"space-between\",\n\t\t\t\talignItems: \"center\",\n\t\t\t},\n\t\t});\n\n\t\tconst statusDiv = DOM.create(\"div\", {\n\t\t\tstyle: { fontWeight: \"bold\" },\n\t\t});\n\n\t\tif (summary === \"full\") {\n\t\t\tstatusDiv.textContent = \"🟢 Full Browser Support\";\n\t\t} else if (summary === \"partial\") {\n\t\t\tstatusDiv.textContent = \"🟡 Partial Browser Support\";\n\t\t} else if (summary === \"none\") {\n\t\t\tstatusDiv.textContent = \"🔴 No Browser Support\";\n\t\t}\n\n\t\tconst detailsButton = DOM.create(\"button\", {\n\t\t\ttype: \"button\",\n\t\t\tstyle: { fontSize: \"14px\" },\n\t\t});\n\n\t\teffect.event(detailsButton, \"click\", () => {\n\t\t\tthis.#details.update((prev) => !prev);\n\t\t});\n\n\t\teffect.effect((effect) => {\n\t\t\tdetailsButton.textContent = effect.get(this.#details) ? \"Details ➖\" : \"Details ➕\";\n\t\t});\n\n\t\tconst closeButton = DOM.create(\n\t\t\t\"button\",\n\t\t\t{\n\t\t\t\ttype: \"button\",\n\t\t\t\tstyle: { fontSize: \"14px\" },\n\t\t\t},\n\t\t\t\"Close ❌\",\n\t\t);\n\n\t\teffect.event(closeButton, \"click\", () => {\n\t\t\tthis.#close.set(true);\n\t\t});\n\n\t\theaderDiv.appendChild(statusDiv);\n\t\theaderDiv.appendChild(detailsButton);\n\t\theaderDiv.appendChild(closeButton);\n\n\t\tparent.appendChild(headerDiv);\n\t\teffect.cleanup(() => parent.removeChild(headerDiv));\n\t}\n\n\t#renderDetails(parent: HTMLDivElement, support: Full, effect: Effect) {\n\t\tconst container = DOM.create(\"div\", {\n\t\t\tstyle: {\n\t\t\t\tdisplay: \"grid\",\n\t\t\t\tgridTemplateColumns: \"1fr 1fr 1fr\",\n\t\t\t\tcolumnGap: \"0.5rem\",\n\t\t\t\trowGap: \"0.2rem\",\n\t\t\t\tbackgroundColor: \"rgba(0, 0, 0, 0.6)\",\n\t\t\t\tborderRadius: \"0.5rem\",\n\t\t\t\tpadding: \"1rem\",\n\t\t\t\tfontSize: \"0.875rem\",\n\t\t\t},\n\t\t});\n\n\t\tconst binary = (value: boolean | undefined) => (value ? \"🟢 Yes\" : \"🔴 No\");\n\t\tconst hardware = (codec: Codec | undefined) =>\n\t\t\tcodec?.hardware ? \"🟢 Hardware\" : codec?.software ? `🟡 Software${isFirefox ? \"*\" : \"\"}` : \"🔴 No\";\n\t\tconst partial = (value: Partial | undefined) =>\n\t\t\tvalue === \"full\" ? \"🟢 Full\" : value === \"partial\" ? \"🟡 Polyfill\" : \"🔴 None\";\n\n\t\tconst addRow = (label: string, col2: string, col3: string) => {\n\t\t\tconst labelDiv = DOM.create(\n\t\t\t\t\"div\",\n\t\t\t\t{\n\t\t\t\t\tstyle: {\n\t\t\t\t\t\tgridColumnStart: \"1\",\n\t\t\t\t\t\tfontWeight: \"bold\",\n\t\t\t\t\t\ttextAlign: \"right\",\n\t\t\t\t\t},\n\t\t\t\t},\n\t\t\t\tlabel,\n\t\t\t);\n\n\t\t\tconst col2Div = DOM.create(\n\t\t\t\t\"div\",\n\t\t\t\t{\n\t\t\t\t\tstyle: {\n\t\t\t\t\t\tgridColumnStart: \"2\",\n\t\t\t\t\t\ttextAlign: \"center\",\n\t\t\t\t\t},\n\t\t\t\t},\n\t\t\t\tcol2,\n\t\t\t);\n\n\t\t\tconst col3Div = DOM.create(\n\t\t\t\t\"div\",\n\t\t\t\t{\n\t\t\t\t\tstyle: { gridColumnStart: \"3\" },\n\t\t\t\t},\n\t\t\t\tcol3,\n\t\t\t);\n\n\t\t\tcontainer.appendChild(labelDiv);\n\t\t\tcontainer.appendChild(col2Div);\n\t\t\tcontainer.appendChild(col3Div);\n\t\t};\n\n\t\taddRow(\"WebTransport\", \"\", partial(support.webtransport));\n\t\taddRow(\"Rendering\", \"Audio\", binary(support.audio.render));\n\t\taddRow(\"\", \"Video\", binary(support.video.render));\n\t\taddRow(\"Decoding\", \"Opus\", partial(support.audio.decoding.opus));\n\t\taddRow(\"\", \"AAC\", binary(support.audio.decoding.aac));\n\t\taddRow(\"\", \"AV1\", hardware(support.video.decoding?.av1));\n\t\taddRow(\"\", \"H.265\", hardware(support.video.decoding?.h265));\n\t\taddRow(\"\", \"H.264\", hardware(support.video.decoding?.h264));\n\t\taddRow(\"\", \"VP9\", hardware(support.video.decoding?.vp9));\n\t\taddRow(\"\", \"VP8\", hardware(support.video.decoding?.vp8));\n\n\t\tif (isFirefox) {\n\t\t\tconst noteDiv = DOM.create(\n\t\t\t\t\"div\",\n\t\t\t\t{\n\t\t\t\t\tstyle: {\n\t\t\t\t\t\tgridColumnStart: \"1\",\n\t\t\t\t\t\tgridColumnEnd: \"4\",\n\t\t\t\t\t\ttextAlign: \"center\",\n\t\t\t\t\t\tfontSize: \"0.875rem\",\n\t\t\t\t\t\tfontStyle: \"italic\",\n\t\t\t\t\t},\n\t\t\t\t},\n\t\t\t\t\"Hardware acceleration is \",\n\t\t\t\tDOM.create(\n\t\t\t\t\t\"a\",\n\t\t\t\t\t{\n\t\t\t\t\t\thref: \"https://github.com/w3c/webcodecs/issues/896\",\n\t\t\t\t\t},\n\t\t\t\t\t\"undetectable\",\n\t\t\t\t),\n\t\t\t\t\" on Firefox.\",\n\t\t\t);\n\t\t\tcontainer.appendChild(noteDiv);\n\t\t}\n\n\t\tparent.appendChild(container);\n\t\teffect.cleanup(() => parent.removeChild(container));\n\t}\n}\n\ncustomElements.define(\"moq-watch-support\", MoqWatchSupport);\n\ndeclare global {\n\tinterface HTMLElementTagNameMap {\n\t\t\"moq-watch-support\": MoqWatchSupport;\n\t}\n}\n"],"names":["create","tagName","options","children","element","style","classList","dataset","attributes","props","key","value","child","isFirefox","OBSERVED","MoqWatchSupport","#show","Signal","#details","#support","#close","#signals","isSupported","s","err","Effect","#render","name","_oldValue","newValue","show","exhaustive","details","#getSummary","support","v","effect","summary","container","DOM.create","#renderHeader","#renderDetails","parent","headerDiv","statusDiv","detailsButton","prev","closeButton","binary","hardware","codec","partial","addRow","label","col2","col3","labelDiv","col2Div","col3Div","noteDiv"],"mappings":";;AAWO,SAASA,EACfC,GACAC,MACGC,GACwB;AAC3B,QAAMC,IAAU,SAAS,cAAcH,CAAO;AAE9C,MAAI,CAACC,EAAS,QAAOE;AAErB,QAAM,EAAE,OAAAC,GAAO,WAAAC,GAAW,SAAAC,GAAS,YAAAC,GAAY,GAAGC,MAAUP;AAG5D,SAAIG,KACH,OAAO,OAAOD,EAAQ,OAAOC,CAAK,GAI/BC,KACHF,EAAQ,UAAU,IAAI,GAAGE,CAAS,GAI/BC,KACH,OAAO,QAAQA,CAAO,EAAE,QAAQ,CAAC,CAACG,GAAKC,CAAK,MAAM;AACjD,IAAAP,EAAQ,QAAQM,CAAG,IAAIC;AAAA,EACxB,CAAC,GAIEH,KACH,OAAO,QAAQA,CAAU,EAAE,QAAQ,CAAC,CAACE,GAAKC,CAAK,MAAM;AACpD,IAAAP,EAAQ,aAAaM,GAAKC,CAAK;AAAA,EAChC,CAAC,GAIER,KACHA,EAAS,QAAQ,CAACS,MAAU;AAC3B,IAAI,OAAOA,KAAU,WACpBR,EAAQ,YAAY,SAAS,eAAeQ,CAAK,CAAC,IAElDR,EAAQ,YAAYQ,CAAK;AAAA,EAE3B,CAAC,GAIF,OAAO,OAAOR,GAASK,CAAK,GAErBL;AACR;ACxDA,MAAMS,IAAY,UAAU,UAAU,YAAA,EAAc,SAAS,SAAS,GAEhEC,IAAW,CAAC,QAAQ,SAAS;AAUnC,MAAqBC,UAAwB,YAAY;AAAA,EACxDC,KAAQ,IAAIC,EAAa,SAAS;AAAA,EAClCC,KAAW,IAAID,EAAgB,EAAK;AAAA,EACpCE,KAAW,IAAIF,EAAyB,MAAS;AAAA,EACjDG,KAAS,IAAIH,EAAgB,EAAK;AAAA,EAElCI;AAAA,EAEA,OAAO,qBAAqBP;AAAA,EAE5B,cAAc;AACb,UAAA,GAEAQ,EAAA,EACE,KAAK,CAACC,MAAM,KAAKJ,GAAS,IAAII,CAAC,CAAC,EAChC,MAAM,CAACC,MAAQ,QAAQ,MAAM,mCAAmCA,CAAG,CAAC;AAAA,EACvE;AAAA,EAEA,oBAAoB;AACnB,SAAKH,KAAW,IAAII,EAAA,GACpB,KAAKJ,GAAS,OAAO,KAAKK,GAAQ,KAAK,IAAI,CAAC;AAAA,EAC7C;AAAA,EAEA,uBAAuB;AACtB,SAAKL,IAAU,MAAA,GACf,KAAKA,KAAW;AAAA,EACjB;AAAA,EAEA,yBAAyBM,GAAgBC,GAA0BC,GAAyB;AAC3F,QAAIF,MAAS,QAAQ;AACpB,YAAMG,IAAOD,KAAY;AACzB,UAAIC,MAAS,YAAYA,MAAS,aAAaA,MAAS,WAAWA,MAAS;AAC3E,aAAK,OAAOA;AAAA;AAEZ,cAAM,IAAI,MAAM,iBAAiBA,CAAI,EAAE;AAAA,IAEzC,WAAWH,MAAS;AACnB,WAAK,UAAUE,MAAa;AAAA,SACtB;AACN,YAAME,IAAoBJ;AAC1B,YAAM,IAAI,MAAM,sBAAsBI,CAAU,EAAE;AAAA,IACnD;AAAA,EACD;AAAA,EAEA,IAAI,OAAa;AAChB,WAAO,KAAKf,GAAM,KAAA;AAAA,EACnB;AAAA,EAEA,IAAI,KAAKc,GAAY;AACpB,SAAKd,GAAM,IAAIc,CAAI;AAAA,EACpB;AAAA,EAEA,IAAI,UAAmB;AACtB,WAAO,KAAKZ,GAAS,KAAA;AAAA,EACtB;AAAA,EAEA,IAAI,QAAQc,GAAkB;AAC7B,SAAKd,GAAS,IAAIc,CAAO;AAAA,EAC1B;AAAA,EAEAC,GAAYC,GAAwB;AAQnC,WAPIA,EAAQ,iBAAiB,UAEzB,CAACA,EAAQ,MAAM,YAAY,CAACA,EAAQ,MAAM,YAC1C,CAACA,EAAQ,MAAM,UAAU,CAACA,EAAQ,MAAM,UAExC,CAAC,OAAO,OAAOA,EAAQ,MAAM,QAAQ,EAAE,KAAK,CAACC,MAAMA,MAAM,MAAQA,MAAM,UAAUA,MAAM,SAAS,KAEhG,CAAC,OAAO,OAAOD,EAAQ,MAAM,QAAQ,EAAE,KAAK,CAACC,MAAMA,EAAE,YAAYA,EAAE,QAAQ,IAAU,SAErF,CAAC,OAAO,OAAOD,EAAQ,MAAM,QAAQ,EAAE,MAAM,CAACC,MAAMA,MAAM,MAAQA,MAAM,MAAM,KAC9E,CAAC,OAAO,OAAOD,EAAQ,MAAM,QAAQ,EAAE,MAAM,CAACC,MAAMA,EAAE,YAAYA,EAAE,QAAQ,IAAU,YAEnF;AAAA,EACR;AAAA,EAEAT,GAAQU,GAAgB;AACvB,UAAMF,IAAUE,EAAO,IAAI,KAAKjB,EAAQ;AAIxC,QAHI,CAACe,KAESE,EAAO,IAAI,KAAKhB,EAAM,EACzB;AAEX,UAAMU,IAAOM,EAAO,IAAI,KAAKpB,EAAK;AAClC,QAAIc,MAAS,QAAS;AAEtB,UAAMO,IAAU,KAAKJ,GAAYC,CAAO;AAMxC,QAHIJ,MAAS,aAAaO,MAAY,UAGlCP,MAAS,WAAWO,MAAY,OAAQ;AAE5C,UAAMC,IAAYC,EAAW,OAAO;AAAA,MACnC,OAAO;AAAA,QACN,QAAQ;AAAA,QACR,UAAU;AAAA,QACV,SAAS;AAAA,MAAA;AAAA,IACV,CACA;AAED,SAAK,YAAYD,CAAS,GAC1BF,EAAO,QAAQ,MAAM,KAAK,YAAYE,CAAS,CAAC,GAEhD,KAAKE,GAAcF,GAAWD,GAASD,CAAM,GAEzCA,EAAO,IAAI,KAAKlB,EAAQ,KAC3B,KAAKuB,GAAeH,GAAWJ,GAASE,CAAM;AAAA,EAEhD;AAAA,EAEAI,GAAcE,GAAwBL,GAAkBD,GAAgB;AACvE,UAAMO,IAAYJ,EAAW,OAAO;AAAA,MACnC,OAAO;AAAA,QACN,SAAS;AAAA,QACT,eAAe;AAAA,QACf,KAAK;AAAA,QACL,UAAU;AAAA,QACV,gBAAgB;AAAA,QAChB,YAAY;AAAA,MAAA;AAAA,IACb,CACA,GAEKK,IAAYL,EAAW,OAAO;AAAA,MACnC,OAAO,EAAE,YAAY,OAAA;AAAA,IAAO,CAC5B;AAED,IAAIF,MAAY,SACfO,EAAU,cAAc,4BACdP,MAAY,YACtBO,EAAU,cAAc,+BACdP,MAAY,WACtBO,EAAU,cAAc;AAGzB,UAAMC,IAAgBN,EAAW,UAAU;AAAA,MAC1C,MAAM;AAAA,MACN,OAAO,EAAE,UAAU,OAAA;AAAA,IAAO,CAC1B;AAED,IAAAH,EAAO,MAAMS,GAAe,SAAS,MAAM;AAC1C,WAAK3B,GAAS,OAAO,CAAC4B,MAAS,CAACA,CAAI;AAAA,IACrC,CAAC,GAEDV,EAAO,OAAO,CAACA,MAAW;AACzB,MAAAS,EAAc,cAAcT,EAAO,IAAI,KAAKlB,EAAQ,IAAI,cAAc;AAAA,IACvE,CAAC;AAED,UAAM6B,IAAcR;AAAAA,MACnB;AAAA,MACA;AAAA,QACC,MAAM;AAAA,QACN,OAAO,EAAE,UAAU,OAAA;AAAA,MAAO;AAAA,MAE3B;AAAA,IAAA;AAGD,IAAAH,EAAO,MAAMW,GAAa,SAAS,MAAM;AACxC,WAAK3B,GAAO,IAAI,EAAI;AAAA,IACrB,CAAC,GAEDuB,EAAU,YAAYC,CAAS,GAC/BD,EAAU,YAAYE,CAAa,GACnCF,EAAU,YAAYI,CAAW,GAEjCL,EAAO,YAAYC,CAAS,GAC5BP,EAAO,QAAQ,MAAMM,EAAO,YAAYC,CAAS,CAAC;AAAA,EACnD;AAAA,EAEAF,GAAeC,GAAwBR,GAAeE,GAAgB;AACrE,UAAME,IAAYC,EAAW,OAAO;AAAA,MACnC,OAAO;AAAA,QACN,SAAS;AAAA,QACT,qBAAqB;AAAA,QACrB,WAAW;AAAA,QACX,QAAQ;AAAA,QACR,iBAAiB;AAAA,QACjB,cAAc;AAAA,QACd,SAAS;AAAA,QACT,UAAU;AAAA,MAAA;AAAA,IACX,CACA,GAEKS,IAAS,CAACrC,MAAgCA,IAAQ,WAAW,SAC7DsC,IAAW,CAACC,MACjBA,GAAO,WAAW,gBAAgBA,GAAO,WAAW,cAAcrC,IAAY,MAAM,EAAE,KAAK,SACtFsC,IAAU,CAACxC,MAChBA,MAAU,SAAS,YAAYA,MAAU,YAAY,gBAAgB,WAEhEyC,IAAS,CAACC,GAAeC,GAAcC,MAAiB;AAC7D,YAAMC,IAAWjB;AAAAA,QAChB;AAAA,QACA;AAAA,UACC,OAAO;AAAA,YACN,iBAAiB;AAAA,YACjB,YAAY;AAAA,YACZ,WAAW;AAAA,UAAA;AAAA,QACZ;AAAA,QAEDc;AAAA,MAAA,GAGKI,IAAUlB;AAAAA,QACf;AAAA,QACA;AAAA,UACC,OAAO;AAAA,YACN,iBAAiB;AAAA,YACjB,WAAW;AAAA,UAAA;AAAA,QACZ;AAAA,QAEDe;AAAA,MAAA,GAGKI,IAAUnB;AAAAA,QACf;AAAA,QACA;AAAA,UACC,OAAO,EAAE,iBAAiB,IAAA;AAAA,QAAI;AAAA,QAE/BgB;AAAA,MAAA;AAGD,MAAAjB,EAAU,YAAYkB,CAAQ,GAC9BlB,EAAU,YAAYmB,CAAO,GAC7BnB,EAAU,YAAYoB,CAAO;AAAA,IAC9B;AAaA,QAXAN,EAAO,gBAAgB,IAAID,EAAQjB,EAAQ,YAAY,CAAC,GACxDkB,EAAO,aAAa,SAASJ,EAAOd,EAAQ,MAAM,MAAM,CAAC,GACzDkB,EAAO,IAAI,SAASJ,EAAOd,EAAQ,MAAM,MAAM,CAAC,GAChDkB,EAAO,YAAY,QAAQD,EAAQjB,EAAQ,MAAM,SAAS,IAAI,CAAC,GAC/DkB,EAAO,IAAI,OAAOJ,EAAOd,EAAQ,MAAM,SAAS,GAAG,CAAC,GACpDkB,EAAO,IAAI,OAAOH,EAASf,EAAQ,MAAM,UAAU,GAAG,CAAC,GACvDkB,EAAO,IAAI,SAASH,EAASf,EAAQ,MAAM,UAAU,IAAI,CAAC,GAC1DkB,EAAO,IAAI,SAASH,EAASf,EAAQ,MAAM,UAAU,IAAI,CAAC,GAC1DkB,EAAO,IAAI,OAAOH,EAASf,EAAQ,MAAM,UAAU,GAAG,CAAC,GACvDkB,EAAO,IAAI,OAAOH,EAASf,EAAQ,MAAM,UAAU,GAAG,CAAC,GAEnDrB,GAAW;AACd,YAAM8C,IAAUpB;AAAAA,QACf;AAAA,QACA;AAAA,UACC,OAAO;AAAA,YACN,iBAAiB;AAAA,YACjB,eAAe;AAAA,YACf,WAAW;AAAA,YACX,UAAU;AAAA,YACV,WAAW;AAAA,UAAA;AAAA,QACZ;AAAA,QAED;AAAA,QACAA;AAAAA,UACC;AAAA,UACA;AAAA,YACC,MAAM;AAAA,UAAA;AAAA,UAEP;AAAA,QAAA;AAAA,QAED;AAAA,MAAA;AAED,MAAAD,EAAU,YAAYqB,CAAO;AAAA,IAC9B;AAEA,IAAAjB,EAAO,YAAYJ,CAAS,GAC5BF,EAAO,QAAQ,MAAMM,EAAO,YAAYJ,CAAS,CAAC;AAAA,EACnD;AACD;AAEA,eAAe,OAAO,qBAAqBvB,CAAe;"}
@@ -0,0 +1,29 @@
1
+ export type Partial = "full" | "partial" | "none";
2
+ export type Codec = {
3
+ hardware?: boolean;
4
+ software: boolean;
5
+ };
6
+ export type Audio = {
7
+ aac: boolean;
8
+ opus: Partial;
9
+ };
10
+ export type Video = {
11
+ h264: Codec;
12
+ h265: Codec;
13
+ vp8: Codec;
14
+ vp9: Codec;
15
+ av1: Codec;
16
+ };
17
+ export type Full = {
18
+ webtransport: Partial;
19
+ audio: {
20
+ decoding: Audio;
21
+ render: boolean;
22
+ };
23
+ video: {
24
+ decoding: Video | undefined;
25
+ render: boolean;
26
+ };
27
+ };
28
+ export declare function isSupported(): Promise<Full>;
29
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/support/index.ts"],"names":[],"mappings":"AAGA,MAAM,MAAM,OAAO,GAAG,MAAM,GAAG,SAAS,GAAG,MAAM,CAAC;AAElD,MAAM,MAAM,KAAK,GAAG;IACnB,QAAQ,CAAC,EAAE,OAAO,CAAC;IACnB,QAAQ,EAAE,OAAO,CAAC;CAClB,CAAC;AAEF,MAAM,MAAM,KAAK,GAAG;IACnB,GAAG,EAAE,OAAO,CAAC;IACb,IAAI,EAAE,OAAO,CAAC;CACd,CAAC;AAEF,MAAM,MAAM,KAAK,GAAG;IACnB,IAAI,EAAE,KAAK,CAAC;IACZ,IAAI,EAAE,KAAK,CAAC;IACZ,GAAG,EAAE,KAAK,CAAC;IACX,GAAG,EAAE,KAAK,CAAC;IACX,GAAG,EAAE,KAAK,CAAC;CACX,CAAC;AAEF,MAAM,MAAM,IAAI,GAAG;IAClB,YAAY,EAAE,OAAO,CAAC;IACtB,KAAK,EAAE;QACN,QAAQ,EAAE,KAAK,CAAC;QAChB,MAAM,EAAE,OAAO,CAAC;KAChB,CAAC;IACF,KAAK,EAAE;QACN,QAAQ,EAAE,KAAK,GAAG,SAAS,CAAC;QAC5B,MAAM,EAAE,OAAO,CAAC;KAChB,CAAC;CACF,CAAC;AA8CF,wBAAsB,WAAW,IAAI,OAAO,CAAC,IAAI,CAAC,CAwBjD"}
@@ -0,0 +1,55 @@
1
+ const d = navigator.userAgent.toLowerCase().includes("firefox"), r = {
2
+ aac: "mp4a.40.2",
3
+ opus: "opus",
4
+ av1: "av01.0.08M.08",
5
+ h264: "avc1.640028",
6
+ h265: "hev1.1.6.L93.B0",
7
+ vp9: "vp09.00.10.08",
8
+ vp8: "vp8"
9
+ };
10
+ async function t(o) {
11
+ return globalThis.AudioDecoder ? (await AudioDecoder.isConfigSupported({
12
+ codec: r[o],
13
+ numberOfChannels: 2,
14
+ sampleRate: 48e3
15
+ })).supported === !0 : !1;
16
+ }
17
+ async function e(o) {
18
+ const a = await VideoDecoder.isConfigSupported({
19
+ codec: r[o],
20
+ hardwareAcceleration: "prefer-software"
21
+ }), n = await VideoDecoder.isConfigSupported({
22
+ codec: r[o],
23
+ hardwareAcceleration: "prefer-hardware"
24
+ });
25
+ return {
26
+ hardware: d || n.config?.hardwareAcceleration !== "prefer-hardware" ? void 0 : n.supported === !0,
27
+ software: a.supported === !0
28
+ };
29
+ }
30
+ async function p() {
31
+ return {
32
+ webtransport: typeof WebTransport < "u" ? "full" : "partial",
33
+ audio: {
34
+ decoding: {
35
+ aac: await t("aac"),
36
+ opus: await t("opus") ? "full" : "partial"
37
+ },
38
+ render: typeof AudioContext < "u" && typeof AudioBufferSourceNode < "u"
39
+ },
40
+ video: {
41
+ decoding: typeof VideoDecoder < "u" ? {
42
+ h264: await e("h264"),
43
+ h265: await e("h265"),
44
+ vp8: await e("vp8"),
45
+ vp9: await e("vp9"),
46
+ av1: await e("av1")
47
+ } : void 0,
48
+ render: typeof OffscreenCanvas < "u" && typeof CanvasRenderingContext2D < "u"
49
+ }
50
+ };
51
+ }
52
+ export {
53
+ p as isSupported
54
+ };
55
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sources":["../../src/support/index.ts"],"sourcesContent":["// https://bugzilla.mozilla.org/show_bug.cgi?id=1967793\nconst isFirefox = navigator.userAgent.toLowerCase().includes(\"firefox\");\n\nexport type Partial = \"full\" | \"partial\" | \"none\";\n\nexport type Codec = {\n\thardware?: boolean; // undefined when we can't detect hardware acceleration\n\tsoftware: boolean;\n};\n\nexport type Audio = {\n\taac: boolean;\n\topus: Partial;\n};\n\nexport type Video = {\n\th264: Codec;\n\th265: Codec;\n\tvp8: Codec;\n\tvp9: Codec;\n\tav1: Codec;\n};\n\nexport type Full = {\n\twebtransport: Partial;\n\taudio: {\n\t\tdecoding: Audio;\n\t\trender: boolean;\n\t};\n\tvideo: {\n\t\tdecoding: Video | undefined;\n\t\trender: boolean;\n\t};\n};\n\n// Pick a codec string for each codec.\n// This is not strictly correct, as browsers may not support every profile or level.\nconst CODECS = {\n\taac: \"mp4a.40.2\",\n\topus: \"opus\",\n\tav1: \"av01.0.08M.08\",\n\th264: \"avc1.640028\",\n\th265: \"hev1.1.6.L93.B0\",\n\tvp9: \"vp09.00.10.08\",\n\tvp8: \"vp8\",\n};\n\nasync function audioDecoderSupported(codec: keyof typeof CODECS): Promise<boolean> {\n\tif (!globalThis.AudioDecoder) return false;\n\n\tconst res = await AudioDecoder.isConfigSupported({\n\t\tcodec: CODECS[codec],\n\t\tnumberOfChannels: 2,\n\t\tsampleRate: 48000,\n\t});\n\n\treturn res.supported === true;\n}\n\nasync function videoDecoderSupported(codec: keyof typeof CODECS): Promise<Codec> {\n\tconst software = await VideoDecoder.isConfigSupported({\n\t\tcodec: CODECS[codec],\n\t\thardwareAcceleration: \"prefer-software\",\n\t});\n\n\tconst hardware = await VideoDecoder.isConfigSupported({\n\t\tcodec: CODECS[codec],\n\t\thardwareAcceleration: \"prefer-hardware\",\n\t});\n\n\t// We can't reliably detect hardware encoding on Firefox: https://github.com/w3c/webcodecs/issues/896\n\tconst unknown = isFirefox || hardware.config?.hardwareAcceleration !== \"prefer-hardware\";\n\n\treturn {\n\t\thardware: unknown ? undefined : hardware.supported === true,\n\t\tsoftware: software.supported === true,\n\t};\n}\n\nexport async function isSupported(): Promise<Full> {\n\treturn {\n\t\twebtransport: typeof WebTransport !== \"undefined\" ? \"full\" : \"partial\",\n\t\taudio: {\n\t\t\tdecoding: {\n\t\t\t\taac: await audioDecoderSupported(\"aac\"),\n\t\t\t\topus: (await audioDecoderSupported(\"opus\")) ? \"full\" : \"partial\",\n\t\t\t},\n\t\t\trender: typeof AudioContext !== \"undefined\" && typeof AudioBufferSourceNode !== \"undefined\",\n\t\t},\n\t\tvideo: {\n\t\t\tdecoding:\n\t\t\t\ttypeof VideoDecoder !== \"undefined\"\n\t\t\t\t\t? {\n\t\t\t\t\t\t\th264: await videoDecoderSupported(\"h264\"),\n\t\t\t\t\t\t\th265: await videoDecoderSupported(\"h265\"),\n\t\t\t\t\t\t\tvp8: await videoDecoderSupported(\"vp8\"),\n\t\t\t\t\t\t\tvp9: await videoDecoderSupported(\"vp9\"),\n\t\t\t\t\t\t\tav1: await videoDecoderSupported(\"av1\"),\n\t\t\t\t\t\t}\n\t\t\t\t\t: undefined,\n\t\t\trender: typeof OffscreenCanvas !== \"undefined\" && typeof CanvasRenderingContext2D !== \"undefined\",\n\t\t},\n\t};\n}\n"],"names":["isFirefox","CODECS","audioDecoderSupported","codec","videoDecoderSupported","software","hardware","isSupported"],"mappings":"AACA,MAAMA,IAAY,UAAU,UAAU,YAAA,EAAc,SAAS,SAAS,GAoChEC,IAAS;AAAA,EACd,KAAK;AAAA,EACL,MAAM;AAAA,EACN,KAAK;AAAA,EACL,MAAM;AAAA,EACN,MAAM;AAAA,EACN,KAAK;AAAA,EACL,KAAK;AACN;AAEA,eAAeC,EAAsBC,GAA8C;AAClF,SAAK,WAAW,gBAEJ,MAAM,aAAa,kBAAkB;AAAA,IAChD,OAAOF,EAAOE,CAAK;AAAA,IACnB,kBAAkB;AAAA,IAClB,YAAY;AAAA,EAAA,CACZ,GAEU,cAAc,KARY;AAStC;AAEA,eAAeC,EAAsBD,GAA4C;AAChF,QAAME,IAAW,MAAM,aAAa,kBAAkB;AAAA,IACrD,OAAOJ,EAAOE,CAAK;AAAA,IACnB,sBAAsB;AAAA,EAAA,CACtB,GAEKG,IAAW,MAAM,aAAa,kBAAkB;AAAA,IACrD,OAAOL,EAAOE,CAAK;AAAA,IACnB,sBAAsB;AAAA,EAAA,CACtB;AAKD,SAAO;AAAA,IACN,UAHeH,KAAaM,EAAS,QAAQ,yBAAyB,oBAGlD,SAAYA,EAAS,cAAc;AAAA,IACvD,UAAUD,EAAS,cAAc;AAAA,EAAA;AAEnC;AAEA,eAAsBE,IAA6B;AAClD,SAAO;AAAA,IACN,cAAc,OAAO,eAAiB,MAAc,SAAS;AAAA,IAC7D,OAAO;AAAA,MACN,UAAU;AAAA,QACT,KAAK,MAAML,EAAsB,KAAK;AAAA,QACtC,MAAO,MAAMA,EAAsB,MAAM,IAAK,SAAS;AAAA,MAAA;AAAA,MAExD,QAAQ,OAAO,eAAiB,OAAe,OAAO,wBAA0B;AAAA,IAAA;AAAA,IAEjF,OAAO;AAAA,MACN,UACC,OAAO,eAAiB,MACrB;AAAA,QACA,MAAM,MAAME,EAAsB,MAAM;AAAA,QACxC,MAAM,MAAMA,EAAsB,MAAM;AAAA,QACxC,KAAK,MAAMA,EAAsB,KAAK;AAAA,QACtC,KAAK,MAAMA,EAAsB,KAAK;AAAA,QACtC,KAAK,MAAMA,EAAsB,KAAK;AAAA,MAAA,IAEtC;AAAA,MACJ,QAAQ,OAAO,kBAAoB,OAAe,OAAO,2BAA6B;AAAA,IAAA;AAAA,EACvF;AAEF;"}
package/sync.d.ts ADDED
@@ -0,0 +1,21 @@
1
+ import { Time } from "@moq/lite";
2
+ import { Effect, Signal } from "@moq/signals";
3
+ export interface SyncProps {
4
+ jitter?: Time.Milli | Signal<Time.Milli>;
5
+ audio?: Time.Milli | Signal<Time.Milli | undefined>;
6
+ video?: Time.Milli | Signal<Time.Milli | undefined>;
7
+ }
8
+ export declare class Sync {
9
+ #private;
10
+ readonly reference: Signal<Time.Milli | undefined>;
11
+ jitter: Signal<Time.Milli>;
12
+ audio: Signal<Time.Milli | undefined>;
13
+ video: Signal<Time.Milli | undefined>;
14
+ readonly latency: Signal<Time.Milli>;
15
+ signals: Effect;
16
+ constructor(props?: SyncProps);
17
+ received(timestamp: Time.Milli): void;
18
+ wait(timestamp: Time.Milli): Promise<void>;
19
+ close(): void;
20
+ }
21
+ //# sourceMappingURL=sync.d.ts.map
package/sync.d.ts.map ADDED
@@ -0,0 +1 @@
1
+ {"version":3,"file":"sync.d.ts","sourceRoot":"","sources":["../src/sync.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAC;AACjC,OAAO,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,cAAc,CAAC;AAE9C,MAAM,WAAW,SAAS;IACzB,MAAM,CAAC,EAAE,IAAI,CAAC,KAAK,GAAG,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IACzC,KAAK,CAAC,EAAE,IAAI,CAAC,KAAK,GAAG,MAAM,CAAC,IAAI,CAAC,KAAK,GAAG,SAAS,CAAC,CAAC;IACpD,KAAK,CAAC,EAAE,IAAI,CAAC,KAAK,GAAG,MAAM,CAAC,IAAI,CAAC,KAAK,GAAG,SAAS,CAAC,CAAC;CACpD;AAED,qBAAa,IAAI;;IAKhB,QAAQ,CAAC,SAAS,EAAE,MAAM,CAAC,IAAI,CAAC,KAAK,GAAG,SAAS,CAAC,CAAmB;IAGrE,MAAM,EAAE,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IAG3B,KAAK,EAAE,MAAM,CAAC,IAAI,CAAC,KAAK,GAAG,SAAS,CAAC,CAAC;IACtC,KAAK,EAAE,MAAM,CAAC,IAAI,CAAC,KAAK,GAAG,SAAS,CAAC,CAAC;IAItC,QAAQ,CAAC,OAAO,EAAE,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,CAAiB;IAOrD,OAAO,SAAgB;gBAEX,KAAK,CAAC,EAAE,SAAS;IA4B7B,QAAQ,CAAC,SAAS,EAAE,IAAI,CAAC,KAAK,GAAG,IAAI;IAgB/B,IAAI,CAAC,SAAS,EAAE,IAAI,CAAC,KAAK,GAAG,OAAO,CAAC,IAAI,CAAC;IAwBhD,KAAK;CAGL"}
@@ -0,0 +1,8 @@
1
+ import { Moq } from "@moq/hang";
2
+ type BufferControlProps = {
3
+ /** Maximum buffer range in milliseconds (default: 5000ms = 5s) */
4
+ max?: Moq.Time.Milli;
5
+ };
6
+ export default function BufferControl(props: BufferControlProps): import("solid-js").JSX.Element;
7
+ export {};
8
+ //# sourceMappingURL=BufferControl.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"BufferControl.d.ts","sourceRoot":"","sources":["../../../src/ui/components/BufferControl.tsx"],"names":[],"mappings":"AAAA,OAAO,EAAE,GAAG,EAAE,MAAM,WAAW,CAAC;AAQhC,KAAK,kBAAkB,GAAG;IACzB,kEAAkE;IAClE,GAAG,CAAC,EAAE,GAAG,CAAC,IAAI,CAAC,KAAK,CAAC;CACrB,CAAC;AAEF,MAAM,CAAC,OAAO,UAAU,aAAa,CAAC,KAAK,EAAE,kBAAkB,kCAsJ9D"}
@@ -0,0 +1,2 @@
1
+ export default function BufferingIndicator(): import("solid-js").JSX.Element;
2
+ //# sourceMappingURL=BufferingIndicator.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"BufferingIndicator.d.ts","sourceRoot":"","sources":["../../../src/ui/components/BufferingIndicator.tsx"],"names":[],"mappings":"AAGA,MAAM,CAAC,OAAO,UAAU,kBAAkB,mCAUzC"}
@@ -0,0 +1,2 @@
1
+ export default function FullscreenButton(): import("solid-js").JSX.Element;
2
+ //# sourceMappingURL=FullscreenButton.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"FullscreenButton.d.ts","sourceRoot":"","sources":["../../../src/ui/components/FullscreenButton.tsx"],"names":[],"mappings":"AAIA,MAAM,CAAC,OAAO,UAAU,gBAAgB,mCAcvC"}
@@ -0,0 +1,2 @@
1
+ export default function LatencySlider(): import("solid-js").JSX.Element;
2
+ //# sourceMappingURL=LatencySlider.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"LatencySlider.d.ts","sourceRoot":"","sources":["../../../src/ui/components/LatencySlider.tsx"],"names":[],"mappings":"AAOA,MAAM,CAAC,OAAO,UAAU,aAAa,mCA0BpC"}
@@ -0,0 +1,2 @@
1
+ export default function PlayPauseButton(): import("solid-js").JSX.Element;
2
+ //# sourceMappingURL=PlayPauseButton.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"PlayPauseButton.d.ts","sourceRoot":"","sources":["../../../src/ui/components/PlayPauseButton.tsx"],"names":[],"mappings":"AAIA,MAAM,CAAC,OAAO,UAAU,eAAe,mCAatC"}
@@ -0,0 +1,3 @@
1
+ import { type JSX } from "solid-js";
2
+ export default function QualitySelector(): JSX.Element;
3
+ //# sourceMappingURL=QualitySelector.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"QualitySelector.d.ts","sourceRoot":"","sources":["../../../src/ui/components/QualitySelector.tsx"],"names":[],"mappings":"AAAA,OAAO,EAAO,KAAK,GAAG,EAAE,MAAM,UAAU,CAAC;AAGzC,MAAM,CAAC,OAAO,UAAU,eAAe,gBA+BtC"}
@@ -0,0 +1,5 @@
1
+ /**
2
+ * Toggle button for showing/hiding stats panel
3
+ */
4
+ export default function StatsButton(): import("solid-js").JSX.Element;
5
+ //# sourceMappingURL=StatsButton.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"StatsButton.d.ts","sourceRoot":"","sources":["../../../src/ui/components/StatsButton.tsx"],"names":[],"mappings":"AAGA;;GAEG;AACH,MAAM,CAAC,OAAO,UAAU,WAAW,mCAYlC"}
@@ -0,0 +1,2 @@
1
+ export default function VolumeSlider(): import("solid-js").JSX.Element;
2
+ //# sourceMappingURL=VolumeSlider.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"VolumeSlider.d.ts","sourceRoot":"","sources":["../../../src/ui/components/VolumeSlider.tsx"],"names":[],"mappings":"AAgBA,MAAM,CAAC,OAAO,UAAU,YAAY,mCAwBnC"}
@@ -0,0 +1,2 @@
1
+ export default function WatchControls(): import("solid-js").JSX.Element;
2
+ //# sourceMappingURL=WatchControls.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"WatchControls.d.ts","sourceRoot":"","sources":["../../../src/ui/components/WatchControls.tsx"],"names":[],"mappings":"AAQA,MAAM,CAAC,OAAO,UAAU,aAAa,mCAgBpC"}
@@ -0,0 +1,2 @@
1
+ export default function WatchStatusIndicator(): import("solid-js").JSX.Element;
2
+ //# sourceMappingURL=WatchStatusIndicator.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"WatchStatusIndicator.d.ts","sourceRoot":"","sources":["../../../src/ui/components/WatchStatusIndicator.tsx"],"names":[],"mappings":"AAGA,MAAM,CAAC,OAAO,UAAU,oBAAoB,mCAgB3C"}
@@ -0,0 +1,41 @@
1
+ import { type Moq } from "@moq/hang";
2
+ import type { JSX } from "solid-js";
3
+ import type { BufferedRanges } from "..";
4
+ import type MoqWatch from "../element";
5
+ type WatchUIContextProviderProps = {
6
+ moqWatch: MoqWatch;
7
+ children: JSX.Element;
8
+ };
9
+ type WatchStatus = "no-url" | "disconnected" | "connecting" | "offline" | "loading" | "live" | "connected";
10
+ export type Rendition = {
11
+ name: string;
12
+ width?: number;
13
+ height?: number;
14
+ };
15
+ export type WatchUIContextValues = {
16
+ moqWatch: MoqWatch;
17
+ watchStatus: () => WatchStatus;
18
+ isPlaying: () => boolean;
19
+ isMuted: () => boolean;
20
+ setVolume: (vol: number) => void;
21
+ currentVolume: () => number;
22
+ togglePlayback: () => void;
23
+ toggleMuted: () => void;
24
+ buffering: () => boolean;
25
+ jitter: () => Moq.Time.Milli;
26
+ setJitter: (value: Moq.Time.Milli) => void;
27
+ availableRenditions: () => Rendition[];
28
+ activeRendition: () => string | undefined;
29
+ setActiveRendition: (name: string | undefined) => void;
30
+ isStatsPanelVisible: () => boolean;
31
+ setIsStatsPanelVisible: (visible: boolean) => void;
32
+ isFullscreen: () => boolean;
33
+ toggleFullscreen: () => void;
34
+ timestamp: () => Moq.Time.Milli | undefined;
35
+ videoBuffered: () => BufferedRanges;
36
+ audioBuffered: () => BufferedRanges;
37
+ };
38
+ export declare const WatchUIContext: import("solid-js").Context<WatchUIContextValues | undefined>;
39
+ export default function WatchUIContextProvider(props: WatchUIContextProviderProps): JSX.Element;
40
+ export {};
41
+ //# sourceMappingURL=context.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"context.d.ts","sourceRoot":"","sources":["../../src/ui/context.tsx"],"names":[],"mappings":"AAAA,OAAO,EAAE,KAAK,GAAG,EAAW,MAAM,WAAW,CAAC;AAE9C,OAAO,KAAK,EAAE,GAAG,EAAE,MAAM,UAAU,CAAC;AAEpC,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,IAAI,CAAC;AACzC,OAAO,KAAK,QAAQ,MAAM,YAAY,CAAC;AAEvC,KAAK,2BAA2B,GAAG;IAClC,QAAQ,EAAE,QAAQ,CAAC;IACnB,QAAQ,EAAE,GAAG,CAAC,OAAO,CAAC;CACtB,CAAC;AAEF,KAAK,WAAW,GAAG,QAAQ,GAAG,cAAc,GAAG,YAAY,GAAG,SAAS,GAAG,SAAS,GAAG,MAAM,GAAG,WAAW,CAAC;AAE3G,MAAM,MAAM,SAAS,GAAG;IACvB,IAAI,EAAE,MAAM,CAAC;IACb,KAAK,CAAC,EAAE,MAAM,CAAC;IACf,MAAM,CAAC,EAAE,MAAM,CAAC;CAChB,CAAC;AAEF,MAAM,MAAM,oBAAoB,GAAG;IAClC,QAAQ,EAAE,QAAQ,CAAC;IACnB,WAAW,EAAE,MAAM,WAAW,CAAC;IAC/B,SAAS,EAAE,MAAM,OAAO,CAAC;IACzB,OAAO,EAAE,MAAM,OAAO,CAAC;IACvB,SAAS,EAAE,CAAC,GAAG,EAAE,MAAM,KAAK,IAAI,CAAC;IACjC,aAAa,EAAE,MAAM,MAAM,CAAC;IAC5B,cAAc,EAAE,MAAM,IAAI,CAAC;IAC3B,WAAW,EAAE,MAAM,IAAI,CAAC;IACxB,SAAS,EAAE,MAAM,OAAO,CAAC;IACzB,MAAM,EAAE,MAAM,GAAG,CAAC,IAAI,CAAC,KAAK,CAAC;IAC7B,SAAS,EAAE,CAAC,KAAK,EAAE,GAAG,CAAC,IAAI,CAAC,KAAK,KAAK,IAAI,CAAC;IAC3C,mBAAmB,EAAE,MAAM,SAAS,EAAE,CAAC;IACvC,eAAe,EAAE,MAAM,MAAM,GAAG,SAAS,CAAC;IAC1C,kBAAkB,EAAE,CAAC,IAAI,EAAE,MAAM,GAAG,SAAS,KAAK,IAAI,CAAC;IACvD,mBAAmB,EAAE,MAAM,OAAO,CAAC;IACnC,sBAAsB,EAAE,CAAC,OAAO,EAAE,OAAO,KAAK,IAAI,CAAC;IACnD,YAAY,EAAE,MAAM,OAAO,CAAC;IAC5B,gBAAgB,EAAE,MAAM,IAAI,CAAC;IAC7B,SAAS,EAAE,MAAM,GAAG,CAAC,IAAI,CAAC,KAAK,GAAG,SAAS,CAAC;IAC5C,aAAa,EAAE,MAAM,cAAc,CAAC;IACpC,aAAa,EAAE,MAAM,cAAc,CAAC;CACpC,CAAC;AAEF,eAAO,MAAM,cAAc,8DAAwC,CAAC;AAEpE,MAAM,CAAC,OAAO,UAAU,sBAAsB,CAAC,KAAK,EAAE,2BAA2B,eAoJhF"}
@@ -0,0 +1,5 @@
1
+ import type MoqWatch from "../element";
2
+ export declare function WatchUI(props: {
3
+ watch: MoqWatch;
4
+ }): import("solid-js").JSX.Element;
5
+ //# sourceMappingURL=element.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"element.d.ts","sourceRoot":"","sources":["../../src/ui/element.tsx"],"names":[],"mappings":"AAGA,OAAO,KAAK,QAAQ,MAAM,YAAY,CAAC;AAOvC,wBAAgB,OAAO,CAAC,KAAK,EAAE;IAAE,KAAK,EAAE,QAAQ,CAAA;CAAE,kCAyBjD"}
@@ -0,0 +1,3 @@
1
+ import { type WatchUIContextValues } from "../context";
2
+ export default function useWatchUIContext(): WatchUIContextValues;
3
+ //# sourceMappingURL=use-watch-ui.d.ts.map