Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Expandable player with unified UI #2907

Merged
merged 43 commits into from
Aug 1, 2020
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
43 commits
Select commit Hold shift + click to select a range
c45514b
All players in one place
avently Dec 29, 2019
fc9b632
Optimizations and fixes of rare situations
avently Dec 31, 2019
bc2dc8d
First block of fixes for review
avently Dec 31, 2019
4519dd0
Second block of fixes for review
avently Jan 3, 2020
e063967
Third block of fixes for review
avently Jan 3, 2020
a2d5314
Fourth block of fixes for review
avently Jan 6, 2020
4c57893
New features and fixes
avently Jan 8, 2020
22bb129
Autoplay enhancement and new button at the top left corner
avently Jan 9, 2020
6fc9131
Changed default autoplay type to "Only on WiFi"
avently Jan 9, 2020
421b821
Fixes of VideoDetailFragment
avently Jan 10, 2020
0c394b1
Another fix of VideoDetailFragment
avently Jan 13, 2020
d1609cb
Enhancements to background playback and media button handling
avently Jan 15, 2020
92ff98d
New logic for handling global orientation
avently Jan 16, 2020
cc438fd
Player's elements positioning is better for tablet and in multiWindow…
avently Jan 17, 2020
26e487c
Hotfix
avently Jan 26, 2020
f334a27
Mini player, ExpandableSurfaceView with ZOOM support, popup
avently Feb 5, 2020
a47e6dd
AppBarLayout scrolling awesomeness, PlayQueue layout touches intercep…
avently Feb 12, 2020
6d7e376
Vertical videos in portrait & fullscreen, UI enhancements for tablets…
avently Feb 24, 2020
5c2ff9b
Better implementation of old code
avently Feb 28, 2020
d87e488
Fix for a ripple effect on a button
avently Feb 29, 2020
398cbe9
Better backstack, better tablet support, switching players confirmati…
avently Mar 10, 2020
a7fbe05
Changes for review
avently Jun 27, 2020
b2164ce
Marked many (too many) variables as final
avently Jul 12, 2020
d2aaa6f
Merged the latest changes
avently Jul 13, 2020
bff2387
Small fixes of issues
avently Jul 13, 2020
d8f7db4
Made checkStyle happy
avently Jul 14, 2020
08412d6
Mini player slide to botom fix, buttons size fix
avently Jul 14, 2020
787b136
NonNull instead of NotNull annotations
avently Jul 14, 2020
d196f8b
New icons
avently Jul 15, 2020
06e70ab
Merged the latest changes
avently Jul 20, 2020
c7ccf9b
AndroidTvUtils -> DeviceUtils
avently Jul 20, 2020
77cd318
Removed unused line
avently Jul 20, 2020
3ecbbea
Better TV support, icons, activity_main refactoring
avently Jul 21, 2020
7aa8a5c
Fixed a situation when background playback could use a video stream i…
avently Jul 21, 2020
91a0257
Fixes for Android API <21
avently Jul 22, 2020
7c79d42
Quality selector for external playback and better fullscreen mode for…
avently Jul 23, 2020
08db1d5
Android TV: ability to select all buttons in the main player, as well…
avently Jul 25, 2020
ec8b000
Merged the latest code
avently Jul 25, 2020
f2e4b69
Another part of UI improvements for Android TV
avently Jul 25, 2020
5293d17
Removed unused files, translations, styles, settings key
avently Jul 25, 2020
68ed738
Renamed files
avently Jul 25, 2020
8562fbd
Merge branch 'dev'
avently Jul 28, 2020
e41218c
Disable starting player service via media button when there is nothin…
avently Jul 28, 2020
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -2,9 +2,9 @@

import android.os.Parcel;

import androidx.test.ext.junit.runners.AndroidJUnit4;
import androidx.test.filters.LargeTest;

import androidx.test.runner.AndroidJUnit4;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.schabi.newpipe.R;
Expand Down
22 changes: 3 additions & 19 deletions app/src/main/AndroidManifest.xml
Original file line number Diff line number Diff line change
Expand Up @@ -43,34 +43,18 @@
</receiver>

<service
android:name=".player.BackgroundPlayer"
android:exported="false">
android:name=".player.MainPlayer"
android:exported="false">
<intent-filter>
<action android:name="android.intent.action.MEDIA_BUTTON" />
</intent-filter>
</service>

<activity
android:name=".player.BackgroundPlayerActivity"
android:label="@string/title_activity_background_player"
android:label="@string/title_activity_play_queue"
android:launchMode="singleTask" />

<activity
android:name=".player.PopupVideoPlayerActivity"
android:label="@string/title_activity_popup_player"
android:launchMode="singleTask" />

<service
android:name=".player.PopupVideoPlayer"
android:exported="false" />

<activity
android:name=".player.MainVideoPlayer"
android:configChanges="keyboard|keyboardHidden|orientation|screenSize"
android:label="@string/app_name"
android:launchMode="singleTask"
android:theme="@style/VideoPlayerTheme" />

<activity
android:name=".settings.SettingsActivity"
android:label="@string/settings" />
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -4,11 +4,14 @@
import android.graphics.Rect;
import android.util.AttributeSet;
import android.view.MotionEvent;
import android.view.View;
import android.view.ViewGroup;
import android.widget.OverScroller;

import androidx.annotation.NonNull;
import androidx.annotation.Nullable;
import androidx.coordinatorlayout.widget.CoordinatorLayout;
import org.schabi.newpipe.R;

import java.lang.reflect.Field;

Expand All @@ -20,6 +23,9 @@ public FlingBehavior(final Context context, final AttributeSet attrs) {
super(context, attrs);
}

private boolean allowScroll = true;
private final Rect globalRect = new Rect();

@Override
public boolean onRequestChildRectangleOnScreen(
@NonNull final CoordinatorLayout coordinatorLayout, @NonNull final AppBarLayout child,
Expand Down Expand Up @@ -55,6 +61,15 @@ public boolean onRequestChildRectangleOnScreen(

public boolean onInterceptTouchEvent(final CoordinatorLayout parent, final AppBarLayout child,
final MotionEvent ev) {
final ViewGroup playQueue = child.findViewById(R.id.playQueuePanel);
if (playQueue != null) {
final boolean visible = playQueue.getGlobalVisibleRect(globalRect);
if (visible && globalRect.contains((int) ev.getRawX(), (int) ev.getRawY())) {
allowScroll = false;
return false;
}
}
allowScroll = true;
switch (ev.getActionMasked()) {
case MotionEvent.ACTION_DOWN:
// remove reference to old nested scrolling child
Expand All @@ -68,6 +83,26 @@ public boolean onInterceptTouchEvent(final CoordinatorLayout parent, final AppBa
return super.onInterceptTouchEvent(parent, child, ev);
}

@Override
public boolean onStartNestedScroll(@NonNull final CoordinatorLayout parent,
@NonNull final AppBarLayout child,
@NonNull final View directTargetChild,
final View target,
final int nestedScrollAxes,
final int type) {
return allowScroll && super.onStartNestedScroll(
parent, child, directTargetChild, target, nestedScrollAxes, type);
}

@Override
public boolean onNestedFling(@NonNull final CoordinatorLayout coordinatorLayout,
@NonNull final AppBarLayout child,
@NonNull final View target, final float velocityX,
final float velocityY, final boolean consumed) {
return allowScroll && super.onNestedFling(
coordinatorLayout, child, target, velocityX, velocityY, consumed);
}

@Nullable
private OverScroller getScrollerField() {
try {
Expand Down
94 changes: 79 additions & 15 deletions app/src/main/java/org/schabi/newpipe/MainActivity.java
Original file line number Diff line number Diff line change
Expand Up @@ -29,6 +29,8 @@
import android.os.Looper;
import android.preference.PreferenceManager;
import android.util.Log;

import android.view.KeyEvent;
import android.view.LayoutInflater;
import android.view.Menu;
import android.view.MenuItem;
Expand All @@ -37,10 +39,10 @@
import android.widget.AdapterView;
import android.widget.ArrayAdapter;
import android.widget.Button;
import android.widget.FrameLayout;
import android.widget.ImageView;
import android.widget.Spinner;
import android.widget.TextView;

import androidx.annotation.NonNull;
import androidx.appcompat.app.ActionBar;
import androidx.appcompat.app.ActionBarDrawerToggle;
Expand All @@ -51,6 +53,7 @@
import androidx.fragment.app.Fragment;
import androidx.fragment.app.FragmentManager;

import com.google.android.material.bottomsheet.BottomSheetBehavior;
import com.google.android.material.navigation.NavigationView;

import org.schabi.newpipe.extractor.NewPipe;
Expand All @@ -61,14 +64,18 @@
import org.schabi.newpipe.fragments.MainFragment;
import org.schabi.newpipe.fragments.detail.VideoDetailFragment;
import org.schabi.newpipe.fragments.list.search.SearchFragment;
import org.schabi.newpipe.player.VideoPlayer;
import org.schabi.newpipe.player.event.OnKeyDownListener;
import org.schabi.newpipe.player.playqueue.PlayQueue;
import org.schabi.newpipe.report.ErrorActivity;
import org.schabi.newpipe.util.AndroidTvUtils;
import org.schabi.newpipe.util.DeviceUtils;
import org.schabi.newpipe.util.Constants;
import org.schabi.newpipe.util.KioskTranslator;
import org.schabi.newpipe.util.Localization;
import org.schabi.newpipe.util.NavigationHelper;
import org.schabi.newpipe.util.PeertubeHelper;
import org.schabi.newpipe.util.PermissionHelper;
import org.schabi.newpipe.util.SerializedCache;
import org.schabi.newpipe.util.ServiceHelper;
import org.schabi.newpipe.util.StateSaver;
import org.schabi.newpipe.util.TLSSocketFactoryCompat;
Expand Down Expand Up @@ -137,7 +144,7 @@ && getSupportFragmentManager().getBackStackEntryCount() == 0) {
ErrorActivity.reportUiError(this, e);
}

if (AndroidTvUtils.isTv(this)) {
if (DeviceUtils.isTv(this)) {
FocusOverlayView.setupFocusObserver(this);
}
}
Expand Down Expand Up @@ -518,25 +525,60 @@ protected void onNewIntent(final Intent intent) {
handleIntent(intent);
}

@Override
public boolean onKeyDown(final int keyCode, final KeyEvent event) {
final Fragment fragment = getSupportFragmentManager()
.findFragmentById(R.id.fragment_player_holder);
if (fragment instanceof OnKeyDownListener
&& !bottomSheetHiddenOrCollapsed()) {
// Provide keyDown event to fragment which then sends this event
// to the main player service
return ((OnKeyDownListener) fragment).onKeyDown(keyCode)
|| super.onKeyDown(keyCode, event);
}
return super.onKeyDown(keyCode, event);
}

@Override
public void onBackPressed() {
if (DEBUG) {
Log.d(TAG, "onBackPressed() called");
}

if (AndroidTvUtils.isTv(this)) {
if (DeviceUtils.isTv(this)) {
View drawerPanel = findViewById(R.id.navigation);
if (drawer.isDrawerOpen(drawerPanel)) {
drawer.closeDrawers();
return;
}
}

Fragment fragment = getSupportFragmentManager().findFragmentById(R.id.fragment_holder);
// If current fragment implements BackPressable (i.e. can/wanna handle back press)
// delegate the back press to it
if (fragment instanceof BackPressable) {
if (((BackPressable) fragment).onBackPressed()) {
// In case bottomSheet is not visible on the screen or collapsed we can assume that the user
// interacts with a fragment inside fragment_holder so all back presses should be
// handled by it
if (bottomSheetHiddenOrCollapsed()) {
final Fragment fragment = getSupportFragmentManager()
.findFragmentById(R.id.fragment_holder);
// If current fragment implements BackPressable (i.e. can/wanna handle back press)
// delegate the back press to it
if (fragment instanceof BackPressable) {
if (((BackPressable) fragment).onBackPressed()) {
return;
}
}

} else {
final Fragment fragmentPlayer = getSupportFragmentManager()
.findFragmentById(R.id.fragment_player_holder);
// If current fragment implements BackPressable (i.e. can/wanna handle back press)
// delegate the back press to it
if (fragmentPlayer instanceof BackPressable) {
if (!((BackPressable) fragmentPlayer).onBackPressed()) {
final FrameLayout bottomSheetLayout =
findViewById(R.id.fragment_player_holder);
BottomSheetBehavior.from(bottomSheetLayout)
.setState(BottomSheetBehavior.STATE_COLLAPSED);
}
return;
}
}
Expand All @@ -563,7 +605,7 @@ public void onRequestPermissionsResult(final int requestCode,
break;
case PermissionHelper.DOWNLOAD_DIALOG_REQUEST_CODE:
Fragment fragment = getSupportFragmentManager()
.findFragmentById(R.id.fragment_holder);
.findFragmentById(R.id.fragment_player_holder);
if (fragment instanceof VideoDetailFragment) {
((VideoDetailFragment) fragment).openDownloadDialog();
}
Expand Down Expand Up @@ -615,10 +657,6 @@ public boolean onCreateOptionsMenu(final Menu menu) {
super.onCreateOptionsMenu(menu);

Fragment fragment = getSupportFragmentManager().findFragmentById(R.id.fragment_holder);
if (!(fragment instanceof VideoDetailFragment)) {
findViewById(R.id.toolbar).findViewById(R.id.toolbar_spinner).setVisibility(View.GONE);
}

if (!(fragment instanceof SearchFragment)) {
findViewById(R.id.toolbar).findViewById(R.id.toolbar_search_container)
.setVisibility(View.GONE);
Expand Down Expand Up @@ -660,6 +698,13 @@ private void initFragments() {
}
StateSaver.clearStateFiles();
if (getIntent() != null && getIntent().hasExtra(Constants.KEY_LINK_TYPE)) {
// When user watch a video inside popup and then tries to open the video in main player
// while the app is closed he will see a blank fragment on place of kiosk.
// Let's open it first
if (getSupportFragmentManager().getBackStackEntryCount() == 0) {
NavigationHelper.openMainFragment(getSupportFragmentManager());
}

handleIntent(getIntent());
} else {
NavigationHelper.gotoMainFragment(getSupportFragmentManager());
Expand Down Expand Up @@ -708,8 +753,14 @@ private void handleIntent(final Intent intent) {
case STREAM:
boolean autoPlay = intent
.getBooleanExtra(VideoDetailFragment.AUTO_PLAY, false);
final String intentCacheKey = intent
.getStringExtra(VideoPlayer.PLAY_QUEUE_KEY);
final PlayQueue playQueue = intentCacheKey != null
? SerializedCache.getInstance()
.take(intentCacheKey, PlayQueue.class)
: null;
NavigationHelper.openVideoDetailFragment(getSupportFragmentManager(),
serviceId, url, title, autoPlay);
serviceId, url, title, autoPlay, playQueue);
break;
case CHANNEL:
NavigationHelper.openChannelFragment(getSupportFragmentManager(),
Expand Down Expand Up @@ -742,4 +793,17 @@ private void handleIntent(final Intent intent) {
ErrorActivity.reportUiError(this, e);
}
}
/*
* Utils
* */

private boolean bottomSheetHiddenOrCollapsed() {
final FrameLayout bottomSheetLayout = findViewById(R.id.fragment_player_holder);
final BottomSheetBehavior<FrameLayout> bottomSheetBehavior =
BottomSheetBehavior.from(bottomSheetLayout);

final int sheetState = bottomSheetBehavior.getState();
return sheetState == BottomSheetBehavior.STATE_HIDDEN
|| sheetState == BottomSheetBehavior.STATE_COLLAPSED;
}
}
13 changes: 9 additions & 4 deletions app/src/main/java/org/schabi/newpipe/RouterActivity.java
Original file line number Diff line number Diff line change
Expand Up @@ -44,7 +44,7 @@
import org.schabi.newpipe.player.playqueue.PlaylistPlayQueue;
import org.schabi.newpipe.player.playqueue.SinglePlayQueue;
import org.schabi.newpipe.report.UserAction;
import org.schabi.newpipe.util.AndroidTvUtils;
import org.schabi.newpipe.util.DeviceUtils;
import org.schabi.newpipe.util.Constants;
import org.schabi.newpipe.util.ExtractorHelper;
import org.schabi.newpipe.util.ListHelper;
Expand Down Expand Up @@ -347,7 +347,7 @@ private void showDialog(final List<AdapterChoiceItem> choices) {

alertDialog.show();

if (AndroidTvUtils.isTv(this)) {
if (DeviceUtils.isTv(this)) {
FocusOverlayView.setupFocusObserver(alertDialog);
}
}
Expand Down Expand Up @@ -701,7 +701,7 @@ public Consumer<Info> getResultHandler(final Choice choice) {
playQueue = new SinglePlayQueue((StreamInfo) info);

if (playerChoice.equals(videoPlayerKey)) {
NavigationHelper.playOnMainPlayer(this, playQueue, true);
openMainPlayer(playQueue, choice);
} else if (playerChoice.equals(backgroundPlayerKey)) {
NavigationHelper.enqueueOnBackgroundPlayer(this, playQueue, true);
} else if (playerChoice.equals(popupPlayerKey)) {
Expand All @@ -716,7 +716,7 @@ public Consumer<Info> getResultHandler(final Choice choice) {
: new PlaylistPlayQueue((PlaylistInfo) info);

if (playerChoice.equals(videoPlayerKey)) {
NavigationHelper.playOnMainPlayer(this, playQueue, true);
openMainPlayer(playQueue, choice);
} else if (playerChoice.equals(backgroundPlayerKey)) {
NavigationHelper.playOnBackgroundPlayer(this, playQueue, true);
} else if (playerChoice.equals(popupPlayerKey)) {
Expand All @@ -726,6 +726,11 @@ public Consumer<Info> getResultHandler(final Choice choice) {
};
}

private void openMainPlayer(final PlayQueue playQueue, final Choice choice) {
NavigationHelper.playOnMainPlayer(this, playQueue, choice.linkType,
choice.url, "", true, true);
}

@Override
public void onDestroy() {
super.onDestroy();
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -13,7 +13,7 @@
import androidx.appcompat.widget.Toolbar;

import org.schabi.newpipe.R;
import org.schabi.newpipe.util.AndroidTvUtils;
import org.schabi.newpipe.util.DeviceUtils;
import org.schabi.newpipe.util.ThemeHelper;
import org.schabi.newpipe.views.FocusOverlayView;

Expand Down Expand Up @@ -57,7 +57,7 @@ public void onGlobalLayout() {
}
});

if (AndroidTvUtils.isTv(this)) {
if (DeviceUtils.isTv(this)) {
FocusOverlayView.setupFocusObserver(this);
}
}
Expand Down
Loading