2017-01-11 12:51:45 +00:00
|
|
|
package com.brentvatne.exoplayer;
|
|
|
|
|
2023-11-18 22:13:54 +09:00
|
|
|
import static androidx.media3.common.C.CONTENT_TYPE_DASH;
|
|
|
|
import static androidx.media3.common.C.CONTENT_TYPE_HLS;
|
|
|
|
import static androidx.media3.common.C.CONTENT_TYPE_OTHER;
|
2024-04-16 10:41:39 +02:00
|
|
|
import static androidx.media3.common.C.CONTENT_TYPE_RTSP;
|
2023-11-18 22:13:54 +09:00
|
|
|
import static androidx.media3.common.C.CONTENT_TYPE_SS;
|
|
|
|
import static androidx.media3.common.C.TIME_END_OF_SOURCE;
|
2023-01-02 22:59:10 +01:00
|
|
|
|
2017-01-11 12:51:45 +00:00
|
|
|
import android.annotation.SuppressLint;
|
2018-05-17 15:42:44 -07:00
|
|
|
import android.app.Activity;
|
2021-11-03 19:14:19 -04:00
|
|
|
import android.app.ActivityManager;
|
2024-10-05 20:04:25 +03:30
|
|
|
import android.app.AlertDialog;
|
2024-05-07 12:30:57 +02:00
|
|
|
import android.content.ComponentName;
|
2017-01-11 12:51:45 +00:00
|
|
|
import android.content.Context;
|
2024-05-07 12:30:57 +02:00
|
|
|
import android.content.Intent;
|
|
|
|
import android.content.ServiceConnection;
|
2017-01-11 12:51:45 +00:00
|
|
|
import android.media.AudioManager;
|
|
|
|
import android.net.Uri;
|
2024-05-07 12:30:57 +02:00
|
|
|
import android.os.Build;
|
2017-01-11 12:51:45 +00:00
|
|
|
import android.os.Handler;
|
2024-05-07 12:30:57 +02:00
|
|
|
import android.os.IBinder;
|
2022-03-30 14:29:08 +03:00
|
|
|
import android.os.Looper;
|
2017-01-11 12:51:45 +00:00
|
|
|
import android.os.Message;
|
|
|
|
import android.text.TextUtils;
|
2018-05-17 15:42:44 -07:00
|
|
|
import android.view.View;
|
2018-06-21 09:08:37 -07:00
|
|
|
import android.view.accessibility.CaptioningManager;
|
2017-01-11 12:51:45 +00:00
|
|
|
import android.widget.FrameLayout;
|
2020-02-20 19:53:23 +05:30
|
|
|
import android.widget.ImageButton;
|
2024-05-20 14:15:18 +03:30
|
|
|
import android.widget.LinearLayout;
|
|
|
|
import android.widget.TextView;
|
2017-01-11 12:51:45 +00:00
|
|
|
|
2023-11-18 22:13:54 +09:00
|
|
|
import androidx.activity.OnBackPressedCallback;
|
2023-10-12 10:36:43 +02:00
|
|
|
import androidx.annotation.NonNull;
|
2024-10-11 22:50:22 +02:00
|
|
|
import androidx.annotation.Nullable;
|
2022-08-06 12:05:07 +02:00
|
|
|
import androidx.annotation.WorkerThread;
|
2023-11-18 22:13:54 +09:00
|
|
|
import androidx.media3.common.AudioAttributes;
|
|
|
|
import androidx.media3.common.C;
|
|
|
|
import androidx.media3.common.Format;
|
|
|
|
import androidx.media3.common.MediaItem;
|
2024-05-07 12:30:57 +02:00
|
|
|
import androidx.media3.common.MediaMetadata;
|
2023-11-18 22:13:54 +09:00
|
|
|
import androidx.media3.common.Metadata;
|
|
|
|
import androidx.media3.common.PlaybackException;
|
|
|
|
import androidx.media3.common.PlaybackParameters;
|
|
|
|
import androidx.media3.common.Player;
|
|
|
|
import androidx.media3.common.StreamKey;
|
|
|
|
import androidx.media3.common.Timeline;
|
|
|
|
import androidx.media3.common.TrackGroup;
|
|
|
|
import androidx.media3.common.TrackSelectionOverride;
|
|
|
|
import androidx.media3.common.Tracks;
|
2024-03-11 21:50:19 +09:00
|
|
|
import androidx.media3.common.text.CueGroup;
|
2023-11-18 22:13:54 +09:00
|
|
|
import androidx.media3.common.util.Util;
|
|
|
|
import androidx.media3.datasource.DataSource;
|
|
|
|
import androidx.media3.datasource.DataSpec;
|
|
|
|
import androidx.media3.datasource.HttpDataSource;
|
|
|
|
import androidx.media3.exoplayer.DefaultLoadControl;
|
|
|
|
import androidx.media3.exoplayer.DefaultRenderersFactory;
|
|
|
|
import androidx.media3.exoplayer.ExoPlayer;
|
|
|
|
import androidx.media3.exoplayer.dash.DashMediaSource;
|
|
|
|
import androidx.media3.exoplayer.dash.DashUtil;
|
|
|
|
import androidx.media3.exoplayer.dash.DefaultDashChunkSource;
|
|
|
|
import androidx.media3.exoplayer.dash.manifest.AdaptationSet;
|
|
|
|
import androidx.media3.exoplayer.dash.manifest.DashManifest;
|
|
|
|
import androidx.media3.exoplayer.dash.manifest.Period;
|
|
|
|
import androidx.media3.exoplayer.dash.manifest.Representation;
|
|
|
|
import androidx.media3.exoplayer.drm.DefaultDrmSessionManager;
|
|
|
|
import androidx.media3.exoplayer.drm.DefaultDrmSessionManagerProvider;
|
|
|
|
import androidx.media3.exoplayer.drm.DrmSessionEventListener;
|
|
|
|
import androidx.media3.exoplayer.drm.DrmSessionManager;
|
|
|
|
import androidx.media3.exoplayer.drm.DrmSessionManagerProvider;
|
|
|
|
import androidx.media3.exoplayer.drm.FrameworkMediaDrm;
|
|
|
|
import androidx.media3.exoplayer.drm.HttpMediaDrmCallback;
|
|
|
|
import androidx.media3.exoplayer.drm.UnsupportedDrmException;
|
|
|
|
import androidx.media3.exoplayer.hls.HlsMediaSource;
|
2023-11-24 13:17:13 +01:00
|
|
|
import androidx.media3.exoplayer.ima.ImaAdsLoader;
|
2023-11-18 22:13:54 +09:00
|
|
|
import androidx.media3.exoplayer.mediacodec.MediaCodecInfo;
|
|
|
|
import androidx.media3.exoplayer.mediacodec.MediaCodecUtil;
|
2024-04-16 10:41:39 +02:00
|
|
|
import androidx.media3.exoplayer.rtsp.RtspMediaSource;
|
2023-11-18 22:13:54 +09:00
|
|
|
import androidx.media3.exoplayer.smoothstreaming.DefaultSsChunkSource;
|
|
|
|
import androidx.media3.exoplayer.smoothstreaming.SsMediaSource;
|
|
|
|
import androidx.media3.exoplayer.source.ClippingMediaSource;
|
|
|
|
import androidx.media3.exoplayer.source.DefaultMediaSourceFactory;
|
|
|
|
import androidx.media3.exoplayer.source.MediaSource;
|
|
|
|
import androidx.media3.exoplayer.source.MergingMediaSource;
|
|
|
|
import androidx.media3.exoplayer.source.ProgressiveMediaSource;
|
|
|
|
import androidx.media3.exoplayer.source.TrackGroupArray;
|
|
|
|
import androidx.media3.exoplayer.source.ads.AdsMediaSource;
|
|
|
|
import androidx.media3.exoplayer.trackselection.AdaptiveTrackSelection;
|
|
|
|
import androidx.media3.exoplayer.trackselection.DefaultTrackSelector;
|
|
|
|
import androidx.media3.exoplayer.trackselection.ExoTrackSelection;
|
|
|
|
import androidx.media3.exoplayer.trackselection.MappingTrackSelector;
|
|
|
|
import androidx.media3.exoplayer.trackselection.TrackSelection;
|
|
|
|
import androidx.media3.exoplayer.trackselection.TrackSelectionArray;
|
|
|
|
import androidx.media3.exoplayer.upstream.BandwidthMeter;
|
2024-08-22 17:47:51 +09:00
|
|
|
import androidx.media3.exoplayer.upstream.CmcdConfiguration;
|
2023-11-18 22:13:54 +09:00
|
|
|
import androidx.media3.exoplayer.upstream.DefaultAllocator;
|
|
|
|
import androidx.media3.exoplayer.upstream.DefaultBandwidthMeter;
|
2024-05-17 15:10:37 +02:00
|
|
|
import androidx.media3.exoplayer.util.EventLogger;
|
2023-11-18 22:13:54 +09:00
|
|
|
import androidx.media3.extractor.metadata.emsg.EventMessage;
|
|
|
|
import androidx.media3.extractor.metadata.id3.Id3Frame;
|
|
|
|
import androidx.media3.extractor.metadata.id3.TextInformationFrame;
|
2024-05-07 12:30:57 +02:00
|
|
|
import androidx.media3.session.MediaSessionService;
|
2023-11-18 22:13:54 +09:00
|
|
|
import androidx.media3.ui.LegacyPlayerControlView;
|
2022-08-18 01:12:08 -07:00
|
|
|
|
2024-10-10 23:53:39 +02:00
|
|
|
import com.brentvatne.common.api.AdsProps;
|
2024-05-06 22:04:40 +02:00
|
|
|
import com.brentvatne.common.api.BufferConfig;
|
2024-05-11 22:02:04 +02:00
|
|
|
import com.brentvatne.common.api.BufferingStrategy;
|
2024-05-20 14:15:18 +03:30
|
|
|
import com.brentvatne.common.api.ControlsConfig;
|
2024-06-10 22:42:18 +02:00
|
|
|
import com.brentvatne.common.api.DRMProps;
|
2023-12-07 08:47:40 +01:00
|
|
|
import com.brentvatne.common.api.ResizeMode;
|
2024-05-11 18:57:59 +02:00
|
|
|
import com.brentvatne.common.api.SideLoadedTextTrack;
|
2024-05-30 08:53:49 +02:00
|
|
|
import com.brentvatne.common.api.Source;
|
2023-12-07 08:47:40 +01:00
|
|
|
import com.brentvatne.common.api.SubtitleStyle;
|
|
|
|
import com.brentvatne.common.api.TimedMetadata;
|
|
|
|
import com.brentvatne.common.api.Track;
|
|
|
|
import com.brentvatne.common.api.VideoTrack;
|
2023-10-12 21:46:40 +02:00
|
|
|
import com.brentvatne.common.react.VideoEventEmitter;
|
2023-10-10 09:47:56 +02:00
|
|
|
import com.brentvatne.common.toolbox.DebugLog;
|
2024-09-13 10:50:33 +02:00
|
|
|
import com.brentvatne.common.toolbox.ReactBridgeUtils;
|
2024-04-16 14:23:19 +02:00
|
|
|
import com.brentvatne.react.BuildConfig;
|
2017-01-11 12:51:45 +00:00
|
|
|
import com.brentvatne.react.R;
|
2024-06-25 08:55:32 +02:00
|
|
|
import com.brentvatne.react.ReactNativeVideoManager;
|
2017-01-11 12:51:45 +00:00
|
|
|
import com.brentvatne.receiver.AudioBecomingNoisyReceiver;
|
|
|
|
import com.brentvatne.receiver.BecomingNoisyListener;
|
|
|
|
import com.facebook.react.bridge.LifecycleEventListener;
|
2024-05-28 02:00:38 -07:00
|
|
|
import com.facebook.react.bridge.Promise;
|
2023-12-01 05:16:24 +09:00
|
|
|
import com.facebook.react.bridge.UiThreadUtil;
|
2017-01-11 12:51:45 +00:00
|
|
|
import com.facebook.react.uimanager.ThemedReactContext;
|
2024-03-21 14:07:53 +01:00
|
|
|
import com.google.ads.interactivemedia.v3.api.AdError;
|
2023-12-02 13:52:01 +01:00
|
|
|
import com.google.ads.interactivemedia.v3.api.AdErrorEvent;
|
2024-09-03 12:46:20 +03:30
|
|
|
import com.google.ads.interactivemedia.v3.api.AdEvent;
|
2024-08-29 13:30:05 +03:00
|
|
|
import com.google.ads.interactivemedia.v3.api.ImaSdkFactory;
|
2024-09-03 12:46:20 +03:30
|
|
|
import com.google.ads.interactivemedia.v3.api.ImaSdkSettings;
|
2022-11-16 11:43:25 +01:00
|
|
|
import com.google.common.collect.ImmutableList;
|
2023-11-18 22:13:54 +09:00
|
|
|
|
2017-01-11 12:51:45 +00:00
|
|
|
import java.net.CookieHandler;
|
|
|
|
import java.net.CookieManager;
|
|
|
|
import java.net.CookiePolicy;
|
2023-11-18 22:13:54 +09:00
|
|
|
import java.util.ArrayList;
|
2024-09-03 12:46:20 +03:30
|
|
|
import java.util.List;
|
2018-07-09 11:36:35 -07:00
|
|
|
import java.util.Locale;
|
2024-07-04 21:01:28 +09:00
|
|
|
import java.util.Map;
|
2024-10-11 22:50:22 +02:00
|
|
|
import java.util.Objects;
|
2021-12-12 13:47:50 +07:00
|
|
|
import java.util.UUID;
|
2023-11-18 22:13:54 +09:00
|
|
|
import java.util.concurrent.Callable;
|
2021-11-09 14:22:32 +02:00
|
|
|
import java.util.concurrent.ExecutorService;
|
|
|
|
import java.util.concurrent.Executors;
|
|
|
|
import java.util.concurrent.Future;
|
2022-03-31 15:37:53 +03:00
|
|
|
import java.util.concurrent.TimeUnit;
|
2017-01-11 12:51:45 +00:00
|
|
|
|
|
|
|
@SuppressLint("ViewConstructor")
|
2023-09-18 13:09:53 +02:00
|
|
|
public class ReactExoplayerView extends FrameLayout implements
|
2017-01-11 12:51:45 +00:00
|
|
|
LifecycleEventListener,
|
2022-06-16 00:24:55 +07:00
|
|
|
Player.Listener,
|
2018-11-01 21:41:57 +05:30
|
|
|
BandwidthMeter.EventListener,
|
2017-01-11 12:51:45 +00:00
|
|
|
BecomingNoisyListener,
|
2022-11-09 14:26:39 +01:00
|
|
|
DrmSessionEventListener,
|
2023-12-02 13:52:01 +01:00
|
|
|
AdEvent.AdEventListener,
|
|
|
|
AdErrorEvent.AdErrorListener {
|
2017-01-11 12:51:45 +00:00
|
|
|
|
2021-11-04 13:54:43 -04:00
|
|
|
public static final double DEFAULT_MAX_HEAP_ALLOCATION_PERCENT = 1;
|
2022-01-21 14:10:22 +02:00
|
|
|
public static final double DEFAULT_MIN_BUFFER_MEMORY_RESERVE = 0;
|
2021-11-04 13:54:43 -04:00
|
|
|
|
2017-01-11 12:51:45 +00:00
|
|
|
private static final String TAG = "ReactExoplayerView";
|
|
|
|
|
|
|
|
private static final CookieManager DEFAULT_COOKIE_MANAGER;
|
|
|
|
private static final int SHOW_PROGRESS = 1;
|
|
|
|
|
|
|
|
static {
|
|
|
|
DEFAULT_COOKIE_MANAGER = new CookieManager();
|
|
|
|
DEFAULT_COOKIE_MANAGER.setCookiePolicy(CookiePolicy.ACCEPT_ORIGINAL_SERVER);
|
|
|
|
}
|
|
|
|
|
2024-07-04 21:01:28 +09:00
|
|
|
protected final VideoEventEmitter eventEmitter;
|
2019-09-16 16:29:31 -04:00
|
|
|
private final ReactExoplayerConfig config;
|
|
|
|
private final DefaultBandwidthMeter bandwidthMeter;
|
2023-11-18 22:13:54 +09:00
|
|
|
private LegacyPlayerControlView playerControlView;
|
2019-02-04 19:18:29 +05:30
|
|
|
private View playPauseControlContainer;
|
2022-06-16 00:24:55 +07:00
|
|
|
private Player.Listener eventListener;
|
2019-09-16 16:29:31 -04:00
|
|
|
|
2017-01-11 12:51:45 +00:00
|
|
|
private ExoPlayerView exoPlayerView;
|
2022-07-14 11:10:18 -07:00
|
|
|
private FullScreenPlayerView fullScreenPlayerView;
|
2022-11-16 20:46:36 +01:00
|
|
|
private ImaAdsLoader adsLoader;
|
2017-01-11 12:51:45 +00:00
|
|
|
|
|
|
|
private DataSource.Factory mediaDataSourceFactory;
|
2022-06-16 00:24:55 +07:00
|
|
|
private ExoPlayer player;
|
2018-08-07 23:10:03 -07:00
|
|
|
private DefaultTrackSelector trackSelector;
|
2017-01-11 12:51:45 +00:00
|
|
|
private boolean playerNeedsSource;
|
2024-05-07 12:30:57 +02:00
|
|
|
private ServiceConnection playbackServiceConnection;
|
|
|
|
private PlaybackServiceBinder playbackServiceBinder;
|
2017-01-11 12:51:45 +00:00
|
|
|
|
2024-05-17 15:10:37 +02:00
|
|
|
// logger to be enable by props
|
|
|
|
private EventLogger debugEventLogger = null;
|
|
|
|
private boolean enableDebug = false;
|
|
|
|
private static final String TAG_EVENT_LOGGER = "RNVExoplayer";
|
|
|
|
|
2017-03-21 20:25:17 +00:00
|
|
|
private int resumeWindow;
|
|
|
|
private long resumePosition;
|
2017-01-11 12:51:45 +00:00
|
|
|
private boolean loadVideoStarted;
|
2018-05-17 15:42:44 -07:00
|
|
|
private boolean isFullscreen;
|
2018-06-09 16:11:18 -07:00
|
|
|
private boolean isInBackground;
|
2018-07-09 16:18:42 -07:00
|
|
|
private boolean isPaused;
|
2017-01-11 12:51:45 +00:00
|
|
|
private boolean isBuffering;
|
2019-08-01 11:33:28 -07:00
|
|
|
private boolean muted = false;
|
2021-06-24 08:00:38 +03:00
|
|
|
private boolean hasAudioFocus = false;
|
2017-06-14 00:45:12 +02:00
|
|
|
private float rate = 1f;
|
2023-04-02 14:02:56 -04:00
|
|
|
private AudioOutput audioOutput = AudioOutput.SPEAKER;
|
2018-11-14 12:33:28 +01:00
|
|
|
private float audioVolume = 1f;
|
2024-05-06 22:04:40 +02:00
|
|
|
private BufferConfig bufferConfig = new BufferConfig();
|
2018-11-26 14:50:31 -08:00
|
|
|
private int maxBitRate = 0;
|
2021-10-13 18:03:29 +03:00
|
|
|
private boolean hasDrmFailed = false;
|
2021-11-09 14:22:32 +02:00
|
|
|
private boolean isUsingContentResolution = false;
|
|
|
|
private boolean selectTrackWhenReady = false;
|
2024-07-10 12:17:22 +02:00
|
|
|
private final Handler mainHandler;
|
2023-11-18 22:13:54 +09:00
|
|
|
private Runnable mainRunnable;
|
2024-05-28 00:29:21 -07:00
|
|
|
private boolean useCache = false;
|
2024-05-20 14:15:18 +03:30
|
|
|
private ControlsConfig controlsConfig = new ControlsConfig();
|
2018-08-01 15:58:02 +02:00
|
|
|
|
2024-07-11 10:08:36 +02:00
|
|
|
/*
|
|
|
|
* When user is seeking first called is on onPositionDiscontinuity -> DISCONTINUITY_REASON_SEEK
|
|
|
|
* Then we set if to false when playback is back in onIsPlayingChanged -> true
|
|
|
|
*/
|
|
|
|
private boolean isSeeking = false;
|
|
|
|
private long seekPosition = -1;
|
|
|
|
|
2017-01-11 12:51:45 +00:00
|
|
|
// Props from React
|
2024-05-30 08:53:49 +02:00
|
|
|
private Source source = new Source();
|
2017-01-11 12:51:45 +00:00
|
|
|
private boolean repeat;
|
2018-07-17 14:14:21 -07:00
|
|
|
private String audioTrackType;
|
2024-03-22 07:58:09 +01:00
|
|
|
private String audioTrackValue;
|
2018-08-24 15:33:46 +05:30
|
|
|
private String videoTrackType;
|
2024-03-22 07:58:09 +01:00
|
|
|
private String videoTrackValue;
|
2024-09-13 10:50:33 +02:00
|
|
|
private String textTrackType = "disabled";
|
2024-03-22 07:58:09 +01:00
|
|
|
private String textTrackValue;
|
2017-01-11 12:51:45 +00:00
|
|
|
private boolean disableFocus;
|
2022-09-26 01:51:18 +01:00
|
|
|
private boolean focusable = true;
|
2024-05-11 22:02:04 +02:00
|
|
|
private BufferingStrategy.BufferingStrategyEnum bufferingStrategy;
|
2021-05-17 13:09:09 +03:00
|
|
|
private boolean disableDisconnectError;
|
2020-06-16 14:31:23 +02:00
|
|
|
private boolean preventsDisplaySleepDuringVideoPlayback = true;
|
2017-03-31 18:15:39 +02:00
|
|
|
private float mProgressUpdateInterval = 250.0f;
|
2017-05-09 06:30:45 +10:00
|
|
|
private boolean playInBackground = false;
|
2018-08-25 21:53:11 +05:30
|
|
|
private boolean mReportBandwidth = false;
|
2019-07-07 22:17:15 +02:00
|
|
|
private boolean controls;
|
2024-05-07 12:30:57 +02:00
|
|
|
|
|
|
|
private boolean showNotificationControls = false;
|
2017-01-11 12:51:45 +00:00
|
|
|
// \ End props
|
|
|
|
|
|
|
|
// React
|
|
|
|
private final ThemedReactContext themedReactContext;
|
|
|
|
private final AudioManager audioManager;
|
|
|
|
private final AudioBecomingNoisyReceiver audioBecomingNoisyReceiver;
|
2023-05-09 11:45:56 +09:00
|
|
|
private final AudioManager.OnAudioFocusChangeListener audioFocusChangeListener;
|
2017-01-11 12:51:45 +00:00
|
|
|
|
2022-04-27 21:26:37 +02:00
|
|
|
// store last progress event values to avoid sending unnecessary messages
|
|
|
|
private long lastPos = -1;
|
|
|
|
private long lastBufferDuration = -1;
|
|
|
|
private long lastDuration = -1;
|
|
|
|
|
2024-06-07 13:54:14 +02:00
|
|
|
private boolean viewHasDropped = false;
|
2024-10-05 20:04:25 +03:30
|
|
|
private int selectedSpeedIndex = 1; // Default is 1.0x
|
2024-06-25 08:55:32 +02:00
|
|
|
|
2024-09-03 12:46:20 +03:30
|
|
|
private final String instanceId = String.valueOf(UUID.randomUUID());
|
2024-06-25 08:55:32 +02:00
|
|
|
|
2024-08-22 17:47:51 +09:00
|
|
|
private CmcdConfiguration.Factory cmcdConfigurationFactory;
|
|
|
|
|
|
|
|
public void setCmcdConfigurationFactory(CmcdConfiguration.Factory factory) {
|
|
|
|
this.cmcdConfigurationFactory = factory;
|
|
|
|
}
|
|
|
|
|
2024-06-03 12:13:52 +02:00
|
|
|
private void updateProgress() {
|
|
|
|
if (player != null) {
|
|
|
|
if (playerControlView != null && isPlayingAd() && controls) {
|
|
|
|
playerControlView.hide();
|
|
|
|
}
|
|
|
|
long bufferedDuration = player.getBufferedPercentage() * player.getDuration() / 100;
|
|
|
|
long duration = player.getDuration();
|
|
|
|
long pos = player.getCurrentPosition();
|
|
|
|
if (pos > duration) {
|
|
|
|
pos = duration;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (lastPos != pos
|
|
|
|
|| lastBufferDuration != bufferedDuration
|
|
|
|
|| lastDuration != duration) {
|
|
|
|
lastPos = pos;
|
|
|
|
lastBufferDuration = bufferedDuration;
|
|
|
|
lastDuration = duration;
|
2024-07-04 21:01:28 +09:00
|
|
|
eventEmitter.onVideoProgress.invoke(pos, bufferedDuration, player.getDuration(), getPositionInFirstPeriodMsForCurrentWindow(pos));
|
2024-06-03 12:13:52 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-06-16 00:24:55 +07:00
|
|
|
private final Handler progressHandler = new Handler(Looper.getMainLooper()) {
|
2017-01-11 12:51:45 +00:00
|
|
|
@Override
|
|
|
|
public void handleMessage(Message msg) {
|
2024-05-22 14:02:55 +02:00
|
|
|
if (msg.what == SHOW_PROGRESS) {
|
2024-06-03 12:13:52 +02:00
|
|
|
updateProgress();
|
|
|
|
msg = obtainMessage(SHOW_PROGRESS);
|
|
|
|
sendMessageDelayed(msg, Math.round(mProgressUpdateInterval));
|
2017-01-11 12:51:45 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
2021-03-18 03:58:04 -07:00
|
|
|
|
2020-05-15 12:55:19 +05:30
|
|
|
public double getPositionInFirstPeriodMsForCurrentWindow(long currentPosition) {
|
|
|
|
Timeline.Window window = new Timeline.Window();
|
2023-08-22 23:30:01 -04:00
|
|
|
if(!player.getCurrentTimeline().isEmpty()) {
|
2022-06-16 00:24:55 +07:00
|
|
|
player.getCurrentTimeline().getWindow(player.getCurrentMediaItemIndex(), window);
|
2020-05-15 12:55:19 +05:30
|
|
|
}
|
|
|
|
return window.windowStartTimeMs + currentPosition;
|
|
|
|
}
|
2017-01-11 12:51:45 +00:00
|
|
|
|
2019-09-16 16:29:31 -04:00
|
|
|
public ReactExoplayerView(ThemedReactContext context, ReactExoplayerConfig config) {
|
2017-01-11 12:51:45 +00:00
|
|
|
super(context);
|
2018-01-29 13:25:58 -07:00
|
|
|
this.themedReactContext = context;
|
2024-07-04 21:01:28 +09:00
|
|
|
this.eventEmitter = new VideoEventEmitter();
|
2019-09-16 16:29:31 -04:00
|
|
|
this.config = config;
|
|
|
|
this.bandwidthMeter = config.getBandwidthMeter();
|
2024-07-10 12:17:22 +02:00
|
|
|
mainHandler = new Handler();
|
2017-01-11 12:51:45 +00:00
|
|
|
createViews();
|
2019-09-16 16:29:31 -04:00
|
|
|
|
2017-01-11 12:51:45 +00:00
|
|
|
audioManager = (AudioManager) context.getSystemService(Context.AUDIO_SERVICE);
|
|
|
|
themedReactContext.addLifecycleEventListener(this);
|
|
|
|
audioBecomingNoisyReceiver = new AudioBecomingNoisyReceiver(themedReactContext);
|
2023-10-18 22:45:19 +02:00
|
|
|
audioFocusChangeListener = new OnAudioFocusChangedListener(this, themedReactContext);
|
2017-01-11 12:51:45 +00:00
|
|
|
}
|
|
|
|
|
2022-11-09 14:26:39 +01:00
|
|
|
private boolean isPlayingAd() {
|
2022-11-16 20:46:36 +01:00
|
|
|
return player != null && player.isPlayingAd();
|
2022-11-09 14:26:39 +01:00
|
|
|
}
|
2017-01-11 12:51:45 +00:00
|
|
|
|
|
|
|
private void createViews() {
|
|
|
|
if (CookieHandler.getDefault() != DEFAULT_COOKIE_MANAGER) {
|
|
|
|
CookieHandler.setDefault(DEFAULT_COOKIE_MANAGER);
|
|
|
|
}
|
|
|
|
|
|
|
|
LayoutParams layoutParams = new LayoutParams(
|
|
|
|
LayoutParams.MATCH_PARENT,
|
|
|
|
LayoutParams.MATCH_PARENT);
|
|
|
|
exoPlayerView = new ExoPlayerView(getContext());
|
|
|
|
exoPlayerView.setLayoutParams(layoutParams);
|
|
|
|
addView(exoPlayerView, 0, layoutParams);
|
Add iOS and Android basic DRM support (#1445)
This PR adds support for DRM streams on iOS (Fairplay) and Android (Playready, Widevine, Clearkey)
I am neither Android nor iOS developer, so feel free to provide feedback to improve this PR.
**Test stream for ANDROID:**
```
testStream = {
uri: 'http://profficialsite.origin.mediaservices.windows.net/c51358ea-9a5e-4322-8951-897d640fdfd7/tearsofsteel_4k.ism/manifest(format=mpd-time-csf)',
type: 'mpd',
drm: {
type: DRMType.PLAYREADY,
licenseServer: 'http://test.playready.microsoft.com/service/rightsmanager.asmx?cfg=(persist:false,sl:150)'
}
};
```
or
```
{
uri: 'https://media.axprod.net/TestVectors/v7-MultiDRM-SingleKey/Manifest_1080p.mpd',
drm: {
type: 'widevine', //or DRMType.WIDEVINE
licenseServer: 'https://drm-widevine-licensing.axtest.net/AcquireLicense',
headers: {
'X-AxDRM-Message': 'eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJ2ZXJzaW9uIjoxLCJjb21fa2V5X2lkIjoiYjMzNjRlYjUtNTFmNi00YWUzLThjOTgtMzNjZWQ1ZTMxYzc4IiwibWVzc2FnZSI6eyJ0eXBlIjoiZW50aXRsZW1lbnRfbWVzc2FnZSIsImZpcnN0X3BsYXlfZXhwaXJhdGlvbiI6NjAsInBsYXlyZWFkeSI6eyJyZWFsX3RpbWVfZXhwaXJhdGlvbiI6dHJ1ZX0sImtleXMiOlt7ImlkIjoiOWViNDA1MGQtZTQ0Yi00ODAyLTkzMmUtMjdkNzUwODNlMjY2IiwiZW5jcnlwdGVkX2tleSI6ImxLM09qSExZVzI0Y3Iya3RSNzRmbnc9PSJ9XX19.FAbIiPxX8BHi9RwfzD7Yn-wugU19ghrkBFKsaCPrZmU'
},
}
}
```
**Test stream for iOS:**
Sorry but I can not provide free streams to test. If anyone can provide test streams, or found some we can use, please let me know to also test them.
It has been tested with a private provider and they work, at least with the `getLicense` override method. (An example implementation is provided in the README)
2020-08-13 03:56:21 +02:00
|
|
|
|
2022-09-26 01:51:18 +01:00
|
|
|
exoPlayerView.setFocusable(this.focusable);
|
2017-01-11 12:51:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// LifecycleEventListener implementation
|
|
|
|
@Override
|
|
|
|
public void onHostResume() {
|
2018-06-09 16:11:18 -07:00
|
|
|
if (!playInBackground || !isInBackground) {
|
|
|
|
setPlayWhenReady(!isPaused);
|
|
|
|
}
|
|
|
|
isInBackground = false;
|
2017-01-11 12:51:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void onHostPause() {
|
2018-06-09 16:11:18 -07:00
|
|
|
isInBackground = true;
|
2017-05-09 06:30:45 +10:00
|
|
|
if (playInBackground) {
|
|
|
|
return;
|
|
|
|
}
|
2017-01-11 12:51:45 +00:00
|
|
|
setPlayWhenReady(false);
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void onHostDestroy() {
|
2024-03-29 20:59:58 +01:00
|
|
|
cleanUpResources();
|
2017-01-11 12:51:45 +00:00
|
|
|
}
|
|
|
|
|
2024-05-07 12:30:57 +02:00
|
|
|
@Override
|
|
|
|
protected void onDetachedFromWindow() {
|
|
|
|
cleanupPlaybackService();
|
|
|
|
super.onDetachedFromWindow();
|
|
|
|
}
|
|
|
|
|
2017-01-11 12:51:45 +00:00
|
|
|
public void cleanUpResources() {
|
|
|
|
stopPlayback();
|
2024-01-20 21:29:29 +09:00
|
|
|
themedReactContext.removeLifecycleEventListener(this);
|
2024-06-07 13:54:14 +02:00
|
|
|
releasePlayer();
|
|
|
|
viewHasDropped = true;
|
2017-01-11 12:51:45 +00:00
|
|
|
}
|
|
|
|
|
2023-08-22 23:30:01 -04:00
|
|
|
//BandwidthMeter.EventListener implementation
|
2018-11-01 21:41:57 +05:30
|
|
|
@Override
|
|
|
|
public void onBandwidthSample(int elapsedMs, long bytes, long bitrate) {
|
2018-12-31 21:33:02 -08:00
|
|
|
if (mReportBandwidth) {
|
2020-05-15 12:55:19 +05:30
|
|
|
if (player == null) {
|
2024-09-17 15:41:02 +03:30
|
|
|
eventEmitter.onVideoBandwidthUpdate.invoke(bitrate, 0, 0, null);
|
2020-05-15 12:55:19 +05:30
|
|
|
} else {
|
|
|
|
Format videoFormat = player.getVideoFormat();
|
2024-09-17 15:41:02 +03:30
|
|
|
boolean isRotatedContent = videoFormat != null && (videoFormat.rotationDegrees == 90 || videoFormat.rotationDegrees == 270);
|
|
|
|
int width = videoFormat != null ? (isRotatedContent ? videoFormat.height : videoFormat.width) : 0;
|
|
|
|
int height = videoFormat != null ? (isRotatedContent ? videoFormat.width : videoFormat.height) : 0;
|
|
|
|
String trackId = videoFormat != null ? videoFormat.id : null;
|
2024-07-04 21:01:28 +09:00
|
|
|
eventEmitter.onVideoBandwidthUpdate.invoke(bitrate, height, width, trackId);
|
2020-05-15 12:55:19 +05:30
|
|
|
}
|
2018-11-01 21:41:57 +05:30
|
|
|
}
|
2018-11-01 15:18:59 +05:30
|
|
|
}
|
2017-01-11 12:51:45 +00:00
|
|
|
|
2018-11-01 21:41:57 +05:30
|
|
|
// Internal methods
|
2019-01-04 14:58:32 +05:30
|
|
|
|
|
|
|
/**
|
2019-09-16 16:29:31 -04:00
|
|
|
* Toggling the visibility of the player control view
|
2019-01-04 14:58:32 +05:30
|
|
|
*/
|
|
|
|
private void togglePlayerControlVisibility() {
|
2023-11-18 22:13:54 +09:00
|
|
|
if (player == null) return;
|
2024-01-26 21:34:07 +01:00
|
|
|
reLayoutControls();
|
2019-02-10 18:15:30 -08:00
|
|
|
if (playerControlView.isVisible()) {
|
2019-01-28 14:50:51 +05:30
|
|
|
playerControlView.hide();
|
2019-01-04 14:58:32 +05:30
|
|
|
} else {
|
2019-01-28 14:50:51 +05:30
|
|
|
playerControlView.show();
|
2019-01-04 14:58:32 +05:30
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2019-01-16 23:47:32 +05:30
|
|
|
* Initializing Player control
|
2019-01-04 14:58:32 +05:30
|
|
|
*/
|
2019-01-16 23:47:32 +05:30
|
|
|
private void initializePlayerControl() {
|
2019-02-10 18:15:30 -08:00
|
|
|
if (playerControlView == null) {
|
2023-11-18 22:13:54 +09:00
|
|
|
playerControlView = new LegacyPlayerControlView(getContext());
|
2024-06-22 02:15:21 -07:00
|
|
|
playerControlView.addVisibilityListener(new LegacyPlayerControlView.VisibilityListener() {
|
|
|
|
@Override
|
|
|
|
public void onVisibilityChange(int visibility) {
|
2024-07-04 21:01:28 +09:00
|
|
|
eventEmitter.onControlsVisibilityChange.invoke(visibility == View.VISIBLE);
|
2024-06-22 02:15:21 -07:00
|
|
|
}
|
|
|
|
});
|
2019-01-25 15:54:53 +05:30
|
|
|
}
|
2019-01-04 14:58:32 +05:30
|
|
|
|
2019-02-10 18:15:30 -08:00
|
|
|
// Setting the player for the playerControlView
|
2019-01-04 14:58:32 +05:30
|
|
|
playerControlView.setPlayer(player);
|
2019-02-04 19:18:29 +05:30
|
|
|
playPauseControlContainer = playerControlView.findViewById(R.id.exo_play_pause_container);
|
2019-01-04 14:58:32 +05:30
|
|
|
|
2019-02-10 18:15:30 -08:00
|
|
|
// Invoking onClick event for exoplayerView
|
2023-11-18 22:13:54 +09:00
|
|
|
exoPlayerView.setOnClickListener((View v) -> {
|
|
|
|
if (!isPlayingAd()) {
|
|
|
|
togglePlayerControlVisibility();
|
2019-01-04 14:58:32 +05:30
|
|
|
}
|
|
|
|
});
|
2019-02-04 19:18:29 +05:30
|
|
|
|
2023-08-22 23:30:01 -04:00
|
|
|
//Handling the playButton click event
|
2020-02-20 19:53:23 +05:30
|
|
|
ImageButton playButton = playerControlView.findViewById(R.id.exo_play);
|
2024-09-29 20:51:02 +02:00
|
|
|
|
2023-11-18 22:13:54 +09:00
|
|
|
playButton.setOnClickListener((View v) -> {
|
|
|
|
if (player != null && player.getPlaybackState() == Player.STATE_ENDED) {
|
|
|
|
player.seekTo(0);
|
2020-02-20 19:53:23 +05:30
|
|
|
}
|
2023-11-18 22:13:54 +09:00
|
|
|
setPausedModifier(false);
|
2020-02-20 19:53:23 +05:30
|
|
|
});
|
|
|
|
|
2024-05-28 00:32:30 -07:00
|
|
|
//Handling the rewind and forward button click events
|
|
|
|
ImageButton exoRewind = playerControlView.findViewById(R.id.exo_rew);
|
|
|
|
ImageButton exoForward = playerControlView.findViewById(R.id.exo_ffwd);
|
|
|
|
exoRewind.setOnClickListener((View v) -> {
|
|
|
|
seekTo(player.getCurrentPosition() - controlsConfig.getSeekIncrementMS());
|
|
|
|
});
|
|
|
|
|
|
|
|
exoForward.setOnClickListener((View v) -> {
|
|
|
|
seekTo(player.getCurrentPosition() + controlsConfig.getSeekIncrementMS());
|
|
|
|
});
|
|
|
|
|
2023-08-22 23:30:01 -04:00
|
|
|
//Handling the pauseButton click event
|
2020-02-20 19:53:23 +05:30
|
|
|
ImageButton pauseButton = playerControlView.findViewById(R.id.exo_pause);
|
2024-05-22 14:02:55 +02:00
|
|
|
pauseButton.setOnClickListener((View v) ->
|
2024-07-04 21:01:28 +09:00
|
|
|
setPausedModifier(true)
|
2024-05-22 14:02:55 +02:00
|
|
|
);
|
2020-02-20 19:53:23 +05:30
|
|
|
|
2024-10-05 20:04:25 +03:30
|
|
|
//Handling the settingButton click event
|
|
|
|
final ImageButton settingButton = playerControlView.findViewById(R.id.exo_settings);
|
|
|
|
settingButton.setOnClickListener(v -> openSettings());
|
|
|
|
|
2023-08-22 23:30:01 -04:00
|
|
|
//Handling the fullScreenButton click event
|
2022-11-03 08:26:06 +01:00
|
|
|
final ImageButton fullScreenButton = playerControlView.findViewById(R.id.exo_fullscreen);
|
2022-07-14 11:10:18 -07:00
|
|
|
fullScreenButton.setOnClickListener(v -> setFullscreen(!isFullscreen));
|
2024-06-11 00:11:26 +03:30
|
|
|
updateFullScreenButtonVisibility();
|
2024-09-29 20:51:02 +02:00
|
|
|
refreshControlsStyles();
|
2022-07-14 11:10:18 -07:00
|
|
|
|
2022-06-16 00:24:55 +07:00
|
|
|
// Invoking onPlaybackStateChanged and onPlayWhenReadyChanged events for Player
|
|
|
|
eventListener = new Player.Listener() {
|
2019-02-04 19:18:29 +05:30
|
|
|
@Override
|
2022-06-16 00:24:55 +07:00
|
|
|
public void onPlaybackStateChanged(int playbackState) {
|
2022-07-14 11:58:50 -07:00
|
|
|
View playButton = playerControlView.findViewById(R.id.exo_play);
|
|
|
|
View pauseButton = playerControlView.findViewById(R.id.exo_pause);
|
|
|
|
if (playButton != null && playButton.getVisibility() == GONE) {
|
|
|
|
playButton.setVisibility(INVISIBLE);
|
|
|
|
}
|
|
|
|
if (pauseButton != null && pauseButton.getVisibility() == GONE) {
|
|
|
|
pauseButton.setVisibility(INVISIBLE);
|
|
|
|
}
|
2024-09-29 20:51:02 +02:00
|
|
|
|
2022-06-16 00:24:55 +07:00
|
|
|
reLayout(playPauseControlContainer);
|
2023-08-22 23:30:01 -04:00
|
|
|
//Remove this eventListener once its executed. since UI will work fine once after the reLayout is done
|
2022-06-16 00:24:55 +07:00
|
|
|
player.removeListener(eventListener);
|
|
|
|
}
|
|
|
|
|
2019-02-04 19:18:29 +05:30
|
|
|
@Override
|
2022-06-16 00:24:55 +07:00
|
|
|
public void onPlayWhenReadyChanged(boolean playWhenReady, int reason) {
|
2019-02-04 19:18:29 +05:30
|
|
|
reLayout(playPauseControlContainer);
|
2023-08-22 23:30:01 -04:00
|
|
|
//Remove this eventListener once its executed. since UI will work fine once after the reLayout is done
|
2019-02-04 19:18:29 +05:30
|
|
|
player.removeListener(eventListener);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
player.addListener(eventListener);
|
|
|
|
}
|
2024-10-05 20:04:25 +03:30
|
|
|
private void openSettings() {
|
|
|
|
AlertDialog.Builder builder = new AlertDialog.Builder(themedReactContext);
|
|
|
|
builder.setTitle(R.string.settings);
|
|
|
|
String[] settingsOptions = {themedReactContext.getString(R.string.playback_speed)};
|
|
|
|
builder.setItems(settingsOptions, (dialog, which) -> {
|
|
|
|
if (which == 0) {
|
|
|
|
showPlaybackSpeedOptions();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
builder.show();
|
|
|
|
}
|
|
|
|
|
|
|
|
private void showPlaybackSpeedOptions() {
|
|
|
|
String[] speedOptions = {"0.5x", "1.0x", "1.5x", "2.0x"};
|
|
|
|
AlertDialog.Builder builder = new AlertDialog.Builder(themedReactContext);
|
|
|
|
builder.setTitle(R.string.select_playback_speed);
|
|
|
|
|
|
|
|
builder.setSingleChoiceItems(speedOptions, selectedSpeedIndex, (dialog, which) -> {
|
|
|
|
selectedSpeedIndex = which;
|
|
|
|
float speed = switch (which) {
|
|
|
|
case 0 -> 0.5f;
|
|
|
|
case 2 -> 1.5f;
|
|
|
|
case 3 -> 2.0f;
|
|
|
|
default -> 1.0f;
|
|
|
|
};
|
|
|
|
setRateModifier(speed);
|
|
|
|
});
|
|
|
|
builder.show();
|
|
|
|
}
|
2019-02-04 19:18:29 +05:30
|
|
|
|
2019-02-06 03:22:06 +05:30
|
|
|
/**
|
|
|
|
* Adding Player control to the frame layout
|
|
|
|
*/
|
|
|
|
private void addPlayerControl() {
|
2023-11-18 22:13:54 +09:00
|
|
|
if (playerControlView == null) return;
|
2019-02-06 03:22:06 +05:30
|
|
|
LayoutParams layoutParams = new LayoutParams(
|
|
|
|
LayoutParams.MATCH_PARENT,
|
|
|
|
LayoutParams.MATCH_PARENT);
|
|
|
|
playerControlView.setLayoutParams(layoutParams);
|
2019-07-07 22:17:15 +02:00
|
|
|
int indexOfPC = indexOfChild(playerControlView);
|
|
|
|
if (indexOfPC != -1) {
|
|
|
|
removeViewAt(indexOfPC);
|
|
|
|
}
|
2019-02-06 03:22:06 +05:30
|
|
|
addView(playerControlView, 1, layoutParams);
|
2022-08-23 13:11:10 -07:00
|
|
|
reLayout(playerControlView);
|
2019-02-06 03:22:06 +05:30
|
|
|
}
|
|
|
|
|
2019-02-04 19:18:29 +05:30
|
|
|
/**
|
|
|
|
* Update the layout
|
2023-08-22 23:30:01 -04:00
|
|
|
* @param view view needs to update layout
|
2019-02-04 19:18:29 +05:30
|
|
|
*
|
2023-10-12 10:36:43 +02:00
|
|
|
* This is a workaround for the open bug in react-native: <a href="https://github.com/facebook/react-native/issues/17968">...</a>
|
2019-02-04 19:18:29 +05:30
|
|
|
*/
|
|
|
|
private void reLayout(View view) {
|
2023-08-22 23:30:01 -04:00
|
|
|
if (view == null) return;
|
2019-02-04 19:18:29 +05:30
|
|
|
view.measure(MeasureSpec.makeMeasureSpec(getMeasuredWidth(), MeasureSpec.EXACTLY),
|
|
|
|
MeasureSpec.makeMeasureSpec(getMeasuredHeight(), MeasureSpec.EXACTLY));
|
|
|
|
view.layout(view.getLeft(), view.getTop(), view.getMeasuredWidth(), view.getMeasuredHeight());
|
2019-01-04 14:58:32 +05:30
|
|
|
}
|
|
|
|
|
2024-10-03 01:07:18 +03:30
|
|
|
private void refreshControlsStyles() {
|
|
|
|
if (playerControlView == null || player == null || !controls) return;
|
|
|
|
updateLiveContent();
|
|
|
|
updatePlayPauseButtons();
|
|
|
|
updateButtonVisibility(controlsConfig.getHideForward(), R.id.exo_ffwd);
|
|
|
|
updateButtonVisibility(controlsConfig.getHideRewind(), R.id.exo_rew);
|
|
|
|
updateButtonVisibility(controlsConfig.getHideNext(), R.id.exo_next);
|
|
|
|
updateButtonVisibility(controlsConfig.getHidePrevious(), R.id.exo_prev);
|
|
|
|
updateViewVisibility(playerControlView.findViewById(R.id.exo_fullscreen), controlsConfig.getHideFullscreen(), GONE);
|
|
|
|
updateViewVisibility(playerControlView.findViewById(R.id.exo_position), controlsConfig.getHidePosition(), GONE);
|
|
|
|
updateViewVisibility(playerControlView.findViewById(R.id.exo_progress), controlsConfig.getHideSeekBar(), INVISIBLE);
|
|
|
|
updateViewVisibility(playerControlView.findViewById(R.id.exo_duration), controlsConfig.getHideDuration(), GONE);
|
2024-10-05 20:04:25 +03:30
|
|
|
updateViewVisibility(playerControlView.findViewById(R.id.exo_settings), controlsConfig.getHideSettingButton(), GONE );
|
2024-10-03 01:07:18 +03:30
|
|
|
}
|
|
|
|
|
|
|
|
private void updateLiveContent() {
|
|
|
|
LinearLayout exoLiveContainer = playerControlView.findViewById(R.id.exo_live_container);
|
|
|
|
TextView exoLiveLabel = playerControlView.findViewById(R.id.exo_live_label);
|
|
|
|
|
|
|
|
boolean isLive = false;
|
|
|
|
Timeline timeline = player.getCurrentTimeline();
|
|
|
|
|
|
|
|
// Determine if the content is live
|
|
|
|
if (!timeline.isEmpty()) {
|
|
|
|
Timeline.Window window = new Timeline.Window();
|
|
|
|
timeline.getWindow(player.getCurrentMediaItemIndex(), window);
|
|
|
|
isLive = window.isLive();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isLive && controlsConfig.getLiveLabel() != null) {
|
|
|
|
exoLiveLabel.setText(controlsConfig.getLiveLabel());
|
|
|
|
exoLiveContainer.setVisibility(VISIBLE);
|
|
|
|
} else {
|
|
|
|
exoLiveContainer.setVisibility(GONE);
|
|
|
|
}
|
|
|
|
}
|
2024-08-21 16:05:40 +08:00
|
|
|
|
2024-10-03 01:07:18 +03:30
|
|
|
private void updatePlayPauseButtons() {
|
2024-09-29 20:51:02 +02:00
|
|
|
final ImageButton playButton = playerControlView.findViewById(R.id.exo_play);
|
|
|
|
final ImageButton pauseButton = playerControlView.findViewById(R.id.exo_pause);
|
2024-10-03 01:07:18 +03:30
|
|
|
|
2024-09-29 20:51:02 +02:00
|
|
|
if (controlsConfig.getHidePlayPause()) {
|
|
|
|
playPauseControlContainer.setAlpha(0);
|
|
|
|
playButton.setClickable(false);
|
|
|
|
pauseButton.setClickable(false);
|
|
|
|
} else {
|
|
|
|
playPauseControlContainer.setAlpha(1.0f);
|
|
|
|
playButton.setClickable(true);
|
|
|
|
pauseButton.setClickable(true);
|
|
|
|
}
|
2024-10-03 01:07:18 +03:30
|
|
|
}
|
2024-09-29 20:51:02 +02:00
|
|
|
|
2024-10-03 01:07:18 +03:30
|
|
|
private void updateButtonVisibility(boolean hide, int buttonID) {
|
|
|
|
ImageButton button = playerControlView.findViewById(buttonID);
|
|
|
|
if (hide) {
|
|
|
|
button.setImageAlpha(0);
|
|
|
|
button.setClickable(false);
|
2024-09-29 20:51:02 +02:00
|
|
|
} else {
|
2024-10-03 01:07:18 +03:30
|
|
|
button.setImageAlpha(255);
|
|
|
|
button.setClickable(true);
|
2024-09-29 20:51:02 +02:00
|
|
|
}
|
2024-10-03 01:07:18 +03:30
|
|
|
}
|
2024-09-29 20:51:02 +02:00
|
|
|
|
2024-10-03 01:07:18 +03:30
|
|
|
private void updateViewVisibility(View view, boolean hide, int hideVisibility) {
|
|
|
|
if (hide) {
|
|
|
|
view.setVisibility(hideVisibility);
|
|
|
|
} else if (view.getVisibility() == hideVisibility) {
|
|
|
|
view.setVisibility(VISIBLE);
|
2024-09-29 20:51:02 +02:00
|
|
|
}
|
2024-10-03 01:07:18 +03:30
|
|
|
}
|
2024-09-29 20:51:02 +02:00
|
|
|
|
|
|
|
|
2024-05-20 14:15:18 +03:30
|
|
|
|
2024-01-26 21:34:07 +01:00
|
|
|
private void reLayoutControls() {
|
|
|
|
reLayout(exoPlayerView);
|
|
|
|
reLayout(playerControlView);
|
|
|
|
}
|
|
|
|
|
2024-09-17 15:57:26 +02:00
|
|
|
/// returns true is adaptive bitrate shall be used
|
|
|
|
public boolean isUsingVideoABR() {
|
|
|
|
return videoTrackType == null || "auto".equals(videoTrackType);
|
|
|
|
}
|
|
|
|
|
2024-05-17 15:10:37 +02:00
|
|
|
public void setDebug(boolean enableDebug) {
|
|
|
|
this.enableDebug = enableDebug;
|
|
|
|
refreshDebugState();
|
|
|
|
}
|
|
|
|
|
|
|
|
private void refreshDebugState() {
|
|
|
|
if (player == null) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (enableDebug) {
|
|
|
|
debugEventLogger = new EventLogger(TAG_EVENT_LOGGER);
|
|
|
|
player.addAnalyticsListener(debugEventLogger);
|
|
|
|
} else if (debugEventLogger != null) {
|
|
|
|
player.removeAnalyticsListener(debugEventLogger);
|
|
|
|
debugEventLogger = null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-06-27 11:58:06 +02:00
|
|
|
public void setViewType(int viewType) {
|
|
|
|
exoPlayerView.updateSurfaceView(viewType);
|
|
|
|
}
|
|
|
|
|
2021-03-18 03:58:04 -07:00
|
|
|
private class RNVLoadControl extends DefaultLoadControl {
|
2023-10-12 10:36:43 +02:00
|
|
|
private final int availableHeapInBytes;
|
|
|
|
private final Runtime runtime;
|
2024-05-06 22:04:40 +02:00
|
|
|
public RNVLoadControl(DefaultAllocator allocator, BufferConfig config) {
|
2021-03-18 03:58:04 -07:00
|
|
|
super(allocator,
|
2024-05-06 22:04:40 +02:00
|
|
|
config.getMinBufferMs() != BufferConfig.Companion.getBufferConfigPropUnsetInt()
|
|
|
|
? config.getMinBufferMs()
|
|
|
|
: DefaultLoadControl.DEFAULT_MIN_BUFFER_MS,
|
|
|
|
config.getMaxBufferMs() != BufferConfig.Companion.getBufferConfigPropUnsetInt()
|
|
|
|
? config.getMaxBufferMs()
|
|
|
|
: DefaultLoadControl.DEFAULT_MAX_BUFFER_MS,
|
|
|
|
config.getBufferForPlaybackMs() != BufferConfig.Companion.getBufferConfigPropUnsetInt()
|
|
|
|
? config.getBufferForPlaybackMs()
|
|
|
|
: DefaultLoadControl.DEFAULT_BUFFER_FOR_PLAYBACK_MS ,
|
|
|
|
config.getBufferForPlaybackAfterRebufferMs() != BufferConfig.Companion.getBufferConfigPropUnsetInt()
|
|
|
|
? config.getBufferForPlaybackAfterRebufferMs()
|
|
|
|
: DefaultLoadControl.DEFAULT_BUFFER_FOR_PLAYBACK_AFTER_REBUFFER_MS,
|
|
|
|
-1,
|
|
|
|
true,
|
|
|
|
config.getBackBufferDurationMs() != BufferConfig.Companion.getBufferConfigPropUnsetInt()
|
|
|
|
? config.getBackBufferDurationMs()
|
|
|
|
: DefaultLoadControl.DEFAULT_BACK_BUFFER_DURATION_MS,
|
|
|
|
DefaultLoadControl.DEFAULT_RETAIN_BACK_BUFFER_FROM_KEYFRAME);
|
2021-11-16 20:35:30 -04:00
|
|
|
runtime = Runtime.getRuntime();
|
2023-10-12 10:36:43 +02:00
|
|
|
ActivityManager activityManager = (ActivityManager) themedReactContext.getSystemService(ThemedReactContext.ACTIVITY_SERVICE);
|
2024-05-06 22:04:40 +02:00
|
|
|
double maxHeap = config.getMaxHeapAllocationPercent() != BufferConfig.Companion.getBufferConfigPropUnsetDouble()
|
|
|
|
? bufferConfig.getMaxHeapAllocationPercent()
|
|
|
|
: DEFAULT_MAX_HEAP_ALLOCATION_PERCENT;
|
|
|
|
availableHeapInBytes = (int) Math.floor(activityManager.getMemoryClass() * maxHeap * 1024 * 1024);
|
2021-03-18 03:58:04 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public boolean shouldContinueLoading(long playbackPositionUs, long bufferedDurationUs, float playbackSpeed) {
|
2024-05-11 22:02:04 +02:00
|
|
|
if (bufferingStrategy == BufferingStrategy.BufferingStrategyEnum.DisableBuffering) {
|
2021-11-16 20:35:30 -04:00
|
|
|
return false;
|
2024-05-11 22:02:04 +02:00
|
|
|
} else if (bufferingStrategy == BufferingStrategy.BufferingStrategyEnum.DependingOnMemory) {
|
|
|
|
// The goal of this algorithm is to pause video loading (increasing the buffer)
|
|
|
|
// when available memory on device become low.
|
|
|
|
int loadedBytes = getAllocator().getTotalBytesAllocated();
|
|
|
|
boolean isHeapReached = availableHeapInBytes > 0 && loadedBytes >= availableHeapInBytes;
|
|
|
|
if (isHeapReached) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
long usedMemory = runtime.totalMemory() - runtime.freeMemory();
|
|
|
|
long freeMemory = runtime.maxMemory() - usedMemory;
|
|
|
|
double minBufferMemoryReservePercent = bufferConfig.getMinBufferMemoryReservePercent() != BufferConfig.Companion.getBufferConfigPropUnsetDouble()
|
|
|
|
? bufferConfig.getMinBufferMemoryReservePercent()
|
|
|
|
: ReactExoplayerView.DEFAULT_MIN_BUFFER_MEMORY_RESERVE;
|
|
|
|
long reserveMemory = (long) minBufferMemoryReservePercent * runtime.maxMemory();
|
|
|
|
long bufferedMs = bufferedDurationUs / (long) 1000;
|
|
|
|
if (reserveMemory > freeMemory && bufferedMs > 2000) {
|
|
|
|
// We don't have enough memory in reserve so we stop buffering to allow other components to use it instead
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (runtime.freeMemory() == 0) {
|
|
|
|
DebugLog.w(TAG, "Free memory reached 0, forcing garbage collection");
|
|
|
|
runtime.gc();
|
|
|
|
return false;
|
|
|
|
}
|
2021-11-16 20:35:30 -04:00
|
|
|
}
|
2024-05-11 22:02:04 +02:00
|
|
|
// "default" case or normal case for "DependingOnMemory"
|
2021-03-18 03:58:04 -07:00
|
|
|
return super.shouldContinueLoading(playbackPositionUs, bufferedDurationUs, playbackSpeed);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-01-11 12:51:45 +00:00
|
|
|
private void initializePlayer() {
|
2019-07-07 10:21:23 +02:00
|
|
|
ReactExoplayerView self = this;
|
2022-04-07 19:36:50 +03:00
|
|
|
Activity activity = themedReactContext.getCurrentActivity();
|
2019-09-16 16:29:31 -04:00
|
|
|
// This ensures all props have been settled, to avoid async racing conditions.
|
2024-10-10 22:59:41 +02:00
|
|
|
Source runningSource = source;
|
2023-11-18 22:13:54 +09:00
|
|
|
mainRunnable = () -> {
|
2024-10-10 22:59:41 +02:00
|
|
|
if (viewHasDropped && runningSource == source) {
|
2024-06-07 13:54:14 +02:00
|
|
|
return;
|
|
|
|
}
|
2023-11-18 22:13:54 +09:00
|
|
|
try {
|
2024-10-10 22:59:41 +02:00
|
|
|
if (runningSource.getUri() == null) {
|
|
|
|
return;
|
|
|
|
}
|
2023-11-18 22:13:54 +09:00
|
|
|
if (player == null) {
|
|
|
|
// Initialize core configuration and listeners
|
|
|
|
initializePlayerCore(self);
|
|
|
|
}
|
2024-10-10 22:59:41 +02:00
|
|
|
if (playerNeedsSource) {
|
|
|
|
// Will force display of shutter view if needed
|
|
|
|
exoPlayerView.updateShutterViewVisibility();
|
2023-11-18 22:13:54 +09:00
|
|
|
exoPlayerView.invalidateAspectRatio();
|
|
|
|
// DRM session manager creation must be done on a different thread to prevent crashes so we start a new thread
|
|
|
|
ExecutorService es = Executors.newSingleThreadExecutor();
|
|
|
|
es.execute(() -> {
|
|
|
|
// DRM initialization must run on a different thread
|
2024-10-10 22:59:41 +02:00
|
|
|
if (viewHasDropped && runningSource == source) {
|
2024-06-07 13:54:14 +02:00
|
|
|
return;
|
|
|
|
}
|
2023-11-18 22:13:54 +09:00
|
|
|
if (activity == null) {
|
2024-05-30 08:53:49 +02:00
|
|
|
DebugLog.e(TAG, "Failed to initialize Player!, null activity");
|
2024-07-04 21:01:28 +09:00
|
|
|
eventEmitter.onVideoError.invoke("Failed to initialize Player!", new Exception("Current Activity is null!"), "1001");
|
2023-11-18 22:13:54 +09:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Initialize handler to run on the main thread
|
|
|
|
activity.runOnUiThread(() -> {
|
2024-10-10 22:59:41 +02:00
|
|
|
if (viewHasDropped && runningSource == source) {
|
2024-06-07 13:54:14 +02:00
|
|
|
return;
|
|
|
|
}
|
2023-11-18 22:13:54 +09:00
|
|
|
try {
|
|
|
|
// Source initialization must run on the main thread
|
2024-10-10 22:59:41 +02:00
|
|
|
initializePlayerSource(runningSource);
|
2023-11-18 22:13:54 +09:00
|
|
|
} catch (Exception ex) {
|
|
|
|
self.playerNeedsSource = true;
|
2024-06-07 13:54:14 +02:00
|
|
|
DebugLog.e(TAG, "Failed to initialize Player! 1");
|
2024-05-11 19:03:44 +02:00
|
|
|
DebugLog.e(TAG, ex.toString());
|
2024-05-30 08:53:49 +02:00
|
|
|
ex.printStackTrace();
|
2024-07-04 21:01:28 +09:00
|
|
|
eventEmitter.onVideoError.invoke(ex.toString(), ex, "1001");
|
2021-10-12 14:56:50 +03:00
|
|
|
}
|
2022-03-30 14:29:08 +03:00
|
|
|
});
|
2023-11-18 22:13:54 +09:00
|
|
|
});
|
2024-10-10 22:59:41 +02:00
|
|
|
} else if (runningSource == source) {
|
|
|
|
initializePlayerSource(runningSource);
|
2019-07-07 10:21:23 +02:00
|
|
|
}
|
2023-11-18 22:13:54 +09:00
|
|
|
} catch (Exception ex) {
|
|
|
|
self.playerNeedsSource = true;
|
2024-06-07 13:54:14 +02:00
|
|
|
DebugLog.e(TAG, "Failed to initialize Player! 2");
|
2024-05-11 19:03:44 +02:00
|
|
|
DebugLog.e(TAG, ex.toString());
|
2024-05-30 08:53:49 +02:00
|
|
|
ex.printStackTrace();
|
2024-07-04 21:01:28 +09:00
|
|
|
eventEmitter.onVideoError.invoke(ex.toString(), ex, "1001");
|
2019-07-07 10:21:23 +02:00
|
|
|
}
|
2023-11-18 22:13:54 +09:00
|
|
|
};
|
|
|
|
mainHandler.postDelayed(mainRunnable, 1);
|
2017-01-11 12:51:45 +00:00
|
|
|
}
|
|
|
|
|
2024-05-28 02:00:38 -07:00
|
|
|
public void getCurrentPosition(Promise promise) {
|
|
|
|
if (player != null) {
|
2024-08-05 13:28:33 +03:30
|
|
|
float currentPosition = player.getCurrentPosition() / 1000.0f;
|
2024-05-28 02:00:38 -07:00
|
|
|
promise.resolve(currentPosition);
|
|
|
|
} else {
|
|
|
|
promise.reject("PLAYER_NOT_AVAILABLE", "Player is not initialized.");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-03-30 14:35:48 +03:00
|
|
|
private void initializePlayerCore(ReactExoplayerView self) {
|
|
|
|
ExoTrackSelection.Factory videoTrackSelectionFactory = new AdaptiveTrackSelection.Factory();
|
2022-08-26 10:32:22 -04:00
|
|
|
self.trackSelector = new DefaultTrackSelector(getContext(), videoTrackSelectionFactory);
|
2022-05-10 00:53:09 +03:00
|
|
|
self.trackSelector.setParameters(trackSelector.buildUponParameters()
|
2022-03-30 14:35:48 +03:00
|
|
|
.setMaxVideoBitrate(maxBitRate == 0 ? Integer.MAX_VALUE : maxBitRate));
|
|
|
|
|
|
|
|
DefaultAllocator allocator = new DefaultAllocator(true, C.DEFAULT_BUFFER_SEGMENT_SIZE);
|
|
|
|
RNVLoadControl loadControl = new RNVLoadControl(
|
|
|
|
allocator,
|
2024-05-06 22:04:40 +02:00
|
|
|
bufferConfig
|
2023-08-22 23:30:01 -04:00
|
|
|
);
|
|
|
|
DefaultRenderersFactory renderersFactory =
|
|
|
|
new DefaultRenderersFactory(getContext())
|
2024-03-12 22:38:24 +01:00
|
|
|
.setExtensionRendererMode(DefaultRenderersFactory.EXTENSION_RENDERER_MODE_OFF)
|
2024-05-31 08:58:29 +02:00
|
|
|
.setEnableDecoderFallback(true)
|
|
|
|
.forceEnableMediaCodecAsynchronousQueueing();
|
2022-11-16 11:43:25 +01:00
|
|
|
|
2024-10-10 22:59:41 +02:00
|
|
|
DefaultMediaSourceFactory mediaSourceFactory = new DefaultMediaSourceFactory(mediaDataSourceFactory);
|
|
|
|
if (useCache) {
|
|
|
|
mediaSourceFactory.setDataSourceFactory(RNVSimpleCache.INSTANCE.getCacheFactory(buildHttpDataSourceFactory(true)));
|
|
|
|
}
|
|
|
|
|
|
|
|
mediaSourceFactory.setLocalAdInsertionComponents(unusedAdTagUri -> adsLoader, exoPlayerView);
|
2022-11-16 11:43:25 +01:00
|
|
|
|
2022-06-16 00:24:55 +07:00
|
|
|
player = new ExoPlayer.Builder(getContext(), renderersFactory)
|
2023-11-18 22:13:54 +09:00
|
|
|
.setTrackSelector(self.trackSelector)
|
|
|
|
.setBandwidthMeter(bandwidthMeter)
|
|
|
|
.setLoadControl(loadControl)
|
|
|
|
.setMediaSourceFactory(mediaSourceFactory)
|
|
|
|
.build();
|
2024-06-25 08:55:32 +02:00
|
|
|
ReactNativeVideoManager.Companion.getInstance().onInstanceCreated(instanceId, player);
|
2024-05-17 15:10:37 +02:00
|
|
|
refreshDebugState();
|
2022-03-30 15:33:17 +03:00
|
|
|
player.addListener(self);
|
2023-11-09 03:39:04 +09:00
|
|
|
player.setVolume(muted ? 0.f : audioVolume * 1);
|
2022-03-30 15:33:17 +03:00
|
|
|
exoPlayerView.setPlayer(player);
|
2024-10-10 22:59:41 +02:00
|
|
|
|
2022-03-30 15:33:17 +03:00
|
|
|
audioBecomingNoisyReceiver.setListener(self);
|
|
|
|
bandwidthMeter.addEventListener(new Handler(), self);
|
2022-03-30 14:35:48 +03:00
|
|
|
setPlayWhenReady(!isPaused);
|
2022-11-16 13:23:50 +01:00
|
|
|
playerNeedsSource = true;
|
2018-06-11 15:23:43 -07:00
|
|
|
|
2022-03-30 14:35:48 +03:00
|
|
|
PlaybackParameters params = new PlaybackParameters(rate, 1f);
|
2022-03-30 15:33:17 +03:00
|
|
|
player.setPlaybackParameters(params);
|
2023-04-12 17:43:52 -04:00
|
|
|
changeAudioOutput(this.audioOutput);
|
2024-05-07 12:30:57 +02:00
|
|
|
|
|
|
|
if(showNotificationControls) {
|
|
|
|
setupPlaybackService();
|
|
|
|
}
|
2022-03-30 14:35:48 +03:00
|
|
|
}
|
|
|
|
|
2024-10-17 21:56:06 +02:00
|
|
|
private AdsMediaSource initializeAds(MediaSource videoSource, Source runningSource) {
|
|
|
|
AdsProps adProps = runningSource.getAdsProps();
|
|
|
|
Uri uri = runningSource.getUri();
|
|
|
|
if (adProps != null && uri != null) {
|
|
|
|
Uri adTagUrl = adProps.getAdTagUrl();
|
|
|
|
if (adTagUrl != null) {
|
|
|
|
exoPlayerView.showAds();
|
|
|
|
// Create an AdsLoader.
|
|
|
|
ImaAdsLoader.Builder imaLoaderBuilder = new ImaAdsLoader
|
|
|
|
.Builder(themedReactContext)
|
|
|
|
.setAdEventListener(this)
|
|
|
|
.setAdErrorListener(this);
|
|
|
|
|
|
|
|
if (adProps.getAdLanguage() != null) {
|
|
|
|
ImaSdkSettings imaSdkSettings = ImaSdkFactory.getInstance().createImaSdkSettings();
|
|
|
|
imaSdkSettings.setLanguage(adProps.getAdLanguage());
|
|
|
|
imaLoaderBuilder.setImaSdkSettings(imaSdkSettings);
|
|
|
|
}
|
|
|
|
adsLoader = imaLoaderBuilder.build();
|
|
|
|
adsLoader.setPlayer(player);
|
|
|
|
if (adsLoader != null) {
|
|
|
|
DefaultMediaSourceFactory mediaSourceFactory = new DefaultMediaSourceFactory(mediaDataSourceFactory)
|
|
|
|
.setLocalAdInsertionComponents(unusedAdTagUri -> adsLoader, exoPlayerView);
|
|
|
|
DataSpec adTagDataSpec = new DataSpec(adTagUrl);
|
|
|
|
return new AdsMediaSource(videoSource,
|
|
|
|
adTagDataSpec,
|
|
|
|
ImmutableList.of(uri, adTagUrl),
|
|
|
|
mediaSourceFactory, adsLoader, exoPlayerView);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
exoPlayerView.hideAds();
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2024-07-10 12:17:22 +02:00
|
|
|
private DrmSessionManager initializePlayerDrm() {
|
2022-03-30 14:29:08 +03:00
|
|
|
DrmSessionManager drmSessionManager = null;
|
2024-07-10 12:17:22 +02:00
|
|
|
DRMProps drmProps = source.getDrmProps();
|
|
|
|
// need to realign UUID in DRM Props from source
|
|
|
|
if (drmProps != null && drmProps.getDrmType() != null) {
|
|
|
|
UUID uuid = Util.getDrmUuid(drmProps.getDrmType());
|
|
|
|
if (uuid != null) {
|
|
|
|
try {
|
|
|
|
DebugLog.w(TAG, "drm buildDrmSessionManager");
|
|
|
|
drmSessionManager = buildDrmSessionManager(uuid, drmProps);
|
|
|
|
} catch (UnsupportedDrmException e) {
|
|
|
|
int errorStringId = Util.SDK_INT < 18 ? R.string.error_drm_not_supported
|
|
|
|
: (e.reason == UnsupportedDrmException.REASON_UNSUPPORTED_SCHEME
|
|
|
|
? R.string.error_drm_unsupported_scheme : R.string.error_drm_unknown);
|
|
|
|
eventEmitter.onVideoError.invoke(getResources().getString(errorStringId), e, "3003");
|
|
|
|
}
|
2022-03-30 14:29:08 +03:00
|
|
|
}
|
|
|
|
}
|
2022-03-30 14:43:04 +03:00
|
|
|
return drmSessionManager;
|
2022-03-30 14:29:08 +03:00
|
|
|
}
|
2017-01-11 12:51:45 +00:00
|
|
|
|
2024-10-10 22:59:41 +02:00
|
|
|
private void initializePlayerSource(Source runningSource) {
|
|
|
|
if (runningSource.getUri() == null) {
|
2024-05-30 08:53:49 +02:00
|
|
|
return;
|
|
|
|
}
|
2024-07-10 12:17:22 +02:00
|
|
|
/// init DRM
|
|
|
|
DrmSessionManager drmSessionManager = initializePlayerDrm();
|
2024-10-10 22:59:41 +02:00
|
|
|
if (drmSessionManager == null && runningSource.getDrmProps() != null && runningSource.getDrmProps().getDrmType() != null) {
|
2024-07-10 12:17:22 +02:00
|
|
|
// Failed to initialize DRM session manager - cannot continue
|
2024-05-30 08:53:49 +02:00
|
|
|
DebugLog.e(TAG, "Failed to initialize DRM Session Manager Framework!");
|
|
|
|
return;
|
|
|
|
}
|
2024-07-10 12:17:22 +02:00
|
|
|
// init source to manage ads and external text tracks
|
2024-10-17 21:56:06 +02:00
|
|
|
MediaSource videoSource = buildMediaSource(runningSource.getUri(),
|
|
|
|
runningSource.getExtension(),
|
|
|
|
drmSessionManager,
|
|
|
|
runningSource.getCropStartMs(),
|
|
|
|
runningSource.getCropEndMs());
|
|
|
|
MediaSource mediaSourceWithAds = initializeAds(videoSource, runningSource);
|
|
|
|
MediaSource mediaSource = Objects.requireNonNullElse(mediaSourceWithAds, videoSource);
|
2024-10-11 22:50:22 +02:00
|
|
|
|
2024-10-17 21:56:06 +02:00
|
|
|
MediaSource subtitlesSource = buildTextSource();
|
|
|
|
if (subtitlesSource != null) {
|
|
|
|
MediaSource[] mediaSourceArray = {mediaSource, subtitlesSource};
|
2024-10-11 22:50:22 +02:00
|
|
|
mediaSource = new MergingMediaSource(mediaSourceArray);
|
2022-03-30 14:29:08 +03:00
|
|
|
}
|
2019-02-06 03:22:06 +05:30
|
|
|
|
2022-04-28 20:17:59 -04:00
|
|
|
// wait for player to be set
|
|
|
|
while (player == null) {
|
|
|
|
try {
|
|
|
|
wait();
|
|
|
|
} catch (InterruptedException ex) {
|
|
|
|
Thread.currentThread().interrupt();
|
2024-05-11 19:03:44 +02:00
|
|
|
DebugLog.e(TAG, ex.toString());
|
2019-07-07 10:21:23 +02:00
|
|
|
}
|
2022-04-28 20:17:59 -04:00
|
|
|
}
|
|
|
|
|
2022-03-30 14:29:08 +03:00
|
|
|
boolean haveResumePosition = resumeWindow != C.INDEX_UNSET;
|
|
|
|
if (haveResumePosition) {
|
|
|
|
player.seekTo(resumeWindow, resumePosition);
|
2024-05-17 21:21:02 +09:00
|
|
|
player.setMediaSource(mediaSource, false);
|
2024-10-10 22:59:41 +02:00
|
|
|
} else if (runningSource.getStartPositionMs() > 0) {
|
|
|
|
player.setMediaSource(mediaSource, runningSource.getStartPositionMs());
|
2023-11-24 20:52:46 +09:00
|
|
|
} else {
|
2024-05-17 21:21:02 +09:00
|
|
|
player.setMediaSource(mediaSource, true);
|
2023-11-24 20:52:46 +09:00
|
|
|
}
|
|
|
|
player.prepare();
|
2022-03-30 14:29:08 +03:00
|
|
|
playerNeedsSource = false;
|
|
|
|
|
2024-01-26 21:34:07 +01:00
|
|
|
reLayoutControls();
|
|
|
|
|
2024-07-04 21:01:28 +09:00
|
|
|
eventEmitter.onVideoLoadStart.invoke();
|
2022-03-30 14:29:08 +03:00
|
|
|
loadVideoStarted = true;
|
|
|
|
|
|
|
|
finishPlayerInitialization();
|
|
|
|
}
|
|
|
|
|
|
|
|
private void finishPlayerInitialization() {
|
|
|
|
// Initializing the playerControlView
|
|
|
|
initializePlayerControl();
|
|
|
|
setControls(controls);
|
|
|
|
applyModifiers();
|
|
|
|
}
|
|
|
|
|
2024-05-07 12:30:57 +02:00
|
|
|
private void setupPlaybackService() {
|
|
|
|
if (!showNotificationControls || player == null) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
playbackServiceConnection = new ServiceConnection() {
|
|
|
|
@Override
|
|
|
|
public void onServiceConnected(ComponentName name, IBinder service) {
|
|
|
|
playbackServiceBinder = (PlaybackServiceBinder) service;
|
|
|
|
|
|
|
|
try {
|
2024-09-14 15:20:50 +02:00
|
|
|
Activity currentActivity = themedReactContext.getCurrentActivity();
|
|
|
|
if (currentActivity != null) {
|
|
|
|
playbackServiceBinder.getService().registerPlayer(player,
|
|
|
|
(Class<Activity>) currentActivity.getClass());
|
|
|
|
} else {
|
|
|
|
// Handle the case where currentActivity is null
|
|
|
|
DebugLog.w(TAG, "Could not register ExoPlayer: currentActivity is null");
|
|
|
|
}
|
2024-05-07 12:30:57 +02:00
|
|
|
} catch (Exception e) {
|
2024-09-14 15:20:50 +02:00
|
|
|
DebugLog.e(TAG, "Could not register ExoPlayer: " + e.getMessage());
|
2024-05-07 12:30:57 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void onServiceDisconnected(ComponentName name) {
|
|
|
|
try {
|
2024-09-14 15:20:50 +02:00
|
|
|
if (playbackServiceBinder != null) {
|
|
|
|
playbackServiceBinder.getService().unregisterPlayer(player);
|
|
|
|
}
|
2024-05-07 12:30:57 +02:00
|
|
|
} catch (Exception ignored) {}
|
|
|
|
|
|
|
|
playbackServiceBinder = null;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void onNullBinding(ComponentName name) {
|
2024-09-14 15:20:50 +02:00
|
|
|
DebugLog.e(TAG, "Could not register ExoPlayer");
|
2024-05-07 12:30:57 +02:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
Intent intent = new Intent(themedReactContext, VideoPlaybackService.class);
|
|
|
|
intent.setAction(MediaSessionService.SERVICE_INTERFACE);
|
|
|
|
|
2024-10-03 01:23:17 -05:00
|
|
|
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
|
|
|
|
themedReactContext.startForegroundService(intent);
|
|
|
|
} else {
|
|
|
|
themedReactContext.startService(intent);
|
|
|
|
}
|
2024-05-07 12:30:57 +02:00
|
|
|
|
|
|
|
int flags;
|
|
|
|
if (Build.VERSION.SDK_INT >= 29) {
|
|
|
|
flags = Context.BIND_AUTO_CREATE | Context.BIND_INCLUDE_CAPABILITIES;
|
|
|
|
} else {
|
|
|
|
flags = Context.BIND_AUTO_CREATE;
|
|
|
|
}
|
|
|
|
|
|
|
|
themedReactContext.bindService(intent, playbackServiceConnection, flags);
|
|
|
|
}
|
|
|
|
|
|
|
|
private void cleanupPlaybackService() {
|
|
|
|
try {
|
|
|
|
if(player != null && playbackServiceBinder != null) {
|
|
|
|
playbackServiceBinder.getService().unregisterPlayer(player);
|
|
|
|
}
|
|
|
|
|
|
|
|
playbackServiceBinder = null;
|
|
|
|
|
|
|
|
if(playbackServiceConnection != null) {
|
|
|
|
themedReactContext.unbindService(playbackServiceConnection);
|
|
|
|
}
|
|
|
|
} catch(Exception e) {
|
|
|
|
DebugLog.w(TAG, "Cloud not cleanup playback service");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-07-10 12:17:22 +02:00
|
|
|
private DrmSessionManager buildDrmSessionManager(UUID uuid, DRMProps drmProps) throws UnsupportedDrmException {
|
|
|
|
return buildDrmSessionManager(uuid, drmProps, 0);
|
2017-01-11 12:51:45 +00:00
|
|
|
}
|
|
|
|
|
2024-07-10 12:17:22 +02:00
|
|
|
private DrmSessionManager buildDrmSessionManager(UUID uuid, DRMProps drmProps, int retryCount) throws UnsupportedDrmException {
|
Add iOS and Android basic DRM support (#1445)
This PR adds support for DRM streams on iOS (Fairplay) and Android (Playready, Widevine, Clearkey)
I am neither Android nor iOS developer, so feel free to provide feedback to improve this PR.
**Test stream for ANDROID:**
```
testStream = {
uri: 'http://profficialsite.origin.mediaservices.windows.net/c51358ea-9a5e-4322-8951-897d640fdfd7/tearsofsteel_4k.ism/manifest(format=mpd-time-csf)',
type: 'mpd',
drm: {
type: DRMType.PLAYREADY,
licenseServer: 'http://test.playready.microsoft.com/service/rightsmanager.asmx?cfg=(persist:false,sl:150)'
}
};
```
or
```
{
uri: 'https://media.axprod.net/TestVectors/v7-MultiDRM-SingleKey/Manifest_1080p.mpd',
drm: {
type: 'widevine', //or DRMType.WIDEVINE
licenseServer: 'https://drm-widevine-licensing.axtest.net/AcquireLicense',
headers: {
'X-AxDRM-Message': 'eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJ2ZXJzaW9uIjoxLCJjb21fa2V5X2lkIjoiYjMzNjRlYjUtNTFmNi00YWUzLThjOTgtMzNjZWQ1ZTMxYzc4IiwibWVzc2FnZSI6eyJ0eXBlIjoiZW50aXRsZW1lbnRfbWVzc2FnZSIsImZpcnN0X3BsYXlfZXhwaXJhdGlvbiI6NjAsInBsYXlyZWFkeSI6eyJyZWFsX3RpbWVfZXhwaXJhdGlvbiI6dHJ1ZX0sImtleXMiOlt7ImlkIjoiOWViNDA1MGQtZTQ0Yi00ODAyLTkzMmUtMjdkNzUwODNlMjY2IiwiZW5jcnlwdGVkX2tleSI6ImxLM09qSExZVzI0Y3Iya3RSNzRmbnc9PSJ9XX19.FAbIiPxX8BHi9RwfzD7Yn-wugU19ghrkBFKsaCPrZmU'
},
}
}
```
**Test stream for iOS:**
Sorry but I can not provide free streams to test. If anyone can provide test streams, or found some we can use, please let me know to also test them.
It has been tested with a private provider and they work, at least with the `getLicense` override method. (An example implementation is provided in the README)
2020-08-13 03:56:21 +02:00
|
|
|
if (Util.SDK_INT < 18) {
|
|
|
|
return null;
|
|
|
|
}
|
2022-03-29 15:21:39 +03:00
|
|
|
try {
|
2024-07-10 12:17:22 +02:00
|
|
|
HttpMediaDrmCallback drmCallback = new HttpMediaDrmCallback(drmProps.getDrmLicenseServer(),
|
2022-03-29 15:21:39 +03:00
|
|
|
buildHttpDataSourceFactory(false));
|
2024-07-10 12:17:22 +02:00
|
|
|
|
|
|
|
String[] keyRequestPropertiesArray = drmProps.getDrmLicenseHeader();
|
|
|
|
for (int i = 0; i < keyRequestPropertiesArray.length - 1; i += 2) {
|
|
|
|
drmCallback.setKeyRequestProperty(keyRequestPropertiesArray[i], keyRequestPropertiesArray[i + 1]);
|
Add iOS and Android basic DRM support (#1445)
This PR adds support for DRM streams on iOS (Fairplay) and Android (Playready, Widevine, Clearkey)
I am neither Android nor iOS developer, so feel free to provide feedback to improve this PR.
**Test stream for ANDROID:**
```
testStream = {
uri: 'http://profficialsite.origin.mediaservices.windows.net/c51358ea-9a5e-4322-8951-897d640fdfd7/tearsofsteel_4k.ism/manifest(format=mpd-time-csf)',
type: 'mpd',
drm: {
type: DRMType.PLAYREADY,
licenseServer: 'http://test.playready.microsoft.com/service/rightsmanager.asmx?cfg=(persist:false,sl:150)'
}
};
```
or
```
{
uri: 'https://media.axprod.net/TestVectors/v7-MultiDRM-SingleKey/Manifest_1080p.mpd',
drm: {
type: 'widevine', //or DRMType.WIDEVINE
licenseServer: 'https://drm-widevine-licensing.axtest.net/AcquireLicense',
headers: {
'X-AxDRM-Message': 'eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJ2ZXJzaW9uIjoxLCJjb21fa2V5X2lkIjoiYjMzNjRlYjUtNTFmNi00YWUzLThjOTgtMzNjZWQ1ZTMxYzc4IiwibWVzc2FnZSI6eyJ0eXBlIjoiZW50aXRsZW1lbnRfbWVzc2FnZSIsImZpcnN0X3BsYXlfZXhwaXJhdGlvbiI6NjAsInBsYXlyZWFkeSI6eyJyZWFsX3RpbWVfZXhwaXJhdGlvbiI6dHJ1ZX0sImtleXMiOlt7ImlkIjoiOWViNDA1MGQtZTQ0Yi00ODAyLTkzMmUtMjdkNzUwODNlMjY2IiwiZW5jcnlwdGVkX2tleSI6ImxLM09qSExZVzI0Y3Iya3RSNzRmbnc9PSJ9XX19.FAbIiPxX8BHi9RwfzD7Yn-wugU19ghrkBFKsaCPrZmU'
},
}
}
```
**Test stream for iOS:**
Sorry but I can not provide free streams to test. If anyone can provide test streams, or found some we can use, please let me know to also test them.
It has been tested with a private provider and they work, at least with the `getLicense` override method. (An example implementation is provided in the README)
2020-08-13 03:56:21 +02:00
|
|
|
}
|
2022-03-29 15:21:39 +03:00
|
|
|
FrameworkMediaDrm mediaDrm = FrameworkMediaDrm.newInstance(uuid);
|
|
|
|
if (hasDrmFailed) {
|
|
|
|
// When DRM fails using L1 we want to switch to L3
|
|
|
|
mediaDrm.setPropertyString("securityLevel", "L3");
|
|
|
|
}
|
2024-05-22 14:02:55 +02:00
|
|
|
return new DefaultDrmSessionManager.Builder()
|
2023-11-18 22:13:54 +09:00
|
|
|
.setUuidAndExoMediaDrmProvider(uuid, (_uuid) -> mediaDrm)
|
|
|
|
.setKeyRequestParameters(null)
|
2024-07-10 12:17:22 +02:00
|
|
|
.setMultiSession(drmProps.getMultiDrm())
|
2023-11-18 22:13:54 +09:00
|
|
|
.build(drmCallback);
|
|
|
|
} catch (UnsupportedDrmException ex) {
|
2022-03-29 15:21:39 +03:00
|
|
|
// Unsupported DRM exceptions are handled by the calling method
|
2022-03-30 14:43:04 +03:00
|
|
|
throw ex;
|
|
|
|
} catch (Exception ex) {
|
2022-03-29 15:21:39 +03:00
|
|
|
if (retryCount < 3) {
|
2023-08-22 23:30:01 -04:00
|
|
|
// Attempt retry 3 times in case where the OS Media DRM Framework fails for whatever reason
|
2024-07-10 12:17:22 +02:00
|
|
|
return buildDrmSessionManager(uuid, drmProps, ++retryCount);
|
Add iOS and Android basic DRM support (#1445)
This PR adds support for DRM streams on iOS (Fairplay) and Android (Playready, Widevine, Clearkey)
I am neither Android nor iOS developer, so feel free to provide feedback to improve this PR.
**Test stream for ANDROID:**
```
testStream = {
uri: 'http://profficialsite.origin.mediaservices.windows.net/c51358ea-9a5e-4322-8951-897d640fdfd7/tearsofsteel_4k.ism/manifest(format=mpd-time-csf)',
type: 'mpd',
drm: {
type: DRMType.PLAYREADY,
licenseServer: 'http://test.playready.microsoft.com/service/rightsmanager.asmx?cfg=(persist:false,sl:150)'
}
};
```
or
```
{
uri: 'https://media.axprod.net/TestVectors/v7-MultiDRM-SingleKey/Manifest_1080p.mpd',
drm: {
type: 'widevine', //or DRMType.WIDEVINE
licenseServer: 'https://drm-widevine-licensing.axtest.net/AcquireLicense',
headers: {
'X-AxDRM-Message': 'eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJ2ZXJzaW9uIjoxLCJjb21fa2V5X2lkIjoiYjMzNjRlYjUtNTFmNi00YWUzLThjOTgtMzNjZWQ1ZTMxYzc4IiwibWVzc2FnZSI6eyJ0eXBlIjoiZW50aXRsZW1lbnRfbWVzc2FnZSIsImZpcnN0X3BsYXlfZXhwaXJhdGlvbiI6NjAsInBsYXlyZWFkeSI6eyJyZWFsX3RpbWVfZXhwaXJhdGlvbiI6dHJ1ZX0sImtleXMiOlt7ImlkIjoiOWViNDA1MGQtZTQ0Yi00ODAyLTkzMmUtMjdkNzUwODNlMjY2IiwiZW5jcnlwdGVkX2tleSI6ImxLM09qSExZVzI0Y3Iya3RSNzRmbnc9PSJ9XX19.FAbIiPxX8BHi9RwfzD7Yn-wugU19ghrkBFKsaCPrZmU'
},
}
}
```
**Test stream for iOS:**
Sorry but I can not provide free streams to test. If anyone can provide test streams, or found some we can use, please let me know to also test them.
It has been tested with a private provider and they work, at least with the `getLicense` override method. (An example implementation is provided in the README)
2020-08-13 03:56:21 +02:00
|
|
|
}
|
2022-03-29 15:21:39 +03:00
|
|
|
// Handle the unknow exception and emit to JS
|
2024-07-04 21:01:28 +09:00
|
|
|
eventEmitter.onVideoError.invoke(ex.toString(), ex, "3006");
|
2022-03-29 15:21:39 +03:00
|
|
|
return null;
|
Add iOS and Android basic DRM support (#1445)
This PR adds support for DRM streams on iOS (Fairplay) and Android (Playready, Widevine, Clearkey)
I am neither Android nor iOS developer, so feel free to provide feedback to improve this PR.
**Test stream for ANDROID:**
```
testStream = {
uri: 'http://profficialsite.origin.mediaservices.windows.net/c51358ea-9a5e-4322-8951-897d640fdfd7/tearsofsteel_4k.ism/manifest(format=mpd-time-csf)',
type: 'mpd',
drm: {
type: DRMType.PLAYREADY,
licenseServer: 'http://test.playready.microsoft.com/service/rightsmanager.asmx?cfg=(persist:false,sl:150)'
}
};
```
or
```
{
uri: 'https://media.axprod.net/TestVectors/v7-MultiDRM-SingleKey/Manifest_1080p.mpd',
drm: {
type: 'widevine', //or DRMType.WIDEVINE
licenseServer: 'https://drm-widevine-licensing.axtest.net/AcquireLicense',
headers: {
'X-AxDRM-Message': 'eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJ2ZXJzaW9uIjoxLCJjb21fa2V5X2lkIjoiYjMzNjRlYjUtNTFmNi00YWUzLThjOTgtMzNjZWQ1ZTMxYzc4IiwibWVzc2FnZSI6eyJ0eXBlIjoiZW50aXRsZW1lbnRfbWVzc2FnZSIsImZpcnN0X3BsYXlfZXhwaXJhdGlvbiI6NjAsInBsYXlyZWFkeSI6eyJyZWFsX3RpbWVfZXhwaXJhdGlvbiI6dHJ1ZX0sImtleXMiOlt7ImlkIjoiOWViNDA1MGQtZTQ0Yi00ODAyLTkzMmUtMjdkNzUwODNlMjY2IiwiZW5jcnlwdGVkX2tleSI6ImxLM09qSExZVzI0Y3Iya3RSNzRmbnc9PSJ9XX19.FAbIiPxX8BHi9RwfzD7Yn-wugU19ghrkBFKsaCPrZmU'
},
}
}
```
**Test stream for iOS:**
Sorry but I can not provide free streams to test. If anyone can provide test streams, or found some we can use, please let me know to also test them.
It has been tested with a private provider and they work, at least with the `getLicense` override method. (An example implementation is provided in the README)
2020-08-13 03:56:21 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-11-24 20:52:46 +09:00
|
|
|
private MediaSource buildMediaSource(Uri uri, String overrideExtension, DrmSessionManager drmSessionManager, long cropStartMs, long cropEndMs) {
|
2021-10-12 14:56:50 +03:00
|
|
|
if (uri == null) {
|
|
|
|
throw new IllegalStateException("Invalid video uri");
|
|
|
|
}
|
2024-04-16 10:41:39 +02:00
|
|
|
int type;
|
|
|
|
if ("rtsp".equals(overrideExtension)) {
|
2024-05-22 14:02:55 +02:00
|
|
|
type = CONTENT_TYPE_RTSP;
|
2024-04-16 10:41:39 +02:00
|
|
|
} else {
|
|
|
|
type = Util.inferContentType(!TextUtils.isEmpty(overrideExtension) ? "." + overrideExtension
|
|
|
|
: uri.getLastPathSegment());
|
|
|
|
}
|
2021-05-17 13:09:09 +03:00
|
|
|
config.setDisableDisconnectError(this.disableDisconnectError);
|
2022-11-16 11:43:25 +01:00
|
|
|
|
2024-05-07 12:30:57 +02:00
|
|
|
MediaItem.Builder mediaItemBuilder = new MediaItem.Builder()
|
|
|
|
.setUri(uri);
|
|
|
|
|
2024-06-07 15:06:49 +02:00
|
|
|
// refresh custom Metadata
|
2024-09-03 12:46:20 +03:30
|
|
|
MediaMetadata customMetadata = ConfigurationUtils.buildCustomMetadata(source.getMetadata());
|
2024-05-07 12:30:57 +02:00
|
|
|
if (customMetadata != null) {
|
|
|
|
mediaItemBuilder.setMediaMetadata(customMetadata);
|
|
|
|
}
|
2024-10-10 23:53:39 +02:00
|
|
|
if (source.getAdsProps() != null) {
|
|
|
|
Uri adTagUrl = source.getAdsProps().getAdTagUrl();
|
|
|
|
if (adTagUrl != null) {
|
|
|
|
mediaItemBuilder.setAdsConfiguration(
|
|
|
|
new MediaItem.AdsConfiguration.Builder(adTagUrl).build()
|
|
|
|
);
|
|
|
|
}
|
2022-11-16 20:46:36 +01:00
|
|
|
}
|
|
|
|
|
2024-05-20 16:18:20 +02:00
|
|
|
MediaItem.LiveConfiguration.Builder liveConfiguration = ConfigurationUtils.getLiveConfiguration(bufferConfig);
|
|
|
|
mediaItemBuilder.setLiveConfiguration(liveConfiguration.build());
|
|
|
|
|
2023-11-18 22:13:54 +09:00
|
|
|
MediaSource.Factory mediaSourceFactory;
|
2023-10-12 10:36:43 +02:00
|
|
|
DrmSessionManagerProvider drmProvider;
|
2024-05-22 14:02:55 +02:00
|
|
|
List<StreamKey> streamKeys = new ArrayList<>();
|
2022-06-16 00:24:55 +07:00
|
|
|
if (drmSessionManager != null) {
|
2023-11-18 22:13:54 +09:00
|
|
|
drmProvider = ((_mediaItem) -> drmSessionManager);
|
2022-08-26 10:32:22 -04:00
|
|
|
} else {
|
|
|
|
drmProvider = new DefaultDrmSessionManagerProvider();
|
2022-06-16 00:24:55 +07:00
|
|
|
}
|
2023-11-18 22:13:54 +09:00
|
|
|
|
2024-10-17 21:56:06 +02:00
|
|
|
|
2017-01-11 12:51:45 +00:00
|
|
|
switch (type) {
|
2023-01-02 22:59:10 +01:00
|
|
|
case CONTENT_TYPE_SS:
|
2024-04-16 14:23:19 +02:00
|
|
|
if(!BuildConfig.USE_EXOPLAYER_SMOOTH_STREAMING) {
|
|
|
|
DebugLog.e("Exo Player Exception", "Smooth Streaming is not enabled!");
|
|
|
|
throw new IllegalStateException("Smooth Streaming is not enabled!");
|
|
|
|
}
|
|
|
|
|
2023-11-18 22:13:54 +09:00
|
|
|
mediaSourceFactory = new SsMediaSource.Factory(
|
2019-09-16 16:29:31 -04:00
|
|
|
new DefaultSsChunkSource.Factory(mediaDataSourceFactory),
|
2023-08-22 23:30:01 -04:00
|
|
|
buildDataSourceFactory(false)
|
2023-11-18 22:13:54 +09:00
|
|
|
);
|
2023-02-09 09:38:05 +02:00
|
|
|
break;
|
2023-01-02 22:59:10 +01:00
|
|
|
case CONTENT_TYPE_DASH:
|
2024-04-16 14:23:19 +02:00
|
|
|
if(!BuildConfig.USE_EXOPLAYER_DASH) {
|
|
|
|
DebugLog.e("Exo Player Exception", "DASH is not enabled!");
|
|
|
|
throw new IllegalStateException("DASH is not enabled!");
|
|
|
|
}
|
|
|
|
|
2023-11-18 22:13:54 +09:00
|
|
|
mediaSourceFactory = new DashMediaSource.Factory(
|
2019-09-16 16:29:31 -04:00
|
|
|
new DefaultDashChunkSource.Factory(mediaDataSourceFactory),
|
2023-08-22 23:30:01 -04:00
|
|
|
buildDataSourceFactory(false)
|
2023-11-18 22:13:54 +09:00
|
|
|
);
|
2023-02-09 09:38:05 +02:00
|
|
|
break;
|
2023-01-02 22:59:10 +01:00
|
|
|
case CONTENT_TYPE_HLS:
|
2024-04-16 14:23:19 +02:00
|
|
|
if (!BuildConfig.USE_EXOPLAYER_HLS) {
|
|
|
|
DebugLog.e("Exo Player Exception", "HLS is not enabled!");
|
|
|
|
throw new IllegalStateException("HLS is not enabled!");
|
|
|
|
}
|
|
|
|
|
2023-11-18 22:13:54 +09:00
|
|
|
mediaSourceFactory = new HlsMediaSource.Factory(
|
2023-08-22 23:30:01 -04:00
|
|
|
mediaDataSourceFactory
|
2024-06-20 11:19:35 +02:00
|
|
|
).setAllowChunklessPreparation(source.getTextTracksAllowChunklessPreparation());
|
2023-02-09 09:38:05 +02:00
|
|
|
break;
|
2023-01-02 22:59:10 +01:00
|
|
|
case CONTENT_TYPE_OTHER:
|
2024-05-30 08:53:49 +02:00
|
|
|
if ("asset".equals(uri.getScheme())) {
|
2024-05-06 21:51:17 +02:00
|
|
|
try {
|
2024-07-11 11:37:05 +03:30
|
|
|
DataSource.Factory assetDataSourceFactory = DataSourceUtil.buildAssetDataSourceFactory(themedReactContext, uri);
|
2024-05-06 21:51:17 +02:00
|
|
|
mediaSourceFactory = new ProgressiveMediaSource.Factory(assetDataSourceFactory);
|
|
|
|
} catch (Exception e) {
|
2024-05-30 08:53:49 +02:00
|
|
|
throw new IllegalStateException("cannot open input file" + uri);
|
2024-05-06 21:51:17 +02:00
|
|
|
}
|
2024-05-30 08:53:49 +02:00
|
|
|
} else if ("file".equals(uri.getScheme()) ||
|
2024-05-28 00:29:21 -07:00
|
|
|
!useCache) {
|
2024-05-01 02:20:34 -07:00
|
|
|
mediaSourceFactory = new ProgressiveMediaSource.Factory(
|
|
|
|
mediaDataSourceFactory
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
mediaSourceFactory = new ProgressiveMediaSource.Factory(
|
2024-05-28 00:29:21 -07:00
|
|
|
RNVSimpleCache.INSTANCE.getCacheFactory(buildHttpDataSourceFactory(true))
|
2024-05-01 02:20:34 -07:00
|
|
|
);
|
|
|
|
|
|
|
|
}
|
2023-02-09 09:38:05 +02:00
|
|
|
break;
|
2024-04-16 10:41:39 +02:00
|
|
|
case CONTENT_TYPE_RTSP:
|
2024-04-16 14:23:19 +02:00
|
|
|
if (!BuildConfig.USE_EXOPLAYER_RTSP) {
|
|
|
|
DebugLog.e("Exo Player Exception", "RTSP is not enabled!");
|
|
|
|
throw new IllegalStateException("RTSP is not enabled!");
|
|
|
|
}
|
|
|
|
|
2024-04-16 10:41:39 +02:00
|
|
|
mediaSourceFactory = new RtspMediaSource.Factory();
|
|
|
|
break;
|
2017-01-11 12:51:45 +00:00
|
|
|
default: {
|
|
|
|
throw new IllegalStateException("Unsupported type: " + type);
|
|
|
|
}
|
|
|
|
}
|
2023-02-09 09:38:05 +02:00
|
|
|
|
2024-08-22 17:47:51 +09:00
|
|
|
if (cmcdConfigurationFactory != null) {
|
|
|
|
mediaSourceFactory = mediaSourceFactory.setCmcdConfigurationFactory(
|
|
|
|
cmcdConfigurationFactory::createCmcdConfiguration
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2023-11-18 22:13:54 +09:00
|
|
|
MediaItem mediaItem = mediaItemBuilder.setStreamKeys(streamKeys).build();
|
|
|
|
MediaSource mediaSource = mediaSourceFactory
|
|
|
|
.setDrmSessionManagerProvider(drmProvider)
|
|
|
|
.setLoadErrorHandlingPolicy(
|
2024-10-16 23:41:22 +02:00
|
|
|
config.buildLoadErrorHandlingPolicy(source.getMinLoadRetryCount())
|
2023-11-18 22:13:54 +09:00
|
|
|
)
|
|
|
|
.createMediaSource(mediaItem);
|
|
|
|
|
2023-11-24 20:52:46 +09:00
|
|
|
if (cropStartMs >= 0 && cropEndMs >= 0) {
|
|
|
|
return new ClippingMediaSource(mediaSource, cropStartMs * 1000, cropEndMs * 1000);
|
|
|
|
} else if (cropStartMs >= 0) {
|
|
|
|
return new ClippingMediaSource(mediaSource, cropStartMs * 1000, TIME_END_OF_SOURCE);
|
|
|
|
} else if (cropEndMs >= 0) {
|
|
|
|
return new ClippingMediaSource(mediaSource, 0, cropEndMs * 1000);
|
2023-02-09 09:38:05 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return mediaSource;
|
2017-01-11 12:51:45 +00:00
|
|
|
}
|
|
|
|
|
2024-10-11 22:50:22 +02:00
|
|
|
@Nullable
|
|
|
|
private MediaSource buildTextSource() {
|
2024-09-13 10:50:33 +02:00
|
|
|
if (source.getSideLoadedTextTracks() == null) {
|
2024-10-11 22:50:22 +02:00
|
|
|
return null;
|
2018-06-11 21:25:58 -07:00
|
|
|
}
|
|
|
|
|
2024-10-11 22:50:22 +02:00
|
|
|
List<MediaItem.SubtitleConfiguration> subtitleConfigurations = new ArrayList<>();
|
2018-06-11 21:25:58 -07:00
|
|
|
|
2024-10-11 22:50:22 +02:00
|
|
|
for (SideLoadedTextTrack track : source.getSideLoadedTextTracks().getTracks()) {
|
|
|
|
MediaItem.SubtitleConfiguration subtitleConfiguration = new MediaItem.SubtitleConfiguration.Builder(track.getUri())
|
|
|
|
.setMimeType(track.getType())
|
|
|
|
.setLanguage(track.getLanguage())
|
2022-06-16 00:24:55 +07:00
|
|
|
.setSelectionFlags(C.SELECTION_FLAG_DEFAULT)
|
|
|
|
.setRoleFlags(C.ROLE_FLAG_SUBTITLE)
|
2024-10-11 22:50:22 +02:00
|
|
|
.setLabel(track.getTitle())
|
2022-06-16 00:24:55 +07:00
|
|
|
.build();
|
2024-10-11 22:50:22 +02:00
|
|
|
subtitleConfigurations.add(subtitleConfiguration);
|
|
|
|
}
|
|
|
|
|
|
|
|
MediaItem subtitlesMediaItem = new MediaItem.Builder()
|
|
|
|
.setUri(source.getUri())
|
|
|
|
.setSubtitleConfigurations(subtitleConfigurations).build();
|
|
|
|
|
|
|
|
return new DefaultMediaSourceFactory(mediaDataSourceFactory).createMediaSource(subtitlesMediaItem);
|
2018-06-11 15:23:43 -07:00
|
|
|
}
|
|
|
|
|
2017-01-11 12:51:45 +00:00
|
|
|
private void releasePlayer() {
|
|
|
|
if (player != null) {
|
2024-05-07 12:30:57 +02:00
|
|
|
if(playbackServiceBinder != null) {
|
|
|
|
playbackServiceBinder.getService().unregisterPlayer(player);
|
|
|
|
themedReactContext.unbindService(playbackServiceConnection);
|
|
|
|
}
|
|
|
|
|
2017-03-21 20:25:17 +00:00
|
|
|
updateResumePosition();
|
2017-01-11 12:51:45 +00:00
|
|
|
player.release();
|
2022-06-16 00:24:55 +07:00
|
|
|
player.removeListener(this);
|
2017-01-11 12:51:45 +00:00
|
|
|
trackSelector = null;
|
2024-05-07 12:30:57 +02:00
|
|
|
|
2024-06-25 08:55:32 +02:00
|
|
|
ReactNativeVideoManager.Companion.getInstance().onInstanceRemoved(instanceId, player);
|
2019-07-07 10:21:23 +02:00
|
|
|
player = null;
|
2017-01-11 12:51:45 +00:00
|
|
|
}
|
2024-05-07 12:30:57 +02:00
|
|
|
|
2023-01-18 14:53:34 +03:00
|
|
|
if (adsLoader != null) {
|
|
|
|
adsLoader.release();
|
2024-10-10 23:53:39 +02:00
|
|
|
adsLoader = null;
|
2023-01-18 14:53:34 +03:00
|
|
|
}
|
2017-01-11 12:51:45 +00:00
|
|
|
progressHandler.removeMessages(SHOW_PROGRESS);
|
|
|
|
audioBecomingNoisyReceiver.removeListener();
|
2019-09-16 16:29:31 -04:00
|
|
|
bandwidthMeter.removeEventListener(this);
|
2023-11-18 22:13:54 +09:00
|
|
|
|
|
|
|
if (mainHandler != null && mainRunnable != null) {
|
|
|
|
mainHandler.removeCallbacks(mainRunnable);
|
|
|
|
mainRunnable = null;
|
|
|
|
}
|
2017-01-11 12:51:45 +00:00
|
|
|
}
|
|
|
|
|
2023-05-09 11:45:56 +09:00
|
|
|
private static class OnAudioFocusChangedListener implements AudioManager.OnAudioFocusChangeListener {
|
|
|
|
private final ReactExoplayerView view;
|
2023-10-18 22:45:19 +02:00
|
|
|
private final ThemedReactContext themedReactContext;
|
2023-05-09 11:45:56 +09:00
|
|
|
|
2023-10-18 22:45:19 +02:00
|
|
|
private OnAudioFocusChangedListener(ReactExoplayerView view, ThemedReactContext themedReactContext) {
|
2023-05-09 11:45:56 +09:00
|
|
|
this.view = view;
|
2023-10-18 22:45:19 +02:00
|
|
|
this.themedReactContext = themedReactContext;
|
2023-05-09 11:45:56 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void onAudioFocusChange(int focusChange) {
|
2024-06-20 11:50:56 +02:00
|
|
|
Activity activity = themedReactContext.getCurrentActivity();
|
|
|
|
|
2023-05-09 11:45:56 +09:00
|
|
|
switch (focusChange) {
|
|
|
|
case AudioManager.AUDIOFOCUS_LOSS:
|
|
|
|
view.hasAudioFocus = false;
|
2024-07-04 21:01:28 +09:00
|
|
|
view.eventEmitter.onAudioFocusChanged.invoke(false);
|
2024-01-29 07:36:13 +01:00
|
|
|
// FIXME this pause can cause issue if content doesn't have pause capability (can happen on live channel)
|
2024-06-20 11:50:56 +02:00
|
|
|
if (activity != null) {
|
|
|
|
activity.runOnUiThread(view::pausePlayback);
|
|
|
|
}
|
2023-05-09 11:45:56 +09:00
|
|
|
view.audioManager.abandonAudioFocus(this);
|
|
|
|
break;
|
|
|
|
case AudioManager.AUDIOFOCUS_LOSS_TRANSIENT:
|
2024-07-04 21:01:28 +09:00
|
|
|
view.eventEmitter.onAudioFocusChanged.invoke(false);
|
2023-05-09 11:45:56 +09:00
|
|
|
break;
|
|
|
|
case AudioManager.AUDIOFOCUS_GAIN:
|
|
|
|
view.hasAudioFocus = true;
|
2024-07-04 21:01:28 +09:00
|
|
|
view.eventEmitter.onAudioFocusChanged.invoke(true);
|
2023-05-09 11:45:56 +09:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2023-10-18 22:45:19 +02:00
|
|
|
if (view.player != null && activity != null) {
|
2023-05-09 11:45:56 +09:00
|
|
|
if (focusChange == AudioManager.AUDIOFOCUS_LOSS_TRANSIENT_CAN_DUCK) {
|
|
|
|
// Lower the volume
|
|
|
|
if (!view.muted) {
|
2024-05-22 14:02:55 +02:00
|
|
|
activity.runOnUiThread(() ->
|
2024-07-04 21:01:28 +09:00
|
|
|
view.player.setVolume(view.audioVolume * 0.8f)
|
2024-05-22 14:02:55 +02:00
|
|
|
);
|
2023-05-09 11:45:56 +09:00
|
|
|
}
|
|
|
|
} else if (focusChange == AudioManager.AUDIOFOCUS_GAIN) {
|
|
|
|
// Raise it back to normal
|
|
|
|
if (!view.muted) {
|
2024-05-22 14:02:55 +02:00
|
|
|
activity.runOnUiThread(() ->
|
2024-07-04 21:01:28 +09:00
|
|
|
view.player.setVolume(view.audioVolume * 1)
|
2024-05-22 14:02:55 +02:00
|
|
|
);
|
2023-05-09 11:45:56 +09:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-01-11 12:51:45 +00:00
|
|
|
private boolean requestAudioFocus() {
|
2024-05-30 08:53:49 +02:00
|
|
|
if (disableFocus || source.getUri() == null || this.hasAudioFocus) {
|
2017-01-11 12:51:45 +00:00
|
|
|
return true;
|
|
|
|
}
|
2023-05-09 11:45:56 +09:00
|
|
|
int result = audioManager.requestAudioFocus(audioFocusChangeListener,
|
2017-01-11 12:51:45 +00:00
|
|
|
AudioManager.STREAM_MUSIC,
|
|
|
|
AudioManager.AUDIOFOCUS_GAIN);
|
|
|
|
return result == AudioManager.AUDIOFOCUS_REQUEST_GRANTED;
|
|
|
|
}
|
|
|
|
|
|
|
|
private void setPlayWhenReady(boolean playWhenReady) {
|
|
|
|
if (player == null) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (playWhenReady) {
|
2021-06-24 08:00:38 +03:00
|
|
|
this.hasAudioFocus = requestAudioFocus();
|
|
|
|
if (this.hasAudioFocus) {
|
2017-01-11 12:51:45 +00:00
|
|
|
player.setPlayWhenReady(true);
|
|
|
|
}
|
|
|
|
} else {
|
2024-09-06 15:11:33 +02:00
|
|
|
player.setPlayWhenReady(false);
|
2017-01-11 12:51:45 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-03-29 20:59:58 +01:00
|
|
|
private void resumePlayback() {
|
2017-01-11 12:51:45 +00:00
|
|
|
if (player != null) {
|
2024-03-29 20:59:58 +01:00
|
|
|
if (!player.getPlayWhenReady()) {
|
|
|
|
setPlayWhenReady(true);
|
2017-01-11 12:51:45 +00:00
|
|
|
}
|
2024-03-29 20:59:58 +01:00
|
|
|
setKeepScreenOn(preventsDisplaySleepDuringVideoPlayback);
|
2017-01-11 12:51:45 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private void pausePlayback() {
|
|
|
|
if (player != null) {
|
|
|
|
if (player.getPlayWhenReady()) {
|
|
|
|
setPlayWhenReady(false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
setKeepScreenOn(false);
|
|
|
|
}
|
|
|
|
|
|
|
|
private void stopPlayback() {
|
|
|
|
onStopPlayback();
|
|
|
|
releasePlayer();
|
|
|
|
}
|
|
|
|
|
|
|
|
private void onStopPlayback() {
|
2023-05-09 11:45:56 +09:00
|
|
|
audioManager.abandonAudioFocus(audioFocusChangeListener);
|
2017-01-11 12:51:45 +00:00
|
|
|
}
|
|
|
|
|
2017-03-21 20:25:17 +00:00
|
|
|
private void updateResumePosition() {
|
2022-06-16 00:24:55 +07:00
|
|
|
resumeWindow = player.getCurrentMediaItemIndex();
|
|
|
|
resumePosition = player.isCurrentMediaItemSeekable() ? Math.max(0, player.getCurrentPosition())
|
2017-03-21 20:25:17 +00:00
|
|
|
: C.TIME_UNSET;
|
|
|
|
}
|
|
|
|
|
|
|
|
private void clearResumePosition() {
|
|
|
|
resumeWindow = C.INDEX_UNSET;
|
|
|
|
resumePosition = C.TIME_UNSET;
|
|
|
|
}
|
|
|
|
|
2017-01-11 12:51:45 +00:00
|
|
|
/**
|
|
|
|
* Returns a new DataSource factory.
|
|
|
|
*
|
2023-08-22 23:30:01 -04:00
|
|
|
* @param useBandwidthMeter Whether to set {@link #bandwidthMeter} as a listener to the new
|
2017-01-11 12:51:45 +00:00
|
|
|
* DataSource factory.
|
|
|
|
* @return A new DataSource factory.
|
|
|
|
*/
|
|
|
|
private DataSource.Factory buildDataSourceFactory(boolean useBandwidthMeter) {
|
2019-09-16 16:29:31 -04:00
|
|
|
return DataSourceUtil.getDefaultDataSourceFactory(this.themedReactContext,
|
2024-05-30 08:53:49 +02:00
|
|
|
useBandwidthMeter ? bandwidthMeter : null, source.getHeaders());
|
2017-01-11 12:51:45 +00:00
|
|
|
}
|
|
|
|
|
Add iOS and Android basic DRM support (#1445)
This PR adds support for DRM streams on iOS (Fairplay) and Android (Playready, Widevine, Clearkey)
I am neither Android nor iOS developer, so feel free to provide feedback to improve this PR.
**Test stream for ANDROID:**
```
testStream = {
uri: 'http://profficialsite.origin.mediaservices.windows.net/c51358ea-9a5e-4322-8951-897d640fdfd7/tearsofsteel_4k.ism/manifest(format=mpd-time-csf)',
type: 'mpd',
drm: {
type: DRMType.PLAYREADY,
licenseServer: 'http://test.playready.microsoft.com/service/rightsmanager.asmx?cfg=(persist:false,sl:150)'
}
};
```
or
```
{
uri: 'https://media.axprod.net/TestVectors/v7-MultiDRM-SingleKey/Manifest_1080p.mpd',
drm: {
type: 'widevine', //or DRMType.WIDEVINE
licenseServer: 'https://drm-widevine-licensing.axtest.net/AcquireLicense',
headers: {
'X-AxDRM-Message': 'eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJ2ZXJzaW9uIjoxLCJjb21fa2V5X2lkIjoiYjMzNjRlYjUtNTFmNi00YWUzLThjOTgtMzNjZWQ1ZTMxYzc4IiwibWVzc2FnZSI6eyJ0eXBlIjoiZW50aXRsZW1lbnRfbWVzc2FnZSIsImZpcnN0X3BsYXlfZXhwaXJhdGlvbiI6NjAsInBsYXlyZWFkeSI6eyJyZWFsX3RpbWVfZXhwaXJhdGlvbiI6dHJ1ZX0sImtleXMiOlt7ImlkIjoiOWViNDA1MGQtZTQ0Yi00ODAyLTkzMmUtMjdkNzUwODNlMjY2IiwiZW5jcnlwdGVkX2tleSI6ImxLM09qSExZVzI0Y3Iya3RSNzRmbnc9PSJ9XX19.FAbIiPxX8BHi9RwfzD7Yn-wugU19ghrkBFKsaCPrZmU'
},
}
}
```
**Test stream for iOS:**
Sorry but I can not provide free streams to test. If anyone can provide test streams, or found some we can use, please let me know to also test them.
It has been tested with a private provider and they work, at least with the `getLicense` override method. (An example implementation is provided in the README)
2020-08-13 03:56:21 +02:00
|
|
|
/**
|
|
|
|
* Returns a new HttpDataSource factory.
|
|
|
|
*
|
2023-08-22 23:30:01 -04:00
|
|
|
* @param useBandwidthMeter Whether to set {@link #bandwidthMeter} as a listener to the new
|
|
|
|
* DataSource factory.
|
Add iOS and Android basic DRM support (#1445)
This PR adds support for DRM streams on iOS (Fairplay) and Android (Playready, Widevine, Clearkey)
I am neither Android nor iOS developer, so feel free to provide feedback to improve this PR.
**Test stream for ANDROID:**
```
testStream = {
uri: 'http://profficialsite.origin.mediaservices.windows.net/c51358ea-9a5e-4322-8951-897d640fdfd7/tearsofsteel_4k.ism/manifest(format=mpd-time-csf)',
type: 'mpd',
drm: {
type: DRMType.PLAYREADY,
licenseServer: 'http://test.playready.microsoft.com/service/rightsmanager.asmx?cfg=(persist:false,sl:150)'
}
};
```
or
```
{
uri: 'https://media.axprod.net/TestVectors/v7-MultiDRM-SingleKey/Manifest_1080p.mpd',
drm: {
type: 'widevine', //or DRMType.WIDEVINE
licenseServer: 'https://drm-widevine-licensing.axtest.net/AcquireLicense',
headers: {
'X-AxDRM-Message': 'eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJ2ZXJzaW9uIjoxLCJjb21fa2V5X2lkIjoiYjMzNjRlYjUtNTFmNi00YWUzLThjOTgtMzNjZWQ1ZTMxYzc4IiwibWVzc2FnZSI6eyJ0eXBlIjoiZW50aXRsZW1lbnRfbWVzc2FnZSIsImZpcnN0X3BsYXlfZXhwaXJhdGlvbiI6NjAsInBsYXlyZWFkeSI6eyJyZWFsX3RpbWVfZXhwaXJhdGlvbiI6dHJ1ZX0sImtleXMiOlt7ImlkIjoiOWViNDA1MGQtZTQ0Yi00ODAyLTkzMmUtMjdkNzUwODNlMjY2IiwiZW5jcnlwdGVkX2tleSI6ImxLM09qSExZVzI0Y3Iya3RSNzRmbnc9PSJ9XX19.FAbIiPxX8BHi9RwfzD7Yn-wugU19ghrkBFKsaCPrZmU'
},
}
}
```
**Test stream for iOS:**
Sorry but I can not provide free streams to test. If anyone can provide test streams, or found some we can use, please let me know to also test them.
It has been tested with a private provider and they work, at least with the `getLicense` override method. (An example implementation is provided in the README)
2020-08-13 03:56:21 +02:00
|
|
|
* @return A new HttpDataSource factory.
|
|
|
|
*/
|
|
|
|
private HttpDataSource.Factory buildHttpDataSourceFactory(boolean useBandwidthMeter) {
|
2024-05-30 08:53:49 +02:00
|
|
|
return DataSourceUtil.getDefaultHttpDataSourceFactory(this.themedReactContext, useBandwidthMeter ? bandwidthMeter : null, source.getHeaders());
|
Add iOS and Android basic DRM support (#1445)
This PR adds support for DRM streams on iOS (Fairplay) and Android (Playready, Widevine, Clearkey)
I am neither Android nor iOS developer, so feel free to provide feedback to improve this PR.
**Test stream for ANDROID:**
```
testStream = {
uri: 'http://profficialsite.origin.mediaservices.windows.net/c51358ea-9a5e-4322-8951-897d640fdfd7/tearsofsteel_4k.ism/manifest(format=mpd-time-csf)',
type: 'mpd',
drm: {
type: DRMType.PLAYREADY,
licenseServer: 'http://test.playready.microsoft.com/service/rightsmanager.asmx?cfg=(persist:false,sl:150)'
}
};
```
or
```
{
uri: 'https://media.axprod.net/TestVectors/v7-MultiDRM-SingleKey/Manifest_1080p.mpd',
drm: {
type: 'widevine', //or DRMType.WIDEVINE
licenseServer: 'https://drm-widevine-licensing.axtest.net/AcquireLicense',
headers: {
'X-AxDRM-Message': 'eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJ2ZXJzaW9uIjoxLCJjb21fa2V5X2lkIjoiYjMzNjRlYjUtNTFmNi00YWUzLThjOTgtMzNjZWQ1ZTMxYzc4IiwibWVzc2FnZSI6eyJ0eXBlIjoiZW50aXRsZW1lbnRfbWVzc2FnZSIsImZpcnN0X3BsYXlfZXhwaXJhdGlvbiI6NjAsInBsYXlyZWFkeSI6eyJyZWFsX3RpbWVfZXhwaXJhdGlvbiI6dHJ1ZX0sImtleXMiOlt7ImlkIjoiOWViNDA1MGQtZTQ0Yi00ODAyLTkzMmUtMjdkNzUwODNlMjY2IiwiZW5jcnlwdGVkX2tleSI6ImxLM09qSExZVzI0Y3Iya3RSNzRmbnc9PSJ9XX19.FAbIiPxX8BHi9RwfzD7Yn-wugU19ghrkBFKsaCPrZmU'
},
}
}
```
**Test stream for iOS:**
Sorry but I can not provide free streams to test. If anyone can provide test streams, or found some we can use, please let me know to also test them.
It has been tested with a private provider and they work, at least with the `getLicense` override method. (An example implementation is provided in the README)
2020-08-13 03:56:21 +02:00
|
|
|
}
|
|
|
|
|
2017-01-11 12:51:45 +00:00
|
|
|
// AudioBecomingNoisyListener implementation
|
|
|
|
@Override
|
|
|
|
public void onAudioBecomingNoisy() {
|
2024-07-04 21:01:28 +09:00
|
|
|
eventEmitter.onVideoAudioBecomingNoisy.invoke();
|
2017-01-11 12:51:45 +00:00
|
|
|
}
|
|
|
|
|
2022-06-16 00:24:55 +07:00
|
|
|
// Player.Listener implementation
|
2017-01-11 12:51:45 +00:00
|
|
|
@Override
|
2022-06-16 00:24:55 +07:00
|
|
|
public void onIsLoadingChanged(boolean isLoading) {
|
2017-01-11 12:51:45 +00:00
|
|
|
// Do nothing.
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
2023-10-12 10:36:43 +02:00
|
|
|
public void onEvents(@NonNull Player player, Player.Events events) {
|
2023-08-22 23:30:01 -04:00
|
|
|
if (events.contains(Player.EVENT_PLAYBACK_STATE_CHANGED) || events.contains(Player.EVENT_PLAY_WHEN_READY_CHANGED)) {
|
2022-06-16 00:24:55 +07:00
|
|
|
int playbackState = player.getPlaybackState();
|
|
|
|
boolean playWhenReady = player.getPlayWhenReady();
|
|
|
|
String text = "onStateChanged: playWhenReady=" + playWhenReady + ", playbackState=";
|
2024-07-04 21:01:28 +09:00
|
|
|
eventEmitter.onPlaybackRateChange.invoke(playWhenReady && playbackState == ExoPlayer.STATE_READY ? 1.0f : 0.0f);
|
2022-06-16 00:24:55 +07:00
|
|
|
switch (playbackState) {
|
|
|
|
case Player.STATE_IDLE:
|
|
|
|
text += "idle";
|
2024-07-04 21:01:28 +09:00
|
|
|
eventEmitter.onVideoIdle.invoke();
|
2022-06-16 00:24:55 +07:00
|
|
|
clearProgressMessageHandler();
|
|
|
|
if (!player.getPlayWhenReady()) {
|
|
|
|
setKeepScreenOn(false);
|
|
|
|
}
|
|
|
|
break;
|
2023-11-18 22:13:54 +09:00
|
|
|
case Player.STATE_BUFFERING:
|
|
|
|
text += "buffering";
|
|
|
|
onBuffering(true);
|
|
|
|
clearProgressMessageHandler();
|
|
|
|
setKeepScreenOn(preventsDisplaySleepDuringVideoPlayback);
|
|
|
|
break;
|
|
|
|
case Player.STATE_READY:
|
|
|
|
text += "ready";
|
2024-07-04 21:01:28 +09:00
|
|
|
eventEmitter.onReadyForDisplay.invoke();
|
2023-11-18 22:13:54 +09:00
|
|
|
onBuffering(false);
|
|
|
|
clearProgressMessageHandler(); // ensure there is no other message
|
|
|
|
startProgressHandler();
|
|
|
|
videoLoaded();
|
|
|
|
if (selectTrackWhenReady && isUsingContentResolution) {
|
|
|
|
selectTrackWhenReady = false;
|
|
|
|
setSelectedTrack(C.TRACK_TYPE_VIDEO, videoTrackType, videoTrackValue);
|
|
|
|
}
|
|
|
|
// Setting the visibility for the playerControlView
|
|
|
|
if (playerControlView != null) {
|
|
|
|
playerControlView.show();
|
|
|
|
}
|
|
|
|
setKeepScreenOn(preventsDisplaySleepDuringVideoPlayback);
|
|
|
|
break;
|
|
|
|
case Player.STATE_ENDED:
|
|
|
|
text += "ended";
|
2024-06-03 12:13:52 +02:00
|
|
|
updateProgress();
|
2024-07-04 21:01:28 +09:00
|
|
|
eventEmitter.onVideoEnd.invoke();
|
2023-11-18 22:13:54 +09:00
|
|
|
onStopPlayback();
|
|
|
|
setKeepScreenOn(false);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
text += "unknown";
|
|
|
|
break;
|
2022-06-16 00:24:55 +07:00
|
|
|
}
|
2023-10-12 10:36:43 +02:00
|
|
|
DebugLog.d(TAG, text);
|
2017-01-11 12:51:45 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private void startProgressHandler() {
|
|
|
|
progressHandler.sendEmptyMessage(SHOW_PROGRESS);
|
|
|
|
}
|
|
|
|
|
2023-11-18 22:13:54 +09:00
|
|
|
/**
|
|
|
|
* The progress message handler will duplicate recursions of the onProgressMessage handler
|
|
|
|
* on change of player state from any state to STATE_READY with playWhenReady is true (when
|
|
|
|
* the video is not paused). This clears all existing messages.
|
2019-09-30 13:27:08 -04:00
|
|
|
*/
|
|
|
|
private void clearProgressMessageHandler() {
|
2023-11-18 22:13:54 +09:00
|
|
|
progressHandler.removeMessages(SHOW_PROGRESS);
|
2019-09-30 13:27:08 -04:00
|
|
|
}
|
|
|
|
|
2017-01-11 12:51:45 +00:00
|
|
|
private void videoLoaded() {
|
2022-11-09 14:26:39 +01:00
|
|
|
if (!player.isPlayingAd() && loadVideoStarted) {
|
2017-01-11 12:51:45 +00:00
|
|
|
loadVideoStarted = false;
|
2022-05-03 14:24:56 +08:00
|
|
|
if (audioTrackType != null) {
|
2022-05-03 14:47:43 +08:00
|
|
|
setSelectedAudioTrack(audioTrackType, audioTrackValue);
|
|
|
|
}
|
|
|
|
if (videoTrackType != null) {
|
|
|
|
setSelectedVideoTrack(videoTrackType, videoTrackValue);
|
|
|
|
}
|
|
|
|
if (textTrackType != null) {
|
|
|
|
setSelectedTextTrack(textTrackType, textTrackValue);
|
|
|
|
}
|
2017-01-11 12:51:45 +00:00
|
|
|
Format videoFormat = player.getVideoFormat();
|
2024-05-31 18:53:24 +02:00
|
|
|
boolean isRotatedContent = videoFormat != null && (videoFormat.rotationDegrees == 90 || videoFormat.rotationDegrees == 270);
|
|
|
|
int width = videoFormat != null ? (isRotatedContent ? videoFormat.height : videoFormat.width) : 0;
|
|
|
|
int height = videoFormat != null ? (isRotatedContent ? videoFormat.width : videoFormat.height) : 0;
|
2024-09-17 15:41:02 +03:30
|
|
|
String trackId = videoFormat != null ? videoFormat.id : null;
|
2022-03-31 15:37:53 +03:00
|
|
|
|
|
|
|
// Properties that must be accessed on the main thread
|
|
|
|
long duration = player.getDuration();
|
|
|
|
long currentPosition = player.getCurrentPosition();
|
2022-08-06 12:05:07 +02:00
|
|
|
ArrayList<Track> audioTracks = getAudioTrackInfo();
|
2023-08-22 23:30:01 -04:00
|
|
|
ArrayList<Track> textTracks = getTextTrackInfo();
|
2022-08-06 12:05:07 +02:00
|
|
|
|
2024-09-14 19:53:54 +02:00
|
|
|
if (source.getContentStartTime() != -1) {
|
2022-08-06 12:05:07 +02:00
|
|
|
ExecutorService es = Executors.newSingleThreadExecutor();
|
2023-11-18 22:13:54 +09:00
|
|
|
es.execute(() -> {
|
|
|
|
// To prevent ANRs caused by getVideoTrackInfo we run this on a different thread and notify the player only when we're done
|
|
|
|
ArrayList<VideoTrack> videoTracks = getVideoTrackInfoFromManifest();
|
|
|
|
if (videoTracks != null) {
|
|
|
|
isUsingContentResolution = true;
|
2022-08-06 12:05:07 +02:00
|
|
|
}
|
2024-07-04 21:01:28 +09:00
|
|
|
eventEmitter.onVideoLoad.invoke(duration, currentPosition, width, height,
|
2023-11-18 22:13:54 +09:00
|
|
|
audioTracks, textTracks, videoTracks, trackId );
|
|
|
|
|
2022-08-06 12:05:07 +02:00
|
|
|
});
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
ArrayList<VideoTrack> videoTracks = getVideoTrackInfo();
|
|
|
|
|
2024-07-04 21:01:28 +09:00
|
|
|
eventEmitter.onVideoLoad.invoke(duration, currentPosition, width, height,
|
2022-08-06 12:05:07 +02:00
|
|
|
audioTracks, textTracks, videoTracks, trackId);
|
2024-10-03 01:07:18 +03:30
|
|
|
refreshControlsStyles();
|
2017-01-11 12:51:45 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-08-06 12:05:07 +02:00
|
|
|
private static boolean isTrackSelected(TrackSelection selection, TrackGroup group,
|
2023-08-22 23:30:01 -04:00
|
|
|
int trackIndex){
|
2022-08-06 12:05:07 +02:00
|
|
|
return selection != null && selection.getTrackGroup() == group
|
2023-08-22 23:30:01 -04:00
|
|
|
&& selection.indexOf( trackIndex ) != C.INDEX_UNSET;
|
2023-08-22 23:42:13 -04:00
|
|
|
}
|
2018-07-17 14:14:21 -07:00
|
|
|
|
2022-08-06 12:05:07 +02:00
|
|
|
private ArrayList<Track> getAudioTrackInfo() {
|
|
|
|
ArrayList<Track> audioTracks = new ArrayList<>();
|
2022-05-10 00:53:09 +03:00
|
|
|
if (trackSelector == null) {
|
|
|
|
// Likely player is unmounting so no audio tracks are available anymore
|
|
|
|
return audioTracks;
|
|
|
|
}
|
|
|
|
|
2018-07-17 14:14:21 -07:00
|
|
|
MappingTrackSelector.MappedTrackInfo info = trackSelector.getCurrentMappedTrackInfo();
|
|
|
|
int index = getTrackRendererIndex(C.TRACK_TYPE_AUDIO);
|
|
|
|
if (info == null || index == C.INDEX_UNSET) {
|
|
|
|
return audioTracks;
|
|
|
|
}
|
|
|
|
TrackGroupArray groups = info.getTrackGroups(index);
|
2022-08-06 12:05:07 +02:00
|
|
|
TrackSelectionArray selectionArray = player.getCurrentTrackSelections();
|
2023-08-22 23:30:01 -04:00
|
|
|
TrackSelection selection = selectionArray.get( C.TRACK_TYPE_AUDIO );
|
2022-08-06 12:05:07 +02:00
|
|
|
|
2018-07-17 14:14:21 -07:00
|
|
|
for (int i = 0; i < groups.length; ++i) {
|
2022-08-06 12:05:07 +02:00
|
|
|
TrackGroup group = groups.get(i);
|
|
|
|
Format format = group.getFormat(0);
|
2023-10-13 17:27:55 +02:00
|
|
|
Track audioTrack = exoplayerTrackToGenericTrack(format, i, selection, group);
|
|
|
|
audioTrack.setBitrate(format.bitrate == Format.NO_VALUE ? 0 : format.bitrate);
|
2022-08-06 12:05:07 +02:00
|
|
|
audioTracks.add(audioTrack);
|
2018-07-17 14:14:21 -07:00
|
|
|
}
|
|
|
|
return audioTracks;
|
|
|
|
}
|
2021-11-09 14:22:32 +02:00
|
|
|
|
2023-10-13 17:27:55 +02:00
|
|
|
private VideoTrack exoplayerVideoTrackToGenericVideoTrack(Format format, int trackIndex) {
|
|
|
|
VideoTrack videoTrack = new VideoTrack();
|
|
|
|
videoTrack.setWidth(format.width == Format.NO_VALUE ? 0 : format.width);
|
|
|
|
videoTrack.setHeight(format.height == Format.NO_VALUE ? 0 : format.height);
|
|
|
|
videoTrack.setBitrate(format.bitrate == Format.NO_VALUE ? 0 : format.bitrate);
|
2024-05-31 18:53:24 +02:00
|
|
|
videoTrack.setRotation(format.rotationDegrees);
|
2023-10-13 17:27:55 +02:00
|
|
|
if (format.codecs != null) videoTrack.setCodecs(format.codecs);
|
2024-05-14 16:39:41 +02:00
|
|
|
videoTrack.setTrackId(format.id == null ? String.valueOf(trackIndex) : format.id);
|
2024-05-22 14:01:55 +02:00
|
|
|
videoTrack.setIndex(trackIndex);
|
2023-10-13 17:27:55 +02:00
|
|
|
return videoTrack;
|
|
|
|
}
|
|
|
|
|
2022-08-06 12:05:07 +02:00
|
|
|
private ArrayList<VideoTrack> getVideoTrackInfo() {
|
|
|
|
ArrayList<VideoTrack> videoTracks = new ArrayList<>();
|
|
|
|
if (trackSelector == null) {
|
2024-05-22 14:01:55 +02:00
|
|
|
// Likely player is unmounting so no video tracks are available anymore
|
2022-08-06 12:05:07 +02:00
|
|
|
return videoTracks;
|
2021-11-09 14:22:32 +02:00
|
|
|
}
|
2018-08-24 15:33:46 +05:30
|
|
|
MappingTrackSelector.MappedTrackInfo info = trackSelector.getCurrentMappedTrackInfo();
|
2022-08-06 12:05:07 +02:00
|
|
|
int index = getTrackRendererIndex(C.TRACK_TYPE_VIDEO);
|
|
|
|
if (info == null || index == C.INDEX_UNSET) {
|
2018-08-24 15:33:46 +05:30
|
|
|
return videoTracks;
|
|
|
|
}
|
|
|
|
|
2022-08-06 12:05:07 +02:00
|
|
|
TrackGroupArray groups = info.getTrackGroups(index);
|
2018-08-24 15:33:46 +05:30
|
|
|
for (int i = 0; i < groups.length; ++i) {
|
|
|
|
TrackGroup group = groups.get(i);
|
|
|
|
|
|
|
|
for (int trackIndex = 0; trackIndex < group.length; trackIndex++) {
|
|
|
|
Format format = group.getFormat(trackIndex);
|
2021-09-26 16:38:10 +03:00
|
|
|
if (isFormatSupported(format)) {
|
2023-10-13 17:27:55 +02:00
|
|
|
VideoTrack videoTrack = exoplayerVideoTrackToGenericVideoTrack(format, trackIndex);
|
2022-08-06 12:05:07 +02:00
|
|
|
videoTracks.add(videoTrack);
|
2021-09-26 16:38:10 +03:00
|
|
|
}
|
2018-08-24 15:33:46 +05:30
|
|
|
}
|
|
|
|
}
|
|
|
|
return videoTracks;
|
|
|
|
}
|
|
|
|
|
2022-08-06 12:05:07 +02:00
|
|
|
private ArrayList<VideoTrack> getVideoTrackInfoFromManifest() {
|
2022-05-31 10:08:14 +03:00
|
|
|
return this.getVideoTrackInfoFromManifest(0);
|
2022-03-31 15:37:53 +03:00
|
|
|
}
|
|
|
|
|
2023-08-22 23:30:01 -04:00
|
|
|
// We need retry count to in case where minefest request fails from poor network conditions
|
2022-08-06 12:05:07 +02:00
|
|
|
@WorkerThread
|
|
|
|
private ArrayList<VideoTrack> getVideoTrackInfoFromManifest(int retryCount) {
|
2021-11-09 14:22:32 +02:00
|
|
|
ExecutorService es = Executors.newSingleThreadExecutor();
|
|
|
|
final DataSource dataSource = this.mediaDataSourceFactory.createDataSource();
|
2024-05-30 08:53:49 +02:00
|
|
|
final Uri sourceUri = source.getUri();
|
2024-09-14 19:53:54 +02:00
|
|
|
final long startTime = source.getContentStartTime() * 1000 - 100; // s -> ms with 100ms offset
|
2021-11-09 14:22:32 +02:00
|
|
|
|
2023-11-18 22:13:54 +09:00
|
|
|
Future<ArrayList<VideoTrack>> result = es.submit(new Callable() {
|
2023-10-12 10:36:43 +02:00
|
|
|
final DataSource ds = dataSource;
|
|
|
|
final Uri uri = sourceUri;
|
|
|
|
final long startTimeUs = startTime * 1000; // ms -> us
|
2021-11-09 14:22:32 +02:00
|
|
|
|
2023-10-12 10:36:43 +02:00
|
|
|
public ArrayList<VideoTrack> call() {
|
2022-08-06 12:05:07 +02:00
|
|
|
ArrayList<VideoTrack> videoTracks = new ArrayList<>();
|
2023-08-22 23:30:01 -04:00
|
|
|
try {
|
2021-11-09 14:22:32 +02:00
|
|
|
DashManifest manifest = DashUtil.loadManifest(this.ds, this.uri);
|
|
|
|
int periodCount = manifest.getPeriodCount();
|
|
|
|
for (int i = 0; i < periodCount; i++) {
|
|
|
|
Period period = manifest.getPeriod(i);
|
2023-08-22 23:30:01 -04:00
|
|
|
for (int adaptationIndex = 0; adaptationIndex < period.adaptationSets.size(); adaptationIndex++) {
|
2021-11-09 14:22:32 +02:00
|
|
|
AdaptationSet adaptation = period.adaptationSets.get(adaptationIndex);
|
|
|
|
if (adaptation.type != C.TRACK_TYPE_VIDEO) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
boolean hasFoundContentPeriod = false;
|
2023-08-22 23:30:01 -04:00
|
|
|
for (int representationIndex = 0; representationIndex < adaptation.representations.size(); representationIndex++) {
|
2021-11-09 14:22:32 +02:00
|
|
|
Representation representation = adaptation.representations.get(representationIndex);
|
|
|
|
Format format = representation.format;
|
|
|
|
if (isFormatSupported(format)) {
|
2022-08-06 12:05:07 +02:00
|
|
|
if (representation.presentationTimeOffsetUs <= startTimeUs) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
hasFoundContentPeriod = true;
|
2023-10-13 17:27:55 +02:00
|
|
|
VideoTrack videoTrack = exoplayerVideoTrackToGenericVideoTrack(format, representationIndex);
|
2022-08-06 12:05:07 +02:00
|
|
|
videoTracks.add(videoTrack);
|
2021-11-09 14:22:32 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if (hasFoundContentPeriod) {
|
|
|
|
return videoTracks;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-10-12 10:36:43 +02:00
|
|
|
} catch (Exception e) {
|
|
|
|
DebugLog.w(TAG, "error in getVideoTrackInfoFromManifest:" + e.getMessage());
|
|
|
|
}
|
2021-11-09 14:22:32 +02:00
|
|
|
return null;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
try {
|
2022-08-06 12:05:07 +02:00
|
|
|
ArrayList<VideoTrack> results = result.get(3000, TimeUnit.MILLISECONDS);
|
2022-03-31 15:37:53 +03:00
|
|
|
if (results == null && retryCount < 1) {
|
2022-05-31 10:08:14 +03:00
|
|
|
return this.getVideoTrackInfoFromManifest(++retryCount);
|
2022-03-31 15:37:53 +03:00
|
|
|
}
|
2021-11-09 14:22:32 +02:00
|
|
|
es.shutdown();
|
|
|
|
return results;
|
2023-10-12 10:36:43 +02:00
|
|
|
} catch (Exception e) {
|
|
|
|
DebugLog.w(TAG, "error in getVideoTrackInfoFromManifest handling request:" + e.getMessage());
|
|
|
|
}
|
2021-11-09 14:22:32 +02:00
|
|
|
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2023-10-13 17:27:55 +02:00
|
|
|
private Track exoplayerTrackToGenericTrack(Format format, int trackIndex, TrackSelection selection, TrackGroup group) {
|
|
|
|
Track track = new Track();
|
|
|
|
track.setIndex(trackIndex);
|
|
|
|
if (format.sampleMimeType != null) track.setMimeType(format.sampleMimeType);
|
|
|
|
if (format.language != null) track.setLanguage(format.language);
|
2024-04-14 14:57:04 +02:00
|
|
|
if (format.label != null) track.setTitle(format.label);
|
2024-07-24 11:38:37 +03:30
|
|
|
track.setSelected(isTrackSelected(selection, group, trackIndex));
|
2023-10-13 17:27:55 +02:00
|
|
|
return track;
|
|
|
|
}
|
|
|
|
|
2022-08-06 12:05:07 +02:00
|
|
|
private ArrayList<Track> getTextTrackInfo() {
|
|
|
|
ArrayList<Track> textTracks = new ArrayList<>();
|
|
|
|
if (trackSelector == null) {
|
|
|
|
return textTracks;
|
|
|
|
}
|
2018-06-11 21:25:58 -07:00
|
|
|
MappingTrackSelector.MappedTrackInfo info = trackSelector.getCurrentMappedTrackInfo();
|
2018-07-17 14:14:21 -07:00
|
|
|
int index = getTrackRendererIndex(C.TRACK_TYPE_TEXT);
|
2018-06-11 21:25:58 -07:00
|
|
|
if (info == null || index == C.INDEX_UNSET) {
|
|
|
|
return textTracks;
|
|
|
|
}
|
2022-08-06 12:05:07 +02:00
|
|
|
TrackSelectionArray selectionArray = player.getCurrentTrackSelections();
|
2023-08-22 23:30:01 -04:00
|
|
|
TrackSelection selection = selectionArray.get( C.TRACK_TYPE_VIDEO );
|
2018-06-11 21:25:58 -07:00
|
|
|
TrackGroupArray groups = info.getTrackGroups(index);
|
2022-08-06 12:05:07 +02:00
|
|
|
|
2018-06-11 21:25:58 -07:00
|
|
|
for (int i = 0; i < groups.length; ++i) {
|
2022-08-06 12:05:07 +02:00
|
|
|
TrackGroup group = groups.get(i);
|
|
|
|
Format format = group.getFormat(0);
|
2023-10-13 17:27:55 +02:00
|
|
|
Track textTrack = exoplayerTrackToGenericTrack(format, i, selection, group);
|
2022-08-06 12:05:07 +02:00
|
|
|
textTracks.add(textTrack);
|
2018-06-11 21:25:58 -07:00
|
|
|
}
|
|
|
|
return textTracks;
|
|
|
|
}
|
|
|
|
|
2017-01-11 12:51:45 +00:00
|
|
|
private void onBuffering(boolean buffering) {
|
|
|
|
if (isBuffering == buffering) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2024-07-11 10:08:36 +02:00
|
|
|
if (isPaused && isSeeking && !buffering) {
|
|
|
|
eventEmitter.onVideoSeek.invoke(player.getCurrentPosition(), seekPosition);
|
|
|
|
isSeeking = false;
|
|
|
|
}
|
|
|
|
|
2017-01-11 12:51:45 +00:00
|
|
|
isBuffering = buffering;
|
2024-07-04 21:01:28 +09:00
|
|
|
eventEmitter.onVideoBuffer.invoke(buffering);
|
2017-01-11 12:51:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
2024-04-22 10:35:51 +02:00
|
|
|
public void onPositionDiscontinuity(@NonNull Player.PositionInfo oldPosition, @NonNull Player.PositionInfo newPosition, @Player.DiscontinuityReason int reason) {
|
|
|
|
if (reason == Player.DISCONTINUITY_REASON_SEEK) {
|
2024-07-11 10:08:36 +02:00
|
|
|
isSeeking = true;
|
|
|
|
seekPosition = newPosition.positionMs;
|
2024-04-22 10:35:51 +02:00
|
|
|
if (isUsingContentResolution) {
|
|
|
|
// We need to update the selected track to make sure that it still matches user selection if track list has changed in this period
|
|
|
|
setSelectedTrack(C.TRACK_TYPE_VIDEO, videoTrackType, videoTrackValue);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-03-21 20:25:17 +00:00
|
|
|
if (playerNeedsSource) {
|
2023-08-22 23:30:01 -04:00
|
|
|
// This will only occur if the user has performed a seek whilst in the error state. Update the
|
|
|
|
// resume position so that if the user then retries, playback will resume from the position to
|
2017-03-21 20:25:17 +00:00
|
|
|
// which they seeked.
|
|
|
|
updateResumePosition();
|
|
|
|
}
|
2021-11-09 14:22:32 +02:00
|
|
|
if (isUsingContentResolution) {
|
2023-08-22 23:30:01 -04:00
|
|
|
// Discontinuity events might have a different track list so we update the selected track
|
2021-11-09 14:22:32 +02:00
|
|
|
setSelectedTrack(C.TRACK_TYPE_VIDEO, videoTrackType, videoTrackValue);
|
|
|
|
selectTrackWhenReady = true;
|
|
|
|
}
|
2023-08-22 23:30:01 -04:00
|
|
|
// When repeat is turned on, reaching the end of the video will not cause a state change
|
2018-05-28 21:26:23 -07:00
|
|
|
// so we need to explicitly detect it.
|
2022-06-16 00:24:55 +07:00
|
|
|
if (reason == Player.DISCONTINUITY_REASON_AUTO_TRANSITION
|
2018-05-28 21:22:47 -07:00
|
|
|
&& player.getRepeatMode() == Player.REPEAT_MODE_ONE) {
|
2024-06-03 12:13:52 +02:00
|
|
|
updateProgress();
|
2024-07-04 21:01:28 +09:00
|
|
|
eventEmitter.onVideoEnd.invoke();
|
2018-05-28 21:22:47 -07:00
|
|
|
}
|
2017-01-11 12:51:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
2023-10-12 10:36:43 +02:00
|
|
|
public void onTimelineChanged(@NonNull Timeline timeline, int reason) {
|
2018-04-03 12:19:04 -05:00
|
|
|
// Do nothing.
|
2017-01-11 12:51:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
2023-10-12 10:36:43 +02:00
|
|
|
public void onTracksChanged(@NonNull Tracks tracks) {
|
2024-07-04 21:01:28 +09:00
|
|
|
eventEmitter.onTextTracks.invoke(getTextTrackInfo());
|
|
|
|
eventEmitter.onAudioTracks.invoke(getAudioTrackInfo());
|
|
|
|
eventEmitter.onVideoTracks.invoke(getVideoTrackInfo());
|
2017-01-11 12:51:45 +00:00
|
|
|
}
|
|
|
|
|
2017-06-14 00:45:12 +02:00
|
|
|
@Override
|
|
|
|
public void onPlaybackParametersChanged(PlaybackParameters params) {
|
2024-07-04 21:01:28 +09:00
|
|
|
eventEmitter.onPlaybackRateChange.invoke(params.speed);
|
2017-06-14 00:45:12 +02:00
|
|
|
}
|
|
|
|
|
2023-11-04 18:11:54 +01:00
|
|
|
@Override
|
|
|
|
public void onVolumeChanged(float volume) {
|
2024-07-04 21:01:28 +09:00
|
|
|
eventEmitter.onVolumeChange.invoke(volume);
|
2023-11-04 18:11:54 +01:00
|
|
|
}
|
|
|
|
|
2017-01-11 12:51:45 +00:00
|
|
|
@Override
|
2022-04-19 12:12:47 -04:00
|
|
|
public void onIsPlayingChanged(boolean isPlaying) {
|
2024-07-11 10:08:36 +02:00
|
|
|
if (isPlaying && isSeeking) {
|
|
|
|
eventEmitter.onVideoSeek.invoke(player.getCurrentPosition(), seekPosition);
|
|
|
|
}
|
|
|
|
|
|
|
|
eventEmitter.onVideoPlaybackStateChanged.invoke(isPlaying, isSeeking);
|
2024-10-03 01:07:18 +03:30
|
|
|
|
2024-07-11 10:08:36 +02:00
|
|
|
if (isPlaying) {
|
|
|
|
isSeeking = false;
|
|
|
|
}
|
2022-04-19 12:12:47 -04:00
|
|
|
}
|
|
|
|
|
2017-01-11 12:51:45 +00:00
|
|
|
@Override
|
2023-10-12 10:36:43 +02:00
|
|
|
public void onPlayerError(@NonNull PlaybackException e) {
|
2022-06-16 00:24:55 +07:00
|
|
|
String errorString = "ExoPlaybackException: " + PlaybackException.getErrorCodeName(e.errorCode);
|
2024-05-22 14:02:55 +02:00
|
|
|
String errorCode = "2" + e.errorCode;
|
2023-08-22 23:30:01 -04:00
|
|
|
switch(e.errorCode) {
|
2022-06-16 00:24:55 +07:00
|
|
|
case PlaybackException.ERROR_CODE_DRM_DEVICE_REVOKED:
|
|
|
|
case PlaybackException.ERROR_CODE_DRM_LICENSE_ACQUISITION_FAILED:
|
|
|
|
case PlaybackException.ERROR_CODE_DRM_PROVISIONING_FAILED:
|
|
|
|
case PlaybackException.ERROR_CODE_DRM_SYSTEM_ERROR:
|
|
|
|
case PlaybackException.ERROR_CODE_DRM_UNSPECIFIED:
|
|
|
|
if (!hasDrmFailed) {
|
2023-08-22 23:30:01 -04:00
|
|
|
// When DRM fails to reach the app level certificate server it will fail with a source error so we assume that it is DRM related and try one more time
|
2022-04-07 19:36:50 +03:00
|
|
|
hasDrmFailed = true;
|
|
|
|
playerNeedsSource = true;
|
|
|
|
updateResumePosition();
|
|
|
|
initializePlayer();
|
2022-04-26 02:44:37 +03:00
|
|
|
setPlayWhenReady(true);
|
2022-04-07 19:36:50 +03:00
|
|
|
return;
|
|
|
|
}
|
2022-06-16 00:24:55 +07:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
2017-12-06 08:56:41 -08:00
|
|
|
}
|
2024-07-04 21:01:28 +09:00
|
|
|
eventEmitter.onVideoError.invoke(errorString, e, errorCode);
|
2017-01-11 12:51:45 +00:00
|
|
|
playerNeedsSource = true;
|
2017-03-21 20:25:17 +00:00
|
|
|
if (isBehindLiveWindow(e)) {
|
|
|
|
clearResumePosition();
|
2024-09-04 16:54:08 +09:00
|
|
|
if (player != null) {
|
|
|
|
player.seekToDefaultPosition();
|
|
|
|
player.prepare();
|
|
|
|
}
|
2017-03-21 20:25:17 +00:00
|
|
|
} else {
|
|
|
|
updateResumePosition();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-06-16 00:24:55 +07:00
|
|
|
private static boolean isBehindLiveWindow(PlaybackException e) {
|
|
|
|
return e.errorCode == PlaybackException.ERROR_CODE_BEHIND_LIVE_WINDOW;
|
2017-01-11 12:51:45 +00:00
|
|
|
}
|
|
|
|
|
2018-07-17 14:14:21 -07:00
|
|
|
public int getTrackRendererIndex(int trackType) {
|
Add iOS and Android basic DRM support (#1445)
This PR adds support for DRM streams on iOS (Fairplay) and Android (Playready, Widevine, Clearkey)
I am neither Android nor iOS developer, so feel free to provide feedback to improve this PR.
**Test stream for ANDROID:**
```
testStream = {
uri: 'http://profficialsite.origin.mediaservices.windows.net/c51358ea-9a5e-4322-8951-897d640fdfd7/tearsofsteel_4k.ism/manifest(format=mpd-time-csf)',
type: 'mpd',
drm: {
type: DRMType.PLAYREADY,
licenseServer: 'http://test.playready.microsoft.com/service/rightsmanager.asmx?cfg=(persist:false,sl:150)'
}
};
```
or
```
{
uri: 'https://media.axprod.net/TestVectors/v7-MultiDRM-SingleKey/Manifest_1080p.mpd',
drm: {
type: 'widevine', //or DRMType.WIDEVINE
licenseServer: 'https://drm-widevine-licensing.axtest.net/AcquireLicense',
headers: {
'X-AxDRM-Message': 'eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJ2ZXJzaW9uIjoxLCJjb21fa2V5X2lkIjoiYjMzNjRlYjUtNTFmNi00YWUzLThjOTgtMzNjZWQ1ZTMxYzc4IiwibWVzc2FnZSI6eyJ0eXBlIjoiZW50aXRsZW1lbnRfbWVzc2FnZSIsImZpcnN0X3BsYXlfZXhwaXJhdGlvbiI6NjAsInBsYXlyZWFkeSI6eyJyZWFsX3RpbWVfZXhwaXJhdGlvbiI6dHJ1ZX0sImtleXMiOlt7ImlkIjoiOWViNDA1MGQtZTQ0Yi00ODAyLTkzMmUtMjdkNzUwODNlMjY2IiwiZW5jcnlwdGVkX2tleSI6ImxLM09qSExZVzI0Y3Iya3RSNzRmbnc9PSJ9XX19.FAbIiPxX8BHi9RwfzD7Yn-wugU19ghrkBFKsaCPrZmU'
},
}
}
```
**Test stream for iOS:**
Sorry but I can not provide free streams to test. If anyone can provide test streams, or found some we can use, please let me know to also test them.
It has been tested with a private provider and they work, at least with the `getLicense` override method. (An example implementation is provided in the README)
2020-08-13 03:56:21 +02:00
|
|
|
if (player != null) {
|
|
|
|
int rendererCount = player.getRendererCount();
|
|
|
|
for (int rendererIndex = 0; rendererIndex < rendererCount; rendererIndex++) {
|
|
|
|
if (player.getRendererType(rendererIndex) == trackType) {
|
|
|
|
return rendererIndex;
|
|
|
|
}
|
2018-06-02 02:24:13 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return C.INDEX_UNSET;
|
|
|
|
}
|
|
|
|
|
2017-02-14 03:38:02 +01:00
|
|
|
@Override
|
2023-10-12 10:36:43 +02:00
|
|
|
public void onMetadata(@NonNull Metadata metadata) {
|
2023-10-12 21:46:40 +02:00
|
|
|
ArrayList<TimedMetadata> metadataArray = new ArrayList<>();
|
|
|
|
for (int i = 0; i < metadata.length(); i++) {
|
|
|
|
Metadata.Entry entry = metadata.get(i);
|
|
|
|
|
|
|
|
if (entry instanceof Id3Frame) {
|
|
|
|
Id3Frame frame = (Id3Frame) metadata.get(i);
|
|
|
|
|
|
|
|
String value = "";
|
|
|
|
|
|
|
|
if (frame instanceof TextInformationFrame) {
|
|
|
|
TextInformationFrame txxxFrame = (TextInformationFrame) frame;
|
|
|
|
value = txxxFrame.value;
|
|
|
|
}
|
2023-10-13 17:27:55 +02:00
|
|
|
TimedMetadata timedMetadata = new TimedMetadata(frame.id, value);
|
2023-10-12 21:46:40 +02:00
|
|
|
metadataArray.add(timedMetadata);
|
2024-05-23 17:35:13 +02:00
|
|
|
} else if (entry instanceof EventMessage) {
|
|
|
|
EventMessage eventMessage = (EventMessage) entry;
|
2023-10-13 17:27:55 +02:00
|
|
|
TimedMetadata timedMetadata = new TimedMetadata(eventMessage.schemeIdUri, eventMessage.value);
|
2023-10-12 21:46:40 +02:00
|
|
|
metadataArray.add(timedMetadata);
|
|
|
|
} else {
|
2024-05-22 14:02:55 +02:00
|
|
|
DebugLog.d(TAG, "unhandled metadata " + entry);
|
2023-10-12 21:46:40 +02:00
|
|
|
}
|
|
|
|
}
|
2024-07-04 21:01:28 +09:00
|
|
|
eventEmitter.onTimedMetadata.invoke(metadataArray);
|
2017-02-14 03:38:02 +01:00
|
|
|
}
|
|
|
|
|
2024-03-11 21:50:19 +09:00
|
|
|
public void onCues(CueGroup cueGroup) {
|
|
|
|
if (!cueGroup.cues.isEmpty() && cueGroup.cues.get(0).text != null) {
|
|
|
|
String subtitleText = cueGroup.cues.get(0).text.toString();
|
2024-07-04 21:01:28 +09:00
|
|
|
eventEmitter.onTextTrackDataChanged.invoke(subtitleText);
|
2024-03-11 21:50:19 +09:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-01-11 12:51:45 +00:00
|
|
|
// ReactExoplayerViewManager public api
|
|
|
|
|
2024-05-30 08:53:49 +02:00
|
|
|
public void setSrc(Source source) {
|
|
|
|
if (source.getUri() != null) {
|
|
|
|
clearResumePosition();
|
|
|
|
boolean isSourceEqual = source.isEquals(this.source);
|
2022-04-26 02:44:37 +03:00
|
|
|
hasDrmFailed = false;
|
2024-05-30 08:53:49 +02:00
|
|
|
this.source = source;
|
2023-08-22 23:30:01 -04:00
|
|
|
this.mediaDataSourceFactory =
|
|
|
|
DataSourceUtil.getDefaultDataSourceFactory(this.themedReactContext, bandwidthMeter,
|
2024-05-30 08:53:49 +02:00
|
|
|
source.getHeaders());
|
|
|
|
|
2024-08-22 17:47:51 +09:00
|
|
|
if (source.getCmcdProps() != null) {
|
|
|
|
CMCDConfig cmcdConfig = new CMCDConfig(source.getCmcdProps());
|
|
|
|
CmcdConfiguration.Factory factory = cmcdConfig.toCmcdConfigurationFactory();
|
|
|
|
this.setCmcdConfigurationFactory(factory);
|
|
|
|
} else {
|
|
|
|
this.setCmcdConfigurationFactory(null);
|
|
|
|
}
|
|
|
|
|
2021-06-30 11:34:02 +09:00
|
|
|
if (!isSourceEqual) {
|
2024-10-17 21:56:06 +02:00
|
|
|
playerNeedsSource = true;
|
|
|
|
initializePlayer();
|
2017-03-21 20:26:23 +00:00
|
|
|
}
|
2024-10-10 22:59:41 +02:00
|
|
|
} else {
|
|
|
|
clearSrc();
|
2017-01-11 12:51:45 +00:00
|
|
|
}
|
|
|
|
}
|
2021-06-30 11:34:02 +09:00
|
|
|
public void clearSrc() {
|
2024-05-30 08:53:49 +02:00
|
|
|
if (source.getUri() != null) {
|
2024-05-17 15:16:48 +02:00
|
|
|
if (player != null) {
|
|
|
|
player.stop();
|
|
|
|
player.clearMediaItems();
|
|
|
|
}
|
2021-06-30 11:34:02 +09:00
|
|
|
}
|
2024-10-10 22:59:41 +02:00
|
|
|
exoPlayerView.hideAds();
|
|
|
|
this.source = new Source();
|
|
|
|
this.mediaDataSourceFactory = null;
|
|
|
|
clearResumePosition();
|
2021-06-30 11:34:02 +09:00
|
|
|
}
|
|
|
|
|
2017-03-31 18:15:39 +02:00
|
|
|
public void setProgressUpdateInterval(final float progressUpdateInterval) {
|
|
|
|
mProgressUpdateInterval = progressUpdateInterval;
|
|
|
|
}
|
|
|
|
|
2018-11-01 21:41:57 +05:30
|
|
|
public void setReportBandwidth(boolean reportBandwidth) {
|
2018-08-25 21:53:11 +05:30
|
|
|
mReportBandwidth = reportBandwidth;
|
2019-09-16 16:29:31 -04:00
|
|
|
}
|
2018-08-25 21:53:11 +05:30
|
|
|
|
2017-01-11 12:51:45 +00:00
|
|
|
public void setResizeModeModifier(@ResizeMode.Mode int resizeMode) {
|
2024-07-10 12:17:22 +02:00
|
|
|
if (exoPlayerView != null) {
|
|
|
|
exoPlayerView.setResizeMode(resizeMode);
|
|
|
|
}
|
2017-01-11 12:51:45 +00:00
|
|
|
}
|
|
|
|
|
2019-07-08 12:47:05 +02:00
|
|
|
private void applyModifiers() {
|
|
|
|
setRepeatModifier(repeat);
|
2019-08-01 11:33:28 -07:00
|
|
|
setMutedModifier(muted);
|
2019-07-08 12:47:05 +02:00
|
|
|
}
|
|
|
|
|
2017-01-11 12:51:45 +00:00
|
|
|
public void setRepeatModifier(boolean repeat) {
|
2018-05-28 21:01:22 -07:00
|
|
|
if (player != null) {
|
|
|
|
if (repeat) {
|
|
|
|
player.setRepeatMode(Player.REPEAT_MODE_ONE);
|
|
|
|
} else {
|
|
|
|
player.setRepeatMode(Player.REPEAT_MODE_OFF);
|
|
|
|
}
|
|
|
|
}
|
2017-01-11 12:51:45 +00:00
|
|
|
this.repeat = repeat;
|
|
|
|
}
|
|
|
|
|
2020-06-16 14:31:23 +02:00
|
|
|
public void setPreventsDisplaySleepDuringVideoPlayback(boolean preventsDisplaySleepDuringVideoPlayback) {
|
|
|
|
this.preventsDisplaySleepDuringVideoPlayback = preventsDisplaySleepDuringVideoPlayback;
|
|
|
|
}
|
|
|
|
|
2023-01-02 21:52:08 +01:00
|
|
|
public void disableTrack(int rendererIndex) {
|
|
|
|
DefaultTrackSelector.Parameters disableParameters = trackSelector.getParameters()
|
|
|
|
.buildUpon()
|
|
|
|
.setRendererDisabled(rendererIndex, true)
|
|
|
|
.build();
|
|
|
|
trackSelector.setParameters(disableParameters);
|
|
|
|
}
|
|
|
|
|
2024-03-22 07:58:09 +01:00
|
|
|
public void setSelectedTrack(int trackType, String type, String value) {
|
2023-08-22 23:30:01 -04:00
|
|
|
if (player == null) return;
|
2018-07-17 14:14:21 -07:00
|
|
|
int rendererIndex = getTrackRendererIndex(trackType);
|
|
|
|
if (rendererIndex == C.INDEX_UNSET) {
|
2018-06-02 02:24:13 -07:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
MappingTrackSelector.MappedTrackInfo info = trackSelector.getCurrentMappedTrackInfo();
|
|
|
|
if (info == null) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-07-17 14:14:21 -07:00
|
|
|
TrackGroupArray groups = info.getTrackGroups(rendererIndex);
|
2018-08-24 15:33:46 +05:30
|
|
|
int groupIndex = C.INDEX_UNSET;
|
2022-06-16 00:24:55 +07:00
|
|
|
List<Integer> tracks = new ArrayList<>();
|
|
|
|
tracks.add(0);
|
2018-07-09 16:18:42 -07:00
|
|
|
|
2018-06-02 02:24:13 -07:00
|
|
|
if (TextUtils.isEmpty(type)) {
|
2018-07-17 14:14:21 -07:00
|
|
|
type = "default";
|
|
|
|
}
|
|
|
|
|
2024-04-25 05:15:32 -04:00
|
|
|
if ("disabled".equals(type)) {
|
2023-01-02 21:52:08 +01:00
|
|
|
disableTrack(rendererIndex);
|
2018-06-02 19:41:50 -07:00
|
|
|
return;
|
2024-04-25 05:15:32 -04:00
|
|
|
} else if ("language".equals(type)) {
|
2018-06-02 02:24:13 -07:00
|
|
|
for (int i = 0; i < groups.length; ++i) {
|
|
|
|
Format format = groups.get(i).getFormat(0);
|
2024-03-22 07:58:09 +01:00
|
|
|
if (format.language != null && format.language.equals(value)) {
|
2018-08-24 15:33:46 +05:30
|
|
|
groupIndex = i;
|
2018-06-02 02:24:13 -07:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2024-04-25 05:15:32 -04:00
|
|
|
} else if ("title".equals(type)) {
|
2018-06-02 02:24:13 -07:00
|
|
|
for (int i = 0; i < groups.length; ++i) {
|
|
|
|
Format format = groups.get(i).getFormat(0);
|
2024-09-02 19:10:39 +02:00
|
|
|
if (format.label != null && format.label.equals(value)) {
|
2018-08-24 15:33:46 +05:30
|
|
|
groupIndex = i;
|
2018-06-02 02:24:13 -07:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2024-04-25 05:15:32 -04:00
|
|
|
} else if ("index".equals(type)) {
|
2024-09-13 10:50:33 +02:00
|
|
|
int iValue = ReactBridgeUtils.safeParseInt(value, -1);
|
|
|
|
if (iValue != -1) {
|
|
|
|
if (trackType == C.TRACK_TYPE_VIDEO && groups.length == 1) {
|
|
|
|
groupIndex = 0;
|
|
|
|
if (iValue < groups.get(groupIndex).length) {
|
|
|
|
tracks.set(0, iValue);
|
|
|
|
}
|
|
|
|
} else if (iValue < groups.length) {
|
|
|
|
groupIndex = iValue;
|
2024-05-22 14:01:55 +02:00
|
|
|
}
|
2018-08-24 15:33:46 +05:30
|
|
|
}
|
2024-04-25 05:15:32 -04:00
|
|
|
} else if ("resolution".equals(type)) {
|
2024-09-13 10:50:33 +02:00
|
|
|
int height = ReactBridgeUtils.safeParseInt(value, -1);
|
|
|
|
if (height != -1) {
|
|
|
|
for (int i = 0; i < groups.length; ++i) { // Search for the exact height
|
|
|
|
TrackGroup group = groups.get(i);
|
|
|
|
Format closestFormat = null;
|
|
|
|
int closestTrackIndex = -1;
|
|
|
|
boolean usingExactMatch = false;
|
|
|
|
for (int j = 0; j < group.length; j++) {
|
|
|
|
Format format = group.getFormat(j);
|
|
|
|
if (format.height == height) {
|
|
|
|
groupIndex = i;
|
|
|
|
tracks.set(0, j);
|
|
|
|
closestFormat = null;
|
|
|
|
closestTrackIndex = -1;
|
|
|
|
usingExactMatch = true;
|
|
|
|
break;
|
|
|
|
} else if (isUsingContentResolution) {
|
|
|
|
// When using content resolution rather than ads, we need to try and find the closest match if there is no exact match
|
|
|
|
if (closestFormat != null) {
|
|
|
|
if ((format.bitrate > closestFormat.bitrate || format.height > closestFormat.height) && format.height < height) {
|
|
|
|
// Higher quality match
|
|
|
|
closestFormat = format;
|
|
|
|
closestTrackIndex = j;
|
|
|
|
}
|
|
|
|
} else if (format.height < height) {
|
2021-11-09 14:22:32 +02:00
|
|
|
closestFormat = format;
|
|
|
|
closestTrackIndex = j;
|
|
|
|
}
|
|
|
|
}
|
2018-08-24 15:33:46 +05:30
|
|
|
}
|
2024-09-13 10:50:33 +02:00
|
|
|
// This is a fallback if the new period contains only higher resolutions than the user has selected
|
|
|
|
if (closestFormat == null && isUsingContentResolution && !usingExactMatch) {
|
|
|
|
// No close match found - so we pick the lowest quality
|
|
|
|
int minHeight = Integer.MAX_VALUE;
|
|
|
|
for (int j = 0; j < group.length; j++) {
|
|
|
|
Format format = group.getFormat(j);
|
|
|
|
if (format.height < minHeight) {
|
|
|
|
minHeight = format.height;
|
|
|
|
groupIndex = i;
|
|
|
|
tracks.set(0, j);
|
|
|
|
}
|
2021-11-09 14:22:32 +02:00
|
|
|
}
|
2018-08-24 15:33:46 +05:30
|
|
|
}
|
2024-09-13 10:50:33 +02:00
|
|
|
// Selecting the closest match found
|
|
|
|
if (closestFormat != null && closestTrackIndex != -1) {
|
|
|
|
// We found the closest match instead of an exact one
|
|
|
|
groupIndex = i;
|
|
|
|
tracks.set(0, closestTrackIndex);
|
|
|
|
}
|
2021-11-09 14:22:32 +02:00
|
|
|
}
|
2018-07-17 14:14:21 -07:00
|
|
|
}
|
2020-10-15 11:17:49 -05:00
|
|
|
} else if (trackType == C.TRACK_TYPE_TEXT && Util.SDK_INT > 18) { // Text default
|
2018-12-31 21:33:02 -08:00
|
|
|
// Use system settings if possible
|
2023-08-22 23:30:01 -04:00
|
|
|
CaptioningManager captioningManager
|
|
|
|
= (CaptioningManager)themedReactContext.getSystemService(Context.CAPTIONING_SERVICE);
|
2018-12-31 21:33:02 -08:00
|
|
|
if (captioningManager != null && captioningManager.isEnabled()) {
|
2018-08-24 15:33:46 +05:30
|
|
|
groupIndex = getGroupIndexForDefaultLocale(groups);
|
2018-07-17 14:14:21 -07:00
|
|
|
}
|
2018-12-31 21:33:02 -08:00
|
|
|
} else if (rendererIndex == C.TRACK_TYPE_AUDIO) { // Audio default
|
|
|
|
groupIndex = getGroupIndexForDefaultLocale(groups);
|
2018-06-02 02:24:13 -07:00
|
|
|
}
|
|
|
|
|
2019-04-03 23:24:02 -07:00
|
|
|
if (groupIndex == C.INDEX_UNSET && trackType == C.TRACK_TYPE_VIDEO && groups.length != 0) { // Video auto
|
|
|
|
// Add all tracks as valid options for ABR to choose from
|
|
|
|
TrackGroup group = groups.get(0);
|
2023-10-12 10:36:43 +02:00
|
|
|
ArrayList<Integer> allTracks = new ArrayList<>(group.length);
|
2019-04-03 23:24:02 -07:00
|
|
|
groupIndex = 0;
|
|
|
|
for (int j = 0; j < group.length; j++) {
|
2022-06-16 00:24:55 +07:00
|
|
|
allTracks.add(j);
|
2021-09-26 16:38:10 +03:00
|
|
|
}
|
2022-09-26 01:51:18 +01:00
|
|
|
|
2021-09-26 16:38:10 +03:00
|
|
|
// Valiate list of all tracks and add only supported formats
|
|
|
|
int supportedFormatLength = 0;
|
2022-06-16 00:24:55 +07:00
|
|
|
for (int g = 0; g < allTracks.size(); g++) {
|
2021-09-26 16:38:10 +03:00
|
|
|
Format format = group.getFormat(g);
|
|
|
|
if (isFormatSupported(format)) {
|
|
|
|
supportedFormatLength++;
|
|
|
|
}
|
|
|
|
}
|
2022-06-16 00:24:55 +07:00
|
|
|
if (allTracks.size() == 1) {
|
2022-06-07 12:23:06 +03:00
|
|
|
// With only one tracks we can't remove any tracks so attempt to play it anyway
|
|
|
|
tracks = allTracks;
|
|
|
|
} else {
|
2023-08-22 23:30:01 -04:00
|
|
|
tracks = new ArrayList<>(supportedFormatLength + 1);
|
2022-06-16 00:24:55 +07:00
|
|
|
for (int k = 0; k < allTracks.size(); k++) {
|
2022-06-07 12:23:06 +03:00
|
|
|
Format format = group.getFormat(k);
|
|
|
|
if (isFormatSupported(format)) {
|
2022-06-16 00:24:55 +07:00
|
|
|
tracks.add(allTracks.get(k));
|
2022-06-07 12:23:06 +03:00
|
|
|
}
|
2021-09-26 16:38:10 +03:00
|
|
|
}
|
2018-12-31 21:33:02 -08:00
|
|
|
}
|
2019-09-16 16:29:31 -04:00
|
|
|
}
|
2019-03-19 13:46:01 -04:00
|
|
|
|
|
|
|
if (groupIndex == C.INDEX_UNSET) {
|
2023-01-02 21:52:08 +01:00
|
|
|
disableTrack(rendererIndex);
|
2018-06-02 02:24:13 -07:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-06-16 00:24:55 +07:00
|
|
|
TrackSelectionOverride selectionOverride = new TrackSelectionOverride(groups.get(groupIndex), tracks);
|
|
|
|
|
2024-09-17 15:57:26 +02:00
|
|
|
DefaultTrackSelector.Parameters.Builder selectionParameters = trackSelector.getParameters()
|
2023-11-18 22:13:54 +09:00
|
|
|
.buildUpon()
|
2024-05-20 15:36:54 +02:00
|
|
|
.setExceedAudioConstraintsIfNecessary(true)
|
|
|
|
.setExceedRendererCapabilitiesIfNecessary(true)
|
|
|
|
.setExceedVideoConstraintsIfNecessary(true)
|
2023-11-18 22:13:54 +09:00
|
|
|
.setRendererDisabled(rendererIndex, false)
|
2024-09-17 15:57:26 +02:00
|
|
|
.clearOverridesOfType(selectionOverride.getType());
|
|
|
|
|
|
|
|
if (trackType == C.TRACK_TYPE_VIDEO && isUsingVideoABR()) {
|
|
|
|
selectionParameters.setMaxVideoBitrate(maxBitRate == 0 ? Integer.MAX_VALUE : maxBitRate);
|
|
|
|
} else {
|
|
|
|
selectionParameters.addOverride(selectionOverride);
|
|
|
|
}
|
|
|
|
|
|
|
|
trackSelector.setParameters(selectionParameters.build());
|
2018-07-17 14:14:21 -07:00
|
|
|
}
|
|
|
|
|
2021-09-26 16:38:10 +03:00
|
|
|
private boolean isFormatSupported(Format format) {
|
|
|
|
int width = format.width == Format.NO_VALUE ? 0 : format.width;
|
|
|
|
int height = format.height == Format.NO_VALUE ? 0 : format.height;
|
|
|
|
float frameRate = format.frameRate == Format.NO_VALUE ? 0 : format.frameRate;
|
|
|
|
String mimeType = format.sampleMimeType;
|
|
|
|
if (mimeType == null) {
|
|
|
|
return true;
|
|
|
|
}
|
2023-10-12 10:36:43 +02:00
|
|
|
boolean isSupported;
|
2021-09-26 16:38:10 +03:00
|
|
|
try {
|
|
|
|
MediaCodecInfo codecInfo = MediaCodecUtil.getDecoderInfo(mimeType, false, false);
|
|
|
|
isSupported = codecInfo.isVideoSizeAndRateSupportedV21(width, height, frameRate);
|
|
|
|
} catch (Exception e) {
|
|
|
|
// Failed to get decoder info - assume it is supported
|
|
|
|
isSupported = true;
|
|
|
|
}
|
|
|
|
return isSupported;
|
|
|
|
}
|
|
|
|
|
2018-08-24 15:33:46 +05:30
|
|
|
private int getGroupIndexForDefaultLocale(TrackGroupArray groups) {
|
2023-08-22 23:30:01 -04:00
|
|
|
if (groups.length == 0){
|
2018-12-31 21:33:02 -08:00
|
|
|
return C.INDEX_UNSET;
|
|
|
|
}
|
|
|
|
|
2018-08-24 15:33:46 +05:30
|
|
|
int groupIndex = 0; // default if no match
|
2018-07-17 14:14:21 -07:00
|
|
|
String locale2 = Locale.getDefault().getLanguage(); // 2 letter code
|
|
|
|
String locale3 = Locale.getDefault().getISO3Language(); // 3 letter code
|
|
|
|
for (int i = 0; i < groups.length; ++i) {
|
|
|
|
Format format = groups.get(i).getFormat(0);
|
|
|
|
String language = format.language;
|
|
|
|
if (language != null && (language.equals(locale2) || language.equals(locale3))) {
|
2018-08-24 15:33:46 +05:30
|
|
|
groupIndex = i;
|
2018-07-17 14:14:21 -07:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2018-08-24 15:33:46 +05:30
|
|
|
return groupIndex;
|
|
|
|
}
|
|
|
|
|
2024-03-22 07:58:09 +01:00
|
|
|
public void setSelectedVideoTrack(String type, String value) {
|
2018-08-24 15:33:46 +05:30
|
|
|
videoTrackType = type;
|
|
|
|
videoTrackValue = value;
|
2024-04-23 04:32:59 +09:00
|
|
|
if (!loadVideoStarted) setSelectedTrack(C.TRACK_TYPE_VIDEO, videoTrackType, videoTrackValue);
|
2018-07-17 14:14:21 -07:00
|
|
|
}
|
|
|
|
|
2024-03-22 07:58:09 +01:00
|
|
|
public void setSelectedAudioTrack(String type, String value) {
|
2018-07-17 14:14:21 -07:00
|
|
|
audioTrackType = type;
|
|
|
|
audioTrackValue = value;
|
|
|
|
setSelectedTrack(C.TRACK_TYPE_AUDIO, audioTrackType, audioTrackValue);
|
|
|
|
}
|
|
|
|
|
2024-03-22 07:58:09 +01:00
|
|
|
public void setSelectedTextTrack(String type, String value) {
|
2018-07-17 14:14:21 -07:00
|
|
|
textTrackType = type;
|
|
|
|
textTrackValue = value;
|
|
|
|
setSelectedTrack(C.TRACK_TYPE_TEXT, textTrackType, textTrackValue);
|
2018-06-02 02:24:13 -07:00
|
|
|
}
|
|
|
|
|
2017-01-11 12:51:45 +00:00
|
|
|
public void setPausedModifier(boolean paused) {
|
|
|
|
isPaused = paused;
|
|
|
|
if (player != null) {
|
|
|
|
if (!paused) {
|
2024-03-29 20:59:58 +01:00
|
|
|
resumePlayback();
|
2017-01-11 12:51:45 +00:00
|
|
|
} else {
|
|
|
|
pausePlayback();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public void setMutedModifier(boolean muted) {
|
2019-08-01 11:33:28 -07:00
|
|
|
this.muted = muted;
|
2017-01-11 12:51:45 +00:00
|
|
|
if (player != null) {
|
2021-05-18 09:20:32 -06:00
|
|
|
player.setVolume(muted ? 0.f : audioVolume);
|
2017-01-11 12:51:45 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-04-12 17:43:52 -04:00
|
|
|
private void changeAudioOutput(AudioOutput output) {
|
|
|
|
if (player != null) {
|
2023-09-05 14:27:10 -04:00
|
|
|
int streamType = output.getStreamType();
|
|
|
|
int usage = Util.getAudioUsageForStreamType(streamType);
|
|
|
|
int contentType = Util.getAudioContentTypeForStreamType(streamType);
|
2023-04-12 17:43:52 -04:00
|
|
|
AudioAttributes audioAttributes = new AudioAttributes.Builder().setUsage(usage)
|
|
|
|
.setContentType(contentType)
|
2023-04-02 14:02:56 -04:00
|
|
|
.build();
|
|
|
|
player.setAudioAttributes(audioAttributes, false);
|
|
|
|
AudioManager audioManager = (AudioManager) themedReactContext.getSystemService(Context.AUDIO_SERVICE);
|
2023-09-05 14:27:10 -04:00
|
|
|
boolean isSpeakerOutput = output == AudioOutput.SPEAKER;
|
2023-04-02 14:02:56 -04:00
|
|
|
audioManager.setMode(
|
2023-09-05 14:27:10 -04:00
|
|
|
isSpeakerOutput ? AudioManager.MODE_NORMAL
|
2023-04-12 17:43:52 -04:00
|
|
|
: AudioManager.MODE_IN_COMMUNICATION);
|
2023-09-05 14:27:10 -04:00
|
|
|
audioManager.setSpeakerphoneOn(isSpeakerOutput);
|
2023-04-02 14:02:56 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-04-12 17:43:52 -04:00
|
|
|
public void setAudioOutput(AudioOutput output) {
|
|
|
|
if (audioOutput != output) {
|
|
|
|
this.audioOutput = output;
|
|
|
|
changeAudioOutput(output);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-01-11 12:51:45 +00:00
|
|
|
public void setVolumeModifier(float volume) {
|
2018-11-14 12:33:28 +01:00
|
|
|
audioVolume = volume;
|
2017-01-11 12:51:45 +00:00
|
|
|
if (player != null) {
|
2018-11-14 12:33:28 +01:00
|
|
|
player.setVolume(audioVolume);
|
2017-01-11 12:51:45 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public void seekTo(long positionMs) {
|
|
|
|
if (player != null) {
|
|
|
|
player.seekTo(positionMs);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-06-14 00:45:12 +02:00
|
|
|
public void setRateModifier(float newRate) {
|
2024-03-21 15:40:25 +01:00
|
|
|
if (newRate <= 0) {
|
|
|
|
DebugLog.w(TAG, "cannot set rate <= 0");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
Add iOS and Android basic DRM support (#1445)
This PR adds support for DRM streams on iOS (Fairplay) and Android (Playready, Widevine, Clearkey)
I am neither Android nor iOS developer, so feel free to provide feedback to improve this PR.
**Test stream for ANDROID:**
```
testStream = {
uri: 'http://profficialsite.origin.mediaservices.windows.net/c51358ea-9a5e-4322-8951-897d640fdfd7/tearsofsteel_4k.ism/manifest(format=mpd-time-csf)',
type: 'mpd',
drm: {
type: DRMType.PLAYREADY,
licenseServer: 'http://test.playready.microsoft.com/service/rightsmanager.asmx?cfg=(persist:false,sl:150)'
}
};
```
or
```
{
uri: 'https://media.axprod.net/TestVectors/v7-MultiDRM-SingleKey/Manifest_1080p.mpd',
drm: {
type: 'widevine', //or DRMType.WIDEVINE
licenseServer: 'https://drm-widevine-licensing.axtest.net/AcquireLicense',
headers: {
'X-AxDRM-Message': 'eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJ2ZXJzaW9uIjoxLCJjb21fa2V5X2lkIjoiYjMzNjRlYjUtNTFmNi00YWUzLThjOTgtMzNjZWQ1ZTMxYzc4IiwibWVzc2FnZSI6eyJ0eXBlIjoiZW50aXRsZW1lbnRfbWVzc2FnZSIsImZpcnN0X3BsYXlfZXhwaXJhdGlvbiI6NjAsInBsYXlyZWFkeSI6eyJyZWFsX3RpbWVfZXhwaXJhdGlvbiI6dHJ1ZX0sImtleXMiOlt7ImlkIjoiOWViNDA1MGQtZTQ0Yi00ODAyLTkzMmUtMjdkNzUwODNlMjY2IiwiZW5jcnlwdGVkX2tleSI6ImxLM09qSExZVzI0Y3Iya3RSNzRmbnc9PSJ9XX19.FAbIiPxX8BHi9RwfzD7Yn-wugU19ghrkBFKsaCPrZmU'
},
}
}
```
**Test stream for iOS:**
Sorry but I can not provide free streams to test. If anyone can provide test streams, or found some we can use, please let me know to also test them.
It has been tested with a private provider and they work, at least with the `getLicense` override method. (An example implementation is provided in the README)
2020-08-13 03:56:21 +02:00
|
|
|
rate = newRate;
|
2017-06-14 00:45:12 +02:00
|
|
|
|
Add iOS and Android basic DRM support (#1445)
This PR adds support for DRM streams on iOS (Fairplay) and Android (Playready, Widevine, Clearkey)
I am neither Android nor iOS developer, so feel free to provide feedback to improve this PR.
**Test stream for ANDROID:**
```
testStream = {
uri: 'http://profficialsite.origin.mediaservices.windows.net/c51358ea-9a5e-4322-8951-897d640fdfd7/tearsofsteel_4k.ism/manifest(format=mpd-time-csf)',
type: 'mpd',
drm: {
type: DRMType.PLAYREADY,
licenseServer: 'http://test.playready.microsoft.com/service/rightsmanager.asmx?cfg=(persist:false,sl:150)'
}
};
```
or
```
{
uri: 'https://media.axprod.net/TestVectors/v7-MultiDRM-SingleKey/Manifest_1080p.mpd',
drm: {
type: 'widevine', //or DRMType.WIDEVINE
licenseServer: 'https://drm-widevine-licensing.axtest.net/AcquireLicense',
headers: {
'X-AxDRM-Message': 'eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJ2ZXJzaW9uIjoxLCJjb21fa2V5X2lkIjoiYjMzNjRlYjUtNTFmNi00YWUzLThjOTgtMzNjZWQ1ZTMxYzc4IiwibWVzc2FnZSI6eyJ0eXBlIjoiZW50aXRsZW1lbnRfbWVzc2FnZSIsImZpcnN0X3BsYXlfZXhwaXJhdGlvbiI6NjAsInBsYXlyZWFkeSI6eyJyZWFsX3RpbWVfZXhwaXJhdGlvbiI6dHJ1ZX0sImtleXMiOlt7ImlkIjoiOWViNDA1MGQtZTQ0Yi00ODAyLTkzMmUtMjdkNzUwODNlMjY2IiwiZW5jcnlwdGVkX2tleSI6ImxLM09qSExZVzI0Y3Iya3RSNzRmbnc9PSJ9XX19.FAbIiPxX8BHi9RwfzD7Yn-wugU19ghrkBFKsaCPrZmU'
},
}
}
```
**Test stream for iOS:**
Sorry but I can not provide free streams to test. If anyone can provide test streams, or found some we can use, please let me know to also test them.
It has been tested with a private provider and they work, at least with the `getLicense` override method. (An example implementation is provided in the README)
2020-08-13 03:56:21 +02:00
|
|
|
if (player != null) {
|
|
|
|
PlaybackParameters params = new PlaybackParameters(rate, 1f);
|
|
|
|
player.setPlaybackParameters(params);
|
|
|
|
}
|
2017-01-11 12:51:45 +00:00
|
|
|
}
|
|
|
|
|
2018-11-26 14:50:31 -08:00
|
|
|
public void setMaxBitRateModifier(int newMaxBitRate) {
|
|
|
|
maxBitRate = newMaxBitRate;
|
2024-09-17 15:57:26 +02:00
|
|
|
if (player != null && isUsingVideoABR()) {
|
|
|
|
// do not apply yet if not auto
|
2018-10-29 09:53:52 -07:00
|
|
|
trackSelector.setParameters(trackSelector.buildUponParameters()
|
2018-12-13 09:50:43 -08:00
|
|
|
.setMaxVideoBitrate(maxBitRate == 0 ? Integer.MAX_VALUE : maxBitRate));
|
2018-10-29 09:53:52 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-01-11 12:51:45 +00:00
|
|
|
public void setPlayInBackground(boolean playInBackground) {
|
2017-05-09 06:30:45 +10:00
|
|
|
this.playInBackground = playInBackground;
|
2017-01-11 12:51:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public void setDisableFocus(boolean disableFocus) {
|
|
|
|
this.disableFocus = disableFocus;
|
|
|
|
}
|
2018-05-17 15:42:44 -07:00
|
|
|
|
2022-09-26 01:51:18 +01:00
|
|
|
public void setFocusable(boolean focusable) {
|
|
|
|
this.focusable = focusable;
|
|
|
|
exoPlayerView.setFocusable(this.focusable);
|
|
|
|
}
|
|
|
|
|
2024-05-07 12:30:57 +02:00
|
|
|
public void setShowNotificationControls(boolean showNotificationControls) {
|
|
|
|
this.showNotificationControls = showNotificationControls;
|
|
|
|
|
|
|
|
if (playbackServiceConnection == null && showNotificationControls) {
|
|
|
|
setupPlaybackService();
|
|
|
|
} else if(!showNotificationControls && playbackServiceConnection != null) {
|
|
|
|
cleanupPlaybackService();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-05-11 22:02:04 +02:00
|
|
|
public void setBufferingStrategy(BufferingStrategy.BufferingStrategyEnum _bufferingStrategy) {
|
|
|
|
bufferingStrategy = _bufferingStrategy;
|
2021-03-18 03:58:04 -07:00
|
|
|
}
|
|
|
|
|
2024-03-22 09:17:00 +01:00
|
|
|
public boolean getPreventsDisplaySleepDuringVideoPlayback() {
|
|
|
|
return preventsDisplaySleepDuringVideoPlayback;
|
|
|
|
}
|
|
|
|
|
2024-06-11 00:11:26 +03:30
|
|
|
private void updateFullScreenButtonVisibility() {
|
2022-11-03 23:06:42 +01:00
|
|
|
if (playerControlView != null) {
|
|
|
|
final ImageButton fullScreenButton = playerControlView.findViewById(R.id.exo_fullscreen);
|
2024-06-11 00:11:26 +03:30
|
|
|
//Handling the fullScreenButton click event
|
|
|
|
if (isFullscreen && fullScreenPlayerView != null && !fullScreenPlayerView.isShowing()) {
|
2022-11-03 23:06:42 +01:00
|
|
|
fullScreenButton.setVisibility(GONE);
|
2024-06-11 00:11:26 +03:30
|
|
|
} else {
|
|
|
|
fullScreenButton.setVisibility(VISIBLE);
|
2022-11-03 23:06:42 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-05-17 13:09:09 +03:00
|
|
|
public void setDisableDisconnectError(boolean disableDisconnectError) {
|
|
|
|
this.disableDisconnectError = disableDisconnectError;
|
|
|
|
}
|
|
|
|
|
2018-05-17 15:42:44 -07:00
|
|
|
public void setFullscreen(boolean fullscreen) {
|
|
|
|
if (fullscreen == isFullscreen) {
|
|
|
|
return; // Avoid generating events when nothing is changing
|
|
|
|
}
|
|
|
|
isFullscreen = fullscreen;
|
|
|
|
|
|
|
|
Activity activity = themedReactContext.getCurrentActivity();
|
|
|
|
if (activity == null) {
|
|
|
|
return;
|
|
|
|
}
|
2022-09-08 00:31:47 -07:00
|
|
|
|
2018-05-17 15:42:44 -07:00
|
|
|
if (isFullscreen) {
|
2024-09-04 09:53:30 +02:00
|
|
|
fullScreenPlayerView = new FullScreenPlayerView(getContext(), exoPlayerView, this, playerControlView, new OnBackPressedCallback(true) {
|
|
|
|
@Override
|
|
|
|
public void handleOnBackPressed() {
|
|
|
|
setFullscreen(false);
|
|
|
|
}
|
|
|
|
}, controlsConfig);
|
2024-07-04 21:01:28 +09:00
|
|
|
eventEmitter.onVideoFullscreenPlayerWillPresent.invoke();
|
2024-06-11 00:11:26 +03:30
|
|
|
if (fullScreenPlayerView != null) {
|
2022-09-13 22:53:49 -07:00
|
|
|
fullScreenPlayerView.show();
|
|
|
|
}
|
2023-12-01 05:16:24 +09:00
|
|
|
UiThreadUtil.runOnUiThread(() -> {
|
2024-07-04 21:01:28 +09:00
|
|
|
eventEmitter.onVideoFullscreenPlayerDidPresent.invoke();
|
2022-09-10 11:52:46 -07:00
|
|
|
});
|
2018-05-17 15:42:44 -07:00
|
|
|
} else {
|
2024-07-04 21:01:28 +09:00
|
|
|
eventEmitter.onVideoFullscreenPlayerWillDismiss.invoke();
|
2024-06-11 00:11:26 +03:30
|
|
|
if (fullScreenPlayerView != null) {
|
2022-09-13 22:53:49 -07:00
|
|
|
fullScreenPlayerView.dismiss();
|
2024-01-26 21:34:07 +01:00
|
|
|
reLayoutControls();
|
2024-06-11 00:11:26 +03:30
|
|
|
setControls(controls);
|
2022-09-13 22:53:49 -07:00
|
|
|
}
|
2023-12-01 05:16:24 +09:00
|
|
|
UiThreadUtil.runOnUiThread(() -> {
|
2024-07-04 21:01:28 +09:00
|
|
|
eventEmitter.onVideoFullscreenPlayerDidDismiss.invoke();
|
2022-09-10 11:52:46 -07:00
|
|
|
});
|
2018-05-17 15:42:44 -07:00
|
|
|
}
|
2023-08-22 23:30:01 -04:00
|
|
|
// need to be done at the end to avoid hiding fullscreen control button when fullScreenPlayerView is shown
|
2024-06-11 00:11:26 +03:30
|
|
|
updateFullScreenButtonVisibility();
|
2018-05-17 15:42:44 -07:00
|
|
|
}
|
2018-06-08 00:01:13 -07:00
|
|
|
|
2018-11-28 14:56:58 +02:00
|
|
|
public void setHideShutterView(boolean hideShutterView) {
|
|
|
|
exoPlayerView.setHideShutterView(hideShutterView);
|
|
|
|
}
|
|
|
|
|
2024-05-06 22:04:40 +02:00
|
|
|
public void setBufferConfig(BufferConfig config) {
|
|
|
|
bufferConfig = config;
|
|
|
|
if (bufferConfig.getCacheSize() > 0) {
|
2024-05-01 02:20:34 -07:00
|
|
|
RNVSimpleCache.INSTANCE.setSimpleCache(
|
|
|
|
this.getContext(),
|
2024-05-28 00:29:21 -07:00
|
|
|
bufferConfig.getCacheSize()
|
2024-05-01 02:20:34 -07:00
|
|
|
);
|
2024-05-28 00:29:21 -07:00
|
|
|
useCache = true;
|
2024-05-01 02:20:34 -07:00
|
|
|
} else {
|
2024-05-28 00:29:21 -07:00
|
|
|
useCache = false;
|
2024-05-01 02:20:34 -07:00
|
|
|
}
|
2018-08-03 15:54:18 -07:00
|
|
|
releasePlayer();
|
2018-08-01 15:58:02 +02:00
|
|
|
initializePlayer();
|
2018-07-31 17:23:20 +02:00
|
|
|
}
|
2019-01-04 14:58:32 +05:30
|
|
|
|
Add iOS and Android basic DRM support (#1445)
This PR adds support for DRM streams on iOS (Fairplay) and Android (Playready, Widevine, Clearkey)
I am neither Android nor iOS developer, so feel free to provide feedback to improve this PR.
**Test stream for ANDROID:**
```
testStream = {
uri: 'http://profficialsite.origin.mediaservices.windows.net/c51358ea-9a5e-4322-8951-897d640fdfd7/tearsofsteel_4k.ism/manifest(format=mpd-time-csf)',
type: 'mpd',
drm: {
type: DRMType.PLAYREADY,
licenseServer: 'http://test.playready.microsoft.com/service/rightsmanager.asmx?cfg=(persist:false,sl:150)'
}
};
```
or
```
{
uri: 'https://media.axprod.net/TestVectors/v7-MultiDRM-SingleKey/Manifest_1080p.mpd',
drm: {
type: 'widevine', //or DRMType.WIDEVINE
licenseServer: 'https://drm-widevine-licensing.axtest.net/AcquireLicense',
headers: {
'X-AxDRM-Message': 'eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJ2ZXJzaW9uIjoxLCJjb21fa2V5X2lkIjoiYjMzNjRlYjUtNTFmNi00YWUzLThjOTgtMzNjZWQ1ZTMxYzc4IiwibWVzc2FnZSI6eyJ0eXBlIjoiZW50aXRsZW1lbnRfbWVzc2FnZSIsImZpcnN0X3BsYXlfZXhwaXJhdGlvbiI6NjAsInBsYXlyZWFkeSI6eyJyZWFsX3RpbWVfZXhwaXJhdGlvbiI6dHJ1ZX0sImtleXMiOlt7ImlkIjoiOWViNDA1MGQtZTQ0Yi00ODAyLTkzMmUtMjdkNzUwODNlMjY2IiwiZW5jcnlwdGVkX2tleSI6ImxLM09qSExZVzI0Y3Iya3RSNzRmbnc9PSJ9XX19.FAbIiPxX8BHi9RwfzD7Yn-wugU19ghrkBFKsaCPrZmU'
},
}
}
```
**Test stream for iOS:**
Sorry but I can not provide free streams to test. If anyone can provide test streams, or found some we can use, please let me know to also test them.
It has been tested with a private provider and they work, at least with the `getLicense` override method. (An example implementation is provided in the README)
2020-08-13 03:56:21 +02:00
|
|
|
@Override
|
2021-06-24 08:01:11 +03:00
|
|
|
public void onDrmKeysLoaded(int windowIndex, MediaSource.MediaPeriodId mediaPeriodId) {
|
2023-10-10 09:47:56 +02:00
|
|
|
DebugLog.d("DRM Info", "onDrmKeysLoaded");
|
Add iOS and Android basic DRM support (#1445)
This PR adds support for DRM streams on iOS (Fairplay) and Android (Playready, Widevine, Clearkey)
I am neither Android nor iOS developer, so feel free to provide feedback to improve this PR.
**Test stream for ANDROID:**
```
testStream = {
uri: 'http://profficialsite.origin.mediaservices.windows.net/c51358ea-9a5e-4322-8951-897d640fdfd7/tearsofsteel_4k.ism/manifest(format=mpd-time-csf)',
type: 'mpd',
drm: {
type: DRMType.PLAYREADY,
licenseServer: 'http://test.playready.microsoft.com/service/rightsmanager.asmx?cfg=(persist:false,sl:150)'
}
};
```
or
```
{
uri: 'https://media.axprod.net/TestVectors/v7-MultiDRM-SingleKey/Manifest_1080p.mpd',
drm: {
type: 'widevine', //or DRMType.WIDEVINE
licenseServer: 'https://drm-widevine-licensing.axtest.net/AcquireLicense',
headers: {
'X-AxDRM-Message': 'eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJ2ZXJzaW9uIjoxLCJjb21fa2V5X2lkIjoiYjMzNjRlYjUtNTFmNi00YWUzLThjOTgtMzNjZWQ1ZTMxYzc4IiwibWVzc2FnZSI6eyJ0eXBlIjoiZW50aXRsZW1lbnRfbWVzc2FnZSIsImZpcnN0X3BsYXlfZXhwaXJhdGlvbiI6NjAsInBsYXlyZWFkeSI6eyJyZWFsX3RpbWVfZXhwaXJhdGlvbiI6dHJ1ZX0sImtleXMiOlt7ImlkIjoiOWViNDA1MGQtZTQ0Yi00ODAyLTkzMmUtMjdkNzUwODNlMjY2IiwiZW5jcnlwdGVkX2tleSI6ImxLM09qSExZVzI0Y3Iya3RSNzRmbnc9PSJ9XX19.FAbIiPxX8BHi9RwfzD7Yn-wugU19ghrkBFKsaCPrZmU'
},
}
}
```
**Test stream for iOS:**
Sorry but I can not provide free streams to test. If anyone can provide test streams, or found some we can use, please let me know to also test them.
It has been tested with a private provider and they work, at least with the `getLicense` override method. (An example implementation is provided in the README)
2020-08-13 03:56:21 +02:00
|
|
|
}
|
|
|
|
|
2023-11-18 22:13:54 +09:00
|
|
|
@Override
|
|
|
|
public void onDrmSessionAcquired(int windowIndex, MediaSource.MediaPeriodId mediaPeriodId, int state) {
|
|
|
|
DebugLog.d("DRM Info", "onDrmSessionAcquired");
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void onDrmSessionReleased(int windowIndex, MediaSource.MediaPeriodId mediaPeriodId) {
|
|
|
|
DebugLog.d("DRM Info", "onDrmSessionReleased");
|
|
|
|
}
|
|
|
|
|
Add iOS and Android basic DRM support (#1445)
This PR adds support for DRM streams on iOS (Fairplay) and Android (Playready, Widevine, Clearkey)
I am neither Android nor iOS developer, so feel free to provide feedback to improve this PR.
**Test stream for ANDROID:**
```
testStream = {
uri: 'http://profficialsite.origin.mediaservices.windows.net/c51358ea-9a5e-4322-8951-897d640fdfd7/tearsofsteel_4k.ism/manifest(format=mpd-time-csf)',
type: 'mpd',
drm: {
type: DRMType.PLAYREADY,
licenseServer: 'http://test.playready.microsoft.com/service/rightsmanager.asmx?cfg=(persist:false,sl:150)'
}
};
```
or
```
{
uri: 'https://media.axprod.net/TestVectors/v7-MultiDRM-SingleKey/Manifest_1080p.mpd',
drm: {
type: 'widevine', //or DRMType.WIDEVINE
licenseServer: 'https://drm-widevine-licensing.axtest.net/AcquireLicense',
headers: {
'X-AxDRM-Message': 'eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJ2ZXJzaW9uIjoxLCJjb21fa2V5X2lkIjoiYjMzNjRlYjUtNTFmNi00YWUzLThjOTgtMzNjZWQ1ZTMxYzc4IiwibWVzc2FnZSI6eyJ0eXBlIjoiZW50aXRsZW1lbnRfbWVzc2FnZSIsImZpcnN0X3BsYXlfZXhwaXJhdGlvbiI6NjAsInBsYXlyZWFkeSI6eyJyZWFsX3RpbWVfZXhwaXJhdGlvbiI6dHJ1ZX0sImtleXMiOlt7ImlkIjoiOWViNDA1MGQtZTQ0Yi00ODAyLTkzMmUtMjdkNzUwODNlMjY2IiwiZW5jcnlwdGVkX2tleSI6ImxLM09qSExZVzI0Y3Iya3RSNzRmbnc9PSJ9XX19.FAbIiPxX8BHi9RwfzD7Yn-wugU19ghrkBFKsaCPrZmU'
},
}
}
```
**Test stream for iOS:**
Sorry but I can not provide free streams to test. If anyone can provide test streams, or found some we can use, please let me know to also test them.
It has been tested with a private provider and they work, at least with the `getLicense` override method. (An example implementation is provided in the README)
2020-08-13 03:56:21 +02:00
|
|
|
@Override
|
2024-05-22 14:02:55 +02:00
|
|
|
public void onDrmSessionManagerError(int windowIndex, MediaSource.MediaPeriodId mediaPeriodId, @NonNull Exception e) {
|
2023-10-10 09:47:56 +02:00
|
|
|
DebugLog.d("DRM Info", "onDrmSessionManagerError");
|
2024-07-04 21:01:28 +09:00
|
|
|
eventEmitter.onVideoError.invoke("onDrmSessionManagerError", e, "3002");
|
Add iOS and Android basic DRM support (#1445)
This PR adds support for DRM streams on iOS (Fairplay) and Android (Playready, Widevine, Clearkey)
I am neither Android nor iOS developer, so feel free to provide feedback to improve this PR.
**Test stream for ANDROID:**
```
testStream = {
uri: 'http://profficialsite.origin.mediaservices.windows.net/c51358ea-9a5e-4322-8951-897d640fdfd7/tearsofsteel_4k.ism/manifest(format=mpd-time-csf)',
type: 'mpd',
drm: {
type: DRMType.PLAYREADY,
licenseServer: 'http://test.playready.microsoft.com/service/rightsmanager.asmx?cfg=(persist:false,sl:150)'
}
};
```
or
```
{
uri: 'https://media.axprod.net/TestVectors/v7-MultiDRM-SingleKey/Manifest_1080p.mpd',
drm: {
type: 'widevine', //or DRMType.WIDEVINE
licenseServer: 'https://drm-widevine-licensing.axtest.net/AcquireLicense',
headers: {
'X-AxDRM-Message': 'eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJ2ZXJzaW9uIjoxLCJjb21fa2V5X2lkIjoiYjMzNjRlYjUtNTFmNi00YWUzLThjOTgtMzNjZWQ1ZTMxYzc4IiwibWVzc2FnZSI6eyJ0eXBlIjoiZW50aXRsZW1lbnRfbWVzc2FnZSIsImZpcnN0X3BsYXlfZXhwaXJhdGlvbiI6NjAsInBsYXlyZWFkeSI6eyJyZWFsX3RpbWVfZXhwaXJhdGlvbiI6dHJ1ZX0sImtleXMiOlt7ImlkIjoiOWViNDA1MGQtZTQ0Yi00ODAyLTkzMmUtMjdkNzUwODNlMjY2IiwiZW5jcnlwdGVkX2tleSI6ImxLM09qSExZVzI0Y3Iya3RSNzRmbnc9PSJ9XX19.FAbIiPxX8BHi9RwfzD7Yn-wugU19ghrkBFKsaCPrZmU'
},
}
}
```
**Test stream for iOS:**
Sorry but I can not provide free streams to test. If anyone can provide test streams, or found some we can use, please let me know to also test them.
It has been tested with a private provider and they work, at least with the `getLicense` override method. (An example implementation is provided in the README)
2020-08-13 03:56:21 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
2021-06-24 08:01:11 +03:00
|
|
|
public void onDrmKeysRestored(int windowIndex, MediaSource.MediaPeriodId mediaPeriodId) {
|
2023-10-10 09:47:56 +02:00
|
|
|
DebugLog.d("DRM Info", "onDrmKeysRestored");
|
Add iOS and Android basic DRM support (#1445)
This PR adds support for DRM streams on iOS (Fairplay) and Android (Playready, Widevine, Clearkey)
I am neither Android nor iOS developer, so feel free to provide feedback to improve this PR.
**Test stream for ANDROID:**
```
testStream = {
uri: 'http://profficialsite.origin.mediaservices.windows.net/c51358ea-9a5e-4322-8951-897d640fdfd7/tearsofsteel_4k.ism/manifest(format=mpd-time-csf)',
type: 'mpd',
drm: {
type: DRMType.PLAYREADY,
licenseServer: 'http://test.playready.microsoft.com/service/rightsmanager.asmx?cfg=(persist:false,sl:150)'
}
};
```
or
```
{
uri: 'https://media.axprod.net/TestVectors/v7-MultiDRM-SingleKey/Manifest_1080p.mpd',
drm: {
type: 'widevine', //or DRMType.WIDEVINE
licenseServer: 'https://drm-widevine-licensing.axtest.net/AcquireLicense',
headers: {
'X-AxDRM-Message': 'eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJ2ZXJzaW9uIjoxLCJjb21fa2V5X2lkIjoiYjMzNjRlYjUtNTFmNi00YWUzLThjOTgtMzNjZWQ1ZTMxYzc4IiwibWVzc2FnZSI6eyJ0eXBlIjoiZW50aXRsZW1lbnRfbWVzc2FnZSIsImZpcnN0X3BsYXlfZXhwaXJhdGlvbiI6NjAsInBsYXlyZWFkeSI6eyJyZWFsX3RpbWVfZXhwaXJhdGlvbiI6dHJ1ZX0sImtleXMiOlt7ImlkIjoiOWViNDA1MGQtZTQ0Yi00ODAyLTkzMmUtMjdkNzUwODNlMjY2IiwiZW5jcnlwdGVkX2tleSI6ImxLM09qSExZVzI0Y3Iya3RSNzRmbnc9PSJ9XX19.FAbIiPxX8BHi9RwfzD7Yn-wugU19ghrkBFKsaCPrZmU'
},
}
}
```
**Test stream for iOS:**
Sorry but I can not provide free streams to test. If anyone can provide test streams, or found some we can use, please let me know to also test them.
It has been tested with a private provider and they work, at least with the `getLicense` override method. (An example implementation is provided in the README)
2020-08-13 03:56:21 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
2021-06-24 08:01:11 +03:00
|
|
|
public void onDrmKeysRemoved(int windowIndex, MediaSource.MediaPeriodId mediaPeriodId) {
|
2023-10-10 09:47:56 +02:00
|
|
|
DebugLog.d("DRM Info", "onDrmKeysRemoved");
|
Add iOS and Android basic DRM support (#1445)
This PR adds support for DRM streams on iOS (Fairplay) and Android (Playready, Widevine, Clearkey)
I am neither Android nor iOS developer, so feel free to provide feedback to improve this PR.
**Test stream for ANDROID:**
```
testStream = {
uri: 'http://profficialsite.origin.mediaservices.windows.net/c51358ea-9a5e-4322-8951-897d640fdfd7/tearsofsteel_4k.ism/manifest(format=mpd-time-csf)',
type: 'mpd',
drm: {
type: DRMType.PLAYREADY,
licenseServer: 'http://test.playready.microsoft.com/service/rightsmanager.asmx?cfg=(persist:false,sl:150)'
}
};
```
or
```
{
uri: 'https://media.axprod.net/TestVectors/v7-MultiDRM-SingleKey/Manifest_1080p.mpd',
drm: {
type: 'widevine', //or DRMType.WIDEVINE
licenseServer: 'https://drm-widevine-licensing.axtest.net/AcquireLicense',
headers: {
'X-AxDRM-Message': 'eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJ2ZXJzaW9uIjoxLCJjb21fa2V5X2lkIjoiYjMzNjRlYjUtNTFmNi00YWUzLThjOTgtMzNjZWQ1ZTMxYzc4IiwibWVzc2FnZSI6eyJ0eXBlIjoiZW50aXRsZW1lbnRfbWVzc2FnZSIsImZpcnN0X3BsYXlfZXhwaXJhdGlvbiI6NjAsInBsYXlyZWFkeSI6eyJyZWFsX3RpbWVfZXhwaXJhdGlvbiI6dHJ1ZX0sImtleXMiOlt7ImlkIjoiOWViNDA1MGQtZTQ0Yi00ODAyLTkzMmUtMjdkNzUwODNlMjY2IiwiZW5jcnlwdGVkX2tleSI6ImxLM09qSExZVzI0Y3Iya3RSNzRmbnc9PSJ9XX19.FAbIiPxX8BHi9RwfzD7Yn-wugU19ghrkBFKsaCPrZmU'
},
}
}
```
**Test stream for iOS:**
Sorry but I can not provide free streams to test. If anyone can provide test streams, or found some we can use, please let me know to also test them.
It has been tested with a private provider and they work, at least with the `getLicense` override method. (An example implementation is provided in the README)
2020-08-13 03:56:21 +02:00
|
|
|
}
|
|
|
|
|
2019-01-04 14:58:32 +05:30
|
|
|
/**
|
|
|
|
* Handling controls prop
|
2019-09-16 16:29:31 -04:00
|
|
|
*
|
2023-08-22 23:30:01 -04:00
|
|
|
* @param controls Controls prop, if true enable controls, if false disable them
|
2019-01-04 14:58:32 +05:30
|
|
|
*/
|
|
|
|
public void setControls(boolean controls) {
|
2019-07-07 22:17:15 +02:00
|
|
|
this.controls = controls;
|
|
|
|
if (controls) {
|
2019-02-10 18:15:30 -08:00
|
|
|
addPlayerControl();
|
2024-06-11 00:11:26 +03:30
|
|
|
updateFullScreenButtonVisibility();
|
2019-07-07 22:17:15 +02:00
|
|
|
} else {
|
|
|
|
int indexOfPC = indexOfChild(playerControlView);
|
|
|
|
if (indexOfPC != -1) {
|
|
|
|
removeViewAt(indexOfPC);
|
|
|
|
}
|
2019-01-04 14:58:32 +05:30
|
|
|
}
|
2024-10-03 01:07:18 +03:30
|
|
|
refreshControlsStyles();
|
2019-01-04 14:58:32 +05:30
|
|
|
}
|
2022-07-05 23:58:30 +02:00
|
|
|
|
|
|
|
public void setSubtitleStyle(SubtitleStyle style) {
|
|
|
|
exoPlayerView.setSubtitleStyle(style);
|
|
|
|
}
|
2022-11-09 14:26:39 +01:00
|
|
|
|
2023-07-23 21:38:26 +05:30
|
|
|
public void setShutterColor(Integer color) {
|
|
|
|
exoPlayerView.setShutterColor(color);
|
|
|
|
}
|
|
|
|
|
2022-11-09 14:26:39 +01:00
|
|
|
@Override
|
|
|
|
public void onAdEvent(AdEvent adEvent) {
|
2023-11-27 21:43:30 +01:00
|
|
|
if (adEvent.getAdData() != null) {
|
2024-07-04 21:01:28 +09:00
|
|
|
eventEmitter.onReceiveAdEvent.invoke(adEvent.getType().name(), adEvent.getAdData());
|
2023-11-27 21:43:30 +01:00
|
|
|
} else {
|
2024-07-04 21:01:28 +09:00
|
|
|
eventEmitter.onReceiveAdEvent.invoke(adEvent.getType().name(), null);
|
2023-11-27 21:43:30 +01:00
|
|
|
}
|
2022-11-09 14:26:39 +01:00
|
|
|
}
|
2023-12-02 13:52:01 +01:00
|
|
|
|
|
|
|
@Override
|
|
|
|
public void onAdError(AdErrorEvent adErrorEvent) {
|
2024-03-21 14:07:53 +01:00
|
|
|
AdError error = adErrorEvent.getError();
|
2024-07-04 21:01:28 +09:00
|
|
|
Map<String, String> errMap = Map.of(
|
|
|
|
"message", error.getMessage(),
|
|
|
|
"code", String.valueOf(error.getErrorCode()),
|
|
|
|
"type", String.valueOf(error.getErrorType())
|
|
|
|
);
|
|
|
|
eventEmitter.onReceiveAdEvent.invoke("ERROR", errMap);
|
2023-12-02 13:52:01 +01:00
|
|
|
}
|
2024-05-20 14:15:18 +03:30
|
|
|
|
|
|
|
public void setControlsStyles(ControlsConfig controlsStyles) {
|
|
|
|
controlsConfig = controlsStyles;
|
2024-09-29 20:51:02 +02:00
|
|
|
refreshControlsStyles();
|
2024-05-20 14:15:18 +03:30
|
|
|
}
|
2024-10-03 01:07:18 +03:30
|
|
|
}
|