@kenzuya/mediabunny 1.26.0 → 1.28.6

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 (238) hide show
  1. package/README.md +1 -1
  2. package/dist/bundles/{mediabunny.mjs → mediabunny.js} +21963 -21390
  3. package/dist/bundles/mediabunny.min.js +490 -0
  4. package/dist/modules/shared/mp3-misc.d.ts.map +1 -1
  5. package/dist/modules/src/adts/adts-demuxer.d.ts +6 -6
  6. package/dist/modules/src/adts/adts-demuxer.d.ts.map +1 -1
  7. package/dist/modules/src/adts/adts-muxer.d.ts +4 -4
  8. package/dist/modules/src/adts/adts-muxer.d.ts.map +1 -1
  9. package/dist/modules/src/adts/adts-reader.d.ts +1 -1
  10. package/dist/modules/src/adts/adts-reader.d.ts.map +1 -1
  11. package/dist/modules/src/avi/avi-demuxer.d.ts +44 -0
  12. package/dist/modules/src/avi/avi-demuxer.d.ts.map +1 -0
  13. package/dist/modules/src/avi/avi-misc.d.ts +88 -0
  14. package/dist/modules/src/avi/avi-misc.d.ts.map +1 -0
  15. package/dist/modules/src/avi/avi-muxer.d.ts +45 -0
  16. package/dist/modules/src/avi/avi-muxer.d.ts.map +1 -0
  17. package/dist/modules/src/avi/riff-writer.d.ts +26 -0
  18. package/dist/modules/src/avi/riff-writer.d.ts.map +1 -0
  19. package/dist/modules/src/codec-data.d.ts +8 -3
  20. package/dist/modules/src/codec-data.d.ts.map +1 -1
  21. package/dist/modules/src/codec.d.ts +10 -10
  22. package/dist/modules/src/codec.d.ts.map +1 -1
  23. package/dist/modules/src/conversion.d.ts +33 -16
  24. package/dist/modules/src/conversion.d.ts.map +1 -1
  25. package/dist/modules/src/custom-coder.d.ts +8 -8
  26. package/dist/modules/src/custom-coder.d.ts.map +1 -1
  27. package/dist/modules/src/demuxer.d.ts +3 -3
  28. package/dist/modules/src/demuxer.d.ts.map +1 -1
  29. package/dist/modules/src/encode.d.ts +8 -8
  30. package/dist/modules/src/encode.d.ts.map +1 -1
  31. package/dist/modules/src/flac/flac-demuxer.d.ts +7 -7
  32. package/dist/modules/src/flac/flac-demuxer.d.ts.map +1 -1
  33. package/dist/modules/src/flac/flac-misc.d.ts +3 -3
  34. package/dist/modules/src/flac/flac-misc.d.ts.map +1 -1
  35. package/dist/modules/src/flac/flac-muxer.d.ts +5 -5
  36. package/dist/modules/src/flac/flac-muxer.d.ts.map +1 -1
  37. package/dist/modules/src/id3.d.ts +3 -3
  38. package/dist/modules/src/id3.d.ts.map +1 -1
  39. package/dist/modules/src/index.d.ts +20 -20
  40. package/dist/modules/src/index.d.ts.map +1 -1
  41. package/dist/modules/src/input-format.d.ts +22 -0
  42. package/dist/modules/src/input-format.d.ts.map +1 -1
  43. package/dist/modules/src/input-track.d.ts +8 -8
  44. package/dist/modules/src/input-track.d.ts.map +1 -1
  45. package/dist/modules/src/input.d.ts +12 -12
  46. package/dist/modules/src/isobmff/isobmff-boxes.d.ts +2 -2
  47. package/dist/modules/src/isobmff/isobmff-boxes.d.ts.map +1 -1
  48. package/dist/modules/src/isobmff/isobmff-demuxer.d.ts +12 -12
  49. package/dist/modules/src/isobmff/isobmff-demuxer.d.ts.map +1 -1
  50. package/dist/modules/src/isobmff/isobmff-misc.d.ts.map +1 -1
  51. package/dist/modules/src/isobmff/isobmff-muxer.d.ts +11 -11
  52. package/dist/modules/src/isobmff/isobmff-muxer.d.ts.map +1 -1
  53. package/dist/modules/src/isobmff/isobmff-reader.d.ts +2 -2
  54. package/dist/modules/src/isobmff/isobmff-reader.d.ts.map +1 -1
  55. package/dist/modules/src/matroska/ebml.d.ts +3 -3
  56. package/dist/modules/src/matroska/ebml.d.ts.map +1 -1
  57. package/dist/modules/src/matroska/matroska-demuxer.d.ts +13 -13
  58. package/dist/modules/src/matroska/matroska-demuxer.d.ts.map +1 -1
  59. package/dist/modules/src/matroska/matroska-input.d.ts +33 -0
  60. package/dist/modules/src/matroska/matroska-input.d.ts.map +1 -0
  61. package/dist/modules/src/matroska/matroska-misc.d.ts.map +1 -1
  62. package/dist/modules/src/matroska/matroska-muxer.d.ts +5 -5
  63. package/dist/modules/src/matroska/matroska-muxer.d.ts.map +1 -1
  64. package/dist/modules/src/media-sink.d.ts +5 -5
  65. package/dist/modules/src/media-sink.d.ts.map +1 -1
  66. package/dist/modules/src/media-source.d.ts +22 -4
  67. package/dist/modules/src/media-source.d.ts.map +1 -1
  68. package/dist/modules/src/metadata.d.ts +2 -2
  69. package/dist/modules/src/metadata.d.ts.map +1 -1
  70. package/dist/modules/src/misc.d.ts +5 -4
  71. package/dist/modules/src/misc.d.ts.map +1 -1
  72. package/dist/modules/src/mp3/mp3-demuxer.d.ts +7 -7
  73. package/dist/modules/src/mp3/mp3-demuxer.d.ts.map +1 -1
  74. package/dist/modules/src/mp3/mp3-muxer.d.ts +4 -4
  75. package/dist/modules/src/mp3/mp3-muxer.d.ts.map +1 -1
  76. package/dist/modules/src/mp3/mp3-reader.d.ts +2 -2
  77. package/dist/modules/src/mp3/mp3-reader.d.ts.map +1 -1
  78. package/dist/modules/src/mp3/mp3-writer.d.ts +1 -1
  79. package/dist/modules/src/mp3/mp3-writer.d.ts.map +1 -1
  80. package/dist/modules/src/muxer.d.ts +4 -4
  81. package/dist/modules/src/muxer.d.ts.map +1 -1
  82. package/dist/modules/src/ogg/ogg-demuxer.d.ts +7 -7
  83. package/dist/modules/src/ogg/ogg-demuxer.d.ts.map +1 -1
  84. package/dist/modules/src/ogg/ogg-misc.d.ts +1 -1
  85. package/dist/modules/src/ogg/ogg-misc.d.ts.map +1 -1
  86. package/dist/modules/src/ogg/ogg-muxer.d.ts +5 -5
  87. package/dist/modules/src/ogg/ogg-muxer.d.ts.map +1 -1
  88. package/dist/modules/src/ogg/ogg-reader.d.ts +1 -1
  89. package/dist/modules/src/ogg/ogg-reader.d.ts.map +1 -1
  90. package/dist/modules/src/output-format.d.ts +51 -6
  91. package/dist/modules/src/output-format.d.ts.map +1 -1
  92. package/dist/modules/src/output.d.ts +13 -13
  93. package/dist/modules/src/output.d.ts.map +1 -1
  94. package/dist/modules/src/packet.d.ts +1 -1
  95. package/dist/modules/src/packet.d.ts.map +1 -1
  96. package/dist/modules/src/pcm.d.ts.map +1 -1
  97. package/dist/modules/src/reader.d.ts +2 -2
  98. package/dist/modules/src/reader.d.ts.map +1 -1
  99. package/dist/modules/src/sample.d.ts +57 -15
  100. package/dist/modules/src/sample.d.ts.map +1 -1
  101. package/dist/modules/src/source.d.ts +3 -3
  102. package/dist/modules/src/source.d.ts.map +1 -1
  103. package/dist/modules/src/subtitles.d.ts +1 -1
  104. package/dist/modules/src/subtitles.d.ts.map +1 -1
  105. package/dist/modules/src/target.d.ts +2 -2
  106. package/dist/modules/src/target.d.ts.map +1 -1
  107. package/dist/modules/src/tsconfig.tsbuildinfo +1 -1
  108. package/dist/modules/src/wave/riff-writer.d.ts +1 -1
  109. package/dist/modules/src/wave/riff-writer.d.ts.map +1 -1
  110. package/dist/modules/src/wave/wave-demuxer.d.ts +6 -6
  111. package/dist/modules/src/wave/wave-demuxer.d.ts.map +1 -1
  112. package/dist/modules/src/wave/wave-muxer.d.ts +4 -4
  113. package/dist/modules/src/wave/wave-muxer.d.ts.map +1 -1
  114. package/dist/modules/src/writer.d.ts +1 -1
  115. package/dist/modules/src/writer.d.ts.map +1 -1
  116. package/dist/packages/eac3/eac3.wasm +0 -0
  117. package/dist/packages/eac3/mediabunny-eac3.js +1058 -0
  118. package/dist/packages/eac3/mediabunny-eac3.min.js +44 -0
  119. package/dist/packages/mp3-encoder/mediabunny-mp3-encoder.js +694 -0
  120. package/dist/packages/mp3-encoder/mediabunny-mp3-encoder.min.js +58 -0
  121. package/dist/packages/mpeg4/mediabunny-mpeg4.js +1198 -0
  122. package/dist/packages/mpeg4/mediabunny-mpeg4.min.js +44 -0
  123. package/dist/packages/mpeg4/xvid.wasm +0 -0
  124. package/package.json +18 -57
  125. package/dist/bundles/mediabunny.cjs +0 -26140
  126. package/dist/bundles/mediabunny.min.cjs +0 -147
  127. package/dist/bundles/mediabunny.min.mjs +0 -146
  128. package/dist/mediabunny.d.ts +0 -3319
  129. package/dist/modules/shared/mp3-misc.js +0 -147
  130. package/dist/modules/src/adts/adts-demuxer.js +0 -239
  131. package/dist/modules/src/adts/adts-muxer.js +0 -80
  132. package/dist/modules/src/adts/adts-reader.js +0 -63
  133. package/dist/modules/src/codec-data.js +0 -1730
  134. package/dist/modules/src/codec.js +0 -869
  135. package/dist/modules/src/conversion.js +0 -1459
  136. package/dist/modules/src/custom-coder.js +0 -117
  137. package/dist/modules/src/demuxer.js +0 -12
  138. package/dist/modules/src/encode.js +0 -442
  139. package/dist/modules/src/flac/flac-demuxer.js +0 -504
  140. package/dist/modules/src/flac/flac-misc.js +0 -135
  141. package/dist/modules/src/flac/flac-muxer.js +0 -222
  142. package/dist/modules/src/id3.js +0 -848
  143. package/dist/modules/src/index.js +0 -28
  144. package/dist/modules/src/input-format.js +0 -480
  145. package/dist/modules/src/input-track.js +0 -372
  146. package/dist/modules/src/input.js +0 -188
  147. package/dist/modules/src/isobmff/isobmff-boxes.js +0 -1480
  148. package/dist/modules/src/isobmff/isobmff-demuxer.js +0 -2618
  149. package/dist/modules/src/isobmff/isobmff-misc.js +0 -20
  150. package/dist/modules/src/isobmff/isobmff-muxer.js +0 -966
  151. package/dist/modules/src/isobmff/isobmff-reader.js +0 -72
  152. package/dist/modules/src/matroska/ebml.js +0 -653
  153. package/dist/modules/src/matroska/matroska-demuxer.js +0 -2133
  154. package/dist/modules/src/matroska/matroska-misc.js +0 -20
  155. package/dist/modules/src/matroska/matroska-muxer.js +0 -1017
  156. package/dist/modules/src/media-sink.js +0 -1736
  157. package/dist/modules/src/media-source.js +0 -1825
  158. package/dist/modules/src/metadata.js +0 -193
  159. package/dist/modules/src/misc.js +0 -623
  160. package/dist/modules/src/mp3/mp3-demuxer.js +0 -285
  161. package/dist/modules/src/mp3/mp3-muxer.js +0 -123
  162. package/dist/modules/src/mp3/mp3-reader.js +0 -26
  163. package/dist/modules/src/mp3/mp3-writer.js +0 -78
  164. package/dist/modules/src/muxer.js +0 -50
  165. package/dist/modules/src/node.d.ts +0 -9
  166. package/dist/modules/src/node.d.ts.map +0 -1
  167. package/dist/modules/src/node.js +0 -9
  168. package/dist/modules/src/ogg/ogg-demuxer.js +0 -763
  169. package/dist/modules/src/ogg/ogg-misc.js +0 -78
  170. package/dist/modules/src/ogg/ogg-muxer.js +0 -353
  171. package/dist/modules/src/ogg/ogg-reader.js +0 -65
  172. package/dist/modules/src/output-format.js +0 -527
  173. package/dist/modules/src/output.js +0 -300
  174. package/dist/modules/src/packet.js +0 -182
  175. package/dist/modules/src/pcm.js +0 -85
  176. package/dist/modules/src/reader.js +0 -236
  177. package/dist/modules/src/sample.js +0 -1056
  178. package/dist/modules/src/source.js +0 -1182
  179. package/dist/modules/src/subtitles.js +0 -575
  180. package/dist/modules/src/target.js +0 -140
  181. package/dist/modules/src/wave/riff-writer.js +0 -30
  182. package/dist/modules/src/wave/wave-demuxer.js +0 -447
  183. package/dist/modules/src/wave/wave-muxer.js +0 -318
  184. package/dist/modules/src/writer.js +0 -370
  185. package/src/adts/adts-demuxer.ts +0 -331
  186. package/src/adts/adts-muxer.ts +0 -111
  187. package/src/adts/adts-reader.ts +0 -85
  188. package/src/codec-data.ts +0 -2078
  189. package/src/codec.ts +0 -1092
  190. package/src/conversion.ts +0 -2112
  191. package/src/custom-coder.ts +0 -197
  192. package/src/demuxer.ts +0 -24
  193. package/src/encode.ts +0 -739
  194. package/src/flac/flac-demuxer.ts +0 -730
  195. package/src/flac/flac-misc.ts +0 -164
  196. package/src/flac/flac-muxer.ts +0 -320
  197. package/src/id3.ts +0 -925
  198. package/src/index.ts +0 -221
  199. package/src/input-format.ts +0 -541
  200. package/src/input-track.ts +0 -529
  201. package/src/input.ts +0 -235
  202. package/src/isobmff/isobmff-boxes.ts +0 -1719
  203. package/src/isobmff/isobmff-demuxer.ts +0 -3190
  204. package/src/isobmff/isobmff-misc.ts +0 -29
  205. package/src/isobmff/isobmff-muxer.ts +0 -1348
  206. package/src/isobmff/isobmff-reader.ts +0 -91
  207. package/src/matroska/ebml.ts +0 -730
  208. package/src/matroska/matroska-demuxer.ts +0 -2481
  209. package/src/matroska/matroska-misc.ts +0 -29
  210. package/src/matroska/matroska-muxer.ts +0 -1276
  211. package/src/media-sink.ts +0 -2179
  212. package/src/media-source.ts +0 -2243
  213. package/src/metadata.ts +0 -320
  214. package/src/misc.ts +0 -798
  215. package/src/mp3/mp3-demuxer.ts +0 -383
  216. package/src/mp3/mp3-muxer.ts +0 -166
  217. package/src/mp3/mp3-reader.ts +0 -34
  218. package/src/mp3/mp3-writer.ts +0 -120
  219. package/src/muxer.ts +0 -88
  220. package/src/node.ts +0 -11
  221. package/src/ogg/ogg-demuxer.ts +0 -1053
  222. package/src/ogg/ogg-misc.ts +0 -116
  223. package/src/ogg/ogg-muxer.ts +0 -497
  224. package/src/ogg/ogg-reader.ts +0 -93
  225. package/src/output-format.ts +0 -945
  226. package/src/output.ts +0 -488
  227. package/src/packet.ts +0 -263
  228. package/src/pcm.ts +0 -112
  229. package/src/reader.ts +0 -323
  230. package/src/sample.ts +0 -1461
  231. package/src/source.ts +0 -1688
  232. package/src/subtitles.ts +0 -711
  233. package/src/target.ts +0 -204
  234. package/src/tsconfig.json +0 -16
  235. package/src/wave/riff-writer.ts +0 -36
  236. package/src/wave/wave-demuxer.ts +0 -529
  237. package/src/wave/wave-muxer.ts +0 -371
  238. package/src/writer.ts +0 -490
@@ -1,117 +0,0 @@
1
- /*!
2
- * Copyright (c) 2025-present, Vanilagy and contributors
3
- *
4
- * This Source Code Form is subject to the terms of the Mozilla Public
5
- * License, v. 2.0. If a copy of the MPL was not distributed with this
6
- * file, You can obtain one at https://mozilla.org/MPL/2.0/.
7
- */
8
- /**
9
- * Base class for custom video decoders. To add your own custom video decoder, extend this class, implement the
10
- * abstract methods and static `supports` method, and register the decoder using {@link registerDecoder}.
11
- * @group Custom coders
12
- * @public
13
- */
14
- export class CustomVideoDecoder {
15
- /** Returns true if and only if the decoder can decode the given codec configuration. */
16
- // eslint-disable-next-line @typescript-eslint/no-unused-vars
17
- static supports(codec, config) {
18
- return false;
19
- }
20
- }
21
- /**
22
- * Base class for custom audio decoders. To add your own custom audio decoder, extend this class, implement the
23
- * abstract methods and static `supports` method, and register the decoder using {@link registerDecoder}.
24
- * @group Custom coders
25
- * @public
26
- */
27
- export class CustomAudioDecoder {
28
- /** Returns true if and only if the decoder can decode the given codec configuration. */
29
- // eslint-disable-next-line @typescript-eslint/no-unused-vars
30
- static supports(codec, config) {
31
- return false;
32
- }
33
- }
34
- /**
35
- * Base class for custom video encoders. To add your own custom video encoder, extend this class, implement the
36
- * abstract methods and static `supports` method, and register the encoder using {@link registerEncoder}.
37
- * @group Custom coders
38
- * @public
39
- */
40
- export class CustomVideoEncoder {
41
- /** Returns true if and only if the encoder can encode the given codec configuration. */
42
- // eslint-disable-next-line @typescript-eslint/no-unused-vars
43
- static supports(codec, config) {
44
- return false;
45
- }
46
- }
47
- /**
48
- * Base class for custom audio encoders. To add your own custom audio encoder, extend this class, implement the
49
- * abstract methods and static `supports` method, and register the encoder using {@link registerEncoder}.
50
- * @group Custom coders
51
- * @public
52
- */
53
- export class CustomAudioEncoder {
54
- /** Returns true if and only if the encoder can encode the given codec configuration. */
55
- // eslint-disable-next-line @typescript-eslint/no-unused-vars
56
- static supports(codec, config) {
57
- return false;
58
- }
59
- }
60
- export const customVideoDecoders = [];
61
- export const customAudioDecoders = [];
62
- export const customVideoEncoders = [];
63
- export const customAudioEncoders = [];
64
- /**
65
- * Registers a custom video or audio decoder. Registered decoders will automatically be used for decoding whenever
66
- * possible.
67
- * @group Custom coders
68
- * @public
69
- */
70
- export const registerDecoder = (decoder) => {
71
- if (decoder.prototype instanceof CustomVideoDecoder) {
72
- const casted = decoder;
73
- if (customVideoDecoders.includes(casted)) {
74
- console.warn('Video decoder already registered.');
75
- return;
76
- }
77
- customVideoDecoders.push(casted);
78
- }
79
- else if (decoder.prototype instanceof CustomAudioDecoder) {
80
- const casted = decoder;
81
- if (customAudioDecoders.includes(casted)) {
82
- console.warn('Audio decoder already registered.');
83
- return;
84
- }
85
- customAudioDecoders.push(casted);
86
- }
87
- else {
88
- throw new TypeError('Decoder must be a CustomVideoDecoder or CustomAudioDecoder.');
89
- }
90
- };
91
- /**
92
- * Registers a custom video or audio encoder. Registered encoders will automatically be used for encoding whenever
93
- * possible.
94
- * @group Custom coders
95
- * @public
96
- */
97
- export const registerEncoder = (encoder) => {
98
- if (encoder.prototype instanceof CustomVideoEncoder) {
99
- const casted = encoder;
100
- if (customVideoEncoders.includes(casted)) {
101
- console.warn('Video encoder already registered.');
102
- return;
103
- }
104
- customVideoEncoders.push(casted);
105
- }
106
- else if (encoder.prototype instanceof CustomAudioEncoder) {
107
- const casted = encoder;
108
- if (customAudioEncoders.includes(casted)) {
109
- console.warn('Audio encoder already registered.');
110
- return;
111
- }
112
- customAudioEncoders.push(casted);
113
- }
114
- else {
115
- throw new TypeError('Encoder must be a CustomVideoEncoder or CustomAudioEncoder.');
116
- }
117
- };
@@ -1,12 +0,0 @@
1
- /*!
2
- * Copyright (c) 2025-present, Vanilagy and contributors
3
- *
4
- * This Source Code Form is subject to the terms of the Mozilla Public
5
- * License, v. 2.0. If a copy of the MPL was not distributed with this
6
- * file, You can obtain one at https://mozilla.org/MPL/2.0/.
7
- */
8
- export class Demuxer {
9
- constructor(input) {
10
- this.input = input;
11
- }
12
- }
@@ -1,442 +0,0 @@
1
- /*!
2
- * Copyright (c) 2025-present, Vanilagy and contributors
3
- *
4
- * This Source Code Form is subject to the terms of the Mozilla Public
5
- * License, v. 2.0. If a copy of the MPL was not distributed with this
6
- * file, You can obtain one at https://mozilla.org/MPL/2.0/.
7
- */
8
- import { AUDIO_CODECS, buildAudioCodecString, buildVideoCodecString, getAudioEncoderConfigExtension, getVideoEncoderConfigExtension, inferCodecFromCodecString, PCM_AUDIO_CODECS, SUBTITLE_CODECS, VIDEO_CODECS, } from './codec.js';
9
- import { customAudioEncoders, customVideoEncoders } from './custom-coder.js';
10
- export const validateVideoEncodingConfig = (config) => {
11
- if (!config || typeof config !== 'object') {
12
- throw new TypeError('Encoding config must be an object.');
13
- }
14
- if (!VIDEO_CODECS.includes(config.codec)) {
15
- throw new TypeError(`Invalid video codec '${config.codec}'. Must be one of: ${VIDEO_CODECS.join(', ')}.`);
16
- }
17
- if (!(config.bitrate instanceof Quality) && (!Number.isInteger(config.bitrate) || config.bitrate <= 0)) {
18
- throw new TypeError('config.bitrate must be a positive integer or a quality.');
19
- }
20
- if (config.keyFrameInterval !== undefined
21
- && (!Number.isFinite(config.keyFrameInterval) || config.keyFrameInterval < 0)) {
22
- throw new TypeError('config.keyFrameInterval, when provided, must be a non-negative number.');
23
- }
24
- if (config.sizeChangeBehavior !== undefined
25
- && !['deny', 'passThrough', 'fill', 'contain', 'cover'].includes(config.sizeChangeBehavior)) {
26
- throw new TypeError('config.sizeChangeBehavior, when provided, must be \'deny\', \'passThrough\', \'fill\', \'contain\''
27
- + ' or \'cover\'.');
28
- }
29
- if (config.onEncodedPacket !== undefined && typeof config.onEncodedPacket !== 'function') {
30
- throw new TypeError('config.onEncodedChunk, when provided, must be a function.');
31
- }
32
- if (config.onEncoderConfig !== undefined && typeof config.onEncoderConfig !== 'function') {
33
- throw new TypeError('config.onEncoderConfig, when provided, must be a function.');
34
- }
35
- validateVideoEncodingAdditionalOptions(config.codec, config);
36
- };
37
- export const validateVideoEncodingAdditionalOptions = (codec, options) => {
38
- if (!options || typeof options !== 'object') {
39
- throw new TypeError('Encoding options must be an object.');
40
- }
41
- if (options.alpha !== undefined && !['discard', 'keep'].includes(options.alpha)) {
42
- throw new TypeError('options.alpha, when provided, must be \'discard\' or \'keep\'.');
43
- }
44
- if (options.bitrateMode !== undefined && !['constant', 'variable'].includes(options.bitrateMode)) {
45
- throw new TypeError('bitrateMode, when provided, must be \'constant\' or \'variable\'.');
46
- }
47
- if (options.latencyMode !== undefined && !['quality', 'realtime'].includes(options.latencyMode)) {
48
- throw new TypeError('latencyMode, when provided, must be \'quality\' or \'realtime\'.');
49
- }
50
- if (options.fullCodecString !== undefined && typeof options.fullCodecString !== 'string') {
51
- throw new TypeError('fullCodecString, when provided, must be a string.');
52
- }
53
- if (options.fullCodecString !== undefined && inferCodecFromCodecString(options.fullCodecString) !== codec) {
54
- throw new TypeError(`fullCodecString, when provided, must be a string that matches the specified codec (${codec}).`);
55
- }
56
- if (options.hardwareAcceleration !== undefined
57
- && !['no-preference', 'prefer-hardware', 'prefer-software'].includes(options.hardwareAcceleration)) {
58
- throw new TypeError('hardwareAcceleration, when provided, must be \'no-preference\', \'prefer-hardware\' or'
59
- + ' \'prefer-software\'.');
60
- }
61
- if (options.scalabilityMode !== undefined && typeof options.scalabilityMode !== 'string') {
62
- throw new TypeError('scalabilityMode, when provided, must be a string.');
63
- }
64
- if (options.contentHint !== undefined && typeof options.contentHint !== 'string') {
65
- throw new TypeError('contentHint, when provided, must be a string.');
66
- }
67
- };
68
- export const buildVideoEncoderConfig = (options) => {
69
- const resolvedBitrate = options.bitrate instanceof Quality
70
- ? options.bitrate._toVideoBitrate(options.codec, options.width, options.height)
71
- : options.bitrate;
72
- return {
73
- codec: options.fullCodecString ?? buildVideoCodecString(options.codec, options.width, options.height, resolvedBitrate),
74
- width: options.width,
75
- height: options.height,
76
- bitrate: resolvedBitrate,
77
- bitrateMode: options.bitrateMode,
78
- alpha: options.alpha ?? 'discard',
79
- framerate: options.framerate,
80
- latencyMode: options.latencyMode,
81
- hardwareAcceleration: options.hardwareAcceleration,
82
- scalabilityMode: options.scalabilityMode,
83
- contentHint: options.contentHint,
84
- ...getVideoEncoderConfigExtension(options.codec),
85
- };
86
- };
87
- export const validateAudioEncodingConfig = (config) => {
88
- if (!config || typeof config !== 'object') {
89
- throw new TypeError('Encoding config must be an object.');
90
- }
91
- if (!AUDIO_CODECS.includes(config.codec)) {
92
- throw new TypeError(`Invalid audio codec '${config.codec}'. Must be one of: ${AUDIO_CODECS.join(', ')}.`);
93
- }
94
- if (config.bitrate === undefined
95
- && (!PCM_AUDIO_CODECS.includes(config.codec) || config.codec === 'flac')) {
96
- throw new TypeError('config.bitrate must be provided for compressed audio codecs.');
97
- }
98
- if (config.bitrate !== undefined
99
- && !(config.bitrate instanceof Quality)
100
- && (!Number.isInteger(config.bitrate) || config.bitrate <= 0)) {
101
- throw new TypeError('config.bitrate, when provided, must be a positive integer or a quality.');
102
- }
103
- if (config.onEncodedPacket !== undefined && typeof config.onEncodedPacket !== 'function') {
104
- throw new TypeError('config.onEncodedChunk, when provided, must be a function.');
105
- }
106
- if (config.onEncoderConfig !== undefined && typeof config.onEncoderConfig !== 'function') {
107
- throw new TypeError('config.onEncoderConfig, when provided, must be a function.');
108
- }
109
- validateAudioEncodingAdditionalOptions(config.codec, config);
110
- };
111
- export const validateAudioEncodingAdditionalOptions = (codec, options) => {
112
- if (!options || typeof options !== 'object') {
113
- throw new TypeError('Encoding options must be an object.');
114
- }
115
- if (options.bitrateMode !== undefined && !['constant', 'variable'].includes(options.bitrateMode)) {
116
- throw new TypeError('bitrateMode, when provided, must be \'constant\' or \'variable\'.');
117
- }
118
- if (options.fullCodecString !== undefined && typeof options.fullCodecString !== 'string') {
119
- throw new TypeError('fullCodecString, when provided, must be a string.');
120
- }
121
- if (options.fullCodecString !== undefined && inferCodecFromCodecString(options.fullCodecString) !== codec) {
122
- throw new TypeError(`fullCodecString, when provided, must be a string that matches the specified codec (${codec}).`);
123
- }
124
- };
125
- export const buildAudioEncoderConfig = (options) => {
126
- const resolvedBitrate = options.bitrate instanceof Quality
127
- ? options.bitrate._toAudioBitrate(options.codec)
128
- : options.bitrate;
129
- return {
130
- codec: options.fullCodecString ?? buildAudioCodecString(options.codec, options.numberOfChannels, options.sampleRate),
131
- numberOfChannels: options.numberOfChannels,
132
- sampleRate: options.sampleRate,
133
- bitrate: resolvedBitrate,
134
- bitrateMode: options.bitrateMode,
135
- ...getAudioEncoderConfigExtension(options.codec),
136
- };
137
- };
138
- /**
139
- * Represents a subjective media quality level.
140
- * @group Encoding
141
- * @public
142
- */
143
- export class Quality {
144
- /** @internal */
145
- constructor(factor) {
146
- this._factor = factor;
147
- }
148
- /** @internal */
149
- _toVideoBitrate(codec, width, height) {
150
- const pixels = width * height;
151
- const codecEfficiencyFactors = {
152
- avc: 1.0, // H.264/AVC (baseline)
153
- hevc: 0.6, // H.265/HEVC (~40% more efficient than AVC)
154
- vp9: 0.6, // Similar to HEVC
155
- av1: 0.4, // ~60% more efficient than AVC
156
- vp8: 1.2, // Slightly less efficient than AVC
157
- };
158
- const referencePixels = 1920 * 1080;
159
- const referenceBitrate = 3000000;
160
- const scaleFactor = Math.pow(pixels / referencePixels, 0.95); // Slight non-linear scaling
161
- const baseBitrate = referenceBitrate * scaleFactor;
162
- const codecAdjustedBitrate = baseBitrate * codecEfficiencyFactors[codec];
163
- const finalBitrate = codecAdjustedBitrate * this._factor;
164
- return Math.ceil(finalBitrate / 1000) * 1000;
165
- }
166
- /** @internal */
167
- _toAudioBitrate(codec) {
168
- if (PCM_AUDIO_CODECS.includes(codec) || codec === 'flac') {
169
- return undefined;
170
- }
171
- const baseRates = {
172
- aac: 128000, // 128kbps base for AAC
173
- opus: 64000, // 64kbps base for Opus
174
- mp3: 160000, // 160kbps base for MP3
175
- vorbis: 64000, // 64kbps base for Vorbis
176
- };
177
- const baseBitrate = baseRates[codec];
178
- if (!baseBitrate) {
179
- throw new Error(`Unhandled codec: ${codec}`);
180
- }
181
- let finalBitrate = baseBitrate * this._factor;
182
- if (codec === 'aac') {
183
- // AAC only works with specific bitrates, let's find the closest
184
- const validRates = [96000, 128000, 160000, 192000];
185
- finalBitrate = validRates.reduce((prev, curr) => Math.abs(curr - finalBitrate) < Math.abs(prev - finalBitrate) ? curr : prev);
186
- }
187
- else if (codec === 'opus' || codec === 'vorbis') {
188
- finalBitrate = Math.max(6000, finalBitrate);
189
- }
190
- else if (codec === 'mp3') {
191
- const validRates = [
192
- 8000, 16000, 24000, 32000, 40000, 48000, 64000, 80000,
193
- 96000, 112000, 128000, 160000, 192000, 224000, 256000, 320000,
194
- ];
195
- finalBitrate = validRates.reduce((prev, curr) => Math.abs(curr - finalBitrate) < Math.abs(prev - finalBitrate) ? curr : prev);
196
- }
197
- return Math.round(finalBitrate / 1000) * 1000;
198
- }
199
- }
200
- /**
201
- * Represents a very low media quality.
202
- * @group Encoding
203
- * @public
204
- */
205
- export const QUALITY_VERY_LOW = /* #__PURE__ */ new Quality(0.3);
206
- /**
207
- * Represents a low media quality.
208
- * @group Encoding
209
- * @public
210
- */
211
- export const QUALITY_LOW = /* #__PURE__ */ new Quality(0.6);
212
- /**
213
- * Represents a medium media quality.
214
- * @group Encoding
215
- * @public
216
- */
217
- export const QUALITY_MEDIUM = /* #__PURE__ */ new Quality(1);
218
- /**
219
- * Represents a high media quality.
220
- * @group Encoding
221
- * @public
222
- */
223
- export const QUALITY_HIGH = /* #__PURE__ */ new Quality(2);
224
- /**
225
- * Represents a very high media quality.
226
- * @group Encoding
227
- * @public
228
- */
229
- export const QUALITY_VERY_HIGH = /* #__PURE__ */ new Quality(4);
230
- /**
231
- * Checks if the browser is able to encode the given codec.
232
- * @group Encoding
233
- * @public
234
- */
235
- export const canEncode = (codec) => {
236
- if (VIDEO_CODECS.includes(codec)) {
237
- return canEncodeVideo(codec);
238
- }
239
- else if (AUDIO_CODECS.includes(codec)) {
240
- return canEncodeAudio(codec);
241
- }
242
- else if (SUBTITLE_CODECS.includes(codec)) {
243
- return canEncodeSubtitles(codec);
244
- }
245
- throw new TypeError(`Unknown codec '${codec}'.`);
246
- };
247
- /**
248
- * Checks if the browser is able to encode the given video codec with the given parameters.
249
- * @group Encoding
250
- * @public
251
- */
252
- export const canEncodeVideo = async (codec, options = {}) => {
253
- const { width = 1280, height = 720, bitrate = 1e6, ...restOptions } = options;
254
- if (!VIDEO_CODECS.includes(codec)) {
255
- return false;
256
- }
257
- if (!Number.isInteger(width) || width <= 0) {
258
- throw new TypeError('width must be a positive integer.');
259
- }
260
- if (!Number.isInteger(height) || height <= 0) {
261
- throw new TypeError('height must be a positive integer.');
262
- }
263
- if (!(bitrate instanceof Quality) && (!Number.isInteger(bitrate) || bitrate <= 0)) {
264
- throw new TypeError('bitrate must be a positive integer or a quality.');
265
- }
266
- validateVideoEncodingAdditionalOptions(codec, restOptions);
267
- let encoderConfig = null;
268
- if (customVideoEncoders.length > 0) {
269
- encoderConfig ??= buildVideoEncoderConfig({
270
- codec,
271
- width,
272
- height,
273
- bitrate,
274
- framerate: undefined,
275
- ...restOptions,
276
- });
277
- if (customVideoEncoders.some(x => x.supports(codec, encoderConfig))) {
278
- // There's a custom encoder
279
- return true;
280
- }
281
- }
282
- if (typeof VideoEncoder === 'undefined') {
283
- return false;
284
- }
285
- const hasOddDimension = width % 2 === 1 || height % 2 === 1;
286
- if (hasOddDimension
287
- && (codec === 'avc' || codec === 'hevc')) {
288
- // Disallow odd dimensions for certain codecs
289
- return false;
290
- }
291
- encoderConfig ??= buildVideoEncoderConfig({
292
- codec,
293
- width,
294
- height,
295
- bitrate,
296
- framerate: undefined,
297
- ...restOptions,
298
- alpha: 'discard', // Since we handle alpha ourselves
299
- });
300
- const support = await VideoEncoder.isConfigSupported(encoderConfig);
301
- return support.supported === true;
302
- };
303
- /**
304
- * Checks if the browser is able to encode the given audio codec with the given parameters.
305
- * @group Encoding
306
- * @public
307
- */
308
- export const canEncodeAudio = async (codec, options = {}) => {
309
- const { numberOfChannels = 2, sampleRate = 48000, bitrate = 128e3, ...restOptions } = options;
310
- if (!AUDIO_CODECS.includes(codec)) {
311
- return false;
312
- }
313
- if (!Number.isInteger(numberOfChannels) || numberOfChannels <= 0) {
314
- throw new TypeError('numberOfChannels must be a positive integer.');
315
- }
316
- if (!Number.isInteger(sampleRate) || sampleRate <= 0) {
317
- throw new TypeError('sampleRate must be a positive integer.');
318
- }
319
- if (!(bitrate instanceof Quality) && (!Number.isInteger(bitrate) || bitrate <= 0)) {
320
- throw new TypeError('bitrate must be a positive integer.');
321
- }
322
- validateAudioEncodingAdditionalOptions(codec, restOptions);
323
- let encoderConfig = null;
324
- if (customAudioEncoders.length > 0) {
325
- encoderConfig ??= buildAudioEncoderConfig({
326
- codec,
327
- numberOfChannels,
328
- sampleRate,
329
- bitrate,
330
- ...restOptions,
331
- });
332
- if (customAudioEncoders.some(x => x.supports(codec, encoderConfig))) {
333
- // There's a custom encoder
334
- return true;
335
- }
336
- }
337
- if (PCM_AUDIO_CODECS.includes(codec)) {
338
- return true; // Because we encode these ourselves
339
- }
340
- if (typeof AudioEncoder === 'undefined') {
341
- return false;
342
- }
343
- encoderConfig ??= buildAudioEncoderConfig({
344
- codec,
345
- numberOfChannels,
346
- sampleRate,
347
- bitrate,
348
- ...restOptions,
349
- });
350
- const support = await AudioEncoder.isConfigSupported(encoderConfig);
351
- return support.supported === true;
352
- };
353
- /**
354
- * Checks if the browser is able to encode the given subtitle codec.
355
- * @group Encoding
356
- * @public
357
- */
358
- export const canEncodeSubtitles = async (codec) => {
359
- if (!SUBTITLE_CODECS.includes(codec)) {
360
- return false;
361
- }
362
- return true;
363
- };
364
- /**
365
- * Returns the list of all media codecs that can be encoded by the browser.
366
- * @group Encoding
367
- * @public
368
- */
369
- export const getEncodableCodecs = async () => {
370
- const [videoCodecs, audioCodecs, subtitleCodecs] = await Promise.all([
371
- getEncodableVideoCodecs(),
372
- getEncodableAudioCodecs(),
373
- getEncodableSubtitleCodecs(),
374
- ]);
375
- return [...videoCodecs, ...audioCodecs, ...subtitleCodecs];
376
- };
377
- /**
378
- * Returns the list of all video codecs that can be encoded by the browser.
379
- * @group Encoding
380
- * @public
381
- */
382
- export const getEncodableVideoCodecs = async (checkedCodecs = VIDEO_CODECS, options) => {
383
- const bools = await Promise.all(checkedCodecs.map(codec => canEncodeVideo(codec, options)));
384
- return checkedCodecs.filter((_, i) => bools[i]);
385
- };
386
- /**
387
- * Returns the list of all audio codecs that can be encoded by the browser.
388
- * @group Encoding
389
- * @public
390
- */
391
- export const getEncodableAudioCodecs = async (checkedCodecs = AUDIO_CODECS, options) => {
392
- const bools = await Promise.all(checkedCodecs.map(codec => canEncodeAudio(codec, options)));
393
- return checkedCodecs.filter((_, i) => bools[i]);
394
- };
395
- /**
396
- * Returns the list of all subtitle codecs that can be encoded by the browser.
397
- * @group Encoding
398
- * @public
399
- */
400
- export const getEncodableSubtitleCodecs = async (checkedCodecs = SUBTITLE_CODECS) => {
401
- const bools = await Promise.all(checkedCodecs.map(canEncodeSubtitles));
402
- return checkedCodecs.filter((_, i) => bools[i]);
403
- };
404
- /**
405
- * Returns the first video codec from the given list that can be encoded by the browser.
406
- * @group Encoding
407
- * @public
408
- */
409
- export const getFirstEncodableVideoCodec = async (checkedCodecs, options) => {
410
- for (const codec of checkedCodecs) {
411
- if (await canEncodeVideo(codec, options)) {
412
- return codec;
413
- }
414
- }
415
- return null;
416
- };
417
- /**
418
- * Returns the first audio codec from the given list that can be encoded by the browser.
419
- * @group Encoding
420
- * @public
421
- */
422
- export const getFirstEncodableAudioCodec = async (checkedCodecs, options) => {
423
- for (const codec of checkedCodecs) {
424
- if (await canEncodeAudio(codec, options)) {
425
- return codec;
426
- }
427
- }
428
- return null;
429
- };
430
- /**
431
- * Returns the first subtitle codec from the given list that can be encoded by the browser.
432
- * @group Encoding
433
- * @public
434
- */
435
- export const getFirstEncodableSubtitleCodec = async (checkedCodecs) => {
436
- for (const codec of checkedCodecs) {
437
- if (await canEncodeSubtitles(codec)) {
438
- return codec;
439
- }
440
- }
441
- return null;
442
- };