diff --git a/android/sdk/src/main/java/org/jitsi/meet/sdk/ExternalAPIModule.java b/android/sdk/src/main/java/org/jitsi/meet/sdk/ExternalAPIModule.java index 6aacbccf2..f1b013765 100644 --- a/android/sdk/src/main/java/org/jitsi/meet/sdk/ExternalAPIModule.java +++ b/android/sdk/src/main/java/org/jitsi/meet/sdk/ExternalAPIModule.java @@ -111,6 +111,34 @@ class ExternalAPIModule extends ReactContextBaseJavaModule { return "ExternalAPI"; } + /** + * The internal processing for the conference URL set on + * a {@link JitsiMeetView} instance. + * + * @param eventName the name of the external API event to be processed. + * @param view the {@link JitsiMeetView} instance. + * @param url the "url" attribute value retrieved from the "data" carried by + * the event. + */ + private void maybeSetConferenceUrlOnTheView( + String eventName, JitsiMeetView view, String url) + { + switch(eventName) { + case "CONFERENCE_WILL_JOIN": + view.setCurrentConferenceUrl(url); + break; + + case "CONFERENCE_FAILED": + case "CONFERENCE_WILL_LEAVE": + case "LOAD_CONFIG_ERROR": + // Abandon the conference only if it's for the current URL + if (url != null && url.equals(view.getCurrentConferenceUrl())) { + view.setCurrentConferenceUrl(null); + } + break; + } + } + /** * Dispatches an event that occurred on JavaScript to the view's listener. * @@ -130,6 +158,8 @@ class ExternalAPIModule extends ReactContextBaseJavaModule { return; } + maybeSetConferenceUrlOnTheView(name, view, data.getString("url")); + JitsiMeetViewListener listener = view.getListener(); if (listener == null) { @@ -141,7 +171,17 @@ class ExternalAPIModule extends ReactContextBaseJavaModule { if (method != null) { try { method.invoke(listener, toHashMap(data)); - } catch (IllegalAccessException | InvocationTargetException e) { + } catch (IllegalAccessException e) { + // FIXME There was a multicatch for IllegalAccessException and + // InvocationTargetException, but Android Studio complained + // with: + // "Multi-catch with these reflection exceptions requires + // API level 19 (current min is 16) because they get compiled to + // the common but new super type ReflectiveOperationException. + // As a workaround either create individual catch statements, or + // catch Exception." + throw new RuntimeException(e); + } catch (InvocationTargetException e) { throw new RuntimeException(e); } } diff --git a/android/sdk/src/main/java/org/jitsi/meet/sdk/JitsiMeetActivity.java b/android/sdk/src/main/java/org/jitsi/meet/sdk/JitsiMeetActivity.java index 3af7d9909..a3643c68a 100644 --- a/android/sdk/src/main/java/org/jitsi/meet/sdk/JitsiMeetActivity.java +++ b/android/sdk/src/main/java/org/jitsi/meet/sdk/JitsiMeetActivity.java @@ -260,7 +260,9 @@ public class JitsiMeetActivity extends AppCompatActivity { @Override protected void onUserLeaveHint() { - JitsiMeetView.onUserLeaveHint(); + if (view != null) { + view.onUserLeaveHint(); + } } /** diff --git a/android/sdk/src/main/java/org/jitsi/meet/sdk/JitsiMeetView.java b/android/sdk/src/main/java/org/jitsi/meet/sdk/JitsiMeetView.java index c38cf5fb9..fd2dc558a 100644 --- a/android/sdk/src/main/java/org/jitsi/meet/sdk/JitsiMeetView.java +++ b/android/sdk/src/main/java/org/jitsi/meet/sdk/JitsiMeetView.java @@ -167,7 +167,7 @@ public class JitsiMeetView extends FrameLayout { DefaultHardwareBackBtnHandler defaultBackButtonImpl) { ReactInstanceManager reactInstanceManager = ReactInstanceManagerHolder.getReactInstanceManager(); - + if (reactInstanceManager != null) { reactInstanceManager.onHostResume(activity, defaultBackButtonImpl); } @@ -205,15 +205,13 @@ public class JitsiMeetView extends FrameLayout { } /** - * Activity lifecycle method which should be called from - * {@code Activity.onUserLeaveHint} so we can do the required internal - * processing. + * Stores the current conference URL. Will have a value when the app is in + * a conference. * - * This is currently not mandatory. + * Currently one thread writes and one thread reads, so it should be fine to + * have this field volatile without additional synchronization. */ - public static void onUserLeaveHint() { - ReactInstanceManagerHolder.emitEvent("onUserLeaveHint", null); - } + private volatile String conferenceUrl; /** * The default base {@code URL} used to join a conference when a partial URL @@ -293,6 +291,16 @@ public class JitsiMeetView extends FrameLayout { } } + /** + * Retrieves the current conferences URL. + * + * @return a string with conference URL if the view is currently in + * a conference or {@code null} otherwise. + */ + public String getCurrentConferenceUrl() { + return conferenceUrl; + } + /** * Gets the default base {@code URL} used to join a conference when a * partial URL (e.g. a room name only) is specified to @@ -458,6 +466,33 @@ public class JitsiMeetView extends FrameLayout { loadURLObject(urlObject); } + /** + * Activity lifecycle method which should be called from + * {@code Activity.onUserLeaveHint} so we can do the required internal + * processing. + * + * This is currently not mandatory, but if used will provide automatic + * handling of the picture in picture mode when user minimizes the app. It + * will be probably the most useful in case the app is using the welcome + * page. + */ + public void onUserLeaveHint() { + if (getPictureInPictureEnabled() && conferenceUrl != null) { + PictureInPictureModule pipModule + = ReactInstanceManagerHolder.getNativeModule( + PictureInPictureModule.class); + + if (pipModule != null) { + try { + pipModule.enterPictureInPicture(); + } catch (RuntimeException exc) { + Log.e( + TAG, "onUserLeaveHint: failed to enter PiP mode", exc); + } + } + } + } + /** * Called when the window containing this view gains or loses focus. * @@ -495,6 +530,17 @@ public class JitsiMeetView extends FrameLayout { } } + /** + * Sets the current conference URL. + * + * @param conferenceUrl a string with new conference URL to set if the view + * is entering the conference or {@code null} if the view is no longer in + * the conference. + */ + void setCurrentConferenceUrl(String conferenceUrl) { + this.conferenceUrl = conferenceUrl; + } + /** * Sets the default base {@code URL} used to join a conference when a * partial URL (e.g. a room name only) is specified to diff --git a/android/sdk/src/main/java/org/jitsi/meet/sdk/PictureInPictureModule.java b/android/sdk/src/main/java/org/jitsi/meet/sdk/PictureInPictureModule.java index bf15edcc6..9fedecbe1 100644 --- a/android/sdk/src/main/java/org/jitsi/meet/sdk/PictureInPictureModule.java +++ b/android/sdk/src/main/java/org/jitsi/meet/sdk/PictureInPictureModule.java @@ -22,6 +22,46 @@ public class PictureInPictureModule extends ReactContextBaseJavaModule { super(reactContext); } + /** + * Enters Picture-in-Picture (mode) for the current {@link Activity}. + * Supported on Android API >= 26 (Oreo) only. + * + * @throws IllegalStateException if {@link #isPictureInPictureSupported()} + * returns {@code false} or if {@link #getCurrentActivity()} returns + * {@code null}. + * @throws RuntimeException if + * {@link Activity#enterPictureInPictureMode(PictureInPictureParams)} fails. + * That method can also throw a {@link RuntimeException} in various cases, + * including when the activity is not visible (paused or stopped), if the + * screen is locked or if the user has an activity pinned. + */ + public void enterPictureInPicture() { + if (!isPictureInPictureSupported()) { + throw new IllegalStateException("Picture-in-Picture not supported"); + } + + Activity currentActivity = getCurrentActivity(); + + if (currentActivity == null) { + throw new IllegalStateException("No current Activity!"); + } + + Log.d(TAG, "Entering Picture-in-Picture"); + + PictureInPictureParams.Builder builder + = new PictureInPictureParams.Builder() + .setAspectRatio(new Rational(1, 1)); + + // https://developer.android.com/reference/android/app/Activity.html#enterPictureInPictureMode(android.app.PictureInPictureParams) + // + // The system may disallow entering picture-in-picture in various cases, + // including when the activity is not visible, if the screen is locked + // or if the user has an activity pinned. + if (!currentActivity.enterPictureInPictureMode(builder.build())) { + throw new RuntimeException("Failed to enter Picture-in-Picture"); + } + } + /** * Enters Picture-in-Picture (mode) for the current {@link Activity}. * Supported on Android API >= 26 (Oreo) only. @@ -31,45 +71,12 @@ public class PictureInPictureModule extends ReactContextBaseJavaModule { */ @ReactMethod public void enterPictureInPicture(Promise promise) { - if (isPictureInPictureSupported()) { - Activity currentActivity = getCurrentActivity(); - - if (currentActivity == null) { - promise.reject(new Exception("No current Activity!")); - return; - } - - Log.d(TAG, "Entering Picture-in-Picture"); - - PictureInPictureParams.Builder builder - = new PictureInPictureParams.Builder() - .setAspectRatio(new Rational(1, 1)); - Throwable error; - - // https://developer.android.com/reference/android/app/Activity.html#enterPictureInPictureMode(android.app.PictureInPictureParams) - // - // The system may disallow entering picture-in-picture in various - // cases, including when the activity is not visible, if the screen - // is locked or if the user has an activity pinned. - try { - error - = currentActivity.enterPictureInPictureMode(builder.build()) - ? null - : new Exception("Failed to enter Picture-in-Picture"); - } catch (RuntimeException re) { - error = re; - } - - if (error == null) { - promise.resolve(null); - } else { - promise.reject(error); - } - - return; + try { + enterPictureInPicture(); + promise.resolve(null); + } catch (RuntimeException re) { + promise.reject(re); } - - promise.reject(new Exception("Picture-in-Picture not supported")); } @Override diff --git a/android/sdk/src/main/java/org/jitsi/meet/sdk/ReactInstanceManagerHolder.java b/android/sdk/src/main/java/org/jitsi/meet/sdk/ReactInstanceManagerHolder.java index 2c88be6aa..69645eab2 100644 --- a/android/sdk/src/main/java/org/jitsi/meet/sdk/ReactInstanceManagerHolder.java +++ b/android/sdk/src/main/java/org/jitsi/meet/sdk/ReactInstanceManagerHolder.java @@ -61,7 +61,7 @@ public class ReactInstanceManagerHolder { @Nullable Object data) { ReactInstanceManager reactInstanceManager = ReactInstanceManagerHolder.getReactInstanceManager(); - + if (reactInstanceManager != null) { ReactContext reactContext = reactInstanceManager.getCurrentReactContext(); @@ -77,6 +77,26 @@ public class ReactInstanceManagerHolder { return false; } + /** + * Finds a native React module for given class. + * + * @param nativeModuleClass the native module's class for which an instance + * is to be retrieved from the {@link #reactInstanceManager}. + * @param the module's type. + * @return {@link NativeModule} instance for given interface type or + * {@code null} if no instance for this interface is available, or if + * {@link #reactInstanceManager} has not been initialized yet. + */ + static T getNativeModule( + Class nativeModuleClass) { + ReactContext reactContext + = reactInstanceManager != null + ? reactInstanceManager.getCurrentReactContext() : null; + + return reactContext != null + ? reactContext.getNativeModule(nativeModuleClass) : null; + } + static ReactInstanceManager getReactInstanceManager() { return reactInstanceManager; } diff --git a/react/features/mobile/picture-in-picture/actionTypes.js b/react/features/mobile/picture-in-picture/actionTypes.js index 0c8c811e0..bc15478d4 100644 --- a/react/features/mobile/picture-in-picture/actionTypes.js +++ b/react/features/mobile/picture-in-picture/actionTypes.js @@ -9,16 +9,3 @@ * @public */ export const ENTER_PICTURE_IN_PICTURE = Symbol('ENTER_PICTURE_IN_PICTURE'); - -/** - * The type of redux action to set the {@code EventEmitter} subscriptions - * utilized by the feature picture-in-picture. - * - * { - * type: _SET_EMITTER_SUBSCRIPTIONS, - * emitterSubscriptions: Array|undefined - * } - * - * @protected - */ -export const _SET_EMITTER_SUBSCRIPTIONS = Symbol('_SET_EMITTER_SUBSCRIPTIONS'); diff --git a/react/features/mobile/picture-in-picture/actions.js b/react/features/mobile/picture-in-picture/actions.js index 0a271a2cf..6386ad9c0 100644 --- a/react/features/mobile/picture-in-picture/actions.js +++ b/react/features/mobile/picture-in-picture/actions.js @@ -4,10 +4,7 @@ import { NativeModules } from 'react-native'; import { Platform } from '../../base/react'; -import { - ENTER_PICTURE_IN_PICTURE, - _SET_EMITTER_SUBSCRIPTIONS -} from './actionTypes'; +import { ENTER_PICTURE_IN_PICTURE } from './actionTypes'; /** * Enters (or rather initiates entering) picture-in-picture. @@ -47,22 +44,3 @@ export function enterPictureInPicture() { } }; } - -/** - * Sets the {@code EventEmitter} subscriptions utilized by the feature - * picture-in-picture. - * - * @param {Array} emitterSubscriptions - The {@code EventEmitter} - * subscriptions to be set. - * @protected - * @returns {{ - * type: _SET_EMITTER_SUBSCRIPTIONS, - * emitterSubscriptions: Array - * }} - */ -export function _setEmitterSubscriptions(emitterSubscriptions: ?Array) { - return { - type: _SET_EMITTER_SUBSCRIPTIONS, - emitterSubscriptions - }; -} diff --git a/react/features/mobile/picture-in-picture/index.js b/react/features/mobile/picture-in-picture/index.js index a29aa08e0..803dacd06 100644 --- a/react/features/mobile/picture-in-picture/index.js +++ b/react/features/mobile/picture-in-picture/index.js @@ -1,6 +1,3 @@ export * from './actions'; export * from './actionTypes'; export * from './components'; - -import './middleware'; -import './reducer'; diff --git a/react/features/mobile/picture-in-picture/middleware.js b/react/features/mobile/picture-in-picture/middleware.js deleted file mode 100644 index d10719d74..000000000 --- a/react/features/mobile/picture-in-picture/middleware.js +++ /dev/null @@ -1,70 +0,0 @@ -// @flow - -import { DeviceEventEmitter } from 'react-native'; - -import { APP_WILL_MOUNT, APP_WILL_UNMOUNT } from '../../app'; -import { MiddlewareRegistry } from '../../base/redux'; - -import { enterPictureInPicture, _setEmitterSubscriptions } from './actions'; -import { _SET_EMITTER_SUBSCRIPTIONS } from './actionTypes'; - -/** - * Middleware that handles Picture-in-Picture requests. Currently it enters - * the native PiP mode on Android, when requested. - * - * @param {Store} store - Redux store. - * @returns {Function} - */ -MiddlewareRegistry.register(store => next => action => { - switch (action.type) { - case APP_WILL_MOUNT: - return _appWillMount(store, next, action); - - case APP_WILL_UNMOUNT: - store.dispatch(_setEmitterSubscriptions(undefined)); - break; - - case _SET_EMITTER_SUBSCRIPTIONS: { - // Remove the current/old EventEmitter subscriptions. - const { emitterSubscriptions } = store.getState()['features/pip']; - - if (emitterSubscriptions) { - for (const emitterSubscription of emitterSubscriptions) { - // XXX We may be removing an EventEmitter subscription which is - // in both the old and new Array of EventEmitter subscriptions! - // Thankfully, we don't have such a practical use case at the - // time of this writing. - emitterSubscription.remove(); - } - } - break; - } - } - - return next(action); -}); - -/** - * Notifies the feature pip that the action {@link APP_WILL_MOUNT} is being - * dispatched within a specific redux {@code store}. - * - * @param {Store} store - The redux store in which the specified {@code action} - * is being dispatched. - * @param {Dispatch} next - The redux dispatch function to dispatch the - * specified {@code action} to the specified {@code store}. - * @param {Action} action - The redux action {@code APP_WILL_MOUNT} which is - * being dispatched in the specified {@code store}. - * @private - * @returns {*} The value returned by {@code next(action)}. - */ -function _appWillMount({ dispatch }, next, action) { - dispatch(_setEmitterSubscriptions([ - - // Android's onUserLeaveHint activity lifecycle callback - DeviceEventEmitter.addListener( - 'onUserLeaveHint', - () => dispatch(enterPictureInPicture())) - ])); - - return next(action); -} diff --git a/react/features/mobile/picture-in-picture/reducer.js b/react/features/mobile/picture-in-picture/reducer.js deleted file mode 100644 index e170f35f7..000000000 --- a/react/features/mobile/picture-in-picture/reducer.js +++ /dev/null @@ -1,17 +0,0 @@ -// @flow - -import { ReducerRegistry } from '../../base/redux'; - -import { _SET_EMITTER_SUBSCRIPTIONS } from './actionTypes'; - -ReducerRegistry.register('features/pip', (state = {}, action) => { - switch (action.type) { - case _SET_EMITTER_SUBSCRIPTIONS: - return { - ...state, - emitterSubscriptions: action.emitterSubscriptions - }; - } - - return state; -});