@xrift/world-components 0.21.12 → 0.22.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.
- package/dist/components/LiveVideoPlayer/components/ControlPanel/LiveIndicator.d.ts +8 -0
- package/dist/components/LiveVideoPlayer/components/ControlPanel/LiveIndicator.d.ts.map +1 -0
- package/dist/components/LiveVideoPlayer/{LiveIndicator.js → components/ControlPanel/LiveIndicator.js} +5 -5
- package/dist/components/LiveVideoPlayer/components/ControlPanel/LiveIndicator.js.map +1 -0
- package/dist/components/LiveVideoPlayer/components/ControlPanel/index.d.ts +16 -0
- package/dist/components/LiveVideoPlayer/components/ControlPanel/index.d.ts.map +1 -0
- package/dist/components/LiveVideoPlayer/components/ControlPanel/index.js +28 -0
- package/dist/components/LiveVideoPlayer/components/ControlPanel/index.js.map +1 -0
- package/dist/components/LiveVideoPlayer/components/LiveVideoTexture.d.ts +14 -0
- package/dist/components/LiveVideoPlayer/components/LiveVideoTexture.d.ts.map +1 -0
- package/dist/components/LiveVideoPlayer/components/LiveVideoTexture.js +18 -0
- package/dist/components/LiveVideoPlayer/components/LiveVideoTexture.js.map +1 -0
- package/dist/components/LiveVideoPlayer/components/index.d.ts +3 -0
- package/dist/components/LiveVideoPlayer/components/index.d.ts.map +1 -0
- package/dist/components/LiveVideoPlayer/components/index.js +3 -0
- package/dist/components/LiveVideoPlayer/components/index.js.map +1 -0
- package/dist/components/LiveVideoPlayer/hooks/useHlsVideo/__tests__/utils.test.d.ts +2 -0
- package/dist/components/LiveVideoPlayer/hooks/useHlsVideo/__tests__/utils.test.d.ts.map +1 -0
- package/dist/components/LiveVideoPlayer/hooks/useHlsVideo/__tests__/utils.test.js +196 -0
- package/dist/components/LiveVideoPlayer/hooks/useHlsVideo/__tests__/utils.test.js.map +1 -0
- package/dist/components/LiveVideoPlayer/hooks/useHlsVideo/classes/HlsJsPlayer.d.ts +17 -0
- package/dist/components/LiveVideoPlayer/hooks/useHlsVideo/classes/HlsJsPlayer.d.ts.map +1 -0
- package/dist/components/LiveVideoPlayer/hooks/useHlsVideo/classes/HlsJsPlayer.js +73 -0
- package/dist/components/LiveVideoPlayer/hooks/useHlsVideo/classes/HlsJsPlayer.js.map +1 -0
- package/dist/components/LiveVideoPlayer/hooks/useHlsVideo/classes/NativeHlsPlayer.d.ts +14 -0
- package/dist/components/LiveVideoPlayer/hooks/useHlsVideo/classes/NativeHlsPlayer.d.ts.map +1 -0
- package/dist/components/LiveVideoPlayer/hooks/useHlsVideo/classes/NativeHlsPlayer.js +46 -0
- package/dist/components/LiveVideoPlayer/hooks/useHlsVideo/classes/NativeHlsPlayer.js.map +1 -0
- package/dist/components/LiveVideoPlayer/hooks/useHlsVideo/classes/RecoveryTracker.d.ts +21 -0
- package/dist/components/LiveVideoPlayer/hooks/useHlsVideo/classes/RecoveryTracker.d.ts.map +1 -0
- package/dist/components/LiveVideoPlayer/hooks/useHlsVideo/classes/RecoveryTracker.js +54 -0
- package/dist/components/LiveVideoPlayer/hooks/useHlsVideo/classes/RecoveryTracker.js.map +1 -0
- package/dist/components/LiveVideoPlayer/hooks/useHlsVideo/classes/__tests__/HlsJsPlayer.test.d.ts +2 -0
- package/dist/components/LiveVideoPlayer/hooks/useHlsVideo/classes/__tests__/HlsJsPlayer.test.d.ts.map +1 -0
- package/dist/components/LiveVideoPlayer/hooks/useHlsVideo/classes/__tests__/HlsJsPlayer.test.js +146 -0
- package/dist/components/LiveVideoPlayer/hooks/useHlsVideo/classes/__tests__/HlsJsPlayer.test.js.map +1 -0
- package/dist/components/LiveVideoPlayer/hooks/useHlsVideo/classes/__tests__/NativeHlsPlayer.test.d.ts +2 -0
- package/dist/components/LiveVideoPlayer/hooks/useHlsVideo/classes/__tests__/NativeHlsPlayer.test.d.ts.map +1 -0
- package/dist/components/LiveVideoPlayer/hooks/useHlsVideo/classes/__tests__/NativeHlsPlayer.test.js +94 -0
- package/dist/components/LiveVideoPlayer/hooks/useHlsVideo/classes/__tests__/NativeHlsPlayer.test.js.map +1 -0
- package/dist/components/LiveVideoPlayer/hooks/useHlsVideo/classes/__tests__/RecoveryTracker.test.d.ts +2 -0
- package/dist/components/LiveVideoPlayer/hooks/useHlsVideo/classes/__tests__/RecoveryTracker.test.d.ts.map +1 -0
- package/dist/components/LiveVideoPlayer/hooks/useHlsVideo/classes/__tests__/RecoveryTracker.test.js +74 -0
- package/dist/components/LiveVideoPlayer/hooks/useHlsVideo/classes/__tests__/RecoveryTracker.test.js.map +1 -0
- package/dist/components/LiveVideoPlayer/hooks/useHlsVideo/index.d.ts +31 -0
- package/dist/components/LiveVideoPlayer/hooks/useHlsVideo/index.d.ts.map +1 -0
- package/dist/components/LiveVideoPlayer/hooks/useHlsVideo/index.js +108 -0
- package/dist/components/LiveVideoPlayer/hooks/useHlsVideo/index.js.map +1 -0
- package/dist/components/LiveVideoPlayer/hooks/useHlsVideo/types.d.ts +34 -0
- package/dist/components/LiveVideoPlayer/hooks/useHlsVideo/types.d.ts.map +1 -0
- package/dist/components/LiveVideoPlayer/hooks/useHlsVideo/types.js +2 -0
- package/dist/components/LiveVideoPlayer/hooks/useHlsVideo/types.js.map +1 -0
- package/dist/components/LiveVideoPlayer/hooks/useHlsVideo/utils.d.ts +15 -0
- package/dist/components/LiveVideoPlayer/hooks/useHlsVideo/utils.d.ts.map +1 -0
- package/dist/components/LiveVideoPlayer/hooks/useHlsVideo/utils.js +62 -0
- package/dist/components/LiveVideoPlayer/hooks/useHlsVideo/utils.js.map +1 -0
- package/dist/components/LiveVideoPlayer/hooks/useLiveVideoPlayer/index.d.ts +25 -0
- package/dist/components/LiveVideoPlayer/hooks/useLiveVideoPlayer/index.d.ts.map +1 -0
- package/dist/components/LiveVideoPlayer/hooks/useLiveVideoPlayer/index.js +97 -0
- package/dist/components/LiveVideoPlayer/hooks/useLiveVideoPlayer/index.js.map +1 -0
- package/dist/components/LiveVideoPlayer/hooks/useLiveVideoPlayer.d.ts +25 -0
- package/dist/components/LiveVideoPlayer/hooks/useLiveVideoPlayer.d.ts.map +1 -0
- package/dist/components/LiveVideoPlayer/hooks/useLiveVideoPlayer.js +97 -0
- package/dist/components/LiveVideoPlayer/hooks/useLiveVideoPlayer.js.map +1 -0
- package/dist/components/LiveVideoPlayer/index.d.ts +20 -3
- package/dist/components/LiveVideoPlayer/index.d.ts.map +1 -1
- package/dist/components/LiveVideoPlayer/index.js +16 -133
- package/dist/components/LiveVideoPlayer/index.js.map +1 -1
- package/dist/components/LiveVideoPlayer/types.d.ts +6 -66
- package/dist/components/LiveVideoPlayer/types.d.ts.map +1 -1
- package/dist/components/Video180Sphere/EyeView.d.ts +14 -0
- package/dist/components/Video180Sphere/EyeView.d.ts.map +1 -0
- package/dist/components/Video180Sphere/EyeView.js +77 -0
- package/dist/components/Video180Sphere/EyeView.js.map +1 -0
- package/dist/components/Video180Sphere/index.d.ts +19 -0
- package/dist/components/Video180Sphere/index.d.ts.map +1 -0
- package/dist/components/Video180Sphere/index.js +90 -0
- package/dist/components/Video180Sphere/index.js.map +1 -0
- package/dist/components/Video180Sphere/types.d.ts +35 -0
- package/dist/components/Video180Sphere/types.d.ts.map +1 -0
- package/dist/components/Video180Sphere/types.js +2 -0
- package/dist/components/Video180Sphere/types.js.map +1 -0
- package/dist/components/VideoPlayer/ControlPanel.d.ts +1 -1
- package/dist/components/VideoPlayer/ControlPanel.d.ts.map +1 -1
- package/dist/components/VideoPlayer/ControlPanel.js +20 -9
- package/dist/components/VideoPlayer/ControlPanel.js.map +1 -1
- package/dist/components/VideoPlayer/ProgressBar.d.ts.map +1 -1
- package/dist/components/VideoPlayer/ProgressBar.js +2 -1
- package/dist/components/VideoPlayer/ProgressBar.js.map +1 -1
- package/dist/components/VideoPlayer/__tests__/utils.test.d.ts +2 -0
- package/dist/components/VideoPlayer/__tests__/utils.test.d.ts.map +1 -0
- package/dist/components/VideoPlayer/__tests__/utils.test.js +23 -0
- package/dist/components/VideoPlayer/__tests__/utils.test.js.map +1 -0
- package/dist/components/VideoPlayer/components/ControlPanel.d.ts +18 -0
- package/dist/components/VideoPlayer/components/ControlPanel.d.ts.map +1 -0
- package/dist/components/VideoPlayer/components/ControlPanel.js +34 -0
- package/dist/components/VideoPlayer/components/ControlPanel.js.map +1 -0
- package/dist/components/VideoPlayer/components/ProgressBar.d.ts +12 -0
- package/dist/components/VideoPlayer/components/ProgressBar.d.ts.map +1 -0
- package/dist/components/VideoPlayer/components/ProgressBar.js +21 -0
- package/dist/components/VideoPlayer/components/ProgressBar.js.map +1 -0
- package/dist/components/VideoPlayer/components/index.d.ts +3 -0
- package/dist/components/VideoPlayer/components/index.d.ts.map +1 -0
- package/dist/components/VideoPlayer/components/index.js +3 -0
- package/dist/components/VideoPlayer/components/index.js.map +1 -0
- package/dist/components/VideoPlayer/index.d.ts +20 -3
- package/dist/components/VideoPlayer/index.d.ts.map +1 -1
- package/dist/components/VideoPlayer/index.js +29 -89
- package/dist/components/VideoPlayer/index.js.map +1 -1
- package/dist/components/VideoPlayer/types.d.ts +1 -28
- package/dist/components/VideoPlayer/types.d.ts.map +1 -1
- package/dist/components/VideoPlayer/utils.d.ts +0 -27
- package/dist/components/VideoPlayer/utils.d.ts.map +1 -1
- package/dist/components/VideoPlayer/utils.js +0 -28
- package/dist/components/VideoPlayer/utils.js.map +1 -1
- package/dist/components/VideoPlayer/utils.test.js +1 -87
- package/dist/components/VideoPlayer/utils.test.js.map +1 -1
- package/dist/components/commons/ErrorBoundary/index.d.ts +18 -0
- package/dist/components/commons/ErrorBoundary/index.d.ts.map +1 -0
- package/dist/components/commons/ErrorBoundary/index.js +22 -0
- package/dist/components/commons/ErrorBoundary/index.js.map +1 -0
- package/dist/components/commons/IconButton/index.d.ts +23 -0
- package/dist/components/commons/IconButton/index.d.ts.map +1 -0
- package/dist/components/commons/IconButton/index.js +12 -0
- package/dist/components/commons/IconButton/index.js.map +1 -0
- package/dist/components/commons/PlaceholderScreen/index.d.ts +9 -0
- package/dist/components/commons/PlaceholderScreen/index.d.ts.map +1 -0
- package/dist/components/commons/PlaceholderScreen/index.js +6 -0
- package/dist/components/commons/PlaceholderScreen/index.js.map +1 -0
- package/dist/components/commons/VideoMesh/index.d.ts +14 -0
- package/dist/components/commons/VideoMesh/index.d.ts.map +1 -0
- package/dist/components/commons/VideoMesh/index.js +91 -0
- package/dist/components/commons/VideoMesh/index.js.map +1 -0
- package/dist/components/commons/VolumeControl/index.d.ts +9 -0
- package/dist/components/commons/VolumeControl/index.d.ts.map +1 -0
- package/dist/components/{LiveVideoPlayer/VolumeControl.js → commons/VolumeControl/index.js} +6 -3
- package/dist/components/commons/VolumeControl/index.js.map +1 -0
- package/dist/components/commons/utils.d.ts +24 -0
- package/dist/components/commons/utils.d.ts.map +1 -0
- package/dist/components/commons/utils.js +23 -0
- package/dist/components/commons/utils.js.map +1 -0
- package/dist/components/commons/utils.test.d.ts +2 -0
- package/dist/components/commons/utils.test.d.ts.map +1 -0
- package/dist/components/commons/utils.test.js +79 -0
- package/dist/components/commons/utils.test.js.map +1 -0
- package/dist/hooks/useSyncState.d.ts +8 -0
- package/dist/hooks/useSyncState.d.ts.map +1 -0
- package/dist/hooks/useSyncState.js +16 -0
- package/dist/hooks/useSyncState.js.map +1 -0
- package/dist/hooks/useVideoElement.d.ts +30 -0
- package/dist/hooks/useVideoElement.d.ts.map +1 -0
- package/dist/hooks/useVideoElement.js +182 -0
- package/dist/hooks/useVideoElement.js.map +1 -0
- package/dist/index.d.ts +3 -2
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +3 -2
- package/dist/index.js.map +1 -1
- package/package.json +2 -1
- package/dist/components/LiveVideoPlayer/ControlPanel.d.ts +0 -3
- package/dist/components/LiveVideoPlayer/ControlPanel.d.ts.map +0 -1
- package/dist/components/LiveVideoPlayer/ControlPanel.js +0 -17
- package/dist/components/LiveVideoPlayer/ControlPanel.js.map +0 -1
- package/dist/components/LiveVideoPlayer/LiveIndicator.d.ts +0 -3
- package/dist/components/LiveVideoPlayer/LiveIndicator.d.ts.map +0 -1
- package/dist/components/LiveVideoPlayer/LiveIndicator.js.map +0 -1
- package/dist/components/LiveVideoPlayer/PlayPauseButton.d.ts +0 -3
- package/dist/components/LiveVideoPlayer/PlayPauseButton.d.ts.map +0 -1
- package/dist/components/LiveVideoPlayer/PlayPauseButton.js +0 -9
- package/dist/components/LiveVideoPlayer/PlayPauseButton.js.map +0 -1
- package/dist/components/LiveVideoPlayer/ReloadButton.d.ts +0 -3
- package/dist/components/LiveVideoPlayer/ReloadButton.d.ts.map +0 -1
- package/dist/components/LiveVideoPlayer/ReloadButton.js +0 -9
- package/dist/components/LiveVideoPlayer/ReloadButton.js.map +0 -1
- package/dist/components/LiveVideoPlayer/UrlInputButton.d.ts +0 -3
- package/dist/components/LiveVideoPlayer/UrlInputButton.d.ts.map +0 -1
- package/dist/components/LiveVideoPlayer/UrlInputButton.js +0 -23
- package/dist/components/LiveVideoPlayer/UrlInputButton.js.map +0 -1
- package/dist/components/LiveVideoPlayer/VolumeControl.d.ts +0 -3
- package/dist/components/LiveVideoPlayer/VolumeControl.d.ts.map +0 -1
- package/dist/components/LiveVideoPlayer/VolumeControl.js.map +0 -1
- package/dist/components/VideoPlayer/PlayPauseButton.d.ts +0 -3
- package/dist/components/VideoPlayer/PlayPauseButton.d.ts.map +0 -1
- package/dist/components/VideoPlayer/PlayPauseButton.js +0 -9
- package/dist/components/VideoPlayer/PlayPauseButton.js.map +0 -1
- package/dist/components/VideoPlayer/StopButton.d.ts +0 -3
- package/dist/components/VideoPlayer/StopButton.d.ts.map +0 -1
- package/dist/components/VideoPlayer/StopButton.js +0 -9
- package/dist/components/VideoPlayer/StopButton.js.map +0 -1
- package/dist/components/VideoPlayer/UrlInputButton.d.ts +0 -3
- package/dist/components/VideoPlayer/UrlInputButton.d.ts.map +0 -1
- package/dist/components/VideoPlayer/UrlInputButton.js +0 -23
- package/dist/components/VideoPlayer/UrlInputButton.js.map +0 -1
- package/dist/components/VideoPlayer/VolumeControl.d.ts +0 -3
- package/dist/components/VideoPlayer/VolumeControl.d.ts.map +0 -1
- package/dist/components/VideoPlayer/VolumeControl.js +0 -26
- package/dist/components/VideoPlayer/VolumeControl.js.map +0 -1
|
@@ -0,0 +1,54 @@
|
|
|
1
|
+
/** リカバリのスロットリング間隔(ms) */
|
|
2
|
+
const RECOVERY_THROTTLE_MS = 5000;
|
|
3
|
+
/** リカバリ失敗とみなすまでの最大試行回数 */
|
|
4
|
+
const MAX_RECOVERY_ATTEMPTS = 3;
|
|
5
|
+
/**
|
|
6
|
+
* エラーリカバリの試行を追跡するクラス
|
|
7
|
+
* スロットリングと試行回数制限を管理
|
|
8
|
+
*/
|
|
9
|
+
export class RecoveryTracker {
|
|
10
|
+
constructor() {
|
|
11
|
+
this.lastRecoveryTime = 0;
|
|
12
|
+
this.recoveryAttempts = 0;
|
|
13
|
+
this.errorReported = false;
|
|
14
|
+
}
|
|
15
|
+
/** 状態をリセット */
|
|
16
|
+
reset() {
|
|
17
|
+
this.lastRecoveryTime = 0;
|
|
18
|
+
this.recoveryAttempts = 0;
|
|
19
|
+
this.errorReported = false;
|
|
20
|
+
}
|
|
21
|
+
/** エラーが既に報告済みかどうか */
|
|
22
|
+
get isErrorReported() {
|
|
23
|
+
return this.errorReported;
|
|
24
|
+
}
|
|
25
|
+
/** エラーを報告済みとしてマーク */
|
|
26
|
+
markErrorReported() {
|
|
27
|
+
this.errorReported = true;
|
|
28
|
+
}
|
|
29
|
+
/**
|
|
30
|
+
* リカバリを試行可能かどうかを判定し、カウンターを更新
|
|
31
|
+
* @returns リカバリを試行すべきかどうか
|
|
32
|
+
*/
|
|
33
|
+
shouldAttemptRecovery() {
|
|
34
|
+
const now = Date.now();
|
|
35
|
+
const timeSinceLastRecovery = now - this.lastRecoveryTime;
|
|
36
|
+
// スロットリング: 5秒以内の連続エラーはカウント増加
|
|
37
|
+
if (timeSinceLastRecovery < RECOVERY_THROTTLE_MS) {
|
|
38
|
+
this.recoveryAttempts++;
|
|
39
|
+
}
|
|
40
|
+
else {
|
|
41
|
+
// 5秒以上経過していればカウントリセット
|
|
42
|
+
this.recoveryAttempts = 1;
|
|
43
|
+
}
|
|
44
|
+
this.lastRecoveryTime = now;
|
|
45
|
+
// 最大試行回数を超えた場合はリカバリ失敗
|
|
46
|
+
if (this.recoveryAttempts > MAX_RECOVERY_ATTEMPTS) {
|
|
47
|
+
console.error(`[RecoveryTracker] Recovery failed after ${MAX_RECOVERY_ATTEMPTS} attempts`);
|
|
48
|
+
return false;
|
|
49
|
+
}
|
|
50
|
+
console.log(`[RecoveryTracker] Attempting recovery (attempt ${this.recoveryAttempts}/${MAX_RECOVERY_ATTEMPTS})`);
|
|
51
|
+
return true;
|
|
52
|
+
}
|
|
53
|
+
}
|
|
54
|
+
//# sourceMappingURL=RecoveryTracker.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"RecoveryTracker.js","sourceRoot":"","sources":["../../../../../../src/components/LiveVideoPlayer/hooks/useHlsVideo/classes/RecoveryTracker.ts"],"names":[],"mappings":"AAAA,yBAAyB;AACzB,MAAM,oBAAoB,GAAG,IAAI,CAAA;AACjC,0BAA0B;AAC1B,MAAM,qBAAqB,GAAG,CAAC,CAAA;AAE/B;;;GAGG;AACH,MAAM,OAAO,eAAe;IAA5B;QACU,qBAAgB,GAAG,CAAC,CAAA;QACpB,qBAAgB,GAAG,CAAC,CAAA;QACpB,kBAAa,GAAG,KAAK,CAAA;IAkD/B,CAAC;IAhDC,cAAc;IACd,KAAK;QACH,IAAI,CAAC,gBAAgB,GAAG,CAAC,CAAA;QACzB,IAAI,CAAC,gBAAgB,GAAG,CAAC,CAAA;QACzB,IAAI,CAAC,aAAa,GAAG,KAAK,CAAA;IAC5B,CAAC;IAED,qBAAqB;IACrB,IAAI,eAAe;QACjB,OAAO,IAAI,CAAC,aAAa,CAAA;IAC3B,CAAC;IAED,qBAAqB;IACrB,iBAAiB;QACf,IAAI,CAAC,aAAa,GAAG,IAAI,CAAA;IAC3B,CAAC;IAED;;;OAGG;IACH,qBAAqB;QACnB,MAAM,GAAG,GAAG,IAAI,CAAC,GAAG,EAAE,CAAA;QACtB,MAAM,qBAAqB,GAAG,GAAG,GAAG,IAAI,CAAC,gBAAgB,CAAA;QAEzD,6BAA6B;QAC7B,IAAI,qBAAqB,GAAG,oBAAoB,EAAE,CAAC;YACjD,IAAI,CAAC,gBAAgB,EAAE,CAAA;QACzB,CAAC;aAAM,CAAC;YACN,sBAAsB;YACtB,IAAI,CAAC,gBAAgB,GAAG,CAAC,CAAA;QAC3B,CAAC;QAED,IAAI,CAAC,gBAAgB,GAAG,GAAG,CAAA;QAE3B,sBAAsB;QACtB,IAAI,IAAI,CAAC,gBAAgB,GAAG,qBAAqB,EAAE,CAAC;YAClD,OAAO,CAAC,KAAK,CACX,2CAA2C,qBAAqB,WAAW,CAC5E,CAAA;YACD,OAAO,KAAK,CAAA;QACd,CAAC;QAED,OAAO,CAAC,GAAG,CACT,kDAAkD,IAAI,CAAC,gBAAgB,IAAI,qBAAqB,GAAG,CACpG,CAAA;QACD,OAAO,IAAI,CAAA;IACb,CAAC;CACF"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"HlsJsPlayer.test.d.ts","sourceRoot":"","sources":["../../../../../../../src/components/LiveVideoPlayer/hooks/useHlsVideo/classes/__tests__/HlsJsPlayer.test.ts"],"names":[],"mappings":""}
|
package/dist/components/LiveVideoPlayer/hooks/useHlsVideo/classes/__tests__/HlsJsPlayer.test.js
ADDED
|
@@ -0,0 +1,146 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @vitest-environment jsdom
|
|
3
|
+
*/
|
|
4
|
+
import { describe, it, expect, vi, beforeEach } from 'vitest';
|
|
5
|
+
import { HlsJsPlayer } from '../HlsJsPlayer';
|
|
6
|
+
import { RecoveryTracker } from '../RecoveryTracker';
|
|
7
|
+
// hls.js のモック
|
|
8
|
+
function createMockHls() {
|
|
9
|
+
const eventHandlers = {};
|
|
10
|
+
const mockHlsInstance = {
|
|
11
|
+
on: vi.fn((event, handler) => {
|
|
12
|
+
eventHandlers[event] = handler;
|
|
13
|
+
}),
|
|
14
|
+
loadSource: vi.fn(),
|
|
15
|
+
attachMedia: vi.fn(),
|
|
16
|
+
destroy: vi.fn(),
|
|
17
|
+
recoverMediaError: vi.fn(),
|
|
18
|
+
startLoad: vi.fn(),
|
|
19
|
+
// イベントをトリガーするヘルパー
|
|
20
|
+
_trigger: (event, ...args) => {
|
|
21
|
+
eventHandlers[event]?.(...args);
|
|
22
|
+
},
|
|
23
|
+
};
|
|
24
|
+
// コンストラクタ関数として使えるクラスを作成
|
|
25
|
+
class MockHlsClass {
|
|
26
|
+
constructor(_config) {
|
|
27
|
+
this.on = mockHlsInstance.on;
|
|
28
|
+
this.loadSource = mockHlsInstance.loadSource;
|
|
29
|
+
this.attachMedia = mockHlsInstance.attachMedia;
|
|
30
|
+
this.destroy = mockHlsInstance.destroy;
|
|
31
|
+
this.recoverMediaError = mockHlsInstance.recoverMediaError;
|
|
32
|
+
this.startLoad = mockHlsInstance.startLoad;
|
|
33
|
+
// コンストラクタの引数は無視
|
|
34
|
+
}
|
|
35
|
+
}
|
|
36
|
+
MockHlsClass.Events = {
|
|
37
|
+
ERROR: 'hlsError',
|
|
38
|
+
FRAG_BUFFERED: 'hlsFragBuffered',
|
|
39
|
+
MANIFEST_PARSED: 'hlsManifestParsed',
|
|
40
|
+
};
|
|
41
|
+
MockHlsClass.ErrorTypes = {
|
|
42
|
+
MEDIA_ERROR: 'mediaError',
|
|
43
|
+
NETWORK_ERROR: 'networkError',
|
|
44
|
+
OTHER_ERROR: 'otherError',
|
|
45
|
+
};
|
|
46
|
+
return { MockHlsClass: MockHlsClass, mockHlsInstance };
|
|
47
|
+
}
|
|
48
|
+
describe('HlsJsPlayer', () => {
|
|
49
|
+
let video;
|
|
50
|
+
let tracker;
|
|
51
|
+
beforeEach(() => {
|
|
52
|
+
video = document.createElement('video');
|
|
53
|
+
tracker = new RecoveryTracker();
|
|
54
|
+
});
|
|
55
|
+
it('初期化時に attachMedia が呼ばれる', () => {
|
|
56
|
+
const { MockHlsClass, mockHlsInstance } = createMockHls();
|
|
57
|
+
new HlsJsPlayer(MockHlsClass, {
|
|
58
|
+
video,
|
|
59
|
+
tracker,
|
|
60
|
+
callbacks: {},
|
|
61
|
+
});
|
|
62
|
+
expect(mockHlsInstance.attachMedia).toHaveBeenCalledWith(video);
|
|
63
|
+
});
|
|
64
|
+
it('load で loadSource が呼ばれる', () => {
|
|
65
|
+
const { MockHlsClass, mockHlsInstance } = createMockHls();
|
|
66
|
+
const player = new HlsJsPlayer(MockHlsClass, {
|
|
67
|
+
video,
|
|
68
|
+
tracker,
|
|
69
|
+
callbacks: {},
|
|
70
|
+
});
|
|
71
|
+
player.load('https://example.com/video.m3u8');
|
|
72
|
+
expect(mockHlsInstance.loadSource).toHaveBeenCalledWith('https://example.com/video.m3u8');
|
|
73
|
+
});
|
|
74
|
+
it('destroy で hls.destroy が呼ばれる', () => {
|
|
75
|
+
const { MockHlsClass, mockHlsInstance } = createMockHls();
|
|
76
|
+
const player = new HlsJsPlayer(MockHlsClass, {
|
|
77
|
+
video,
|
|
78
|
+
tracker,
|
|
79
|
+
callbacks: {},
|
|
80
|
+
});
|
|
81
|
+
player.destroy();
|
|
82
|
+
expect(mockHlsInstance.destroy).toHaveBeenCalled();
|
|
83
|
+
});
|
|
84
|
+
it('MANIFEST_PARSED イベントで onManifestParsed コールバックが呼ばれる', () => {
|
|
85
|
+
const { MockHlsClass, mockHlsInstance } = createMockHls();
|
|
86
|
+
const onManifestParsed = vi.fn();
|
|
87
|
+
new HlsJsPlayer(MockHlsClass, {
|
|
88
|
+
video,
|
|
89
|
+
tracker,
|
|
90
|
+
callbacks: { onManifestParsed },
|
|
91
|
+
});
|
|
92
|
+
mockHlsInstance._trigger('hlsManifestParsed');
|
|
93
|
+
expect(onManifestParsed).toHaveBeenCalled();
|
|
94
|
+
});
|
|
95
|
+
it('FRAG_BUFFERED イベントで onBufferingChange(false) が呼ばれる', () => {
|
|
96
|
+
const { MockHlsClass, mockHlsInstance } = createMockHls();
|
|
97
|
+
const onBufferingChange = vi.fn();
|
|
98
|
+
new HlsJsPlayer(MockHlsClass, {
|
|
99
|
+
video,
|
|
100
|
+
tracker,
|
|
101
|
+
callbacks: { onBufferingChange },
|
|
102
|
+
});
|
|
103
|
+
mockHlsInstance._trigger('hlsFragBuffered');
|
|
104
|
+
expect(onBufferingChange).toHaveBeenCalledWith(false);
|
|
105
|
+
});
|
|
106
|
+
it('非致命的エラーは無視される', () => {
|
|
107
|
+
const { MockHlsClass, mockHlsInstance } = createMockHls();
|
|
108
|
+
const onError = vi.fn();
|
|
109
|
+
new HlsJsPlayer(MockHlsClass, {
|
|
110
|
+
video,
|
|
111
|
+
tracker,
|
|
112
|
+
callbacks: { onError },
|
|
113
|
+
});
|
|
114
|
+
mockHlsInstance._trigger('hlsError', {}, { fatal: false, details: 'someError' });
|
|
115
|
+
expect(onError).not.toHaveBeenCalled();
|
|
116
|
+
});
|
|
117
|
+
it('MEDIA_ERROR 時に recoverMediaError が呼ばれる', () => {
|
|
118
|
+
const { MockHlsClass, mockHlsInstance } = createMockHls();
|
|
119
|
+
new HlsJsPlayer(MockHlsClass, {
|
|
120
|
+
video,
|
|
121
|
+
tracker,
|
|
122
|
+
callbacks: {},
|
|
123
|
+
});
|
|
124
|
+
mockHlsInstance._trigger('hlsError', {}, {
|
|
125
|
+
fatal: true,
|
|
126
|
+
type: 'mediaError',
|
|
127
|
+
details: 'bufferStalledError',
|
|
128
|
+
});
|
|
129
|
+
expect(mockHlsInstance.recoverMediaError).toHaveBeenCalled();
|
|
130
|
+
});
|
|
131
|
+
it('NETWORK_ERROR 時に startLoad が呼ばれる', () => {
|
|
132
|
+
const { MockHlsClass, mockHlsInstance } = createMockHls();
|
|
133
|
+
new HlsJsPlayer(MockHlsClass, {
|
|
134
|
+
video,
|
|
135
|
+
tracker,
|
|
136
|
+
callbacks: {},
|
|
137
|
+
});
|
|
138
|
+
mockHlsInstance._trigger('hlsError', {}, {
|
|
139
|
+
fatal: true,
|
|
140
|
+
type: 'networkError',
|
|
141
|
+
details: 'networkError',
|
|
142
|
+
});
|
|
143
|
+
expect(mockHlsInstance.startLoad).toHaveBeenCalled();
|
|
144
|
+
});
|
|
145
|
+
});
|
|
146
|
+
//# sourceMappingURL=HlsJsPlayer.test.js.map
|
package/dist/components/LiveVideoPlayer/hooks/useHlsVideo/classes/__tests__/HlsJsPlayer.test.js.map
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"HlsJsPlayer.test.js","sourceRoot":"","sources":["../../../../../../../src/components/LiveVideoPlayer/hooks/useHlsVideo/classes/__tests__/HlsJsPlayer.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"NativeHlsPlayer.test.d.ts","sourceRoot":"","sources":["../../../../../../../src/components/LiveVideoPlayer/hooks/useHlsVideo/classes/__tests__/NativeHlsPlayer.test.ts"],"names":[],"mappings":""}
|
package/dist/components/LiveVideoPlayer/hooks/useHlsVideo/classes/__tests__/NativeHlsPlayer.test.js
ADDED
|
@@ -0,0 +1,94 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @vitest-environment jsdom
|
|
3
|
+
*/
|
|
4
|
+
import { describe, it, expect, vi, beforeEach } from 'vitest';
|
|
5
|
+
import { NativeHlsPlayer } from '../NativeHlsPlayer';
|
|
6
|
+
import { RecoveryTracker } from '../RecoveryTracker';
|
|
7
|
+
// MediaError 定数(jsdom にないため定義)
|
|
8
|
+
const MEDIA_ERR_DECODE = 3;
|
|
9
|
+
const MEDIA_ERR_NETWORK = 2;
|
|
10
|
+
describe('NativeHlsPlayer', () => {
|
|
11
|
+
let video;
|
|
12
|
+
let tracker;
|
|
13
|
+
beforeEach(() => {
|
|
14
|
+
video = document.createElement('video');
|
|
15
|
+
tracker = new RecoveryTracker();
|
|
16
|
+
});
|
|
17
|
+
it('load で video.src が設定される', () => {
|
|
18
|
+
const player = new NativeHlsPlayer({
|
|
19
|
+
video,
|
|
20
|
+
tracker,
|
|
21
|
+
callbacks: {},
|
|
22
|
+
});
|
|
23
|
+
player.load('https://example.com/video.m3u8');
|
|
24
|
+
expect(video.src).toContain('video.m3u8');
|
|
25
|
+
});
|
|
26
|
+
it('destroy でエラーイベントリスナーが削除される', () => {
|
|
27
|
+
const removeEventListenerSpy = vi.spyOn(video, 'removeEventListener');
|
|
28
|
+
const player = new NativeHlsPlayer({
|
|
29
|
+
video,
|
|
30
|
+
tracker,
|
|
31
|
+
callbacks: {},
|
|
32
|
+
});
|
|
33
|
+
player.destroy();
|
|
34
|
+
expect(removeEventListenerSpy).toHaveBeenCalledWith('error', expect.any(Function));
|
|
35
|
+
});
|
|
36
|
+
it('デコードエラー時に currentTime をスキップしてリカバリを試みる', () => {
|
|
37
|
+
const playSpy = vi.spyOn(video, 'play').mockResolvedValue();
|
|
38
|
+
new NativeHlsPlayer({
|
|
39
|
+
video,
|
|
40
|
+
tracker,
|
|
41
|
+
callbacks: {},
|
|
42
|
+
});
|
|
43
|
+
// video.currentTime を設定
|
|
44
|
+
Object.defineProperty(video, 'currentTime', {
|
|
45
|
+
value: 10,
|
|
46
|
+
writable: true,
|
|
47
|
+
});
|
|
48
|
+
// MEDIA_ERR_DECODE エラーをシミュレート
|
|
49
|
+
Object.defineProperty(video, 'error', {
|
|
50
|
+
value: { code: MEDIA_ERR_DECODE, message: 'Decode error' },
|
|
51
|
+
});
|
|
52
|
+
// エラーイベントを発火
|
|
53
|
+
video.dispatchEvent(new Event('error'));
|
|
54
|
+
// currentTime が 0.5 秒進んでいることを確認
|
|
55
|
+
expect(video.currentTime).toBe(10.5);
|
|
56
|
+
expect(playSpy).toHaveBeenCalled();
|
|
57
|
+
});
|
|
58
|
+
it('リカバリ上限を超えると onError が呼ばれる', () => {
|
|
59
|
+
const onError = vi.fn();
|
|
60
|
+
new NativeHlsPlayer({
|
|
61
|
+
video,
|
|
62
|
+
tracker,
|
|
63
|
+
callbacks: { onError },
|
|
64
|
+
});
|
|
65
|
+
// MEDIA_ERR_DECODE エラーを設定
|
|
66
|
+
Object.defineProperty(video, 'error', {
|
|
67
|
+
value: { code: MEDIA_ERR_DECODE, message: 'Decode error' },
|
|
68
|
+
});
|
|
69
|
+
// リカバリ試行を消費
|
|
70
|
+
tracker.shouldAttemptRecovery(); // 1
|
|
71
|
+
tracker.shouldAttemptRecovery(); // 2
|
|
72
|
+
tracker.shouldAttemptRecovery(); // 3
|
|
73
|
+
tracker.shouldAttemptRecovery(); // 4 - 失敗
|
|
74
|
+
// エラーイベントを発火
|
|
75
|
+
video.dispatchEvent(new Event('error'));
|
|
76
|
+
expect(onError).toHaveBeenCalledWith(expect.any(Error));
|
|
77
|
+
});
|
|
78
|
+
it('デコードエラー以外はすぐに onError が呼ばれる', () => {
|
|
79
|
+
const onError = vi.fn();
|
|
80
|
+
new NativeHlsPlayer({
|
|
81
|
+
video,
|
|
82
|
+
tracker,
|
|
83
|
+
callbacks: { onError },
|
|
84
|
+
});
|
|
85
|
+
// MEDIA_ERR_NETWORK エラーを設定
|
|
86
|
+
Object.defineProperty(video, 'error', {
|
|
87
|
+
value: { code: MEDIA_ERR_NETWORK, message: 'Network error' },
|
|
88
|
+
});
|
|
89
|
+
// エラーイベントを発火
|
|
90
|
+
video.dispatchEvent(new Event('error'));
|
|
91
|
+
expect(onError).toHaveBeenCalledWith(expect.any(Error));
|
|
92
|
+
});
|
|
93
|
+
});
|
|
94
|
+
//# sourceMappingURL=NativeHlsPlayer.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"NativeHlsPlayer.test.js","sourceRoot":"","sources":["../../../../../../../src/components/LiveVideoPlayer/hooks/useHlsVideo/classes/__tests__/NativeHlsPlayer.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"RecoveryTracker.test.d.ts","sourceRoot":"","sources":["../../../../../../../src/components/LiveVideoPlayer/hooks/useHlsVideo/classes/__tests__/RecoveryTracker.test.ts"],"names":[],"mappings":""}
|
package/dist/components/LiveVideoPlayer/hooks/useHlsVideo/classes/__tests__/RecoveryTracker.test.js
ADDED
|
@@ -0,0 +1,74 @@
|
|
|
1
|
+
import { describe, it, expect, vi, beforeEach, afterEach } from 'vitest';
|
|
2
|
+
import { RecoveryTracker } from '../RecoveryTracker';
|
|
3
|
+
describe('RecoveryTracker', () => {
|
|
4
|
+
beforeEach(() => {
|
|
5
|
+
vi.useFakeTimers();
|
|
6
|
+
});
|
|
7
|
+
afterEach(() => {
|
|
8
|
+
vi.useRealTimers();
|
|
9
|
+
});
|
|
10
|
+
describe('shouldAttemptRecovery', () => {
|
|
11
|
+
it('初回のリカバリ試行は true を返す', () => {
|
|
12
|
+
const tracker = new RecoveryTracker();
|
|
13
|
+
expect(tracker.shouldAttemptRecovery()).toBe(true);
|
|
14
|
+
});
|
|
15
|
+
it('5秒以内の連続リカバリは試行回数をカウントアップ', () => {
|
|
16
|
+
const tracker = new RecoveryTracker();
|
|
17
|
+
expect(tracker.shouldAttemptRecovery()).toBe(true); // 1回目
|
|
18
|
+
vi.advanceTimersByTime(1000); // 1秒後
|
|
19
|
+
expect(tracker.shouldAttemptRecovery()).toBe(true); // 2回目
|
|
20
|
+
vi.advanceTimersByTime(1000); // さらに1秒後
|
|
21
|
+
expect(tracker.shouldAttemptRecovery()).toBe(true); // 3回目
|
|
22
|
+
vi.advanceTimersByTime(1000); // さらに1秒後
|
|
23
|
+
expect(tracker.shouldAttemptRecovery()).toBe(false); // 4回目: 上限超過
|
|
24
|
+
});
|
|
25
|
+
it('5秒以上経過するとカウントがリセットされる', () => {
|
|
26
|
+
const tracker = new RecoveryTracker();
|
|
27
|
+
expect(tracker.shouldAttemptRecovery()).toBe(true); // 1回目
|
|
28
|
+
vi.advanceTimersByTime(1000);
|
|
29
|
+
expect(tracker.shouldAttemptRecovery()).toBe(true); // 2回目
|
|
30
|
+
vi.advanceTimersByTime(1000);
|
|
31
|
+
expect(tracker.shouldAttemptRecovery()).toBe(true); // 3回目
|
|
32
|
+
// 5秒以上経過
|
|
33
|
+
vi.advanceTimersByTime(6000);
|
|
34
|
+
expect(tracker.shouldAttemptRecovery()).toBe(true); // リセットされて1回目として扱われる
|
|
35
|
+
});
|
|
36
|
+
});
|
|
37
|
+
describe('reset', () => {
|
|
38
|
+
it('リカバリカウントをリセット', () => {
|
|
39
|
+
const tracker = new RecoveryTracker();
|
|
40
|
+
// 3回試行
|
|
41
|
+
tracker.shouldAttemptRecovery();
|
|
42
|
+
vi.advanceTimersByTime(100);
|
|
43
|
+
tracker.shouldAttemptRecovery();
|
|
44
|
+
vi.advanceTimersByTime(100);
|
|
45
|
+
tracker.shouldAttemptRecovery();
|
|
46
|
+
vi.advanceTimersByTime(100);
|
|
47
|
+
// 4回目は失敗するはず
|
|
48
|
+
expect(tracker.shouldAttemptRecovery()).toBe(false);
|
|
49
|
+
// リセット
|
|
50
|
+
tracker.reset();
|
|
51
|
+
// リセット後は再度成功
|
|
52
|
+
expect(tracker.shouldAttemptRecovery()).toBe(true);
|
|
53
|
+
});
|
|
54
|
+
it('エラー報告フラグもリセット', () => {
|
|
55
|
+
const tracker = new RecoveryTracker();
|
|
56
|
+
tracker.markErrorReported();
|
|
57
|
+
expect(tracker.isErrorReported).toBe(true);
|
|
58
|
+
tracker.reset();
|
|
59
|
+
expect(tracker.isErrorReported).toBe(false);
|
|
60
|
+
});
|
|
61
|
+
});
|
|
62
|
+
describe('isErrorReported / markErrorReported', () => {
|
|
63
|
+
it('初期状態は false', () => {
|
|
64
|
+
const tracker = new RecoveryTracker();
|
|
65
|
+
expect(tracker.isErrorReported).toBe(false);
|
|
66
|
+
});
|
|
67
|
+
it('markErrorReported で true になる', () => {
|
|
68
|
+
const tracker = new RecoveryTracker();
|
|
69
|
+
tracker.markErrorReported();
|
|
70
|
+
expect(tracker.isErrorReported).toBe(true);
|
|
71
|
+
});
|
|
72
|
+
});
|
|
73
|
+
});
|
|
74
|
+
//# sourceMappingURL=RecoveryTracker.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"RecoveryTracker.test.js","sourceRoot":"","sources":["../../../../../../../src/components/LiveVideoPlayer/hooks/useHlsVideo/classes/__tests__/RecoveryTracker.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,31 @@
|
|
|
1
|
+
import { VideoTexture } from 'three';
|
|
2
|
+
export interface UseHlsVideoOptions {
|
|
3
|
+
/** HLS動画URL */
|
|
4
|
+
url: string;
|
|
5
|
+
/** キャッシュバスター用キー */
|
|
6
|
+
cacheKey?: number;
|
|
7
|
+
/** 再生中かどうか */
|
|
8
|
+
playing: boolean;
|
|
9
|
+
/** 音量 0〜1 */
|
|
10
|
+
volume: number;
|
|
11
|
+
/** エラー発生時のコールバック */
|
|
12
|
+
onError?: (error: Error) => void;
|
|
13
|
+
/** バッファリング状態変更時のコールバック */
|
|
14
|
+
onBufferingChange?: (buffering: boolean) => void;
|
|
15
|
+
}
|
|
16
|
+
export interface UseHlsVideoReturn {
|
|
17
|
+
/** 動画テクスチャ */
|
|
18
|
+
texture: VideoTexture;
|
|
19
|
+
/** 動画要素への参照 */
|
|
20
|
+
videoRef: React.RefObject<HTMLVideoElement>;
|
|
21
|
+
}
|
|
22
|
+
/**
|
|
23
|
+
* HLSストリーム専用のビデオ要素管理フック
|
|
24
|
+
*
|
|
25
|
+
* - hls.js を使用(Safari以外)
|
|
26
|
+
* - Native HLS support を使用(Safari)
|
|
27
|
+
* - recoverMediaError() によるエラーリカバリ
|
|
28
|
+
* - リカバリ失敗時のみ完全リロード
|
|
29
|
+
*/
|
|
30
|
+
export declare function useHlsVideo({ url, cacheKey, playing, volume, onError, onBufferingChange, }: UseHlsVideoOptions): UseHlsVideoReturn;
|
|
31
|
+
//# sourceMappingURL=index.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../../../src/components/LiveVideoPlayer/hooks/useHlsVideo/index.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,YAAY,EAAE,MAAM,OAAO,CAAA;AAMpC,MAAM,WAAW,kBAAkB;IACjC,eAAe;IACf,GAAG,EAAE,MAAM,CAAA;IACX,mBAAmB;IACnB,QAAQ,CAAC,EAAE,MAAM,CAAA;IACjB,cAAc;IACd,OAAO,EAAE,OAAO,CAAA;IAChB,aAAa;IACb,MAAM,EAAE,MAAM,CAAA;IACd,oBAAoB;IACpB,OAAO,CAAC,EAAE,CAAC,KAAK,EAAE,KAAK,KAAK,IAAI,CAAA;IAChC,0BAA0B;IAC1B,iBAAiB,CAAC,EAAE,CAAC,SAAS,EAAE,OAAO,KAAK,IAAI,CAAA;CACjD;AAED,MAAM,WAAW,iBAAiB;IAChC,cAAc;IACd,OAAO,EAAE,YAAY,CAAA;IACrB,eAAe;IACf,QAAQ,EAAE,KAAK,CAAC,SAAS,CAAC,gBAAgB,CAAC,CAAA;CAC5C;AAED;;;;;;;GAOG;AACH,wBAAgB,WAAW,CAAC,EAC1B,GAAG,EACH,QAAY,EACZ,OAAO,EACP,MAAM,EACN,OAAO,EACP,iBAAiB,GAClB,EAAE,kBAAkB,GAAG,iBAAiB,CA8GxC"}
|
|
@@ -0,0 +1,108 @@
|
|
|
1
|
+
import { useEffect, useRef, useState } from 'react';
|
|
2
|
+
import { useWebAudioVolume } from '../../../../hooks/useWebAudioVolume';
|
|
3
|
+
import { appendCacheKey, createVideoTexture, createHlsPlayer } from './utils';
|
|
4
|
+
import { RecoveryTracker } from './classes/RecoveryTracker';
|
|
5
|
+
/**
|
|
6
|
+
* HLSストリーム専用のビデオ要素管理フック
|
|
7
|
+
*
|
|
8
|
+
* - hls.js を使用(Safari以外)
|
|
9
|
+
* - Native HLS support を使用(Safari)
|
|
10
|
+
* - recoverMediaError() によるエラーリカバリ
|
|
11
|
+
* - リカバリ失敗時のみ完全リロード
|
|
12
|
+
*/
|
|
13
|
+
export function useHlsVideo({ url, cacheKey = 0, playing, volume, onError, onBufferingChange, }) {
|
|
14
|
+
const videoRef = useRef(null);
|
|
15
|
+
const textureRef = useRef(null);
|
|
16
|
+
const playerRef = useRef(null);
|
|
17
|
+
const recoveryTrackerRef = useRef(new RecoveryTracker());
|
|
18
|
+
// テクスチャを状態として保持(初回レンダリング時に同期的に作成)
|
|
19
|
+
const [texture] = useState(() => {
|
|
20
|
+
const { video, texture: tex } = createVideoTexture();
|
|
21
|
+
videoRef.current = video;
|
|
22
|
+
textureRef.current = tex;
|
|
23
|
+
return tex;
|
|
24
|
+
});
|
|
25
|
+
// プレイヤーの初期化
|
|
26
|
+
useEffect(() => {
|
|
27
|
+
const video = videoRef.current;
|
|
28
|
+
if (!video)
|
|
29
|
+
return;
|
|
30
|
+
const urlWithCacheKey = appendCacheKey(url, cacheKey);
|
|
31
|
+
const tracker = recoveryTrackerRef.current;
|
|
32
|
+
tracker.reset();
|
|
33
|
+
let player = null;
|
|
34
|
+
const init = async () => {
|
|
35
|
+
const result = await createHlsPlayer({
|
|
36
|
+
video,
|
|
37
|
+
tracker,
|
|
38
|
+
callbacks: {
|
|
39
|
+
onError,
|
|
40
|
+
onBufferingChange,
|
|
41
|
+
onManifestParsed: () => {
|
|
42
|
+
if (playing) {
|
|
43
|
+
video.play().catch((err) => console.error('[useHlsVideo] Play error after manifest parsed:', err));
|
|
44
|
+
}
|
|
45
|
+
},
|
|
46
|
+
},
|
|
47
|
+
});
|
|
48
|
+
if (result.type === 'unsupported') {
|
|
49
|
+
onError?.(result.error);
|
|
50
|
+
return;
|
|
51
|
+
}
|
|
52
|
+
player = result.player;
|
|
53
|
+
playerRef.current = player;
|
|
54
|
+
player.load(urlWithCacheKey);
|
|
55
|
+
};
|
|
56
|
+
init();
|
|
57
|
+
return () => {
|
|
58
|
+
if (player) {
|
|
59
|
+
player.destroy();
|
|
60
|
+
playerRef.current = null;
|
|
61
|
+
}
|
|
62
|
+
video.pause();
|
|
63
|
+
video.src = '';
|
|
64
|
+
video.load();
|
|
65
|
+
};
|
|
66
|
+
}, [url, cacheKey, onError, onBufferingChange, playing]);
|
|
67
|
+
// 再生/停止制御
|
|
68
|
+
useEffect(() => {
|
|
69
|
+
const video = videoRef.current;
|
|
70
|
+
if (!video)
|
|
71
|
+
return;
|
|
72
|
+
if (playing) {
|
|
73
|
+
video.play().catch((err) => console.error('[useHlsVideo] Play error:', err));
|
|
74
|
+
}
|
|
75
|
+
else {
|
|
76
|
+
video.pause();
|
|
77
|
+
}
|
|
78
|
+
}, [playing]);
|
|
79
|
+
// バッファリング状態の監視
|
|
80
|
+
useEffect(() => {
|
|
81
|
+
const video = videoRef.current;
|
|
82
|
+
if (!video)
|
|
83
|
+
return;
|
|
84
|
+
const handleWaiting = () => onBufferingChange?.(true);
|
|
85
|
+
const handlePlaying = () => onBufferingChange?.(false);
|
|
86
|
+
const handleCanPlay = () => onBufferingChange?.(false);
|
|
87
|
+
video.addEventListener('waiting', handleWaiting);
|
|
88
|
+
video.addEventListener('playing', handlePlaying);
|
|
89
|
+
video.addEventListener('canplay', handleCanPlay);
|
|
90
|
+
return () => {
|
|
91
|
+
video.removeEventListener('waiting', handleWaiting);
|
|
92
|
+
video.removeEventListener('playing', handlePlaying);
|
|
93
|
+
video.removeEventListener('canplay', handleCanPlay);
|
|
94
|
+
};
|
|
95
|
+
}, [onBufferingChange]);
|
|
96
|
+
// Web Audio API を使用した音量制御(iOS対応)
|
|
97
|
+
useWebAudioVolume(videoRef.current, volume);
|
|
98
|
+
// クリーンアップ
|
|
99
|
+
useEffect(() => {
|
|
100
|
+
return () => {
|
|
101
|
+
if (textureRef.current) {
|
|
102
|
+
textureRef.current.dispose();
|
|
103
|
+
}
|
|
104
|
+
};
|
|
105
|
+
}, []);
|
|
106
|
+
return { texture, videoRef };
|
|
107
|
+
}
|
|
108
|
+
//# sourceMappingURL=index.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../../../src/components/LiveVideoPlayer/hooks/useHlsVideo/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,MAAM,EAAE,QAAQ,EAAE,MAAM,OAAO,CAAA;AAEnD,OAAO,EAAE,iBAAiB,EAAE,MAAM,qCAAqC,CAAA;AACvE,OAAO,EAAE,cAAc,EAAE,kBAAkB,EAAE,eAAe,EAAE,MAAM,SAAS,CAAA;AAC7E,OAAO,EAAE,eAAe,EAAE,MAAM,2BAA2B,CAAA;AAyB3D;;;;;;;GAOG;AACH,MAAM,UAAU,WAAW,CAAC,EAC1B,GAAG,EACH,QAAQ,GAAG,CAAC,EACZ,OAAO,EACP,MAAM,EACN,OAAO,EACP,iBAAiB,GACE;IACnB,MAAM,QAAQ,GAAG,MAAM,CAAmB,IAAK,CAAC,CAAA;IAChD,MAAM,UAAU,GAAG,MAAM,CAAsB,IAAI,CAAC,CAAA;IACpD,MAAM,SAAS,GAAG,MAAM,CAA2B,IAAI,CAAC,CAAA;IACxD,MAAM,kBAAkB,GAAG,MAAM,CAAC,IAAI,eAAe,EAAE,CAAC,CAAA;IAExD,kCAAkC;IAClC,MAAM,CAAC,OAAO,CAAC,GAAG,QAAQ,CAAe,GAAG,EAAE;QAC5C,MAAM,EAAE,KAAK,EAAE,OAAO,EAAE,GAAG,EAAE,GAAG,kBAAkB,EAAE,CAAA;QACpD,QAAQ,CAAC,OAAO,GAAG,KAAK,CAAA;QACxB,UAAU,CAAC,OAAO,GAAG,GAAG,CAAA;QACxB,OAAO,GAAG,CAAA;IACZ,CAAC,CAAC,CAAA;IAEF,YAAY;IACZ,SAAS,CAAC,GAAG,EAAE;QACb,MAAM,KAAK,GAAG,QAAQ,CAAC,OAAO,CAAA;QAC9B,IAAI,CAAC,KAAK;YAAE,OAAM;QAElB,MAAM,eAAe,GAAG,cAAc,CAAC,GAAG,EAAE,QAAQ,CAAC,CAAA;QACrD,MAAM,OAAO,GAAG,kBAAkB,CAAC,OAAO,CAAA;QAC1C,OAAO,CAAC,KAAK,EAAE,CAAA;QAEf,IAAI,MAAM,GAA6B,IAAI,CAAA;QAE3C,MAAM,IAAI,GAAG,KAAK,IAAI,EAAE;YACtB,MAAM,MAAM,GAAG,MAAM,eAAe,CAAC;gBACnC,KAAK;gBACL,OAAO;gBACP,SAAS,EAAE;oBACT,OAAO;oBACP,iBAAiB;oBACjB,gBAAgB,EAAE,GAAG,EAAE;wBACrB,IAAI,OAAO,EAAE,CAAC;4BACZ,KAAK,CAAC,IAAI,EAAE,CAAC,KAAK,CAAC,CAAC,GAAG,EAAE,EAAE,CACzB,OAAO,CAAC,KAAK,CAAC,iDAAiD,EAAE,GAAG,CAAC,CACtE,CAAA;wBACH,CAAC;oBACH,CAAC;iBACF;aACF,CAAC,CAAA;YAEF,IAAI,MAAM,CAAC,IAAI,KAAK,aAAa,EAAE,CAAC;gBAClC,OAAO,EAAE,CAAC,MAAM,CAAC,KAAK,CAAC,CAAA;gBACvB,OAAM;YACR,CAAC;YAED,MAAM,GAAG,MAAM,CAAC,MAAM,CAAA;YACtB,SAAS,CAAC,OAAO,GAAG,MAAM,CAAA;YAC1B,MAAM,CAAC,IAAI,CAAC,eAAe,CAAC,CAAA;QAC9B,CAAC,CAAA;QAED,IAAI,EAAE,CAAA;QAEN,OAAO,GAAG,EAAE;YACV,IAAI,MAAM,EAAE,CAAC;gBACX,MAAM,CAAC,OAAO,EAAE,CAAA;gBAChB,SAAS,CAAC,OAAO,GAAG,IAAI,CAAA;YAC1B,CAAC;YACD,KAAK,CAAC,KAAK,EAAE,CAAA;YACb,KAAK,CAAC,GAAG,GAAG,EAAE,CAAA;YACd,KAAK,CAAC,IAAI,EAAE,CAAA;QACd,CAAC,CAAA;IACH,CAAC,EAAE,CAAC,GAAG,EAAE,QAAQ,EAAE,OAAO,EAAE,iBAAiB,EAAE,OAAO,CAAC,CAAC,CAAA;IAExD,UAAU;IACV,SAAS,CAAC,GAAG,EAAE;QACb,MAAM,KAAK,GAAG,QAAQ,CAAC,OAAO,CAAA;QAC9B,IAAI,CAAC,KAAK;YAAE,OAAM;QAElB,IAAI,OAAO,EAAE,CAAC;YACZ,KAAK,CAAC,IAAI,EAAE,CAAC,KAAK,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,OAAO,CAAC,KAAK,CAAC,2BAA2B,EAAE,GAAG,CAAC,CAAC,CAAA;QAC9E,CAAC;aAAM,CAAC;YACN,KAAK,CAAC,KAAK,EAAE,CAAA;QACf,CAAC;IACH,CAAC,EAAE,CAAC,OAAO,CAAC,CAAC,CAAA;IAEb,eAAe;IACf,SAAS,CAAC,GAAG,EAAE;QACb,MAAM,KAAK,GAAG,QAAQ,CAAC,OAAO,CAAA;QAC9B,IAAI,CAAC,KAAK;YAAE,OAAM;QAElB,MAAM,aAAa,GAAG,GAAG,EAAE,CAAC,iBAAiB,EAAE,CAAC,IAAI,CAAC,CAAA;QACrD,MAAM,aAAa,GAAG,GAAG,EAAE,CAAC,iBAAiB,EAAE,CAAC,KAAK,CAAC,CAAA;QACtD,MAAM,aAAa,GAAG,GAAG,EAAE,CAAC,iBAAiB,EAAE,CAAC,KAAK,CAAC,CAAA;QAEtD,KAAK,CAAC,gBAAgB,CAAC,SAAS,EAAE,aAAa,CAAC,CAAA;QAChD,KAAK,CAAC,gBAAgB,CAAC,SAAS,EAAE,aAAa,CAAC,CAAA;QAChD,KAAK,CAAC,gBAAgB,CAAC,SAAS,EAAE,aAAa,CAAC,CAAA;QAEhD,OAAO,GAAG,EAAE;YACV,KAAK,CAAC,mBAAmB,CAAC,SAAS,EAAE,aAAa,CAAC,CAAA;YACnD,KAAK,CAAC,mBAAmB,CAAC,SAAS,EAAE,aAAa,CAAC,CAAA;YACnD,KAAK,CAAC,mBAAmB,CAAC,SAAS,EAAE,aAAa,CAAC,CAAA;QACrD,CAAC,CAAA;IACH,CAAC,EAAE,CAAC,iBAAiB,CAAC,CAAC,CAAA;IAEvB,iCAAiC;IACjC,iBAAiB,CAAC,QAAQ,CAAC,OAAO,EAAE,MAAM,CAAC,CAAA;IAE3C,UAAU;IACV,SAAS,CAAC,GAAG,EAAE;QACb,OAAO,GAAG,EAAE;YACV,IAAI,UAAU,CAAC,OAAO,EAAE,CAAC;gBACvB,UAAU,CAAC,OAAO,CAAC,OAAO,EAAE,CAAA;YAC9B,CAAC;QACH,CAAC,CAAA;IACH,CAAC,EAAE,EAAE,CAAC,CAAA;IAEN,OAAO,EAAE,OAAO,EAAE,QAAQ,EAAE,CAAA;AAC9B,CAAC"}
|
|
@@ -0,0 +1,34 @@
|
|
|
1
|
+
import type { VideoTexture } from 'three';
|
|
2
|
+
import type { RecoveryTracker } from './classes/RecoveryTracker';
|
|
3
|
+
export interface HlsPlayerCallbacks {
|
|
4
|
+
onError?: (error: Error) => void;
|
|
5
|
+
onBufferingChange?: (buffering: boolean) => void;
|
|
6
|
+
onManifestParsed?: () => void;
|
|
7
|
+
}
|
|
8
|
+
export interface HlsPlayerOptions {
|
|
9
|
+
video: HTMLVideoElement;
|
|
10
|
+
tracker: RecoveryTracker;
|
|
11
|
+
callbacks: HlsPlayerCallbacks;
|
|
12
|
+
}
|
|
13
|
+
/** HLS プレイヤーの共通インターフェース */
|
|
14
|
+
export interface HlsPlayerStrategy {
|
|
15
|
+
/** ソースを読み込む */
|
|
16
|
+
load(url: string): void;
|
|
17
|
+
/** リソースを解放 */
|
|
18
|
+
destroy(): void;
|
|
19
|
+
/** メディアエラーからのリカバリを試行 */
|
|
20
|
+
attemptRecovery(): boolean;
|
|
21
|
+
}
|
|
22
|
+
export interface VideoTextureResult {
|
|
23
|
+
video: HTMLVideoElement;
|
|
24
|
+
texture: VideoTexture;
|
|
25
|
+
}
|
|
26
|
+
export type CreatePlayerResult = {
|
|
27
|
+
player: HlsPlayerStrategy;
|
|
28
|
+
type: 'hlsjs' | 'native';
|
|
29
|
+
} | {
|
|
30
|
+
player: null;
|
|
31
|
+
type: 'unsupported';
|
|
32
|
+
error: Error;
|
|
33
|
+
};
|
|
34
|
+
//# sourceMappingURL=types.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"types.d.ts","sourceRoot":"","sources":["../../../../../src/components/LiveVideoPlayer/hooks/useHlsVideo/types.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,OAAO,CAAA;AACzC,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,2BAA2B,CAAA;AAEhE,MAAM,WAAW,kBAAkB;IACjC,OAAO,CAAC,EAAE,CAAC,KAAK,EAAE,KAAK,KAAK,IAAI,CAAA;IAChC,iBAAiB,CAAC,EAAE,CAAC,SAAS,EAAE,OAAO,KAAK,IAAI,CAAA;IAChD,gBAAgB,CAAC,EAAE,MAAM,IAAI,CAAA;CAC9B;AAED,MAAM,WAAW,gBAAgB;IAC/B,KAAK,EAAE,gBAAgB,CAAA;IACvB,OAAO,EAAE,eAAe,CAAA;IACxB,SAAS,EAAE,kBAAkB,CAAA;CAC9B;AAED,2BAA2B;AAC3B,MAAM,WAAW,iBAAiB;IAChC,eAAe;IACf,IAAI,CAAC,GAAG,EAAE,MAAM,GAAG,IAAI,CAAA;IACvB,cAAc;IACd,OAAO,IAAI,IAAI,CAAA;IACf,wBAAwB;IACxB,eAAe,IAAI,OAAO,CAAA;CAC3B;AAED,MAAM,WAAW,kBAAkB;IACjC,KAAK,EAAE,gBAAgB,CAAA;IACvB,OAAO,EAAE,YAAY,CAAA;CACtB;AAED,MAAM,MAAM,kBAAkB,GAC1B;IAAE,MAAM,EAAE,iBAAiB,CAAC;IAAC,IAAI,EAAE,OAAO,GAAG,QAAQ,CAAA;CAAE,GACvD;IAAE,MAAM,EAAE,IAAI,CAAC;IAAC,IAAI,EAAE,aAAa,CAAC;IAAC,KAAK,EAAE,KAAK,CAAA;CAAE,CAAA"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"types.js","sourceRoot":"","sources":["../../../../../src/components/LiveVideoPlayer/hooks/useHlsVideo/types.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
import type { HlsPlayerOptions, CreatePlayerResult, VideoTextureResult } from './types';
|
|
2
|
+
/** URLがHLSストリームかどうかを判定 */
|
|
3
|
+
export declare function isHlsUrl(url: string): boolean;
|
|
4
|
+
/** Safari(native HLS対応ブラウザ)かどうかを判定 */
|
|
5
|
+
export declare function canPlayHlsNatively(): boolean;
|
|
6
|
+
/** URLにキャッシュバスター用のキーを付与 */
|
|
7
|
+
export declare function appendCacheKey(url: string, cacheKey: number): string;
|
|
8
|
+
/** HLS再生用のvideo要素とテクスチャを作成 */
|
|
9
|
+
export declare function createVideoTexture(): VideoTextureResult;
|
|
10
|
+
/**
|
|
11
|
+
* 環境に応じて適切な HLS プレイヤーを作成
|
|
12
|
+
* hls.js を優先し、利用できない場合はネイティブ HLS にフォールバック
|
|
13
|
+
*/
|
|
14
|
+
export declare function createHlsPlayer(options: HlsPlayerOptions): Promise<CreatePlayerResult>;
|
|
15
|
+
//# sourceMappingURL=utils.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"utils.d.ts","sourceRoot":"","sources":["../../../../../src/components/LiveVideoPlayer/hooks/useHlsVideo/utils.ts"],"names":[],"mappings":"AAGA,OAAO,KAAK,EAAE,gBAAgB,EAAE,kBAAkB,EAAE,kBAAkB,EAAE,MAAM,SAAS,CAAA;AAEvF,0BAA0B;AAC1B,wBAAgB,QAAQ,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO,CAE7C;AAED,sCAAsC;AACtC,wBAAgB,kBAAkB,IAAI,OAAO,CAI5C;AAED,2BAA2B;AAC3B,wBAAgB,cAAc,CAAC,GAAG,EAAE,MAAM,EAAE,QAAQ,EAAE,MAAM,GAAG,MAAM,CAEpE;AAED,8BAA8B;AAC9B,wBAAgB,kBAAkB,IAAI,kBAAkB,CAYvD;AAED;;;GAGG;AACH,wBAAsB,eAAe,CACnC,OAAO,EAAE,gBAAgB,GACxB,OAAO,CAAC,kBAAkB,CAAC,CA4B7B"}
|