* perf: ensure we do not provide callback to native if no callback provided from app * chore: rework bufferConfig to make it more generic and reduce ReactExoplayerView code size * chore: improve issue template * fix(android): avoid video view flickering at playback startup * chore(android): refactor DRM props into a dedicated class * Update android/src/main/java/com/brentvatne/exoplayer/ReactExoplayerView.java * chore: fix linter * fix: ensure drm prop is correctly cleaned * feat(android): move viewType (secure texture) & drm inside the source The origianl behavior has been kept for interoperability, but marked as deprecated in doc * chore: fix linter * chore(ios): move drm prop in source like on android * chore: fix linter * chore: clean log * fix: allow to disable secure View * chore: fix viewType resolution (source value was not handled) * chore: use contentDeepEquals instead of manual checks * chore: fix linter * fix: ensure player doesn't start when view is unmounted * Fix/ensure view drop stop playback startup (#3875) * fix: ensure player doesn't start when view is unmounted * chore: revert change * chore: add warning in case of invalid Surface configuration * chore: code clean * fix: simplify surface management * chore: restore previous code * chore: fix typo * chore: code cleanup * feat(android): add multiDrm flag support * docs: update docs * chore: fix ios build * chore: fix deprecated declaration --------- Co-authored-by: Krzysztof Moch <krzysmoch.programs@gmail.com>
		
			
				
	
	
		
			419 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			419 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
| /* eslint-disable @typescript-eslint/ban-types */
 | |
| import type {HostComponent, ViewProps} from 'react-native';
 | |
| import {NativeModules, requireNativeComponent} from 'react-native';
 | |
| import type {
 | |
|   DirectEventHandler,
 | |
|   Double,
 | |
|   Float,
 | |
|   Int32,
 | |
|   WithDefault,
 | |
| } from 'react-native/Libraries/Types/CodegenTypes';
 | |
| 
 | |
| // -------- There are types for native component (future codegen) --------
 | |
| // if you are looking for types for react component, see src/types/video.ts
 | |
| 
 | |
| type Headers = ReadonlyArray<
 | |
|   Readonly<{
 | |
|     key: string;
 | |
|     value: string;
 | |
|   }>
 | |
| >;
 | |
| 
 | |
| type VideoMetadata = Readonly<{
 | |
|   title?: string;
 | |
|   subtitle?: string;
 | |
|   description?: string;
 | |
|   imageUri?: string;
 | |
| }>;
 | |
| 
 | |
| export type VideoSrc = Readonly<{
 | |
|   uri?: string;
 | |
|   isNetwork?: boolean;
 | |
|   isAsset?: boolean;
 | |
|   shouldCache?: boolean;
 | |
|   type?: string;
 | |
|   mainVer?: Int32;
 | |
|   patchVer?: Int32;
 | |
|   requestHeaders?: Headers;
 | |
|   startPosition?: Float;
 | |
|   cropStart?: Float;
 | |
|   cropEnd?: Float;
 | |
|   metadata?: VideoMetadata;
 | |
|   drm?: Drm;
 | |
|   textTracksAllowChunklessPreparation?: boolean; // android
 | |
| }>;
 | |
| 
 | |
| type DRMType = WithDefault<string, 'widevine'>;
 | |
| 
 | |
| type DebugConfig = Readonly<{
 | |
|   enable?: boolean;
 | |
|   thread?: boolean;
 | |
| }>;
 | |
| 
 | |
| type Drm = Readonly<{
 | |
|   type?: DRMType;
 | |
|   licenseServer?: string;
 | |
|   headers?: Headers;
 | |
|   contentId?: string; // ios
 | |
|   certificateUrl?: string; // ios
 | |
|   base64Certificate?: boolean; // ios default: false
 | |
|   useExternalGetLicense?: boolean; // ios
 | |
|   multiDrm?: boolean; // android
 | |
| }>;
 | |
| 
 | |
| type TextTracks = ReadonlyArray<
 | |
|   Readonly<{
 | |
|     title: string;
 | |
|     language: string;
 | |
|     type: string;
 | |
|     uri: string;
 | |
|   }>
 | |
| >;
 | |
| 
 | |
| type SelectedTextTrackType = WithDefault<string, 'system'>;
 | |
| 
 | |
| type SelectedAudioTrackType = WithDefault<string, 'system'>;
 | |
| 
 | |
| type SelectedTextTrack = Readonly<{
 | |
|   type?: SelectedTextTrackType;
 | |
|   value?: string;
 | |
| }>;
 | |
| 
 | |
| type SelectedAudioTrack = Readonly<{
 | |
|   type?: SelectedAudioTrackType;
 | |
|   value?: string;
 | |
| }>;
 | |
| 
 | |
| type SelectedVideoTrackType = WithDefault<string, 'auto'>;
 | |
| 
 | |
| type SelectedVideoTrack = Readonly<{
 | |
|   type?: SelectedVideoTrackType;
 | |
|   value?: string;
 | |
| }>;
 | |
| 
 | |
| export type Seek = Readonly<{
 | |
|   time: Float;
 | |
|   tolerance?: Float;
 | |
| }>;
 | |
| 
 | |
| type BufferConfigLive = Readonly<{
 | |
|   maxPlaybackSpeed?: Float;
 | |
|   minPlaybackSpeed?: Float;
 | |
|   maxOffsetMs?: Int32;
 | |
|   minOffsetMs?: Int32;
 | |
|   targetOffsetMs?: Int32;
 | |
| }>;
 | |
| 
 | |
| type BufferingStrategyType = WithDefault<string, 'Default'>;
 | |
| 
 | |
| type BufferConfig = Readonly<{
 | |
|   minBufferMs?: Float;
 | |
|   maxBufferMs?: Float;
 | |
|   bufferForPlaybackMs?: Float;
 | |
|   bufferForPlaybackAfterRebufferMs?: Float;
 | |
|   maxHeapAllocationPercent?: Float;
 | |
|   backBufferDurationMs?: Float; // Android
 | |
|   minBackBufferMemoryReservePercent?: Float;
 | |
|   minBufferMemoryReservePercent?: Float;
 | |
|   cacheSizeMB?: Float;
 | |
|   live?: BufferConfigLive;
 | |
| }>;
 | |
| 
 | |
| type SubtitleStyle = Readonly<{
 | |
|   fontSize?: Float;
 | |
|   paddingTop?: WithDefault<Float, 0>;
 | |
|   paddingBottom?: WithDefault<Float, 0>;
 | |
|   paddingLeft?: WithDefault<Float, 0>;
 | |
|   paddingRight?: WithDefault<Float, 0>;
 | |
|   opacity?: WithDefault<Float, 1>;
 | |
| }>;
 | |
| 
 | |
| type OnLoadData = Readonly<{
 | |
|   currentTime: Float;
 | |
|   duration: Float;
 | |
|   naturalSize: Readonly<{
 | |
|     width: Float;
 | |
|     height: Float;
 | |
|     orientation: WithDefault<string, 'landscape'>;
 | |
|   }>;
 | |
|   audioTracks: {
 | |
|     index: Int32;
 | |
|     title?: string;
 | |
|     language?: string;
 | |
|     bitrate?: Float;
 | |
|     type?: string;
 | |
|     selected?: boolean;
 | |
|   }[];
 | |
|   textTracks: {
 | |
|     index: Int32;
 | |
|     title?: string;
 | |
|     language?: string;
 | |
|     /**
 | |
|      * iOS only supports VTT, Android supports all 3
 | |
|      */
 | |
|     type?: WithDefault<string, 'srt'>;
 | |
|     selected?: boolean;
 | |
|   }[];
 | |
| }>;
 | |
| 
 | |
| export type OnLoadStartData = Readonly<{
 | |
|   isNetwork: boolean;
 | |
|   type: string;
 | |
|   uri: string;
 | |
| }>;
 | |
| 
 | |
| export type OnVideoAspectRatioData = Readonly<{
 | |
|   width: Float;
 | |
|   height: Float;
 | |
| }>;
 | |
| 
 | |
| export type OnBufferData = Readonly<{isBuffering: boolean}>;
 | |
| 
 | |
| export type OnProgressData = Readonly<{
 | |
|   currentTime: Float;
 | |
|   playableDuration: Float;
 | |
|   seekableDuration: Float;
 | |
| }>;
 | |
| 
 | |
| export type OnBandwidthUpdateData = Readonly<{
 | |
|   bitrate: Int32;
 | |
|   width?: Float;
 | |
|   height?: Float;
 | |
|   trackId?: Int32;
 | |
| }>;
 | |
| 
 | |
| export type OnSeekData = Readonly<{
 | |
|   currentTime: Float;
 | |
|   seekTime: Float;
 | |
| }>;
 | |
| 
 | |
| export type OnPlaybackStateChangedData = Readonly<{
 | |
|   isPlaying: boolean;
 | |
| }>;
 | |
| 
 | |
| export type OnTimedMetadataData = Readonly<{
 | |
|   metadata: {
 | |
|     value?: string;
 | |
|     identifier: string;
 | |
|   }[];
 | |
| }>;
 | |
| 
 | |
| export type OnAudioTracksData = Readonly<{
 | |
|   audioTracks: {
 | |
|     index: Int32;
 | |
|     title?: string;
 | |
|     language?: string;
 | |
|     bitrate?: Float;
 | |
|     type?: string;
 | |
|     selected?: boolean;
 | |
|   }[];
 | |
| }>;
 | |
| 
 | |
| type OnTextTracksData = Readonly<{
 | |
|   textTracks: {
 | |
|     index: Int32;
 | |
|     title?: string;
 | |
|     language?: string;
 | |
|     /**
 | |
|      * iOS only supports VTT, Android supports all 3
 | |
|      */
 | |
|     type?: WithDefault<string, 'srt'>;
 | |
|     selected?: boolean;
 | |
|   }[];
 | |
| }>;
 | |
| 
 | |
| export type OnTextTrackDataChangedData = Readonly<{
 | |
|   subtitleTracks: string;
 | |
| }>;
 | |
| 
 | |
| export type OnVideoTracksData = Readonly<{
 | |
|   videoTracks: {
 | |
|     index: Int32;
 | |
|     tracksId?: string;
 | |
|     codecs?: string;
 | |
|     width?: Float;
 | |
|     height?: Float;
 | |
|     bitrate?: Float;
 | |
|     selected?: boolean;
 | |
|   }[];
 | |
| }>;
 | |
| 
 | |
| export type OnPlaybackRateChangeData = Readonly<{
 | |
|   playbackRate: Float;
 | |
| }>;
 | |
| 
 | |
| export type OnVolumeChangeData = Readonly<{
 | |
|   volume: Float;
 | |
| }>;
 | |
| 
 | |
| export type OnExternalPlaybackChangeData = Readonly<{
 | |
|   isExternalPlaybackActive: boolean;
 | |
| }>;
 | |
| 
 | |
| export type OnGetLicenseData = Readonly<{
 | |
|   licenseUrl: string;
 | |
|   loadedLicenseUrl: string;
 | |
|   contentId: string;
 | |
|   spcBase64: string;
 | |
| }>;
 | |
| 
 | |
| export type OnPictureInPictureStatusChangedData = Readonly<{
 | |
|   isActive: boolean;
 | |
| }>;
 | |
| 
 | |
| type OnReceiveAdEventData = Readonly<{
 | |
|   data?: {};
 | |
|   event: WithDefault<string, 'AD_BREAK_ENDED'>;
 | |
| }>;
 | |
| 
 | |
| export type OnVideoErrorData = Readonly<{
 | |
|   error: Readonly<{
 | |
|     errorString?: string; // android
 | |
|     errorException?: string; // android
 | |
|     errorStackTrace?: string; // android
 | |
|     errorCode?: string; // android
 | |
|     error?: string; // ios
 | |
|     code?: Int32; // ios
 | |
|     localizedDescription?: string; // ios
 | |
|     localizedFailureReason?: string; // ios
 | |
|     localizedRecoverySuggestion?: string; // ios
 | |
|     domain?: string; // ios
 | |
|   }>;
 | |
|   target?: Int32; // ios
 | |
| }>;
 | |
| 
 | |
| export type OnAudioFocusChangedData = Readonly<{
 | |
|   hasAudioFocus: boolean;
 | |
| }>;
 | |
| 
 | |
| type ControlsStyles = Readonly<{
 | |
|   hideSeekBar?: boolean;
 | |
|   seekIncrementMS?: number;
 | |
| }>;
 | |
| 
 | |
| export type OnControlsVisibilityChange = Readonly<{
 | |
|   isVisible: boolean;
 | |
| }>;
 | |
| 
 | |
| export interface VideoNativeProps extends ViewProps {
 | |
|   src?: VideoSrc;
 | |
|   adTagUrl?: string;
 | |
|   allowsExternalPlayback?: boolean; // ios, true
 | |
|   maxBitRate?: Float;
 | |
|   resizeMode?: WithDefault<string, 'none'>;
 | |
|   repeat?: boolean;
 | |
|   automaticallyWaitsToMinimizeStalling?: boolean;
 | |
|   textTracks?: TextTracks;
 | |
|   selectedTextTrack?: SelectedTextTrack;
 | |
|   selectedAudioTrack?: SelectedAudioTrack;
 | |
|   selectedVideoTrack?: SelectedVideoTrack; // android
 | |
|   paused?: boolean;
 | |
|   muted?: boolean;
 | |
|   controls?: boolean;
 | |
|   filter?: WithDefault<string, ''>;
 | |
|   filterEnabled?: boolean;
 | |
|   volume?: Float; // default 1.0
 | |
|   playInBackground?: boolean;
 | |
|   preventsDisplaySleepDuringVideoPlayback?: boolean;
 | |
|   preferredForwardBufferDuration?: Float; //ios, 0
 | |
|   playWhenInactive?: boolean; // ios, false
 | |
|   pictureInPicture?: boolean; // ios, false
 | |
|   ignoreSilentSwitch?: WithDefault<string, 'inherit'>; // ios, 'inherit'
 | |
|   mixWithOthers?: WithDefault<string, 'inherit'>; // ios, 'inherit'
 | |
|   rate?: Float;
 | |
|   fullscreen?: boolean; // ios, false
 | |
|   fullscreenAutorotate?: boolean;
 | |
|   fullscreenOrientation?: WithDefault<string, 'all'>;
 | |
|   progressUpdateInterval?: Float;
 | |
|   restoreUserInterfaceForPIPStopCompletionHandler?: boolean;
 | |
|   localSourceEncryptionKeyScheme?: string;
 | |
|   debug?: DebugConfig;
 | |
|   showNotificationControls?: WithDefault<boolean, false>; // Android, iOS
 | |
|   bufferConfig?: BufferConfig; // Android
 | |
|   contentStartTime?: Int32; // Android
 | |
|   currentPlaybackTime?: Double; // Android
 | |
|   disableDisconnectError?: boolean; // Android
 | |
|   focusable?: boolean; // Android
 | |
|   hideShutterView?: boolean; //	Android
 | |
|   minLoadRetryCount?: Int32; // Android
 | |
|   reportBandwidth?: boolean; //Android
 | |
|   subtitleStyle?: SubtitleStyle; // android
 | |
|   viewType?: Int32; // Android
 | |
|   bufferingStrategy?: BufferingStrategyType; // Android
 | |
|   controlsStyles?: ControlsStyles; // Android
 | |
|   onControlsVisibilityChange?: DirectEventHandler<OnControlsVisibilityChange>;
 | |
|   onVideoLoad?: DirectEventHandler<OnLoadData>;
 | |
|   onVideoLoadStart?: DirectEventHandler<OnLoadStartData>;
 | |
|   onVideoAspectRatio?: DirectEventHandler<OnVideoAspectRatioData>;
 | |
|   onVideoBuffer?: DirectEventHandler<OnBufferData>;
 | |
|   onVideoError?: DirectEventHandler<OnVideoErrorData>;
 | |
|   onVideoProgress?: DirectEventHandler<OnProgressData>;
 | |
|   onVideoBandwidthUpdate?: DirectEventHandler<OnBandwidthUpdateData>;
 | |
|   onVideoSeek?: DirectEventHandler<OnSeekData>;
 | |
|   onVideoEnd?: DirectEventHandler<{}>; // all
 | |
|   onVideoAudioBecomingNoisy?: DirectEventHandler<{}>;
 | |
|   onVideoFullscreenPlayerWillPresent?: DirectEventHandler<{}>; // ios, android
 | |
|   onVideoFullscreenPlayerDidPresent?: DirectEventHandler<{}>; // ios, android
 | |
|   onVideoFullscreenPlayerWillDismiss?: DirectEventHandler<{}>; // ios, android
 | |
|   onVideoFullscreenPlayerDidDismiss?: DirectEventHandler<{}>; // ios, android
 | |
|   onReadyForDisplay?: DirectEventHandler<{}>;
 | |
|   onPlaybackRateChange?: DirectEventHandler<OnPlaybackRateChangeData>; // all
 | |
|   onVolumeChange?: DirectEventHandler<OnVolumeChangeData>; // android, ios
 | |
|   onVideoExternalPlaybackChange?: DirectEventHandler<OnExternalPlaybackChangeData>;
 | |
|   onGetLicense?: DirectEventHandler<OnGetLicenseData>;
 | |
|   onPictureInPictureStatusChanged?: DirectEventHandler<OnPictureInPictureStatusChangedData>;
 | |
|   onRestoreUserInterfaceForPictureInPictureStop?: DirectEventHandler<{}>;
 | |
|   onReceiveAdEvent?: DirectEventHandler<OnReceiveAdEventData>;
 | |
|   onVideoPlaybackStateChanged?: DirectEventHandler<OnPlaybackStateChangedData>; // android only
 | |
|   onVideoIdle?: DirectEventHandler<{}>; // android only (nowhere in document, so do not use as props. just type declaration)
 | |
|   onAudioFocusChanged?: DirectEventHandler<OnAudioFocusChangedData>; // android only (nowhere in document, so do not use as props. just type declaration)
 | |
|   onTimedMetadata?: DirectEventHandler<OnTimedMetadataData>; // ios, android
 | |
|   onAudioTracks?: DirectEventHandler<OnAudioTracksData>; // android
 | |
|   onTextTracks?: DirectEventHandler<OnTextTracksData>; // android
 | |
|   onTextTrackDataChanged?: DirectEventHandler<OnTextTrackDataChangedData>; // iOS
 | |
|   onVideoTracks?: DirectEventHandler<OnVideoTracksData>; // android
 | |
| }
 | |
| 
 | |
| export type VideoComponentType = HostComponent<VideoNativeProps>;
 | |
| 
 | |
| export type VideoSaveData = {
 | |
|   uri: string;
 | |
| };
 | |
| 
 | |
| export interface VideoManagerType {
 | |
|   save: (option: object, reactTag: number) => Promise<VideoSaveData>;
 | |
|   seek: (option: Seek, reactTag: number) => Promise<void>;
 | |
|   setPlayerPauseState: (paused: boolean, reactTag: number) => Promise<void>;
 | |
|   setLicenseResult: (
 | |
|     result: string,
 | |
|     licenseUrl: string,
 | |
|     reactTag: number,
 | |
|   ) => Promise<void>;
 | |
|   setLicenseResultError: (
 | |
|     error: string,
 | |
|     licenseUrl: string,
 | |
|     reactTag: number,
 | |
|   ) => Promise<void>;
 | |
|   setVolume: (volume: number, reactTag: number) => Promise<void>;
 | |
|   getCurrentPosition: (reactTag: number) => Promise<number>;
 | |
|   setFullScreen: (fullScreen: boolean, reactTag: number) => Promise<void>;
 | |
| }
 | |
| 
 | |
| export interface VideoDecoderPropertiesType {
 | |
|   getWidevineLevel: () => Promise<number>;
 | |
|   isCodecSupported: (
 | |
|     mimeType: string,
 | |
|     width: number,
 | |
|     height: number,
 | |
|   ) => Promise<'unsupported' | 'hardware' | 'software'>;
 | |
|   isHEVCSupported: () => Promise<'unsupported' | 'hardware' | 'software'>;
 | |
| }
 | |
| 
 | |
| export const VideoManager = NativeModules.VideoManager as VideoManagerType;
 | |
| export const VideoDecoderProperties =
 | |
|   NativeModules.VideoDecoderProperties as VideoDecoderPropertiesType;
 | |
| 
 | |
| export default requireNativeComponent<VideoNativeProps>(
 | |
|   'RCTVideo',
 | |
| ) as VideoComponentType;
 |