Merge branch 'dev' into LongTapInSubs
| @@ -49,12 +49,13 @@ ext { | |||||||
|     icepickLibVersion = '3.2.0' |     icepickLibVersion = '3.2.0' | ||||||
|     stethoLibVersion = '1.5.0' |     stethoLibVersion = '1.5.0' | ||||||
| } | } | ||||||
|  |  | ||||||
| dependencies { | dependencies { | ||||||
|     androidTestImplementation('com.android.support.test.espresso:espresso-core:2.2.2') { |     androidTestImplementation('com.android.support.test.espresso:espresso-core:2.2.2') { | ||||||
|         exclude module: 'support-annotations' |         exclude module: 'support-annotations' | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     implementation 'com.github.TeamNewPipe:NewPipeExtractor:1eff8c5708' |     implementation 'com.github.TeamNewPipe:NewPipeExtractor:fef71aeccc37' | ||||||
|  |  | ||||||
|     testImplementation 'junit:junit:4.12' |     testImplementation 'junit:junit:4.12' | ||||||
|     testImplementation 'org.mockito:mockito-core:2.8.9' |     testImplementation 'org.mockito:mockito-core:2.8.9' | ||||||
| @@ -93,6 +94,9 @@ dependencies { | |||||||
|     debugImplementation "com.squareup.leakcanary:leakcanary-android:$leakCanaryLibVersion" |     debugImplementation "com.squareup.leakcanary:leakcanary-android:$leakCanaryLibVersion" | ||||||
|     releaseImplementation "com.squareup.leakcanary:leakcanary-android-no-op:$leakCanaryLibVersion" |     releaseImplementation "com.squareup.leakcanary:leakcanary-android-no-op:$leakCanaryLibVersion" | ||||||
|  |  | ||||||
|  |  | ||||||
|     implementation "com.squareup.okhttp3:okhttp:$okHttpLibVersion" |     implementation "com.squareup.okhttp3:okhttp:$okHttpLibVersion" | ||||||
|     debugImplementation "com.facebook.stetho:stetho-okhttp3:$stethoLibVersion" |     debugImplementation "com.facebook.stetho:stetho-okhttp3:$stethoLibVersion" | ||||||
|  |     implementation 'com.android.support.constraint:constraint-layout:1.1.2' | ||||||
|  |     implementation 'com.android.support:cardview-v7:27.1.1' | ||||||
| } | } | ||||||
|   | |||||||
| @@ -76,10 +76,6 @@ | |||||||
|             android:name=".about.AboutActivity" |             android:name=".about.AboutActivity" | ||||||
|             android:label="@string/title_activity_about"/> |             android:label="@string/title_activity_about"/> | ||||||
|  |  | ||||||
|         <activity |  | ||||||
|             android:name=".history.HistoryActivity" |  | ||||||
|             android:label="@string/title_activity_history"/> |  | ||||||
|  |  | ||||||
|         <service android:name=".local.subscription.services.SubscriptionsImportService"/> |         <service android:name=".local.subscription.services.SubscriptionsImportService"/> | ||||||
|         <service android:name=".local.subscription.services.SubscriptionsExportService"/> |         <service android:name=".local.subscription.services.SubscriptionsExportService"/> | ||||||
|  |  | ||||||
| @@ -122,6 +118,7 @@ | |||||||
|         <activity |         <activity | ||||||
|             android:name=".ReCaptchaActivity" |             android:name=".ReCaptchaActivity" | ||||||
|             android:label="@string/reCaptchaActivity"/> |             android:label="@string/reCaptchaActivity"/> | ||||||
|  |         <activity android:name=".download.ExtSDDownloadFailedActivity" /> | ||||||
|  |  | ||||||
|         <provider |         <provider | ||||||
|             android:name="android.support.v4.content.FileProvider" |             android:name="android.support.v4.content.FileProvider" | ||||||
|   | |||||||
| @@ -4,6 +4,7 @@ import android.content.Context; | |||||||
| import android.os.Bundle; | import android.os.Bundle; | ||||||
| import android.support.annotation.NonNull; | import android.support.annotation.NonNull; | ||||||
| import android.support.v4.app.Fragment; | import android.support.v4.app.Fragment; | ||||||
|  | import android.support.v4.app.FragmentManager; | ||||||
| import android.support.v7.app.AppCompatActivity; | import android.support.v7.app.AppCompatActivity; | ||||||
| import android.util.Log; | import android.util.Log; | ||||||
| import android.view.View; | import android.view.View; | ||||||
| @@ -11,7 +12,10 @@ import android.view.View; | |||||||
| import com.nostra13.universalimageloader.core.ImageLoader; | import com.nostra13.universalimageloader.core.ImageLoader; | ||||||
| import com.squareup.leakcanary.RefWatcher; | import com.squareup.leakcanary.RefWatcher; | ||||||
|  |  | ||||||
|  | import org.schabi.newpipe.report.UserAction; | ||||||
|  |  | ||||||
| import icepick.Icepick; | import icepick.Icepick; | ||||||
|  | import icepick.State; | ||||||
|  |  | ||||||
| public abstract class BaseFragment extends Fragment { | public abstract class BaseFragment extends Fragment { | ||||||
|     protected final String TAG = getClass().getSimpleName() + "@" + Integer.toHexString(hashCode()); |     protected final String TAG = getClass().getSimpleName() + "@" + Integer.toHexString(hashCode()); | ||||||
| @@ -20,6 +24,15 @@ public abstract class BaseFragment extends Fragment { | |||||||
|     protected AppCompatActivity activity; |     protected AppCompatActivity activity; | ||||||
|     public static final ImageLoader imageLoader = ImageLoader.getInstance(); |     public static final ImageLoader imageLoader = ImageLoader.getInstance(); | ||||||
|  |  | ||||||
|  |     //These values are used for controlling framgents when they are part of the frontpage | ||||||
|  |     @State | ||||||
|  |     protected boolean useAsFrontPage = false; | ||||||
|  |     protected boolean mIsVisibleToUser = false; | ||||||
|  |  | ||||||
|  |     public void useAsFrontPage(boolean value) { | ||||||
|  |         useAsFrontPage = value; | ||||||
|  |     } | ||||||
|  |  | ||||||
|     /*////////////////////////////////////////////////////////////////////////// |     /*////////////////////////////////////////////////////////////////////////// | ||||||
|     // Fragment's Lifecycle |     // Fragment's Lifecycle | ||||||
|     //////////////////////////////////////////////////////////////////////////*/ |     //////////////////////////////////////////////////////////////////////////*/ | ||||||
| @@ -72,6 +85,12 @@ public abstract class BaseFragment extends Fragment { | |||||||
|         if (refWatcher != null) refWatcher.watch(this); |         if (refWatcher != null) refWatcher.watch(this); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public void setUserVisibleHint(boolean isVisibleToUser) { | ||||||
|  |         super.setUserVisibleHint(isVisibleToUser); | ||||||
|  |         mIsVisibleToUser = isVisibleToUser; | ||||||
|  |     } | ||||||
|  |  | ||||||
|     /*////////////////////////////////////////////////////////////////////////// |     /*////////////////////////////////////////////////////////////////////////// | ||||||
|     // Init |     // Init | ||||||
|     //////////////////////////////////////////////////////////////////////////*/ |     //////////////////////////////////////////////////////////////////////////*/ | ||||||
| @@ -88,8 +107,15 @@ public abstract class BaseFragment extends Fragment { | |||||||
|  |  | ||||||
|     public void setTitle(String title) { |     public void setTitle(String title) { | ||||||
|         if (DEBUG) Log.d(TAG, "setTitle() called with: title = [" + title + "]"); |         if (DEBUG) Log.d(TAG, "setTitle() called with: title = [" + title + "]"); | ||||||
|         if (activity != null && activity.getSupportActionBar() != null) { |         if((!useAsFrontPage || mIsVisibleToUser) | ||||||
|  |             && (activity != null && activity.getSupportActionBar() != null)) { | ||||||
|             activity.getSupportActionBar().setTitle(title); |             activity.getSupportActionBar().setTitle(title); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     protected FragmentManager getFM() { | ||||||
|  |         return getParentFragment() == null | ||||||
|  |                 ? getFragmentManager() | ||||||
|  |                 : getParentFragment().getFragmentManager(); | ||||||
|  |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -24,6 +24,7 @@ import android.content.Intent; | |||||||
| import android.content.SharedPreferences; | import android.content.SharedPreferences; | ||||||
| import android.content.pm.PackageManager; | import android.content.pm.PackageManager; | ||||||
| import android.net.Uri; | import android.net.Uri; | ||||||
|  | import android.os.Build; | ||||||
| import android.os.Bundle; | import android.os.Bundle; | ||||||
| import android.os.Handler; | import android.os.Handler; | ||||||
| import android.os.Looper; | import android.os.Looper; | ||||||
| @@ -43,18 +44,22 @@ import android.view.Menu; | |||||||
| import android.view.MenuInflater; | import android.view.MenuInflater; | ||||||
| import android.view.MenuItem; | import android.view.MenuItem; | ||||||
| import android.view.View; | import android.view.View; | ||||||
|  | import android.view.Window; | ||||||
|  | import android.view.WindowManager; | ||||||
| import android.widget.Button; | import android.widget.Button; | ||||||
| import android.widget.ImageButton; | import android.widget.ImageView; | ||||||
| import android.widget.TextView; | import android.widget.TextView; | ||||||
|  |  | ||||||
| import org.schabi.newpipe.extractor.NewPipe; | import org.schabi.newpipe.extractor.NewPipe; | ||||||
| import org.schabi.newpipe.extractor.StreamingService; | import org.schabi.newpipe.extractor.StreamingService; | ||||||
|  | import org.schabi.newpipe.extractor.exceptions.ExtractionException; | ||||||
| import org.schabi.newpipe.fragments.BackPressable; | import org.schabi.newpipe.fragments.BackPressable; | ||||||
| import org.schabi.newpipe.fragments.MainFragment; | import org.schabi.newpipe.fragments.MainFragment; | ||||||
| import org.schabi.newpipe.fragments.detail.VideoDetailFragment; | import org.schabi.newpipe.fragments.detail.VideoDetailFragment; | ||||||
| import org.schabi.newpipe.fragments.list.search.SearchFragment; | import org.schabi.newpipe.fragments.list.search.SearchFragment; | ||||||
| import org.schabi.newpipe.report.ErrorActivity; | import org.schabi.newpipe.report.ErrorActivity; | ||||||
| import org.schabi.newpipe.util.Constants; | import org.schabi.newpipe.util.Constants; | ||||||
|  | import org.schabi.newpipe.util.KioskTranslator; | ||||||
| import org.schabi.newpipe.util.NavigationHelper; | import org.schabi.newpipe.util.NavigationHelper; | ||||||
| import org.schabi.newpipe.util.PermissionHelper; | import org.schabi.newpipe.util.PermissionHelper; | ||||||
| import org.schabi.newpipe.util.ServiceHelper; | import org.schabi.newpipe.util.ServiceHelper; | ||||||
| @@ -72,6 +77,19 @@ public class MainActivity extends AppCompatActivity { | |||||||
|     private NavigationView drawerItems = null; |     private NavigationView drawerItems = null; | ||||||
|     private TextView headerServiceView = null; |     private TextView headerServiceView = null; | ||||||
|  |  | ||||||
|  |     private boolean servicesShown = false; | ||||||
|  |     private ImageView serviceArrow; | ||||||
|  |  | ||||||
|  |     private static final int ITEM_ID_SUBSCRIPTIONS = - 1; | ||||||
|  |     private static final int ITEM_ID_FEED = - 2; | ||||||
|  |     private static final int ITEM_ID_BOOKMARKS = - 3; | ||||||
|  |     private static final int ITEM_ID_DOWNLOADS = - 4; | ||||||
|  |     private static final int ITEM_ID_HISTORY = - 5; | ||||||
|  |     private static final int ITEM_ID_SETTINGS = 0; | ||||||
|  |     private static final int ITEM_ID_ABOUT = 1; | ||||||
|  |  | ||||||
|  |     private static final int ORDER = 0; | ||||||
|  |  | ||||||
|     /*////////////////////////////////////////////////////////////////////////// |     /*////////////////////////////////////////////////////////////////////////// | ||||||
|     // Activity's LifeCycle |     // Activity's LifeCycle | ||||||
|     //////////////////////////////////////////////////////////////////////////*/ |     //////////////////////////////////////////////////////////////////////////*/ | ||||||
| @@ -85,42 +103,66 @@ public class MainActivity extends AppCompatActivity { | |||||||
|         super.onCreate(savedInstanceState); |         super.onCreate(savedInstanceState); | ||||||
|         setContentView(R.layout.activity_main); |         setContentView(R.layout.activity_main); | ||||||
|  |  | ||||||
|  |         if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) { | ||||||
|  |             Window w = getWindow(); | ||||||
|  |             w.setFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS, WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS); | ||||||
|  |         } | ||||||
|  |  | ||||||
|         if (getSupportFragmentManager() != null && getSupportFragmentManager().getBackStackEntryCount() == 0) { |         if (getSupportFragmentManager() != null && getSupportFragmentManager().getBackStackEntryCount() == 0) { | ||||||
|             initFragments(); |             initFragments(); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         setSupportActionBar(findViewById(R.id.toolbar)); |         setSupportActionBar(findViewById(R.id.toolbar)); | ||||||
|  |         try { | ||||||
|             setupDrawer(); |             setupDrawer(); | ||||||
|  |         } catch (Exception e) { | ||||||
|  |             ErrorActivity.reportUiError(this, e); | ||||||
|  |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private void setupDrawer() { |     private void setupDrawer() throws Exception { | ||||||
|         final Toolbar toolbar = findViewById(R.id.toolbar); |         final Toolbar toolbar = findViewById(R.id.toolbar); | ||||||
|         drawer = findViewById(R.id.drawer_layout); |         drawer = findViewById(R.id.drawer_layout); | ||||||
|         drawerItems = findViewById(R.id.navigation); |         drawerItems = findViewById(R.id.navigation); | ||||||
|  |  | ||||||
|         for(StreamingService s : NewPipe.getServices()) { |         //Tabs | ||||||
|             final String title = s.getServiceInfo().getName() + |         int currentServiceId = ServiceHelper.getSelectedServiceId(this); | ||||||
|                     (ServiceHelper.isBeta(s) ? " (beta)" : ""); |         StreamingService service = NewPipe.getService(currentServiceId); | ||||||
|             final MenuItem item = drawerItems.getMenu() |  | ||||||
|                     .add(R.id.menu_services_group, s.getServiceId(), 0, title); |         int kioskId = 0; | ||||||
|             item.setIcon(ServiceHelper.getIcon(s.getServiceId())); |  | ||||||
|  |         for (final String ks : service.getKioskList().getAvailableKiosks()) { | ||||||
|  |             drawerItems.getMenu() | ||||||
|  |                     .add(R.id.menu_tabs_group, kioskId, 0, KioskTranslator.getTranslatedKioskName(ks, this)) | ||||||
|  |                     .setIcon(KioskTranslator.getKioskIcons(ks, this)); | ||||||
|  |             kioskId ++; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         drawerItems.getMenu().getItem(ServiceHelper.getSelectedServiceId(this)).setChecked(true); |         drawerItems.getMenu() | ||||||
|  |                 .add(R.id.menu_tabs_group, ITEM_ID_SUBSCRIPTIONS, ORDER, R.string.tab_subscriptions) | ||||||
|  |                 .setIcon(ThemeHelper.resolveResourceIdFromAttr(this, R.attr.ic_channel)); | ||||||
|  |         drawerItems.getMenu() | ||||||
|  |                 .add(R.id.menu_tabs_group, ITEM_ID_FEED, ORDER, R.string.fragment_whats_new) | ||||||
|  |                 .setIcon(ThemeHelper.resolveResourceIdFromAttr(this, R.attr.rss)); | ||||||
|  |         drawerItems.getMenu() | ||||||
|  |                 .add(R.id.menu_tabs_group, ITEM_ID_BOOKMARKS, ORDER, R.string.tab_bookmarks) | ||||||
|  |                 .setIcon(ThemeHelper.resolveResourceIdFromAttr(this, R.attr.ic_bookmark)); | ||||||
|  |         drawerItems.getMenu() | ||||||
|  |                 .add(R.id.menu_tabs_group, ITEM_ID_DOWNLOADS, ORDER, R.string.downloads) | ||||||
|  |                 .setIcon(ThemeHelper.resolveResourceIdFromAttr(this, R.attr.download)); | ||||||
|  |         drawerItems.getMenu() | ||||||
|  |                 .add(R.id.menu_tabs_group, ITEM_ID_HISTORY, ORDER, R.string.action_history) | ||||||
|  |                 .setIcon(ThemeHelper.resolveResourceIdFromAttr(this, R.attr.history)); | ||||||
|  |  | ||||||
|         toggle = new ActionBarDrawerToggle(this, drawer, toolbar, |         //Settings and About | ||||||
|                 R.string.drawer_open, R.string.drawer_close) { |         drawerItems.getMenu() | ||||||
|             @Override |                 .add(R.id.menu_options_about_group, ITEM_ID_SETTINGS, ORDER, R.string.settings) | ||||||
|             public void onDrawerClosed(View view) { super.onDrawerClosed(view); } |                 .setIcon(ThemeHelper.resolveResourceIdFromAttr(this, R.attr.settings)); | ||||||
|  |         drawerItems.getMenu() | ||||||
|  |                 .add(R.id.menu_options_about_group, ITEM_ID_ABOUT, ORDER, R.string.tab_about) | ||||||
|  |                 .setIcon(ThemeHelper.resolveResourceIdFromAttr(this, R.attr.info)); | ||||||
|  |  | ||||||
|             @Override |         toggle = new ActionBarDrawerToggle(this, drawer, toolbar, R.string.drawer_open, R.string.drawer_close); | ||||||
|             public void onDrawerOpened(View drawerView) { super.onDrawerOpened(drawerView); } |  | ||||||
|  |  | ||||||
|             @Override |  | ||||||
|             public void onDrawerSlide(View drawerView, float slideOffset) { |  | ||||||
|                 super.onDrawerSlide(drawerView, 0); |  | ||||||
|             } |  | ||||||
|         }; |  | ||||||
|         toggle.syncState(); |         toggle.syncState(); | ||||||
|         drawer.addDrawerListener(toggle); |         drawer.addDrawerListener(toggle); | ||||||
|         drawer.addDrawerListener(new DrawerLayout.SimpleDrawerListener() { |         drawer.addDrawerListener(new DrawerLayout.SimpleDrawerListener() { | ||||||
| @@ -133,51 +175,179 @@ public class MainActivity extends AppCompatActivity { | |||||||
|  |  | ||||||
|             @Override |             @Override | ||||||
|             public void onDrawerClosed(View drawerView) { |             public void onDrawerClosed(View drawerView) { | ||||||
|  |                 if(servicesShown) { | ||||||
|  |                     toggleServices(); | ||||||
|  |                 } | ||||||
|                 if (lastService != ServiceHelper.getSelectedServiceId(MainActivity.this)) { |                 if (lastService != ServiceHelper.getSelectedServiceId(MainActivity.this)) { | ||||||
|                     new Handler(Looper.getMainLooper()).post(MainActivity.this::recreate); |                     new Handler(Looper.getMainLooper()).post(MainActivity.this::recreate); | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|         }); |         }); | ||||||
|  |  | ||||||
|         drawerItems.setNavigationItemSelectedListener(this::changeService); |         drawerItems.setNavigationItemSelectedListener(this::drawerItemSelected); | ||||||
|  |  | ||||||
|         setupDrawerFooter(); |  | ||||||
|         setupDrawerHeader(); |         setupDrawerHeader(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     private boolean drawerItemSelected(MenuItem item) { | ||||||
|     private boolean changeService(MenuItem item) { |         switch (item.getGroupId()) { | ||||||
|         if (item.getGroupId() != R.id.menu_services_group) |             case R.id.menu_services_group: | ||||||
|  |                 changeService(item); | ||||||
|  |                 break; | ||||||
|  |             case R.id.menu_tabs_group: | ||||||
|  |                 try { | ||||||
|  |                     tabSelected(item); | ||||||
|  |                 } catch (Exception e) { | ||||||
|  |                     ErrorActivity.reportUiError(this, e); | ||||||
|  |                 } | ||||||
|  |                 break; | ||||||
|  |             case R.id.menu_options_about_group: | ||||||
|  |                 optionsAboutSelected(item); | ||||||
|  |                 break; | ||||||
|  |             default: | ||||||
|                 return false; |                 return false; | ||||||
|         drawerItems.getMenu().getItem(ServiceHelper.getSelectedServiceId(this)).setChecked(false); |         } | ||||||
|         ServiceHelper.setSelectedServiceId(this, item.getItemId()); |  | ||||||
|         drawerItems.getMenu().getItem(ServiceHelper.getSelectedServiceId(this)).setChecked(true); |  | ||||||
|         drawer.closeDrawers(); |         drawer.closeDrawers(); | ||||||
|         return true; |         return true; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private void setupDrawerFooter() { |     private  void changeService(MenuItem item) { | ||||||
|         ImageButton settings = findViewById(R.id.drawer_settings); |         drawerItems.getMenu().getItem(ServiceHelper.getSelectedServiceId(this)).setChecked(false); | ||||||
|         ImageButton downloads = findViewById(R.id.drawer_downloads); |         ServiceHelper.setSelectedServiceId(this, item.getItemId()); | ||||||
|         ImageButton history = findViewById(R.id.drawer_history); |         drawerItems.getMenu().getItem(ServiceHelper.getSelectedServiceId(this)).setChecked(true); | ||||||
|  |     } | ||||||
|  |  | ||||||
|         settings.setOnClickListener(view -> NavigationHelper.openSettings(this)); |     private void tabSelected(MenuItem item) throws ExtractionException { | ||||||
|         downloads.setOnClickListener(view ->NavigationHelper.openDownloads(this)); |         switch(item.getItemId()) { | ||||||
|         history.setOnClickListener(view -> |             case ITEM_ID_SUBSCRIPTIONS: | ||||||
|                 NavigationHelper.openStatisticFragment(getSupportFragmentManager())); |                 NavigationHelper.openSubscriptionFragment(getSupportFragmentManager()); | ||||||
|  |                 break; | ||||||
|  |             case ITEM_ID_FEED: | ||||||
|  |                 NavigationHelper.openWhatsNewFragment(getSupportFragmentManager()); | ||||||
|  |                 break; | ||||||
|  |             case ITEM_ID_BOOKMARKS: | ||||||
|  |                 NavigationHelper.openBookmarksFragment(getSupportFragmentManager()); | ||||||
|  |                 break; | ||||||
|  |             case ITEM_ID_DOWNLOADS: | ||||||
|  |                 NavigationHelper.openDownloads(this); | ||||||
|  |                 break; | ||||||
|  |             case ITEM_ID_HISTORY: | ||||||
|  |                 NavigationHelper.openStatisticFragment(getSupportFragmentManager()); | ||||||
|  |                 break; | ||||||
|  |             default: | ||||||
|  |                 int currentServiceId = ServiceHelper.getSelectedServiceId(this); | ||||||
|  |                 StreamingService service = NewPipe.getService(currentServiceId); | ||||||
|  |                 String serviceName = ""; | ||||||
|  |  | ||||||
|  |                 int kioskId = 0; | ||||||
|  |                 for (final String ks : service.getKioskList().getAvailableKiosks()) { | ||||||
|  |                     if(kioskId == item.getItemId()) { | ||||||
|  |                         serviceName = ks; | ||||||
|  |                     } | ||||||
|  |                     kioskId ++; | ||||||
|  |                 } | ||||||
|  |  | ||||||
|  |                 NavigationHelper.openKioskFragment(getSupportFragmentManager(), currentServiceId, serviceName); | ||||||
|  |                 break; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private void optionsAboutSelected(MenuItem item) { | ||||||
|  |         switch(item.getItemId()) { | ||||||
|  |             case ITEM_ID_SETTINGS: | ||||||
|  |                 NavigationHelper.openSettings(this); | ||||||
|  |                 break; | ||||||
|  |             case ITEM_ID_ABOUT: | ||||||
|  |                 NavigationHelper.openAbout(this); | ||||||
|  |                 break; | ||||||
|  |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private void setupDrawerHeader() { |     private void setupDrawerHeader() { | ||||||
|         headerServiceView = findViewById(R.id.drawer_header_service_view); |         NavigationView navigationView = findViewById(R.id.navigation); | ||||||
|         Button action = findViewById(R.id.drawer_header_action_button); |         View hView =  navigationView.getHeaderView(0); | ||||||
|  |  | ||||||
|  |         serviceArrow = hView.findViewById(R.id.drawer_arrow); | ||||||
|  |         headerServiceView = hView.findViewById(R.id.drawer_header_service_view); | ||||||
|  |         Button action = hView.findViewById(R.id.drawer_header_action_button); | ||||||
|         action.setOnClickListener(view -> { |         action.setOnClickListener(view -> { | ||||||
|             Intent intent = new Intent(Intent.ACTION_VIEW); |             toggleServices(); | ||||||
|             intent.setData(Uri.parse("https://newpipe.schabi.org/blog/")); |  | ||||||
|             startActivity(intent); |  | ||||||
|             drawer.closeDrawers(); |  | ||||||
|         }); |         }); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     private void toggleServices() { | ||||||
|  |         servicesShown = !servicesShown; | ||||||
|  |  | ||||||
|  |         drawerItems.getMenu().removeGroup(R.id.menu_services_group); | ||||||
|  |         drawerItems.getMenu().removeGroup(R.id.menu_tabs_group); | ||||||
|  |         drawerItems.getMenu().removeGroup(R.id.menu_options_about_group); | ||||||
|  |  | ||||||
|  |         if(servicesShown) { | ||||||
|  |             showServices(); | ||||||
|  |         } else { | ||||||
|  |             try { | ||||||
|  |                 showTabs(); | ||||||
|  |             } catch (Exception e) { | ||||||
|  |                 ErrorActivity.reportUiError(this, e); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private void showServices() { | ||||||
|  |         serviceArrow.setImageResource(R.drawable.ic_arrow_up_white); | ||||||
|  |  | ||||||
|  |         for(StreamingService s : NewPipe.getServices()) { | ||||||
|  |             final String title = s.getServiceInfo().getName() + | ||||||
|  |                     (ServiceHelper.isBeta(s) ? " (beta)" : ""); | ||||||
|  |  | ||||||
|  |             drawerItems.getMenu() | ||||||
|  |                     .add(R.id.menu_services_group, s.getServiceId(), ORDER, title) | ||||||
|  |                     .setIcon(ServiceHelper.getIcon(s.getServiceId())); | ||||||
|  |         } | ||||||
|  |         drawerItems.getMenu().getItem(ServiceHelper.getSelectedServiceId(this)).setChecked(true); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private void showTabs() throws ExtractionException { | ||||||
|  |         serviceArrow.setImageResource(R.drawable.ic_arrow_down_white); | ||||||
|  |  | ||||||
|  |         //Tabs | ||||||
|  |         int currentServiceId = ServiceHelper.getSelectedServiceId(this); | ||||||
|  |         StreamingService service = NewPipe.getService(currentServiceId); | ||||||
|  |  | ||||||
|  |         int kioskId = 0; | ||||||
|  |  | ||||||
|  |         for (final String ks : service.getKioskList().getAvailableKiosks()) { | ||||||
|  |             drawerItems.getMenu() | ||||||
|  |                     .add(R.id.menu_tabs_group, kioskId, ORDER, KioskTranslator.getTranslatedKioskName(ks, this)) | ||||||
|  |                     .setIcon(KioskTranslator.getKioskIcons(ks, this)); | ||||||
|  |             kioskId ++; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         drawerItems.getMenu() | ||||||
|  |                 .add(R.id.menu_tabs_group, ITEM_ID_SUBSCRIPTIONS, ORDER, R.string.tab_subscriptions) | ||||||
|  |                 .setIcon(ThemeHelper.resolveResourceIdFromAttr(this, R.attr.ic_channel)); | ||||||
|  |         drawerItems.getMenu() | ||||||
|  |                 .add(R.id.menu_tabs_group, ITEM_ID_FEED, ORDER, R.string.fragment_whats_new) | ||||||
|  |                 .setIcon(ThemeHelper.resolveResourceIdFromAttr(this, R.attr.rss)); | ||||||
|  |         drawerItems.getMenu() | ||||||
|  |                 .add(R.id.menu_tabs_group, ITEM_ID_BOOKMARKS, ORDER, R.string.tab_bookmarks) | ||||||
|  |                 .setIcon(ThemeHelper.resolveResourceIdFromAttr(this, R.attr.ic_bookmark)); | ||||||
|  |         drawerItems.getMenu() | ||||||
|  |                 .add(R.id.menu_tabs_group, ITEM_ID_DOWNLOADS, ORDER, R.string.downloads) | ||||||
|  |                 .setIcon(ThemeHelper.resolveResourceIdFromAttr(this, R.attr.download)); | ||||||
|  |         drawerItems.getMenu() | ||||||
|  |                 .add(R.id.menu_tabs_group, ITEM_ID_HISTORY, ORDER, R.string.action_history) | ||||||
|  |                 .setIcon(ThemeHelper.resolveResourceIdFromAttr(this, R.attr.history)); | ||||||
|  |  | ||||||
|  |         //Settings and About | ||||||
|  |         drawerItems.getMenu() | ||||||
|  |                 .add(R.id.menu_options_about_group, ITEM_ID_SETTINGS, ORDER, R.string.settings) | ||||||
|  |                 .setIcon(ThemeHelper.resolveResourceIdFromAttr(this, R.attr.settings)); | ||||||
|  |         drawerItems.getMenu() | ||||||
|  |                 .add(R.id.menu_options_about_group, ITEM_ID_ABOUT, ORDER, R.string.tab_about) | ||||||
|  |                 .setIcon(ThemeHelper.resolveResourceIdFromAttr(this, R.attr.info)); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     protected void onDestroy() { |     protected void onDestroy() { | ||||||
|         super.onDestroy(); |         super.onDestroy(); | ||||||
| @@ -345,9 +515,6 @@ public class MainActivity extends AppCompatActivity { | |||||||
|             case R.id.action_history: |             case R.id.action_history: | ||||||
|                     NavigationHelper.openStatisticFragment(getSupportFragmentManager()); |                     NavigationHelper.openStatisticFragment(getSupportFragmentManager()); | ||||||
|                     return true; |                     return true; | ||||||
|             case R.id.action_about: |  | ||||||
|                 NavigationHelper.openAbout(this); |  | ||||||
|                 return true; |  | ||||||
|             case R.id.action_settings: |             case R.id.action_settings: | ||||||
|                     NavigationHelper.openSettings(this); |                     NavigationHelper.openSettings(this); | ||||||
|                     return true; |                     return true; | ||||||
|   | |||||||
| @@ -0,0 +1,38 @@ | |||||||
|  | package org.schabi.newpipe.download; | ||||||
|  |  | ||||||
|  | import android.app.AlertDialog; | ||||||
|  | import android.content.DialogInterface; | ||||||
|  | import android.os.Bundle; | ||||||
|  | import android.support.annotation.Nullable; | ||||||
|  | import android.support.v7.app.AppCompatActivity; | ||||||
|  |  | ||||||
|  | import org.schabi.newpipe.R; | ||||||
|  | import org.schabi.newpipe.settings.NewPipeSettings; | ||||||
|  | import org.schabi.newpipe.util.ServiceHelper; | ||||||
|  | import org.schabi.newpipe.util.ThemeHelper; | ||||||
|  |  | ||||||
|  | public class ExtSDDownloadFailedActivity extends AppCompatActivity { | ||||||
|  |     @Override | ||||||
|  |     protected void onCreate(@Nullable Bundle savedInstanceState) { | ||||||
|  |         super.onCreate(savedInstanceState); | ||||||
|  |         ThemeHelper.setTheme(this, ServiceHelper.getSelectedServiceId(this)); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     protected void onStart() { | ||||||
|  |         super.onStart(); | ||||||
|  |         new AlertDialog.Builder(this) | ||||||
|  |                 .setTitle(R.string.download_to_sdcard_error_title) | ||||||
|  |                 .setMessage(R.string.download_to_sdcard_error_message) | ||||||
|  |                 .setPositiveButton(R.string.yes, (DialogInterface dialogInterface, int i) -> { | ||||||
|  |                     NewPipeSettings.resetDownloadFolders(this); | ||||||
|  |                     finish(); | ||||||
|  |                 }) | ||||||
|  |                 .setNegativeButton(R.string.cancel, (DialogInterface dialogInterface, int i) -> { | ||||||
|  |                     dialogInterface.dismiss(); | ||||||
|  |                     finish(); | ||||||
|  |                 }) | ||||||
|  |                 .create() | ||||||
|  |                 .show(); | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -51,9 +51,6 @@ public abstract class BaseStateFragment<I> extends BaseFragment implements ViewC | |||||||
|     protected Button errorButtonRetry; |     protected Button errorButtonRetry; | ||||||
|     protected TextView errorTextView; |     protected TextView errorTextView; | ||||||
|  |  | ||||||
|     @State |  | ||||||
|     protected boolean useAsFrontPage = false; |  | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public void onViewCreated(View rootView, Bundle savedInstanceState) { |     public void onViewCreated(View rootView, Bundle savedInstanceState) { | ||||||
|         super.onViewCreated(rootView, savedInstanceState); |         super.onViewCreated(rootView, savedInstanceState); | ||||||
| @@ -66,9 +63,6 @@ public abstract class BaseStateFragment<I> extends BaseFragment implements ViewC | |||||||
|         wasLoading.set(isLoading.get()); |         wasLoading.set(isLoading.get()); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public void useAsFrontPage(boolean value) { |  | ||||||
|         useAsFrontPage = value; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /*////////////////////////////////////////////////////////////////////////// |     /*////////////////////////////////////////////////////////////////////////// | ||||||
|     // Init |     // Init | ||||||
| @@ -93,12 +87,7 @@ public abstract class BaseStateFragment<I> extends BaseFragment implements ViewC | |||||||
|         RxView.clicks(errorButtonRetry) |         RxView.clicks(errorButtonRetry) | ||||||
|                 .debounce(300, TimeUnit.MILLISECONDS) |                 .debounce(300, TimeUnit.MILLISECONDS) | ||||||
|                 .observeOn(AndroidSchedulers.mainThread()) |                 .observeOn(AndroidSchedulers.mainThread()) | ||||||
|                 .subscribe(new Consumer<Object>() { |                 .subscribe(o -> onRetryButtonClicked()); | ||||||
|                     @Override |  | ||||||
|                     public void accept(Object o) throws Exception { |  | ||||||
|                         onRetryButtonClicked(); |  | ||||||
|                     } |  | ||||||
|                 }); |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     protected void onRetryButtonClicked() { |     protected void onRetryButtonClicked() { | ||||||
|   | |||||||
| @@ -14,24 +14,16 @@ public class BlankFragment extends BaseFragment { | |||||||
|     @Nullable |     @Nullable | ||||||
|     @Override |     @Override | ||||||
|     public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container, Bundle savedInstanceState) { |     public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container, Bundle savedInstanceState) { | ||||||
|         if(activity != null && activity.getSupportActionBar() != null) { |         setTitle("NewPipe"); | ||||||
|             activity.getSupportActionBar() |  | ||||||
|                     .setTitle("NewPipe"); |  | ||||||
|         } |  | ||||||
|         return inflater.inflate(R.layout.fragment_blank, container, false); |         return inflater.inflate(R.layout.fragment_blank, container, false); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public void setUserVisibleHint(boolean isVisibleToUser) { |     public void setUserVisibleHint(boolean isVisibleToUser) { | ||||||
|         super.setUserVisibleHint(isVisibleToUser); |         super.setUserVisibleHint(isVisibleToUser); | ||||||
|         if(isVisibleToUser) { |         setTitle("NewPipe"); | ||||||
|             if(activity != null && activity.getSupportActionBar() != null) { |  | ||||||
|                 activity.getSupportActionBar() |  | ||||||
|                         .setTitle("NewPipe"); |  | ||||||
|             } |  | ||||||
|         // leave this inline. Will make it harder for copy cats. |         // leave this inline. Will make it harder for copy cats. | ||||||
|         // If you are a Copy cat FUCK YOU. |         // If you are a Copy cat FUCK YOU. | ||||||
|         // I WILL FIND YOU, AND I WILL ... |         // I WILL FIND YOU, AND I WILL ... | ||||||
|     } |     } | ||||||
| } | } | ||||||
| } |  | ||||||
|   | |||||||
| @@ -1,5 +1,6 @@ | |||||||
| package org.schabi.newpipe.fragments; | package org.schabi.newpipe.fragments; | ||||||
|  |  | ||||||
|  | import android.content.Context; | ||||||
| import android.content.SharedPreferences; | import android.content.SharedPreferences; | ||||||
| import android.os.Bundle; | import android.os.Bundle; | ||||||
| import android.support.annotation.NonNull; | import android.support.annotation.NonNull; | ||||||
| @@ -17,20 +18,16 @@ import android.view.LayoutInflater; | |||||||
| import android.view.Menu; | import android.view.Menu; | ||||||
| import android.view.MenuInflater; | import android.view.MenuInflater; | ||||||
| import android.view.MenuItem; | import android.view.MenuItem; | ||||||
| import android.view.SubMenu; |  | ||||||
| import android.view.View; | import android.view.View; | ||||||
| import android.view.ViewGroup; | import android.view.ViewGroup; | ||||||
|  |  | ||||||
| import org.schabi.newpipe.BaseFragment; | import org.schabi.newpipe.BaseFragment; | ||||||
| import org.schabi.newpipe.R; | import org.schabi.newpipe.R; | ||||||
| import org.schabi.newpipe.extractor.NewPipe; |  | ||||||
| import org.schabi.newpipe.extractor.ServiceList; |  | ||||||
| import org.schabi.newpipe.extractor.StreamingService; |  | ||||||
| import org.schabi.newpipe.extractor.kiosk.KioskList; |  | ||||||
| import org.schabi.newpipe.fragments.list.channel.ChannelFragment; | import org.schabi.newpipe.fragments.list.channel.ChannelFragment; | ||||||
| import org.schabi.newpipe.local.feed.FeedFragment; |  | ||||||
| import org.schabi.newpipe.fragments.list.kiosk.KioskFragment; | import org.schabi.newpipe.fragments.list.kiosk.KioskFragment; | ||||||
| import org.schabi.newpipe.local.bookmark.BookmarkFragment; | import org.schabi.newpipe.local.bookmark.BookmarkFragment; | ||||||
|  | import org.schabi.newpipe.local.feed.FeedFragment; | ||||||
|  | import org.schabi.newpipe.local.history.StatisticsPlaylistFragment; | ||||||
| import org.schabi.newpipe.local.subscription.SubscriptionFragment; | import org.schabi.newpipe.local.subscription.SubscriptionFragment; | ||||||
| import org.schabi.newpipe.report.ErrorActivity; | import org.schabi.newpipe.report.ErrorActivity; | ||||||
| import org.schabi.newpipe.report.UserAction; | import org.schabi.newpipe.report.UserAction; | ||||||
| @@ -39,20 +36,29 @@ import org.schabi.newpipe.util.NavigationHelper; | |||||||
| import org.schabi.newpipe.util.ServiceHelper; | import org.schabi.newpipe.util.ServiceHelper; | ||||||
| import org.schabi.newpipe.util.ThemeHelper; | import org.schabi.newpipe.util.ThemeHelper; | ||||||
|  |  | ||||||
|  | import java.util.ArrayList; | ||||||
|  | import java.util.Collections; | ||||||
|  | import java.util.List; | ||||||
|  |  | ||||||
| public class MainFragment extends BaseFragment implements TabLayout.OnTabSelectedListener { | public class MainFragment extends BaseFragment implements TabLayout.OnTabSelectedListener { | ||||||
|  |  | ||||||
|     public int currentServiceId = -1; |     public int currentServiceId = -1; | ||||||
|     private ViewPager viewPager; |     private ViewPager viewPager; | ||||||
|  |     private List<String> tabs = new ArrayList<>(); | ||||||
|  |     static PagerAdapter adapter; | ||||||
|  |     TabLayout tabLayout; | ||||||
|  |     private SharedPreferences prefs; | ||||||
|  |     private Bundle savedInstanceStateBundle; | ||||||
|  |  | ||||||
|     /*////////////////////////////////////////////////////////////////////////// |     private static final String TAB_NUMBER_BLANK = "0"; | ||||||
|     // Constants |     private static final String TAB_NUMBER_KIOSK = "1"; | ||||||
|     //////////////////////////////////////////////////////////////////////////*/ |     private static final String TAB_NUMBER_SUBSCIRPTIONS = "2"; | ||||||
|  |     private static final String TAB_NUMBER_FEED = "3"; | ||||||
|  |     private static final String TAB_NUMBER_BOOKMARKS = "4"; | ||||||
|  |     private static final String TAB_NUMBER_HISTORY = "5"; | ||||||
|  |     private static final String TAB_NUMBER_CHANNEL = "6"; | ||||||
|  |  | ||||||
|     private static final int FALLBACK_SERVICE_ID = ServiceList.YouTube.getServiceId(); |     SharedPreferences.OnSharedPreferenceChangeListener listener; | ||||||
|     private static final String FALLBACK_CHANNEL_URL = "https://www.youtube.com/channel/UC-9-kyTW8ZkZNDHQJ6FgpwQ"; |  | ||||||
|     private static final String FALLBACK_CHANNEL_NAME = "Music"; |  | ||||||
|     private static final String FALLBACK_KIOSK_ID = "Trending"; |  | ||||||
|     private static final int KIOSK_MENU_OFFSET = 2000; |  | ||||||
|  |  | ||||||
|     /*////////////////////////////////////////////////////////////////////////// |     /*////////////////////////////////////////////////////////////////////////// | ||||||
|     // Fragment's LifeCycle |     // Fragment's LifeCycle | ||||||
| @@ -60,13 +66,23 @@ public class MainFragment extends BaseFragment implements TabLayout.OnTabSelecte | |||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public void onCreate(Bundle savedInstanceState) { |     public void onCreate(Bundle savedInstanceState) { | ||||||
|  |         savedInstanceStateBundle = savedInstanceState; | ||||||
|         super.onCreate(savedInstanceState); |         super.onCreate(savedInstanceState); | ||||||
|         setHasOptionsMenu(true); |         setHasOptionsMenu(true); | ||||||
|  |         listener = (prefs, key) -> { | ||||||
|  |             if(key.equals("saveUsedTabs")) { | ||||||
|  |                 mainPageChanged(); | ||||||
|  |             } | ||||||
|  |         }; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public View onCreateView(@NonNull LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) { |     public View onCreateView(@NonNull LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) { | ||||||
|         currentServiceId = ServiceHelper.getSelectedServiceId(activity); |         currentServiceId = ServiceHelper.getSelectedServiceId(activity); | ||||||
|  |  | ||||||
|  |         prefs = PreferenceManager.getDefaultSharedPreferences(getContext()); | ||||||
|  |         prefs.registerOnSharedPreferenceChangeListener(listener); | ||||||
|  |  | ||||||
|         return inflater.inflate(R.layout.fragment_main, container, false); |         return inflater.inflate(R.layout.fragment_main, container, false); | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -74,28 +90,114 @@ public class MainFragment extends BaseFragment implements TabLayout.OnTabSelecte | |||||||
|     protected void initViews(View rootView, Bundle savedInstanceState) { |     protected void initViews(View rootView, Bundle savedInstanceState) { | ||||||
|         super.initViews(rootView, savedInstanceState); |         super.initViews(rootView, savedInstanceState); | ||||||
|  |  | ||||||
|         TabLayout tabLayout = rootView.findViewById(R.id.main_tab_layout); |         tabLayout = rootView.findViewById(R.id.main_tab_layout); | ||||||
|         viewPager = rootView.findViewById(R.id.pager); |         viewPager = rootView.findViewById(R.id.pager); | ||||||
|  |  | ||||||
|         /*  Nested fragment, use child fragment here to maintain backstack in view pager. */ |         /*  Nested fragment, use child fragment here to maintain backstack in view pager. */ | ||||||
|         PagerAdapter adapter = new PagerAdapter(getChildFragmentManager()); |         adapter = new PagerAdapter(getChildFragmentManager()); | ||||||
|         viewPager.setAdapter(adapter); |         viewPager.setAdapter(adapter); | ||||||
|         viewPager.setOffscreenPageLimit(adapter.getCount()); |  | ||||||
|  |  | ||||||
|         tabLayout.setupWithViewPager(viewPager); |         tabLayout.setupWithViewPager(viewPager); | ||||||
|  |  | ||||||
|         int channelIcon = ThemeHelper.resolveResourceIdFromAttr(activity, R.attr.ic_channel); |         mainPageChanged(); | ||||||
|         int whatsHotIcon = ThemeHelper.resolveResourceIdFromAttr(activity, R.attr.ic_hot); |  | ||||||
|         int bookmarkIcon = ThemeHelper.resolveResourceIdFromAttr(activity, R.attr.ic_bookmark); |  | ||||||
|  |  | ||||||
|         if (isSubscriptionsPageOnlySelected()) { |  | ||||||
|             tabLayout.getTabAt(0).setIcon(channelIcon); |  | ||||||
|             tabLayout.getTabAt(1).setIcon(bookmarkIcon); |  | ||||||
|         } else { |  | ||||||
|             tabLayout.getTabAt(0).setIcon(whatsHotIcon); |  | ||||||
|             tabLayout.getTabAt(1).setIcon(channelIcon); |  | ||||||
|             tabLayout.getTabAt(2).setIcon(bookmarkIcon); |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     public void mainPageChanged() { | ||||||
|  |         getTabOrder(); | ||||||
|  |         adapter.notifyDataSetChanged(); | ||||||
|  |         viewPager.setOffscreenPageLimit(adapter.getCount()); | ||||||
|  |         setIcons(); | ||||||
|  |         setFirstTitle(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private void setFirstTitle() { | ||||||
|  |         if((tabs.size() > 0) | ||||||
|  |                 && activity != null) { | ||||||
|  |             String tabInformation = tabs.get(0); | ||||||
|  |                 if (tabInformation.startsWith(TAB_NUMBER_KIOSK + "\t")) { | ||||||
|  |                     String kiosk[] = tabInformation.split("\t"); | ||||||
|  |                     if (kiosk.length == 3) { | ||||||
|  |                         setTitle(kiosk[1]); | ||||||
|  |                     } | ||||||
|  |                 } else if (tabInformation.startsWith(TAB_NUMBER_CHANNEL + "\t")) { | ||||||
|  |  | ||||||
|  |                     String channelInfo[] = tabInformation.split("\t"); | ||||||
|  |                     if(channelInfo.length==4) { | ||||||
|  |                         setTitle(channelInfo[2]); | ||||||
|  |                     } | ||||||
|  |                 } else { | ||||||
|  |                     switch (tabInformation) { | ||||||
|  |                         case TAB_NUMBER_BLANK: | ||||||
|  |                             setTitle(getString(R.string.app_name)); | ||||||
|  |                             break; | ||||||
|  |                         case TAB_NUMBER_SUBSCIRPTIONS: | ||||||
|  |                             setTitle(getString(R.string.tab_subscriptions)); | ||||||
|  |                             break; | ||||||
|  |                         case TAB_NUMBER_FEED: | ||||||
|  |                             setTitle(getString(R.string.fragment_whats_new)); | ||||||
|  |                             break; | ||||||
|  |                         case TAB_NUMBER_BOOKMARKS: | ||||||
|  |                             setTitle(getString(R.string.tab_bookmarks)); | ||||||
|  |                             break; | ||||||
|  |                         case TAB_NUMBER_HISTORY: | ||||||
|  |                             setTitle(getString(R.string.title_activity_history)); | ||||||
|  |                             break; | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private void setIcons() { | ||||||
|  |         for (int i = 0; i < tabs.size(); i++) { | ||||||
|  |             String tabInformation = tabs.get(i); | ||||||
|  |  | ||||||
|  |             TabLayout.Tab tabToSet = tabLayout.getTabAt(i); | ||||||
|  |             Context c = getContext(); | ||||||
|  |  | ||||||
|  |             if (tabToSet != null && c != null) { | ||||||
|  |  | ||||||
|  |                 if (tabInformation.startsWith(TAB_NUMBER_KIOSK + "\t")) { | ||||||
|  |                     String kiosk[] = tabInformation.split("\t"); | ||||||
|  |                     if (kiosk.length == 3) { | ||||||
|  |                         tabToSet.setIcon(KioskTranslator.getKioskIcons(kiosk[1], getContext())); | ||||||
|  |                     } | ||||||
|  |                 } else if (tabInformation.startsWith(TAB_NUMBER_CHANNEL + "\t")) { | ||||||
|  |                     tabToSet.setIcon(ThemeHelper.resolveResourceIdFromAttr(getContext(), R.attr.ic_channel)); | ||||||
|  |                 } else { | ||||||
|  |                     switch (tabInformation) { | ||||||
|  |                         case TAB_NUMBER_BLANK: | ||||||
|  |                             tabToSet.setIcon(ThemeHelper.resolveResourceIdFromAttr(getContext(), R.attr.ic_hot)); | ||||||
|  |                             break; | ||||||
|  |                         case TAB_NUMBER_SUBSCIRPTIONS: | ||||||
|  |                             tabToSet.setIcon(ThemeHelper.resolveResourceIdFromAttr(getContext(), R.attr.ic_channel)); | ||||||
|  |                             break; | ||||||
|  |                         case TAB_NUMBER_FEED: | ||||||
|  |                             tabToSet.setIcon(ThemeHelper.resolveResourceIdFromAttr(getContext(), R.attr.rss)); | ||||||
|  |                             break; | ||||||
|  |                         case TAB_NUMBER_BOOKMARKS: | ||||||
|  |                             tabToSet.setIcon(ThemeHelper.resolveResourceIdFromAttr(getContext(), R.attr.ic_bookmark)); | ||||||
|  |                             break; | ||||||
|  |                         case TAB_NUMBER_HISTORY: | ||||||
|  |                             tabToSet.setIcon(ThemeHelper.resolveResourceIdFromAttr(getContext(), R.attr.history)); | ||||||
|  |                             break; | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |  | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     private void getTabOrder() { | ||||||
|  |         tabs.clear(); | ||||||
|  |  | ||||||
|  |         String save = prefs.getString("saveUsedTabs", "1\tTrending\t0\n2\n4\n"); | ||||||
|  |         String tabsArray[] = save.trim().split("\n"); | ||||||
|  |  | ||||||
|  |         Collections.addAll(tabs, tabsArray); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /*////////////////////////////////////////////////////////////////////////// |     /*////////////////////////////////////////////////////////////////////////// | ||||||
| @@ -107,16 +209,6 @@ public class MainFragment extends BaseFragment implements TabLayout.OnTabSelecte | |||||||
|         super.onCreateOptionsMenu(menu, inflater); |         super.onCreateOptionsMenu(menu, inflater); | ||||||
|         if (DEBUG) Log.d(TAG, "onCreateOptionsMenu() called with: menu = [" + menu + "], inflater = [" + inflater + "]"); |         if (DEBUG) Log.d(TAG, "onCreateOptionsMenu() called with: menu = [" + menu + "], inflater = [" + inflater + "]"); | ||||||
|         inflater.inflate(R.menu.main_fragment_menu, menu); |         inflater.inflate(R.menu.main_fragment_menu, menu); | ||||||
|         SubMenu kioskMenu = menu.addSubMenu(Menu.NONE, Menu.NONE, 200, getString(R.string.kiosk)); |  | ||||||
|         try { |  | ||||||
|             createKioskMenu(kioskMenu, inflater); |  | ||||||
|         } catch (Exception e) { |  | ||||||
|             ErrorActivity.reportError(activity, e, |  | ||||||
|                     activity.getClass(), |  | ||||||
|                     null, |  | ||||||
|                     ErrorActivity.ErrorInfo.make(UserAction.UI_ERROR, |  | ||||||
|                             "none", "", R.string.app_ui_crash)); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         ActionBar supportActionBar = activity.getSupportActionBar(); |         ActionBar supportActionBar = activity.getSupportActionBar(); | ||||||
|         if (supportActionBar != null) { |         if (supportActionBar != null) { | ||||||
| @@ -165,115 +257,77 @@ public class MainFragment extends BaseFragment implements TabLayout.OnTabSelecte | |||||||
|  |  | ||||||
|         @Override |         @Override | ||||||
|         public Fragment getItem(int position) { |         public Fragment getItem(int position) { | ||||||
|             switch (position) { |             String tabInformation = tabs.get(position); | ||||||
|                 case 0: |  | ||||||
|                     return isSubscriptionsPageOnlySelected() ? new SubscriptionFragment() : getMainPageFragment(); |  | ||||||
|                 case 1: |  | ||||||
|                     if(PreferenceManager.getDefaultSharedPreferences(getActivity()) |  | ||||||
|                             .getString(getString(R.string.main_page_content_key), getString(R.string.blank_page_key)) |  | ||||||
|                             .equals(getString(R.string.subscription_page_key))) { |  | ||||||
|                         return new BookmarkFragment(); |  | ||||||
|                     } else { |  | ||||||
|                         return new SubscriptionFragment(); |  | ||||||
|                     } |  | ||||||
|                 case 2: |  | ||||||
|                     return new BookmarkFragment(); |  | ||||||
|                 default: |  | ||||||
|                     return new BlankFragment(); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         @Override |  | ||||||
|         public CharSequence getPageTitle(int position) { |  | ||||||
|             //return getString(this.tabTitles[position]); |  | ||||||
|             return ""; |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         @Override |  | ||||||
|         public int getCount() { |  | ||||||
|             return isSubscriptionsPageOnlySelected() ? 2 : 3; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /*////////////////////////////////////////////////////////////////////////// |  | ||||||
|     // Main page content |  | ||||||
|     //////////////////////////////////////////////////////////////////////////*/ |  | ||||||
|  |  | ||||||
|     private boolean isSubscriptionsPageOnlySelected() { |  | ||||||
|         return PreferenceManager.getDefaultSharedPreferences(activity) |  | ||||||
|                 .getString(getString(R.string.main_page_content_key), getString(R.string.blank_page_key)) |  | ||||||
|                 .equals(getString(R.string.subscription_page_key)); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     private Fragment getMainPageFragment() { |  | ||||||
|         if (getActivity() == null) return new BlankFragment(); |  | ||||||
|  |  | ||||||
|  |             if(tabInformation.startsWith(TAB_NUMBER_KIOSK + "\t")) { | ||||||
|  |                 String kiosk[] = tabInformation.split("\t"); | ||||||
|  |                 if(kiosk.length==3) { | ||||||
|  |                     KioskFragment fragment = null; | ||||||
|                     try { |                     try { | ||||||
|             SharedPreferences preferences = |                         fragment = KioskFragment.getInstance(Integer.parseInt(kiosk[2]), kiosk[1]); | ||||||
|                     PreferenceManager.getDefaultSharedPreferences(getActivity()); |  | ||||||
|             final String setMainPage = preferences.getString(getString(R.string.main_page_content_key), |  | ||||||
|                     getString(R.string.main_page_selectd_kiosk_id)); |  | ||||||
|             if (setMainPage.equals(getString(R.string.blank_page_key))) { |  | ||||||
|                 return new BlankFragment(); |  | ||||||
|             } else if (setMainPage.equals(getString(R.string.kiosk_page_key))) { |  | ||||||
|                 int serviceId = preferences.getInt(getString(R.string.main_page_selected_service), |  | ||||||
|                         FALLBACK_SERVICE_ID); |  | ||||||
|                 String kioskId = preferences.getString(getString(R.string.main_page_selectd_kiosk_id), |  | ||||||
|                         FALLBACK_KIOSK_ID); |  | ||||||
|                 KioskFragment fragment = KioskFragment.getInstance(serviceId, kioskId); |  | ||||||
|                         fragment.useAsFrontPage(true); |                         fragment.useAsFrontPage(true); | ||||||
|                         return fragment; |                         return fragment; | ||||||
|             } else if (setMainPage.equals(getString(R.string.feed_page_key))) { |  | ||||||
|                 FeedFragment fragment = new FeedFragment(); |  | ||||||
|                 fragment.useAsFrontPage(true); |  | ||||||
|                 return fragment; |  | ||||||
|             } else if (setMainPage.equals(getString(R.string.channel_page_key))) { |  | ||||||
|                 int serviceId = preferences.getInt(getString(R.string.main_page_selected_service), |  | ||||||
|                         FALLBACK_SERVICE_ID); |  | ||||||
|                 String url = preferences.getString(getString(R.string.main_page_selected_channel_url), |  | ||||||
|                         FALLBACK_CHANNEL_URL); |  | ||||||
|                 String name = preferences.getString(getString(R.string.main_page_selected_channel_name), |  | ||||||
|                         FALLBACK_CHANNEL_NAME); |  | ||||||
|                 ChannelFragment fragment = ChannelFragment.getInstance(serviceId, |  | ||||||
|                         url, |  | ||||||
|                         name); |  | ||||||
|                 fragment.useAsFrontPage(true); |  | ||||||
|                 return fragment; |  | ||||||
|             } else { |  | ||||||
|                 return new BlankFragment(); |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|                     } catch (Exception e) { |                     } catch (Exception e) { | ||||||
|                         ErrorActivity.reportError(activity, e, |                         ErrorActivity.reportError(activity, e, | ||||||
|                                 activity.getClass(), |                                 activity.getClass(), | ||||||
|                                 null, |                                 null, | ||||||
|                                 ErrorActivity.ErrorInfo.make(UserAction.UI_ERROR, |                                 ErrorActivity.ErrorInfo.make(UserAction.UI_ERROR, | ||||||
|                                         "none", "", R.string.app_ui_crash)); |                                         "none", "", R.string.app_ui_crash)); | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |             } else if(tabInformation.startsWith(TAB_NUMBER_CHANNEL + "\t")) { | ||||||
|  |                 String channelInfo[] = tabInformation.split("\t"); | ||||||
|  |                 if(channelInfo.length==4) { | ||||||
|  |                     ChannelFragment fragment = ChannelFragment.getInstance(Integer.parseInt(channelInfo[3]), channelInfo[1], channelInfo[2]); | ||||||
|  |                     fragment.useAsFrontPage(true); | ||||||
|  |                     return fragment; | ||||||
|  |                 } else { | ||||||
|                     return new BlankFragment(); |                     return new BlankFragment(); | ||||||
|                 } |                 } | ||||||
|  |             } else { | ||||||
|  |                     switch (tabInformation) { | ||||||
|  |                         case TAB_NUMBER_BLANK: | ||||||
|  |                             return new BlankFragment(); | ||||||
|  |                         case TAB_NUMBER_SUBSCIRPTIONS: | ||||||
|  |                             SubscriptionFragment sFragment = new SubscriptionFragment(); | ||||||
|  |                             sFragment.useAsFrontPage(true); | ||||||
|  |                             return sFragment; | ||||||
|  |                         case TAB_NUMBER_FEED: | ||||||
|  |                             FeedFragment fFragment = new FeedFragment(); | ||||||
|  |                             fFragment.useAsFrontPage(true); | ||||||
|  |                             return fFragment; | ||||||
|  |                         case TAB_NUMBER_BOOKMARKS: | ||||||
|  |                             BookmarkFragment bFragment = new BookmarkFragment(); | ||||||
|  |                             bFragment.useAsFrontPage(true); | ||||||
|  |                             return bFragment; | ||||||
|  |                         case TAB_NUMBER_HISTORY: | ||||||
|  |                             StatisticsPlaylistFragment cFragment = new StatisticsPlaylistFragment(); | ||||||
|  |                             cFragment.useAsFrontPage(true); | ||||||
|  |                             return cFragment; | ||||||
|  |                     } | ||||||
|                 } |                 } | ||||||
|  |  | ||||||
|     /*////////////////////////////////////////////////////////////////////////// |             return new BlankFragment(); | ||||||
|     // Select Kiosk |             } | ||||||
|     //////////////////////////////////////////////////////////////////////////*/ |  | ||||||
|  |  | ||||||
|     private void createKioskMenu(Menu menu, MenuInflater menuInflater) |         @Override | ||||||
|             throws Exception { |         public int getItemPosition(Object object) { | ||||||
|         StreamingService service = NewPipe.getService(currentServiceId); |             // Causes adapter to reload all Fragments when | ||||||
|         KioskList kl = service.getKioskList(); |             // notifyDataSetChanged is called | ||||||
|         int i = 0; |             return POSITION_NONE; | ||||||
|         for (final String ks : kl.getAvailableKiosks()) { |  | ||||||
|             menu.add(0, KIOSK_MENU_OFFSET + i, Menu.NONE, |  | ||||||
|                     KioskTranslator.getTranslatedKioskName(ks, getContext())) |  | ||||||
|                     .setOnMenuItemClickListener(menuItem -> { |  | ||||||
|                             try { |  | ||||||
|                                 NavigationHelper.openKioskFragment(getFragmentManager(), currentServiceId, ks); |  | ||||||
|                             } catch (Exception e) { |  | ||||||
|                                 ErrorActivity.reportUiError((AppCompatActivity) getActivity(), e); |  | ||||||
|         } |         } | ||||||
|                             return true; |  | ||||||
|                     }); |         @Override | ||||||
|             i++; |         public int getCount() { | ||||||
|  |             return tabs.size(); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         @Override | ||||||
|  |         public void destroyItem(ViewGroup container, int position, Object object) { | ||||||
|  |             getFragmentManager() | ||||||
|  |                     .beginTransaction() | ||||||
|  |                     .remove((Fragment)object) | ||||||
|  |                     .commitNowAllowingStateLoss(); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1227,10 +1227,10 @@ public class VideoDetailFragment | |||||||
|                 spinnerToolbar.setVisibility(View.GONE); |                 spinnerToolbar.setVisibility(View.GONE); | ||||||
|                 break; |                 break; | ||||||
|             default: |             default: | ||||||
|  |                 if(info.getAudioStreams().isEmpty()) detailControlsBackground.setVisibility(View.GONE); | ||||||
|                 if (!info.getVideoStreams().isEmpty() |                 if (!info.getVideoStreams().isEmpty() | ||||||
|                         || !info.getVideoOnlyStreams().isEmpty()) break; |                         || !info.getVideoOnlyStreams().isEmpty()) break; | ||||||
|  |  | ||||||
|                 detailControlsBackground.setVisibility(View.GONE); |  | ||||||
|                 detailControlsPopup.setVisibility(View.GONE); |                 detailControlsPopup.setVisibility(View.GONE); | ||||||
|                 spinnerToolbar.setVisibility(View.GONE); |                 spinnerToolbar.setVisibility(View.GONE); | ||||||
|                 thumbnailPlayButton.setImageResource(R.drawable.ic_headset_white_24dp); |                 thumbnailPlayButton.setImageResource(R.drawable.ic_headset_white_24dp); | ||||||
|   | |||||||
| @@ -156,9 +156,7 @@ public abstract class BaseListFragment<I, N> extends BaseStateFragment<I> implem | |||||||
|             public void selected(ChannelInfoItem selectedItem) { |             public void selected(ChannelInfoItem selectedItem) { | ||||||
|                 try { |                 try { | ||||||
|                     onItemSelected(selectedItem); |                     onItemSelected(selectedItem); | ||||||
|                     NavigationHelper.openChannelFragment(useAsFrontPage ? |                     NavigationHelper.openChannelFragment(getFM(), | ||||||
|                                     getParentFragment().getFragmentManager() |  | ||||||
|                                     : getFragmentManager(), |  | ||||||
|                             selectedItem.getServiceId(), |                             selectedItem.getServiceId(), | ||||||
|                             selectedItem.getUrl(), |                             selectedItem.getUrl(), | ||||||
|                             selectedItem.getName()); |                             selectedItem.getName()); | ||||||
| @@ -173,10 +171,7 @@ public abstract class BaseListFragment<I, N> extends BaseStateFragment<I> implem | |||||||
|             public void selected(PlaylistInfoItem selectedItem) { |             public void selected(PlaylistInfoItem selectedItem) { | ||||||
|                 try { |                 try { | ||||||
|                     onItemSelected(selectedItem); |                     onItemSelected(selectedItem); | ||||||
|                     NavigationHelper.openPlaylistFragment( |                     NavigationHelper.openPlaylistFragment(getFM(), | ||||||
|                             useAsFrontPage |  | ||||||
|                                     ? getParentFragment().getFragmentManager() |  | ||||||
|                                     : getFragmentManager(), |  | ||||||
|                             selectedItem.getServiceId(), |                             selectedItem.getServiceId(), | ||||||
|                             selectedItem.getUrl(), |                             selectedItem.getUrl(), | ||||||
|                             selectedItem.getName()); |                             selectedItem.getName()); | ||||||
| @@ -197,9 +192,7 @@ public abstract class BaseListFragment<I, N> extends BaseStateFragment<I> implem | |||||||
|  |  | ||||||
|     private void onStreamSelected(StreamInfoItem selectedItem) { |     private void onStreamSelected(StreamInfoItem selectedItem) { | ||||||
|         onItemSelected(selectedItem); |         onItemSelected(selectedItem); | ||||||
|         NavigationHelper.openVideoDetailFragment(useAsFrontPage |         NavigationHelper.openVideoDetailFragment(getFM(), | ||||||
|                         ? getParentFragment().getFragmentManager() |  | ||||||
|                         : getFragmentManager(), |  | ||||||
|                 selectedItem.getServiceId(), selectedItem.getUrl(), selectedItem.getName()); |                 selectedItem.getServiceId(), selectedItem.getUrl(), selectedItem.getName()); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -91,6 +91,8 @@ public class ChannelFragment extends BaseListInfoFragment<ChannelInfo> { | |||||||
|  |  | ||||||
|     private MenuItem menuRssButton; |     private MenuItem menuRssButton; | ||||||
|  |  | ||||||
|  |     private boolean mIsVisibleToUser = false; | ||||||
|  |  | ||||||
|     public static ChannelFragment getInstance(int serviceId, String url, String name) { |     public static ChannelFragment getInstance(int serviceId, String url, String name) { | ||||||
|         ChannelFragment instance = new ChannelFragment(); |         ChannelFragment instance = new ChannelFragment(); | ||||||
|         instance.setInitialData(serviceId, url, name); |         instance.setInitialData(serviceId, url, name); | ||||||
| @@ -104,6 +106,7 @@ public class ChannelFragment extends BaseListInfoFragment<ChannelInfo> { | |||||||
|     @Override |     @Override | ||||||
|     public void setUserVisibleHint(boolean isVisibleToUser) { |     public void setUserVisibleHint(boolean isVisibleToUser) { | ||||||
|         super.setUserVisibleHint(isVisibleToUser); |         super.setUserVisibleHint(isVisibleToUser); | ||||||
|  |         mIsVisibleToUser = isVisibleToUser; | ||||||
|         if(activity != null |         if(activity != null | ||||||
|                 && useAsFrontPage |                 && useAsFrontPage | ||||||
|                 && isVisibleToUser) { |                 && isVisibleToUser) { | ||||||
| @@ -166,9 +169,7 @@ public class ChannelFragment extends BaseListInfoFragment<ChannelInfo> { | |||||||
|                 context.getResources().getString(R.string.share) |                 context.getResources().getString(R.string.share) | ||||||
|         }; |         }; | ||||||
|  |  | ||||||
|         final DialogInterface.OnClickListener actions = new DialogInterface.OnClickListener() { |         final DialogInterface.OnClickListener actions = (DialogInterface dialogInterface, int i) -> { | ||||||
|             @Override |  | ||||||
|             public void onClick(DialogInterface dialogInterface, int i) { |  | ||||||
|             final int index = Math.max(infoListAdapter.getItemsList().indexOf(item), 0); |             final int index = Math.max(infoListAdapter.getItemsList().indexOf(item), 0); | ||||||
|             switch (i) { |             switch (i) { | ||||||
|                 case 0: |                 case 0: | ||||||
| @@ -198,7 +199,6 @@ public class ChannelFragment extends BaseListInfoFragment<ChannelInfo> { | |||||||
|                 default: |                 default: | ||||||
|                     break; |                     break; | ||||||
|             } |             } | ||||||
|             } |  | ||||||
|         }; |         }; | ||||||
|  |  | ||||||
|         new InfoItemDialog(getActivity(), item, commands, actions).show(); |         new InfoItemDialog(getActivity(), item, commands, actions).show(); | ||||||
| @@ -255,12 +255,12 @@ public class ChannelFragment extends BaseListInfoFragment<ChannelInfo> { | |||||||
|     private static final int BUTTON_DEBOUNCE_INTERVAL = 100; |     private static final int BUTTON_DEBOUNCE_INTERVAL = 100; | ||||||
|  |  | ||||||
|     private void monitorSubscription(final ChannelInfo info) { |     private void monitorSubscription(final ChannelInfo info) { | ||||||
|         final Consumer<Throwable> onError = new Consumer<Throwable>() { |         final Consumer<Throwable> onError = (Throwable throwable) -> { | ||||||
|             @Override |  | ||||||
|             public void accept(Throwable throwable) throws Exception { |  | ||||||
|                 animateView(headerSubscribeButton, false, 100); |                 animateView(headerSubscribeButton, false, 100); | ||||||
|                 showSnackBarError(throwable, UserAction.SUBSCRIPTION, NewPipe.getNameOfService(currentInfo.getServiceId()), "Get subscription status", 0); |                 showSnackBarError(throwable, UserAction.SUBSCRIPTION, | ||||||
|             } |                         NewPipe.getNameOfService(currentInfo.getServiceId()), | ||||||
|  |                         "Get subscription status", | ||||||
|  |                         0); | ||||||
|         }; |         }; | ||||||
|  |  | ||||||
|         final Observable<List<SubscriptionEntity>> observable = subscriptionService.subscriptionTable() |         final Observable<List<SubscriptionEntity>> observable = subscriptionService.subscriptionTable() | ||||||
| @@ -276,50 +276,38 @@ public class ChannelFragment extends BaseListInfoFragment<ChannelInfo> { | |||||||
|                 // so only update the UI for the latest emission ("sync" the subscribe button's state) |                 // so only update the UI for the latest emission ("sync" the subscribe button's state) | ||||||
|                 .debounce(100, TimeUnit.MILLISECONDS) |                 .debounce(100, TimeUnit.MILLISECONDS) | ||||||
|                 .observeOn(AndroidSchedulers.mainThread()) |                 .observeOn(AndroidSchedulers.mainThread()) | ||||||
|                 .subscribe(new Consumer<List<SubscriptionEntity>>() { |                 .subscribe((List<SubscriptionEntity> subscriptionEntities) -> | ||||||
|                     @Override |                         updateSubscribeButton(!subscriptionEntities.isEmpty()) | ||||||
|                     public void accept(List<SubscriptionEntity> subscriptionEntities) throws Exception { |                         , onError)); | ||||||
|                         updateSubscribeButton(!subscriptionEntities.isEmpty()); |  | ||||||
|                     } |  | ||||||
|                 }, onError)); |  | ||||||
|  |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private Function<Object, Object> mapOnSubscribe(final SubscriptionEntity subscription) { |     private Function<Object, Object> mapOnSubscribe(final SubscriptionEntity subscription) { | ||||||
|         return new Function<Object, Object>() { |         return (@NonNull Object o) -> { | ||||||
|             @Override |  | ||||||
|             public Object apply(@NonNull Object o) throws Exception { |  | ||||||
|             subscriptionService.subscriptionTable().insert(subscription); |             subscriptionService.subscriptionTable().insert(subscription); | ||||||
|             return o; |             return o; | ||||||
|             } |  | ||||||
|         }; |         }; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private Function<Object, Object> mapOnUnsubscribe(final SubscriptionEntity subscription) { |     private Function<Object, Object> mapOnUnsubscribe(final SubscriptionEntity subscription) { | ||||||
|         return new Function<Object, Object>() { |         return (@NonNull Object o) -> { | ||||||
|             @Override |  | ||||||
|             public Object apply(@NonNull Object o) throws Exception { |  | ||||||
|             subscriptionService.subscriptionTable().delete(subscription); |             subscriptionService.subscriptionTable().delete(subscription); | ||||||
|             return o; |             return o; | ||||||
|             } |  | ||||||
|         }; |         }; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private void updateSubscription(final ChannelInfo info) { |     private void updateSubscription(final ChannelInfo info) { | ||||||
|         if (DEBUG) Log.d(TAG, "updateSubscription() called with: info = [" + info + "]"); |         if (DEBUG) Log.d(TAG, "updateSubscription() called with: info = [" + info + "]"); | ||||||
|         final Action onComplete = new Action() { |         final Action onComplete = () -> { | ||||||
|             @Override |  | ||||||
|             public void run() throws Exception { |  | ||||||
|                 if (DEBUG) Log.d(TAG, "Updated subscription: " + info.getUrl()); |                 if (DEBUG) Log.d(TAG, "Updated subscription: " + info.getUrl()); | ||||||
|             } |  | ||||||
|         }; |         }; | ||||||
|  |  | ||||||
|         final Consumer<Throwable> onError = new Consumer<Throwable>() { |         final Consumer<Throwable> onError = (@NonNull Throwable throwable) -> | ||||||
|             @Override |                 onUnrecoverableError(throwable, | ||||||
|             public void accept(@NonNull Throwable throwable) throws Exception { |                         UserAction.SUBSCRIPTION, | ||||||
|                 onUnrecoverableError(throwable, UserAction.SUBSCRIPTION, NewPipe.getNameOfService(info.getServiceId()), "Updating Subscription for " + info.getUrl(), R.string.subscription_update_failed); |                         NewPipe.getNameOfService(info.getServiceId()), | ||||||
|             } |                         "Updating Subscription for " + info.getUrl(), | ||||||
|         }; |                         R.string.subscription_update_failed); | ||||||
|  |  | ||||||
|         disposables.add(subscriptionService.updateChannelInfo(info) |         disposables.add(subscriptionService.updateChannelInfo(info) | ||||||
|                 .subscribeOn(Schedulers.io()) |                 .subscribeOn(Schedulers.io()) | ||||||
| @@ -328,19 +316,16 @@ public class ChannelFragment extends BaseListInfoFragment<ChannelInfo> { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     private Disposable monitorSubscribeButton(final Button subscribeButton, final Function<Object, Object> action) { |     private Disposable monitorSubscribeButton(final Button subscribeButton, final Function<Object, Object> action) { | ||||||
|         final Consumer<Object> onNext = new Consumer<Object>() { |         final Consumer<Object> onNext = (@NonNull Object o) -> { | ||||||
|             @Override |  | ||||||
|             public void accept(@NonNull Object o) throws Exception { |  | ||||||
|                 if (DEBUG) Log.d(TAG, "Changed subscription status to this channel!"); |                 if (DEBUG) Log.d(TAG, "Changed subscription status to this channel!"); | ||||||
|             } |  | ||||||
|         }; |         }; | ||||||
|  |  | ||||||
|         final Consumer<Throwable> onError = new Consumer<Throwable>() { |         final Consumer<Throwable> onError = (@NonNull Throwable throwable) -> | ||||||
|             @Override |                 onUnrecoverableError(throwable, | ||||||
|             public void accept(@NonNull Throwable throwable) throws Exception { |                         UserAction.SUBSCRIPTION, | ||||||
|                 onUnrecoverableError(throwable, UserAction.SUBSCRIPTION, NewPipe.getNameOfService(currentInfo.getServiceId()), "Subscription Change", R.string.subscription_change_failed); |                         NewPipe.getNameOfService(currentInfo.getServiceId()), | ||||||
|             } |                         "Subscription Change", | ||||||
|         }; |                         R.string.subscription_change_failed); | ||||||
|  |  | ||||||
|         /* Emit clicks from main thread unto io thread */ |         /* Emit clicks from main thread unto io thread */ | ||||||
|         return RxView.clicks(subscribeButton) |         return RxView.clicks(subscribeButton) | ||||||
| @@ -352,9 +337,7 @@ public class ChannelFragment extends BaseListInfoFragment<ChannelInfo> { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     private Consumer<List<SubscriptionEntity>> getSubscribeUpdateMonitor(final ChannelInfo info) { |     private Consumer<List<SubscriptionEntity>> getSubscribeUpdateMonitor(final ChannelInfo info) { | ||||||
|         return new Consumer<List<SubscriptionEntity>>() { |         return (List<SubscriptionEntity> subscriptionEntities) -> { | ||||||
|             @Override |  | ||||||
|             public void accept(List<SubscriptionEntity> subscriptionEntities) throws Exception { |  | ||||||
|             if (DEBUG) |             if (DEBUG) | ||||||
|                 Log.d(TAG, "subscriptionService.subscriptionTable.doOnNext() called with: subscriptionEntities = [" + subscriptionEntities + "]"); |                 Log.d(TAG, "subscriptionService.subscriptionTable.doOnNext() called with: subscriptionEntities = [" + subscriptionEntities + "]"); | ||||||
|             if (subscribeButtonMonitor != null) subscribeButtonMonitor.dispose(); |             if (subscribeButtonMonitor != null) subscribeButtonMonitor.dispose(); | ||||||
| @@ -364,14 +347,16 @@ public class ChannelFragment extends BaseListInfoFragment<ChannelInfo> { | |||||||
|                 SubscriptionEntity channel = new SubscriptionEntity(); |                 SubscriptionEntity channel = new SubscriptionEntity(); | ||||||
|                 channel.setServiceId(info.getServiceId()); |                 channel.setServiceId(info.getServiceId()); | ||||||
|                 channel.setUrl(info.getUrl()); |                 channel.setUrl(info.getUrl()); | ||||||
|                     channel.setData(info.getName(), info.getAvatarUrl(), info.getDescription(), info.getSubscriberCount()); |                 channel.setData(info.getName(), | ||||||
|  |                         info.getAvatarUrl(), | ||||||
|  |                         info.getDescription(), | ||||||
|  |                         info.getSubscriberCount()); | ||||||
|                 subscribeButtonMonitor = monitorSubscribeButton(headerSubscribeButton, mapOnSubscribe(channel)); |                 subscribeButtonMonitor = monitorSubscribeButton(headerSubscribeButton, mapOnSubscribe(channel)); | ||||||
|             } else { |             } else { | ||||||
|                 if (DEBUG) Log.d(TAG, "Found subscription to this channel!"); |                 if (DEBUG) Log.d(TAG, "Found subscription to this channel!"); | ||||||
|                 final SubscriptionEntity subscription = subscriptionEntities.get(0); |                 final SubscriptionEntity subscription = subscriptionEntities.get(0); | ||||||
|                 subscribeButtonMonitor = monitorSubscribeButton(headerSubscribeButton, mapOnUnsubscribe(subscription)); |                 subscribeButtonMonitor = monitorSubscribeButton(headerSubscribeButton, mapOnUnsubscribe(subscription)); | ||||||
|             } |             } | ||||||
|             } |  | ||||||
|         }; |         }; | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -488,8 +473,11 @@ public class ChannelFragment extends BaseListInfoFragment<ChannelInfo> { | |||||||
|         super.handleNextItems(result); |         super.handleNextItems(result); | ||||||
|  |  | ||||||
|         if (!result.getErrors().isEmpty()) { |         if (!result.getErrors().isEmpty()) { | ||||||
|             showSnackBarError(result.getErrors(), UserAction.REQUESTED_CHANNEL, NewPipe.getNameOfService(serviceId), |             showSnackBarError(result.getErrors(), | ||||||
|                     "Get next page of: " + url, R.string.general_error); |                     UserAction.REQUESTED_CHANNEL, | ||||||
|  |                     NewPipe.getNameOfService(serviceId), | ||||||
|  |                     "Get next page of: " + url, | ||||||
|  |                     R.string.general_error); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -517,6 +505,6 @@ public class ChannelFragment extends BaseListInfoFragment<ChannelInfo> { | |||||||
|     @Override |     @Override | ||||||
|     public void setTitle(String title) { |     public void setTitle(String title) { | ||||||
|         super.setTitle(title); |         super.setTitle(title); | ||||||
|         headerTitleView.setText(title); |         if (!useAsFrontPage) headerTitleView.setText(title); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -57,6 +57,7 @@ public class KioskFragment extends BaseListInfoFragment<KioskInfo> { | |||||||
|     protected String kioskId = ""; |     protected String kioskId = ""; | ||||||
|     protected String kioskTranslatedName; |     protected String kioskTranslatedName; | ||||||
|  |  | ||||||
|  |  | ||||||
|     /*////////////////////////////////////////////////////////////////////////// |     /*////////////////////////////////////////////////////////////////////////// | ||||||
|     // Views |     // Views | ||||||
|     //////////////////////////////////////////////////////////////////////////*/ |     //////////////////////////////////////////////////////////////////////////*/ | ||||||
| @@ -167,7 +168,9 @@ public class KioskFragment extends BaseListInfoFragment<KioskInfo> { | |||||||
|         super.handleResult(result); |         super.handleResult(result); | ||||||
|  |  | ||||||
|         name = kioskTranslatedName; |         name = kioskTranslatedName; | ||||||
|  |         if(!useAsFrontPage) { | ||||||
|             setTitle(kioskTranslatedName); |             setTitle(kioskTranslatedName); | ||||||
|  |         } | ||||||
|  |  | ||||||
|         if (!result.getErrors().isEmpty()) { |         if (!result.getErrors().isEmpty()) { | ||||||
|             showSnackBarError(result.getErrors(), |             showSnackBarError(result.getErrors(), | ||||||
|   | |||||||
| @@ -365,7 +365,7 @@ public class SearchFragment | |||||||
|         int itemId = 0; |         int itemId = 0; | ||||||
|         boolean isFirstItem = true; |         boolean isFirstItem = true; | ||||||
|         final Context c = getContext(); |         final Context c = getContext(); | ||||||
|         for(String filter : service.getSearchQIHFactory().getAvailableContentFilter()) { |         for(String filter : service.getSearchQHFactory().getAvailableContentFilter()) { | ||||||
|             menuItemToFilterName.put(itemId, filter); |             menuItemToFilterName.put(itemId, filter); | ||||||
|             MenuItem item = menu.add(1, |             MenuItem item = menu.add(1, | ||||||
|                     itemId++, |                     itemId++, | ||||||
| @@ -575,8 +575,7 @@ public class SearchFragment | |||||||
|                                             .onNext(searchEditText.getText().toString()), |                                             .onNext(searchEditText.getText().toString()), | ||||||
|                                     throwable -> showSnackBarError(throwable, |                                     throwable -> showSnackBarError(throwable, | ||||||
|                                             UserAction.DELETE_FROM_HISTORY, "none", |                                             UserAction.DELETE_FROM_HISTORY, "none", | ||||||
|                                             "Deleting item failed", R.string.general_error) |                                             "Deleting item failed", R.string.general_error)); | ||||||
|                             ); |  | ||||||
|                     disposables.add(onDelete); |                     disposables.add(onDelete); | ||||||
|                 }) |                 }) | ||||||
|                 .show(); |                 .show(); | ||||||
| @@ -837,7 +836,10 @@ public class SearchFragment | |||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public void handleResult(@NonNull SearchInfo result) { |     public void handleResult(@NonNull SearchInfo result) { | ||||||
|         if (!result.getErrors().isEmpty()) { |         final List<Throwable> exceptions = result.getErrors(); | ||||||
|  |         if (!exceptions.isEmpty() | ||||||
|  |             && !(exceptions.size() == 1 | ||||||
|  |                 && exceptions.get(0) instanceof SearchExtractor.NothingFoundException)){ | ||||||
|             showSnackBarError(result.getErrors(), UserAction.SEARCHED, |             showSnackBarError(result.getErrors(), UserAction.SEARCHED, | ||||||
|                     NewPipe.getNameOfService(serviceId), searchString, 0); |                     NewPipe.getNameOfService(serviceId), searchString, 0); | ||||||
|         } |         } | ||||||
| @@ -864,6 +866,7 @@ public class SearchFragment | |||||||
|         showListFooter(false); |         showListFooter(false); | ||||||
|         currentPageUrl = result.getNextPageUrl(); |         currentPageUrl = result.getNextPageUrl(); | ||||||
|         infoListAdapter.addInfoItemList(result.getItems()); |         infoListAdapter.addInfoItemList(result.getItems()); | ||||||
|  |         nextPageUrl = result.getNextPageUrl(); | ||||||
|  |  | ||||||
|         if (!result.getErrors().isEmpty()) { |         if (!result.getErrors().isEmpty()) { | ||||||
|             showSnackBarError(result.getErrors(), UserAction.SEARCHED, |             showSnackBarError(result.getErrors(), UserAction.SEARCHED, | ||||||
|   | |||||||
| @@ -6,7 +6,6 @@ import android.os.Parcelable; | |||||||
| import android.support.annotation.NonNull; | import android.support.annotation.NonNull; | ||||||
| import android.support.annotation.Nullable; | import android.support.annotation.Nullable; | ||||||
| import android.support.v4.app.FragmentManager; | import android.support.v4.app.FragmentManager; | ||||||
| import android.support.v7.app.AppCompatActivity; |  | ||||||
| import android.view.LayoutInflater; | import android.view.LayoutInflater; | ||||||
| import android.view.View; | import android.view.View; | ||||||
| import android.view.ViewGroup; | import android.view.ViewGroup; | ||||||
| @@ -20,11 +19,9 @@ import org.schabi.newpipe.database.LocalItem; | |||||||
| import org.schabi.newpipe.database.playlist.PlaylistLocalItem; | import org.schabi.newpipe.database.playlist.PlaylistLocalItem; | ||||||
| import org.schabi.newpipe.database.playlist.PlaylistMetadataEntry; | import org.schabi.newpipe.database.playlist.PlaylistMetadataEntry; | ||||||
| import org.schabi.newpipe.database.playlist.model.PlaylistRemoteEntity; | import org.schabi.newpipe.database.playlist.model.PlaylistRemoteEntity; | ||||||
| import org.schabi.newpipe.extractor.NewPipe; |  | ||||||
| import org.schabi.newpipe.local.BaseLocalListFragment; | import org.schabi.newpipe.local.BaseLocalListFragment; | ||||||
| import org.schabi.newpipe.local.playlist.LocalPlaylistManager; | import org.schabi.newpipe.local.playlist.LocalPlaylistManager; | ||||||
| import org.schabi.newpipe.local.playlist.RemotePlaylistManager; | import org.schabi.newpipe.local.playlist.RemotePlaylistManager; | ||||||
| import org.schabi.newpipe.report.ErrorActivity; |  | ||||||
| import org.schabi.newpipe.report.UserAction; | import org.schabi.newpipe.report.UserAction; | ||||||
| import org.schabi.newpipe.util.NavigationHelper; | import org.schabi.newpipe.util.NavigationHelper; | ||||||
| import org.schabi.newpipe.util.OnClickGesture; | import org.schabi.newpipe.util.OnClickGesture; | ||||||
| @@ -69,11 +66,10 @@ public final class BookmarkFragment | |||||||
|     public View onCreateView(@NonNull LayoutInflater inflater, |     public View onCreateView(@NonNull LayoutInflater inflater, | ||||||
|                              @Nullable ViewGroup container, |                              @Nullable ViewGroup container, | ||||||
|                              Bundle savedInstanceState) { |                              Bundle savedInstanceState) { | ||||||
|         if (activity != null && activity.getSupportActionBar() != null) { |  | ||||||
|             activity.getSupportActionBar().setDisplayShowTitleEnabled(true); |  | ||||||
|             activity.setTitle(R.string.tab_subscriptions); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|  |         if(!useAsFrontPage) { | ||||||
|  |             setTitle(activity.getString(R.string.tab_bookmarks)); | ||||||
|  |         } | ||||||
|         return inflater.inflate(R.layout.fragment_bookmarks, container, false); |         return inflater.inflate(R.layout.fragment_bookmarks, container, false); | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -102,10 +98,7 @@ public final class BookmarkFragment | |||||||
|         itemListAdapter.setSelectedListener(new OnClickGesture<LocalItem>() { |         itemListAdapter.setSelectedListener(new OnClickGesture<LocalItem>() { | ||||||
|             @Override |             @Override | ||||||
|             public void selected(LocalItem selectedItem) { |             public void selected(LocalItem selectedItem) { | ||||||
|                 try { |                 final FragmentManager fragmentManager = getFM(); | ||||||
|                     // Requires the parent fragment to find holder for fragment replacement |  | ||||||
|                     if (getParentFragment() == null) return; |  | ||||||
|                     final FragmentManager fragmentManager = getParentFragment().getFragmentManager(); |  | ||||||
|  |  | ||||||
|                 if (selectedItem instanceof PlaylistMetadataEntry) { |                 if (selectedItem instanceof PlaylistMetadataEntry) { | ||||||
|                     final PlaylistMetadataEntry entry = ((PlaylistMetadataEntry) selectedItem); |                     final PlaylistMetadataEntry entry = ((PlaylistMetadataEntry) selectedItem); | ||||||
| @@ -120,9 +113,6 @@ public final class BookmarkFragment | |||||||
|                             entry.getUrl(), |                             entry.getUrl(), | ||||||
|                             entry.getName()); |                             entry.getName()); | ||||||
|                 } |                 } | ||||||
|                 } catch (Exception e) { |  | ||||||
|                     ErrorActivity.reportUiError((AppCompatActivity) getActivity(), e); |  | ||||||
|                 } |  | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             @Override |             @Override | ||||||
|   | |||||||
| @@ -71,6 +71,10 @@ public class FeedFragment extends BaseListFragment<List<SubscriptionEntity>, Voi | |||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container, Bundle savedInstanceState) { |     public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container, Bundle savedInstanceState) { | ||||||
|  |  | ||||||
|  |         if(!useAsFrontPage) { | ||||||
|  |             setTitle(activity.getString(R.string.fragment_whats_new)); | ||||||
|  |         } | ||||||
|         return inflater.inflate(R.layout.fragment_feed, container, false); |         return inflater.inflate(R.layout.fragment_feed, container, false); | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -105,20 +109,19 @@ public class FeedFragment extends BaseListFragment<List<SubscriptionEntity>, Voi | |||||||
|         super.onDestroyView(); |         super.onDestroyView(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /*@Override |     @Override | ||||||
|     protected RecyclerView.LayoutManager getListLayoutManager() { |     public void setUserVisibleHint(boolean isVisibleToUser) { | ||||||
|         boolean isPortrait = getResources().getDisplayMetrics().heightPixels > getResources().getDisplayMetrics().widthPixels; |         super.setUserVisibleHint(isVisibleToUser); | ||||||
|         return new GridLayoutManager(activity, isPortrait ? 1 : 2); |         if (activity != null && isVisibleToUser) { | ||||||
|     }*/ |             setTitle(activity.getString(R.string.fragment_whats_new)); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public void onCreateOptionsMenu(Menu menu, MenuInflater inflater) { |     public void onCreateOptionsMenu(Menu menu, MenuInflater inflater) { | ||||||
|         super.onCreateOptionsMenu(menu, inflater); |         super.onCreateOptionsMenu(menu, inflater); | ||||||
|  |  | ||||||
|         ActionBar supportActionBar = activity.getSupportActionBar(); |         ActionBar supportActionBar = activity.getSupportActionBar(); | ||||||
|         if (supportActionBar != null) { |  | ||||||
|             supportActionBar.setTitle(R.string.fragment_whats_new); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         if(useAsFrontPage) { |         if(useAsFrontPage) { | ||||||
|             supportActionBar.setDisplayShowTitleEnabled(true); |             supportActionBar.setDisplayShowTitleEnabled(true); | ||||||
| @@ -176,19 +179,9 @@ public class FeedFragment extends BaseListFragment<List<SubscriptionEntity>, Voi | |||||||
|         showLoading(); |         showLoading(); | ||||||
|         showListFooter(true); |         showListFooter(true); | ||||||
|         subscriptionObserver = subscriptionService.getSubscription() |         subscriptionObserver = subscriptionService.getSubscription() | ||||||
|                 .onErrorReturnItem(Collections.<SubscriptionEntity>emptyList()) |                 .onErrorReturnItem(Collections.emptyList()) | ||||||
|                 .observeOn(AndroidSchedulers.mainThread()) |                 .observeOn(AndroidSchedulers.mainThread()) | ||||||
|                 .subscribe(new Consumer<List<SubscriptionEntity>>() { |                 .subscribe(this::handleResult, this::onError); | ||||||
|                     @Override |  | ||||||
|                     public void accept(List<SubscriptionEntity> subscriptionEntities) throws Exception { |  | ||||||
|                         handleResult(subscriptionEntities); |  | ||||||
|                     } |  | ||||||
|                 }, new Consumer<Throwable>() { |  | ||||||
|                     @Override |  | ||||||
|                     public void accept(Throwable throwable) throws Exception { |  | ||||||
|                         onError(throwable); |  | ||||||
|                     } |  | ||||||
|                 }); |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
| @@ -239,13 +232,12 @@ public class FeedFragment extends BaseListFragment<List<SubscriptionEntity>, Voi | |||||||
|                 if (!itemsLoaded.contains(subscriptionEntity.getServiceId() + subscriptionEntity.getUrl())) { |                 if (!itemsLoaded.contains(subscriptionEntity.getServiceId() + subscriptionEntity.getUrl())) { | ||||||
|                     subscriptionService.getChannelInfo(subscriptionEntity) |                     subscriptionService.getChannelInfo(subscriptionEntity) | ||||||
|                             .observeOn(AndroidSchedulers.mainThread()) |                             .observeOn(AndroidSchedulers.mainThread()) | ||||||
|                             .onErrorComplete(new Predicate<Throwable>() { |                             .onErrorComplete( | ||||||
|                                 @Override |                                     (@io.reactivex.annotations.NonNull Throwable throwable) -> | ||||||
|                                 public boolean test(@io.reactivex.annotations.NonNull Throwable throwable) throws Exception { |                                             FeedFragment.super.onError(throwable)) | ||||||
|                                     return FeedFragment.super.onError(throwable); |                             .subscribe( | ||||||
|                                 } |                                     getChannelInfoObserver(subscriptionEntity.getServiceId(), | ||||||
|                             }) |                                             subscriptionEntity.getUrl())); | ||||||
|                             .subscribe(getChannelInfoObserver(subscriptionEntity.getServiceId(), subscriptionEntity.getUrl())); |  | ||||||
|                 } else { |                 } else { | ||||||
|                     requestFeed(1); |                     requestFeed(1); | ||||||
|                 } |                 } | ||||||
| @@ -316,7 +308,10 @@ public class FeedFragment extends BaseListFragment<List<SubscriptionEntity>, Voi | |||||||
|  |  | ||||||
|             @Override |             @Override | ||||||
|             public void onError(Throwable exception) { |             public void onError(Throwable exception) { | ||||||
|                 showSnackBarError(exception, UserAction.SUBSCRIPTION, NewPipe.getNameOfService(serviceId), url, 0); |                 showSnackBarError(exception, | ||||||
|  |                         UserAction.SUBSCRIPTION, | ||||||
|  |                         NewPipe.getNameOfService(serviceId), | ||||||
|  |                         url, 0); | ||||||
|                 requestFeed(1); |                 requestFeed(1); | ||||||
|                 onDone(); |                 onDone(); | ||||||
|             } |             } | ||||||
| @@ -361,12 +356,7 @@ public class FeedFragment extends BaseListFragment<List<SubscriptionEntity>, Voi | |||||||
|         delayHandler.removeCallbacksAndMessages(null); |         delayHandler.removeCallbacksAndMessages(null); | ||||||
|         // Add a little of a delay when requesting more items because the cache is so fast, |         // Add a little of a delay when requesting more items because the cache is so fast, | ||||||
|         // that the view seems stuck to the user when he scroll to the bottom |         // that the view seems stuck to the user when he scroll to the bottom | ||||||
|         delayHandler.postDelayed(new Runnable() { |         delayHandler.postDelayed(() -> requestFeed(FEED_LOAD_COUNT), 300); | ||||||
|             @Override |  | ||||||
|             public void run() { |  | ||||||
|                 requestFeed(FEED_LOAD_COUNT); |  | ||||||
|             } |  | ||||||
|         }, 300); |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
| @@ -423,7 +413,9 @@ public class FeedFragment extends BaseListFragment<List<SubscriptionEntity>, Voi | |||||||
|         int heightPixels = getResources().getDisplayMetrics().heightPixels; |         int heightPixels = getResources().getDisplayMetrics().heightPixels; | ||||||
|         int itemHeightPixels = activity.getResources().getDimensionPixelSize(R.dimen.video_item_search_height); |         int itemHeightPixels = activity.getResources().getDimensionPixelSize(R.dimen.video_item_search_height); | ||||||
|  |  | ||||||
|         int items = itemHeightPixels > 0 ? heightPixels / itemHeightPixels + OFF_SCREEN_ITEMS_COUNT : MIN_ITEMS_INITIAL_LOAD; |         int items = itemHeightPixels > 0 | ||||||
|  |                 ? heightPixels / itemHeightPixels + OFF_SCREEN_ITEMS_COUNT | ||||||
|  |                 : MIN_ITEMS_INITIAL_LOAD; | ||||||
|         return Math.max(MIN_ITEMS_INITIAL_LOAD, items); |         return Math.max(MIN_ITEMS_INITIAL_LOAD, items); | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -441,8 +433,14 @@ public class FeedFragment extends BaseListFragment<List<SubscriptionEntity>, Voi | |||||||
|     protected boolean onError(Throwable exception) { |     protected boolean onError(Throwable exception) { | ||||||
|         if (super.onError(exception)) return true; |         if (super.onError(exception)) return true; | ||||||
|  |  | ||||||
|         int errorId = exception instanceof ExtractionException ? R.string.parsing_error : R.string.general_error; |         int errorId = exception instanceof ExtractionException | ||||||
|         onUnrecoverableError(exception, UserAction.SOMETHING_ELSE, "none", "Requesting feed", errorId); |                 ? R.string.parsing_error | ||||||
|  |                 : R.string.general_error; | ||||||
|  |         onUnrecoverableError(exception, | ||||||
|  |                 UserAction.SOMETHING_ELSE, | ||||||
|  |                 "none", | ||||||
|  |                 "Requesting feed", | ||||||
|  |                 errorId); | ||||||
|         return true; |         return true; | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -21,6 +21,7 @@ import org.schabi.newpipe.R; | |||||||
| import org.schabi.newpipe.database.LocalItem; | import org.schabi.newpipe.database.LocalItem; | ||||||
| import org.schabi.newpipe.database.stream.StreamStatisticsEntry; | import org.schabi.newpipe.database.stream.StreamStatisticsEntry; | ||||||
| import org.schabi.newpipe.extractor.stream.StreamInfoItem; | import org.schabi.newpipe.extractor.stream.StreamInfoItem; | ||||||
|  | import org.schabi.newpipe.fragments.list.BaseListFragment; | ||||||
| import org.schabi.newpipe.local.BaseLocalListFragment; | import org.schabi.newpipe.local.BaseLocalListFragment; | ||||||
| import org.schabi.newpipe.info_list.InfoItemDialog; | import org.schabi.newpipe.info_list.InfoItemDialog; | ||||||
| import org.schabi.newpipe.player.playqueue.PlayQueue; | import org.schabi.newpipe.player.playqueue.PlayQueue; | ||||||
| @@ -73,7 +74,7 @@ public class StatisticsPlaylistFragment | |||||||
|                 return results; |                 return results; | ||||||
|             case MOST_PLAYED: |             case MOST_PLAYED: | ||||||
|                 Collections.sort(results, (left, right) -> |                 Collections.sort(results, (left, right) -> | ||||||
|                         ((Long) right.watchCount).compareTo(left.watchCount)); |                         Long.compare(right.watchCount, left.watchCount)); | ||||||
|                 return results; |                 return results; | ||||||
|             default: return null; |             default: return null; | ||||||
|         } |         } | ||||||
| @@ -96,6 +97,14 @@ public class StatisticsPlaylistFragment | |||||||
|         return inflater.inflate(R.layout.fragment_playlist, container, false); |         return inflater.inflate(R.layout.fragment_playlist, container, false); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public void setUserVisibleHint(boolean isVisibleToUser) { | ||||||
|  |         super.setUserVisibleHint(isVisibleToUser); | ||||||
|  |         if (activity != null && isVisibleToUser) { | ||||||
|  |             setTitle(activity.getString(R.string.title_activity_history)); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|     /////////////////////////////////////////////////////////////////////////// |     /////////////////////////////////////////////////////////////////////////// | ||||||
|     // Fragment LifeCycle - Views |     // Fragment LifeCycle - Views | ||||||
|     /////////////////////////////////////////////////////////////////////////// |     /////////////////////////////////////////////////////////////////////////// | ||||||
| @@ -103,8 +112,10 @@ public class StatisticsPlaylistFragment | |||||||
|     @Override |     @Override | ||||||
|     protected void initViews(View rootView, Bundle savedInstanceState) { |     protected void initViews(View rootView, Bundle savedInstanceState) { | ||||||
|         super.initViews(rootView, savedInstanceState); |         super.initViews(rootView, savedInstanceState); | ||||||
|  |         if(!useAsFrontPage) { | ||||||
|             setTitle(getString(R.string.title_last_played)); |             setTitle(getString(R.string.title_last_played)); | ||||||
|         } |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     protected View getListHeader() { |     protected View getListHeader() { | ||||||
| @@ -129,8 +140,10 @@ public class StatisticsPlaylistFragment | |||||||
|             public void selected(LocalItem selectedItem) { |             public void selected(LocalItem selectedItem) { | ||||||
|                 if (selectedItem instanceof StreamStatisticsEntry) { |                 if (selectedItem instanceof StreamStatisticsEntry) { | ||||||
|                     final StreamStatisticsEntry item = (StreamStatisticsEntry) selectedItem; |                     final StreamStatisticsEntry item = (StreamStatisticsEntry) selectedItem; | ||||||
|                     NavigationHelper.openVideoDetailFragment(getFragmentManager(), |                     NavigationHelper.openVideoDetailFragment(getFM(), | ||||||
|                             item.serviceId, item.url, item.title); |                             item.serviceId, | ||||||
|  |                             item.url, | ||||||
|  |                             item.title); | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|  |  | ||||||
| @@ -341,7 +354,7 @@ public class StatisticsPlaylistFragment | |||||||
|             final Disposable onDelete = recordManager.deleteStreamHistory(entry.streamId) |             final Disposable onDelete = recordManager.deleteStreamHistory(entry.streamId) | ||||||
|                     .observeOn(AndroidSchedulers.mainThread()) |                     .observeOn(AndroidSchedulers.mainThread()) | ||||||
|                     .subscribe( |                     .subscribe( | ||||||
|                             howManyDelted -> { |                             howManyDeleted -> { | ||||||
|                                 if(getView() != null) { |                                 if(getView() != null) { | ||||||
|                                     Snackbar.make(getView(), R.string.one_item_deleted, |                                     Snackbar.make(getView(), R.string.one_item_deleted, | ||||||
|                                             Snackbar.LENGTH_SHORT).show(); |                                             Snackbar.LENGTH_SHORT).show(); | ||||||
|   | |||||||
| @@ -16,6 +16,7 @@ import android.os.Parcelable; | |||||||
| import android.support.annotation.DrawableRes; | import android.support.annotation.DrawableRes; | ||||||
| import android.support.annotation.NonNull; | import android.support.annotation.NonNull; | ||||||
| import android.support.annotation.Nullable; | import android.support.annotation.Nullable; | ||||||
|  | import android.support.v4.app.FragmentManager; | ||||||
| import android.support.v4.content.LocalBroadcastManager; | import android.support.v4.content.LocalBroadcastManager; | ||||||
| import android.support.v7.app.ActionBar; | import android.support.v7.app.ActionBar; | ||||||
| import android.support.v7.app.AppCompatActivity; | import android.support.v7.app.AppCompatActivity; | ||||||
| @@ -212,7 +213,8 @@ public class SubscriptionFragment extends BaseStateFragment<List<SubscriptionEnt | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     private void setupImportFromItems(final ViewGroup listHolder) { |     private void setupImportFromItems(final ViewGroup listHolder) { | ||||||
|         final View previousBackupItem = addItemView(getString(R.string.previous_export), ThemeHelper.resolveResourceIdFromAttr(getContext(), R.attr.ic_backup), listHolder); |         final View previousBackupItem = addItemView(getString(R.string.previous_export), | ||||||
|  |                 ThemeHelper.resolveResourceIdFromAttr(getContext(), R.attr.ic_backup), listHolder); | ||||||
|         previousBackupItem.setOnClickListener(item -> onImportPreviousSelected()); |         previousBackupItem.setOnClickListener(item -> onImportPreviousSelected()); | ||||||
|  |  | ||||||
|         final int iconColor = ThemeHelper.isLightThemeSelected(getContext()) ? Color.BLACK : Color.WHITE; |         final int iconColor = ThemeHelper.isLightThemeSelected(getContext()) ? Color.BLACK : Color.WHITE; | ||||||
| @@ -244,8 +246,8 @@ public class SubscriptionFragment extends BaseStateFragment<List<SubscriptionEnt | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     private void onImportFromServiceSelected(int serviceId) { |     private void onImportFromServiceSelected(int serviceId) { | ||||||
|         if (getParentFragment() == null) return; |         FragmentManager fragmentManager = getFM(); | ||||||
|         NavigationHelper.openSubscriptionsImportFragment(getParentFragment().getFragmentManager(), serviceId); |         NavigationHelper.openSubscriptionsImportFragment(fragmentManager, serviceId); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private void onImportPreviousSelected() { |     private void onImportPreviousSelected() { | ||||||
| @@ -323,15 +325,11 @@ public class SubscriptionFragment extends BaseStateFragment<List<SubscriptionEnt | |||||||
|         infoListAdapter.setOnChannelSelectedListener(new OnClickGesture<ChannelInfoItem>() { |         infoListAdapter.setOnChannelSelectedListener(new OnClickGesture<ChannelInfoItem>() { | ||||||
|  |  | ||||||
|             public void selected(ChannelInfoItem selectedItem) { |             public void selected(ChannelInfoItem selectedItem) { | ||||||
|                 try { |                 final FragmentManager fragmentManager = getFM(); | ||||||
|                     // Requires the parent fragment to find holder for fragment replacement |                 NavigationHelper.openChannelFragment(fragmentManager, | ||||||
|                     NavigationHelper.openChannelFragment(getParentFragment().getFragmentManager(), |  | ||||||
|                         selectedItem.getServiceId(), |                         selectedItem.getServiceId(), | ||||||
|                         selectedItem.getUrl(), |                         selectedItem.getUrl(), | ||||||
|                         selectedItem.getName()); |                         selectedItem.getName()); | ||||||
|                 } catch (Exception e) { |  | ||||||
|                     ErrorActivity.reportUiError((AppCompatActivity) getActivity(), e); |  | ||||||
|                 } |  | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             public void held(ChannelInfoItem selectedItem) { |             public void held(ChannelInfoItem selectedItem) { | ||||||
| @@ -341,8 +339,10 @@ public class SubscriptionFragment extends BaseStateFragment<List<SubscriptionEnt | |||||||
|         }); |         }); | ||||||
|  |  | ||||||
|         //noinspection ConstantConditions |         //noinspection ConstantConditions | ||||||
|         whatsNewItemListHeader.setOnClickListener(v -> |         whatsNewItemListHeader.setOnClickListener(v -> { | ||||||
|                 NavigationHelper.openWhatsNewFragment(getParentFragment().getFragmentManager())); |             FragmentManager fragmentManager = getFM(); | ||||||
|  |             NavigationHelper.openWhatsNewFragment(fragmentManager); | ||||||
|  |         }); | ||||||
|         importExportListHeader.setOnClickListener(v -> importExportOptions.switchState()); |         importExportListHeader.setOnClickListener(v -> importExportOptions.switchState()); | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -492,10 +492,13 @@ public class SubscriptionFragment extends BaseStateFragment<List<SubscriptionEnt | |||||||
|  |  | ||||||
|     private List<InfoItem> getSubscriptionItems(List<SubscriptionEntity> subscriptions) { |     private List<InfoItem> getSubscriptionItems(List<SubscriptionEntity> subscriptions) { | ||||||
|         List<InfoItem> items = new ArrayList<>(); |         List<InfoItem> items = new ArrayList<>(); | ||||||
|         for (final SubscriptionEntity subscription : subscriptions) items.add(subscription.toChannelInfoItem()); |         for (final SubscriptionEntity subscription : subscriptions) { | ||||||
|  |             items.add(subscription.toChannelInfoItem()); | ||||||
|  |         } | ||||||
|  |  | ||||||
|         Collections.sort(items, |         Collections.sort(items, | ||||||
|                 (InfoItem o1, InfoItem o2) -> o1.getName().compareToIgnoreCase(o2.getName())); |                 (InfoItem o1, InfoItem o2) -> | ||||||
|  |                         o1.getName().compareToIgnoreCase(o2.getName())); | ||||||
|         return items; |         return items; | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -524,7 +527,11 @@ public class SubscriptionFragment extends BaseStateFragment<List<SubscriptionEnt | |||||||
|         resetFragment(); |         resetFragment(); | ||||||
|         if (super.onError(exception)) return true; |         if (super.onError(exception)) return true; | ||||||
|  |  | ||||||
|         onUnrecoverableError(exception, UserAction.SOMETHING_ELSE, "none", "Subscriptions", R.string.general_error); |         onUnrecoverableError(exception, | ||||||
|  |                 UserAction.SOMETHING_ELSE, | ||||||
|  |                 "none", | ||||||
|  |                 "Subscriptions", | ||||||
|  |                 R.string.general_error); | ||||||
|         return true; |         return true; | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -26,6 +26,7 @@ import android.content.Context; | |||||||
| import android.content.Intent; | import android.content.Intent; | ||||||
| import android.content.IntentFilter; | import android.content.IntentFilter; | ||||||
| import android.graphics.Bitmap; | import android.graphics.Bitmap; | ||||||
|  | import android.os.Build; | ||||||
| import android.os.IBinder; | import android.os.IBinder; | ||||||
| import android.support.annotation.NonNull; | import android.support.annotation.NonNull; | ||||||
| import android.support.annotation.Nullable; | import android.support.annotation.Nullable; | ||||||
| @@ -343,6 +344,7 @@ public final class BackgroundPlayer extends Service { | |||||||
|  |  | ||||||
|             if (!shouldUpdateOnProgress) return; |             if (!shouldUpdateOnProgress) return; | ||||||
|             resetNotification(); |             resetNotification(); | ||||||
|  |             if(Build.VERSION.SDK_INT >= 26 /*Oreo*/) updateNotificationThumbnail(); | ||||||
|             if (bigNotRemoteView != null) { |             if (bigNotRemoteView != null) { | ||||||
|                 bigNotRemoteView.setProgressBar(R.id.notificationProgressBar, duration, currentProgress, false); |                 bigNotRemoteView.setProgressBar(R.id.notificationProgressBar, duration, currentProgress, false); | ||||||
|                 bigNotRemoteView.setTextViewText(R.id.notificationTime, getTimeString(currentProgress) + " / " + getTimeString(duration)); |                 bigNotRemoteView.setTextViewText(R.id.notificationTime, getTimeString(currentProgress) + " / " + getTimeString(duration)); | ||||||
|   | |||||||
| @@ -51,6 +51,7 @@ import com.nostra13.universalimageloader.core.ImageLoader; | |||||||
| import com.nostra13.universalimageloader.core.assist.FailReason; | import com.nostra13.universalimageloader.core.assist.FailReason; | ||||||
| import com.nostra13.universalimageloader.core.listener.ImageLoadingListener; | import com.nostra13.universalimageloader.core.listener.ImageLoadingListener; | ||||||
|  |  | ||||||
|  | import org.schabi.newpipe.BuildConfig; | ||||||
| import org.schabi.newpipe.Downloader; | import org.schabi.newpipe.Downloader; | ||||||
| import org.schabi.newpipe.R; | import org.schabi.newpipe.R; | ||||||
| import org.schabi.newpipe.extractor.stream.StreamInfo; | import org.schabi.newpipe.extractor.stream.StreamInfo; | ||||||
| @@ -69,6 +70,7 @@ import org.schabi.newpipe.player.playqueue.PlayQueue; | |||||||
| import org.schabi.newpipe.player.playqueue.PlayQueueAdapter; | import org.schabi.newpipe.player.playqueue.PlayQueueAdapter; | ||||||
| import org.schabi.newpipe.player.playqueue.PlayQueueItem; | import org.schabi.newpipe.player.playqueue.PlayQueueItem; | ||||||
| import org.schabi.newpipe.player.resolver.MediaSourceTag; | import org.schabi.newpipe.player.resolver.MediaSourceTag; | ||||||
|  | import org.schabi.newpipe.report.ErrorActivity; | ||||||
| import org.schabi.newpipe.util.ImageDisplayConstants; | import org.schabi.newpipe.util.ImageDisplayConstants; | ||||||
| import org.schabi.newpipe.util.SerializedCache; | import org.schabi.newpipe.util.SerializedCache; | ||||||
|  |  | ||||||
| @@ -86,6 +88,7 @@ import static com.google.android.exoplayer2.Player.DISCONTINUITY_REASON_INTERNAL | |||||||
| import static com.google.android.exoplayer2.Player.DISCONTINUITY_REASON_PERIOD_TRANSITION; | import static com.google.android.exoplayer2.Player.DISCONTINUITY_REASON_PERIOD_TRANSITION; | ||||||
| import static com.google.android.exoplayer2.Player.DISCONTINUITY_REASON_SEEK; | import static com.google.android.exoplayer2.Player.DISCONTINUITY_REASON_SEEK; | ||||||
| import static com.google.android.exoplayer2.Player.DISCONTINUITY_REASON_SEEK_ADJUSTMENT; | import static com.google.android.exoplayer2.Player.DISCONTINUITY_REASON_SEEK_ADJUSTMENT; | ||||||
|  | import static org.schabi.newpipe.report.UserAction.PLAY_STREAM; | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * Base for the players, joining the common properties |  * Base for the players, joining the common properties | ||||||
| @@ -96,7 +99,7 @@ import static com.google.android.exoplayer2.Player.DISCONTINUITY_REASON_SEEK_ADJ | |||||||
| public abstract class BasePlayer implements | public abstract class BasePlayer implements | ||||||
|         Player.EventListener, PlaybackListener, ImageLoadingListener { |         Player.EventListener, PlaybackListener, ImageLoadingListener { | ||||||
|  |  | ||||||
|     public static final boolean DEBUG = true; |     public static final boolean DEBUG = !BuildConfig.BUILD_TYPE.equals("release"); | ||||||
|     @NonNull public static final String TAG = "BasePlayer"; |     @NonNull public static final String TAG = "BasePlayer"; | ||||||
|  |  | ||||||
|     @NonNull final protected Context context; |     @NonNull final protected Context context; | ||||||
| @@ -363,7 +366,10 @@ public abstract class BasePlayer implements | |||||||
|         try { |         try { | ||||||
|             context.unregisterReceiver(broadcastReceiver); |             context.unregisterReceiver(broadcastReceiver); | ||||||
|         } catch (final IllegalArgumentException unregisteredException) { |         } catch (final IllegalArgumentException unregisteredException) { | ||||||
|             Log.e(TAG, "Broadcast receiver already unregistered.", unregisteredException); |             ErrorActivity.reportError(context, unregisteredException, null, null, | ||||||
|  |                     ErrorActivity.ErrorInfo.make(PLAY_STREAM, | ||||||
|  |                             "none", | ||||||
|  |                             "play stream", R.string.general_error)); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -1001,6 +1007,8 @@ public abstract class BasePlayer implements | |||||||
|         try { |         try { | ||||||
|             metadata = (MediaSourceTag) simpleExoPlayer.getCurrentTag(); |             metadata = (MediaSourceTag) simpleExoPlayer.getCurrentTag(); | ||||||
|         } catch (IndexOutOfBoundsException | ClassCastException error) { |         } catch (IndexOutOfBoundsException | ClassCastException error) { | ||||||
|  |             if(DEBUG) Log.d(TAG, "Could not update metadata: " + error.getMessage()); | ||||||
|  |             if(DEBUG) error.printStackTrace(); | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|  |  | ||||||
| @@ -1087,6 +1095,9 @@ public abstract class BasePlayer implements | |||||||
|             return simpleExoPlayer.isCurrentWindowDynamic(); |             return simpleExoPlayer.isCurrentWindowDynamic(); | ||||||
|         } catch (@NonNull IndexOutOfBoundsException ignored) { |         } catch (@NonNull IndexOutOfBoundsException ignored) { | ||||||
|             // Why would this even happen =( |             // Why would this even happen =( | ||||||
|  |             // But lets log it anyway. Save is save | ||||||
|  |             if(DEBUG) Log.d(TAG, "Could not update metadata: " + ignored.getMessage()); | ||||||
|  |             if(DEBUG) ignored.printStackTrace(); | ||||||
|             return false; |             return false; | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -36,6 +36,7 @@ import android.support.annotation.NonNull; | |||||||
| import android.support.annotation.Nullable; | import android.support.annotation.Nullable; | ||||||
| import android.support.v4.app.ActivityCompat; | import android.support.v4.app.ActivityCompat; | ||||||
| import android.support.v7.app.AppCompatActivity; | import android.support.v7.app.AppCompatActivity; | ||||||
|  | import android.support.v7.content.res.AppCompatResources; | ||||||
| import android.support.v7.widget.RecyclerView; | import android.support.v7.widget.RecyclerView; | ||||||
| import android.support.v7.widget.helper.ItemTouchHelper; | import android.support.v7.widget.helper.ItemTouchHelper; | ||||||
| import android.util.DisplayMetrics; | import android.util.DisplayMetrics; | ||||||
| @@ -46,7 +47,9 @@ import android.view.MotionEvent; | |||||||
| import android.view.View; | import android.view.View; | ||||||
| import android.view.WindowManager; | import android.view.WindowManager; | ||||||
| import android.widget.ImageButton; | import android.widget.ImageButton; | ||||||
|  | import android.widget.ImageView; | ||||||
| import android.widget.PopupMenu; | import android.widget.PopupMenu; | ||||||
|  | import android.widget.ProgressBar; | ||||||
| import android.widget.RelativeLayout; | import android.widget.RelativeLayout; | ||||||
| import android.widget.SeekBar; | import android.widget.SeekBar; | ||||||
| import android.widget.TextView; | import android.widget.TextView; | ||||||
| @@ -82,6 +85,7 @@ import java.util.UUID; | |||||||
| import static org.schabi.newpipe.player.BasePlayer.STATE_PLAYING; | import static org.schabi.newpipe.player.BasePlayer.STATE_PLAYING; | ||||||
| import static org.schabi.newpipe.player.VideoPlayer.DEFAULT_CONTROLS_DURATION; | import static org.schabi.newpipe.player.VideoPlayer.DEFAULT_CONTROLS_DURATION; | ||||||
| import static org.schabi.newpipe.player.VideoPlayer.DEFAULT_CONTROLS_HIDE_TIME; | import static org.schabi.newpipe.player.VideoPlayer.DEFAULT_CONTROLS_HIDE_TIME; | ||||||
|  | import static org.schabi.newpipe.util.AnimationUtils.Type.SCALE_AND_ALPHA; | ||||||
| import static org.schabi.newpipe.util.AnimationUtils.Type.SLIDE_AND_ALPHA; | import static org.schabi.newpipe.util.AnimationUtils.Type.SLIDE_AND_ALPHA; | ||||||
| import static org.schabi.newpipe.util.AnimationUtils.animateRotation; | import static org.schabi.newpipe.util.AnimationUtils.animateRotation; | ||||||
| import static org.schabi.newpipe.util.AnimationUtils.animateView; | import static org.schabi.newpipe.util.AnimationUtils.animateView; | ||||||
| @@ -365,10 +369,16 @@ public final class MainVideoPlayer extends AppCompatActivity | |||||||
|  |  | ||||||
|     @SuppressWarnings({"unused", "WeakerAccess"}) |     @SuppressWarnings({"unused", "WeakerAccess"}) | ||||||
|     private class VideoPlayerImpl extends VideoPlayer { |     private class VideoPlayerImpl extends VideoPlayer { | ||||||
|  |         private final float MAX_GESTURE_LENGTH = 0.75f; | ||||||
|  |  | ||||||
|         private TextView titleTextView; |         private TextView titleTextView; | ||||||
|         private TextView channelTextView; |         private TextView channelTextView; | ||||||
|         private TextView volumeTextView; |         private RelativeLayout volumeRelativeLayout; | ||||||
|         private TextView brightnessTextView; |         private ProgressBar volumeProgressBar; | ||||||
|  |         private ImageView volumeImageView; | ||||||
|  |         private RelativeLayout brightnessRelativeLayout; | ||||||
|  |         private ProgressBar brightnessProgressBar; | ||||||
|  |         private ImageView brightnessImageView; | ||||||
|         private ImageButton queueButton; |         private ImageButton queueButton; | ||||||
|         private ImageButton repeatButton; |         private ImageButton repeatButton; | ||||||
|         private ImageButton shuffleButton; |         private ImageButton shuffleButton; | ||||||
| @@ -392,6 +402,8 @@ public final class MainVideoPlayer extends AppCompatActivity | |||||||
|         private RelativeLayout windowRootLayout; |         private RelativeLayout windowRootLayout; | ||||||
|         private View secondaryControls; |         private View secondaryControls; | ||||||
|  |  | ||||||
|  |         private int maxGestureLength; | ||||||
|  |  | ||||||
|         VideoPlayerImpl(final Context context) { |         VideoPlayerImpl(final Context context) { | ||||||
|             super("VideoPlayerImpl" + MainVideoPlayer.TAG, context); |             super("VideoPlayerImpl" + MainVideoPlayer.TAG, context); | ||||||
|         } |         } | ||||||
| @@ -401,8 +413,12 @@ public final class MainVideoPlayer extends AppCompatActivity | |||||||
|             super.initViews(rootView); |             super.initViews(rootView); | ||||||
|             this.titleTextView = rootView.findViewById(R.id.titleTextView); |             this.titleTextView = rootView.findViewById(R.id.titleTextView); | ||||||
|             this.channelTextView = rootView.findViewById(R.id.channelTextView); |             this.channelTextView = rootView.findViewById(R.id.channelTextView); | ||||||
|             this.volumeTextView = rootView.findViewById(R.id.volumeTextView); |             this.volumeRelativeLayout = rootView.findViewById(R.id.volumeRelativeLayout); | ||||||
|             this.brightnessTextView = rootView.findViewById(R.id.brightnessTextView); |             this.volumeProgressBar = rootView.findViewById(R.id.volumeProgressBar); | ||||||
|  |             this.volumeImageView = rootView.findViewById(R.id.volumeImageView); | ||||||
|  |             this.brightnessRelativeLayout = rootView.findViewById(R.id.brightnessRelativeLayout); | ||||||
|  |             this.brightnessProgressBar = rootView.findViewById(R.id.brightnessProgressBar); | ||||||
|  |             this.brightnessImageView = rootView.findViewById(R.id.brightnessImageView); | ||||||
|             this.queueButton = rootView.findViewById(R.id.queueButton); |             this.queueButton = rootView.findViewById(R.id.queueButton); | ||||||
|             this.repeatButton = rootView.findViewById(R.id.repeatButton); |             this.repeatButton = rootView.findViewById(R.id.repeatButton); | ||||||
|             this.shuffleButton = rootView.findViewById(R.id.shuffleButton); |             this.shuffleButton = rootView.findViewById(R.id.shuffleButton); | ||||||
| @@ -461,6 +477,20 @@ public final class MainVideoPlayer extends AppCompatActivity | |||||||
|             toggleOrientationButton.setOnClickListener(this); |             toggleOrientationButton.setOnClickListener(this); | ||||||
|             switchBackgroundButton.setOnClickListener(this); |             switchBackgroundButton.setOnClickListener(this); | ||||||
|             switchPopupButton.setOnClickListener(this); |             switchPopupButton.setOnClickListener(this); | ||||||
|  |  | ||||||
|  |             getRootView().addOnLayoutChangeListener((view, l, t, r, b, ol, ot, or, ob) -> { | ||||||
|  |                 if (l != ol || t != ot || r != or || b != ob) { | ||||||
|  |                     // Use smaller value to be consistent between screen orientations | ||||||
|  |                     // (and to make usage easier) | ||||||
|  |                     int width = r - l, height = b - t; | ||||||
|  |                     maxGestureLength = (int) (Math.min(width, height) * MAX_GESTURE_LENGTH); | ||||||
|  |  | ||||||
|  |                     if (DEBUG) Log.d(TAG, "maxGestureLength = " + maxGestureLength); | ||||||
|  |  | ||||||
|  |                     volumeProgressBar.setMax(maxGestureLength); | ||||||
|  |                     brightnessProgressBar.setMax(maxGestureLength); | ||||||
|  |                 } | ||||||
|  |             }); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         public void minimize() { |         public void minimize() { | ||||||
| @@ -872,12 +902,28 @@ public final class MainVideoPlayer extends AppCompatActivity | |||||||
|             return channelTextView; |             return channelTextView; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         public TextView getVolumeTextView() { |         public RelativeLayout getVolumeRelativeLayout() { | ||||||
|             return volumeTextView; |             return volumeRelativeLayout; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         public TextView getBrightnessTextView() { |         public ProgressBar getVolumeProgressBar() { | ||||||
|             return brightnessTextView; |             return volumeProgressBar; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         public ImageView getVolumeImageView() { | ||||||
|  |             return volumeImageView; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         public RelativeLayout getBrightnessRelativeLayout() { | ||||||
|  |             return brightnessRelativeLayout; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         public ProgressBar getBrightnessProgressBar() { | ||||||
|  |             return brightnessProgressBar; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         public ImageView getBrightnessImageView() { | ||||||
|  |             return brightnessImageView; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         public ImageButton getRepeatButton() { |         public ImageButton getRepeatButton() { | ||||||
| @@ -887,6 +933,10 @@ public final class MainVideoPlayer extends AppCompatActivity | |||||||
|         public ImageButton getPlayPauseButton() { |         public ImageButton getPlayPauseButton() { | ||||||
|             return playPauseButton; |             return playPauseButton; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|  |         public int getMaxGestureLength() { | ||||||
|  |             return maxGestureLength; | ||||||
|  |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private class MySimpleOnGestureListener extends GestureDetector.SimpleOnGestureListener implements View.OnTouchListener { |     private class MySimpleOnGestureListener extends GestureDetector.SimpleOnGestureListener implements View.OnTouchListener { | ||||||
| @@ -930,23 +980,10 @@ public final class MainVideoPlayer extends AppCompatActivity | |||||||
|  |  | ||||||
|         private final boolean isPlayerGestureEnabled = PlayerHelper.isPlayerGestureEnabled(getApplicationContext()); |         private final boolean isPlayerGestureEnabled = PlayerHelper.isPlayerGestureEnabled(getApplicationContext()); | ||||||
|  |  | ||||||
|         private final float stepsBrightness = 15, stepBrightness = (1f / stepsBrightness), minBrightness = .01f; |         private final int maxVolume = playerImpl.getAudioReactor().getMaxVolume(); | ||||||
|         private float currentBrightness = getWindow().getAttributes().screenBrightness > 0 |  | ||||||
|                 ? getWindow().getAttributes().screenBrightness |  | ||||||
|                 : 0.5f; |  | ||||||
|  |  | ||||||
|         private int currentVolume, maxVolume = playerImpl.getAudioReactor().getMaxVolume(); |  | ||||||
|         private final float stepsVolume = 15, stepVolume = (float) Math.ceil(maxVolume / stepsVolume), minVolume = 0; |  | ||||||
|  |  | ||||||
|         private final String brightnessUnicode = new String(Character.toChars(0x2600)); |  | ||||||
|         private final String volumeUnicode = new String(Character.toChars(0x1F508)); |  | ||||||
|  |  | ||||||
|         private final int MOVEMENT_THRESHOLD = 40; |         private final int MOVEMENT_THRESHOLD = 40; | ||||||
|         private final int eventsThreshold = 8; |  | ||||||
|         private boolean triggered = false; |  | ||||||
|         private int eventsNum; |  | ||||||
|  |  | ||||||
|         // TODO: Improve video gesture controls |  | ||||||
|         @Override |         @Override | ||||||
|         public boolean onScroll(MotionEvent e1, MotionEvent e2, float distanceX, float distanceY) { |         public boolean onScroll(MotionEvent e1, MotionEvent e2, float distanceX, float distanceY) { | ||||||
|             if (!isPlayerGestureEnabled) return false; |             if (!isPlayerGestureEnabled) return false; | ||||||
| @@ -956,63 +993,77 @@ public final class MainVideoPlayer extends AppCompatActivity | |||||||
|                     ", e1.getRaw = [" + e1.getRawX() + ", " + e1.getRawY() + "]" + |                     ", e1.getRaw = [" + e1.getRawX() + ", " + e1.getRawY() + "]" + | ||||||
|                     ", e2.getRaw = [" + e2.getRawX() + ", " + e2.getRawY() + "]" + |                     ", e2.getRaw = [" + e2.getRawX() + ", " + e2.getRawY() + "]" + | ||||||
|                     ", distanceXy = [" + distanceX + ", " + distanceY + "]"); |                     ", distanceXy = [" + distanceX + ", " + distanceY + "]"); | ||||||
|             float abs = Math.abs(e2.getY() - e1.getY()); |  | ||||||
|             if (!triggered) { |             if (!isMoving && ( | ||||||
|                 triggered = abs > MOVEMENT_THRESHOLD; |                     Math.abs(e2.getY() - e1.getY()) <= MOVEMENT_THRESHOLD | ||||||
|  |                             || Math.abs(distanceX) > Math.abs(distanceY) | ||||||
|  |             ) || playerImpl.getCurrentState() == BasePlayer.STATE_COMPLETED) | ||||||
|                 return false; |                 return false; | ||||||
|             } |  | ||||||
|  |  | ||||||
|             if (eventsNum++ % eventsThreshold != 0 || playerImpl.getCurrentState() == BasePlayer.STATE_COMPLETED) return false; |  | ||||||
|             isMoving = true; |             isMoving = true; | ||||||
| //            boolean up = !((e2.getY() - e1.getY()) > 0) && distanceY > 0; // Android's origin point is on top |  | ||||||
|             boolean up = distanceY > 0; |  | ||||||
|  |  | ||||||
|  |  | ||||||
|             if (e1.getX() > playerImpl.getRootView().getWidth() / 2) { |             if (e1.getX() > playerImpl.getRootView().getWidth() / 2) { | ||||||
|                 double floor = Math.floor(up ? stepVolume : -stepVolume); |                 playerImpl.getVolumeProgressBar().incrementProgressBy((int) distanceY); | ||||||
|                 currentVolume = (int) (playerImpl.getAudioReactor().getVolume() + floor); |                 float currentProgressPercent = | ||||||
|                 if (currentVolume >= maxVolume) currentVolume = maxVolume; |                         (float) playerImpl.getVolumeProgressBar().getProgress() / playerImpl.getMaxGestureLength(); | ||||||
|                 if (currentVolume <= minVolume) currentVolume = (int) minVolume; |                 int currentVolume = (int) (maxVolume * currentProgressPercent); | ||||||
|                 playerImpl.getAudioReactor().setVolume(currentVolume); |                 playerImpl.getAudioReactor().setVolume(currentVolume); | ||||||
|  |  | ||||||
|                 currentVolume = playerImpl.getAudioReactor().getVolume(); |  | ||||||
|                 if (DEBUG) Log.d(TAG, "onScroll().volumeControl, currentVolume = " + currentVolume); |                 if (DEBUG) Log.d(TAG, "onScroll().volumeControl, currentVolume = " + currentVolume); | ||||||
|                 final String volumeText = volumeUnicode + " " + Math.round((((float) currentVolume) / maxVolume) * 100) + "%"; |  | ||||||
|                 playerImpl.getVolumeTextView().setText(volumeText); |  | ||||||
|  |  | ||||||
|                 if (playerImpl.getVolumeTextView().getVisibility() != View.VISIBLE) animateView(playerImpl.getVolumeTextView(), true, 200); |                 final int resId = | ||||||
|                 if (playerImpl.getBrightnessTextView().getVisibility() == View.VISIBLE) playerImpl.getBrightnessTextView().setVisibility(View.GONE); |                         currentProgressPercent <= 0 ? R.drawable.ic_volume_off_white_72dp | ||||||
|  |                         : currentProgressPercent < 0.25 ? R.drawable.ic_volume_mute_white_72dp | ||||||
|  |                         : currentProgressPercent < 0.75 ? R.drawable.ic_volume_down_white_72dp | ||||||
|  |                         : R.drawable.ic_volume_up_white_72dp; | ||||||
|  |  | ||||||
|  |                 playerImpl.getVolumeImageView().setImageDrawable( | ||||||
|  |                         AppCompatResources.getDrawable(getApplicationContext(), resId) | ||||||
|  |                 ); | ||||||
|  |  | ||||||
|  |                 if (playerImpl.getVolumeRelativeLayout().getVisibility() != View.VISIBLE) { | ||||||
|  |                     animateView(playerImpl.getVolumeRelativeLayout(), SCALE_AND_ALPHA, true, 200); | ||||||
|  |                 } | ||||||
|  |                 if (playerImpl.getBrightnessRelativeLayout().getVisibility() == View.VISIBLE) { | ||||||
|  |                     playerImpl.getBrightnessRelativeLayout().setVisibility(View.GONE); | ||||||
|  |                 } | ||||||
|             } else { |             } else { | ||||||
|                 WindowManager.LayoutParams lp = getWindow().getAttributes(); |                 playerImpl.getBrightnessProgressBar().incrementProgressBy((int) distanceY); | ||||||
|                 currentBrightness += up ? stepBrightness : -stepBrightness; |                 float currentProgressPercent = | ||||||
|                 if (currentBrightness >= 1f) currentBrightness = 1f; |                         (float) playerImpl.getBrightnessProgressBar().getProgress() / playerImpl.getMaxGestureLength(); | ||||||
|                 if (currentBrightness <= minBrightness) currentBrightness = minBrightness; |                 WindowManager.LayoutParams layoutParams = getWindow().getAttributes(); | ||||||
|  |                 layoutParams.screenBrightness = currentProgressPercent; | ||||||
|  |                 getWindow().setAttributes(layoutParams); | ||||||
|  |  | ||||||
|                 lp.screenBrightness = currentBrightness; |                 if (DEBUG) Log.d(TAG, "onScroll().brightnessControl, currentBrightness = " + currentProgressPercent); | ||||||
|                 getWindow().setAttributes(lp); |  | ||||||
|                 if (DEBUG) Log.d(TAG, "onScroll().brightnessControl, currentBrightness = " + currentBrightness); |  | ||||||
|                 int brightnessNormalized = Math.round(currentBrightness * 100); |  | ||||||
|  |  | ||||||
|                 final String brightnessText = brightnessUnicode + " " + (brightnessNormalized == 1 ? 0 : brightnessNormalized) + "%"; |                 final int resId = | ||||||
|                 playerImpl.getBrightnessTextView().setText(brightnessText); |                         currentProgressPercent < 0.25 ? R.drawable.ic_brightness_low_white_72dp | ||||||
|  |                         : currentProgressPercent < 0.75 ? R.drawable.ic_brightness_medium_white_72dp | ||||||
|  |                         : R.drawable.ic_brightness_high_white_72dp; | ||||||
|  |  | ||||||
|                 if (playerImpl.getBrightnessTextView().getVisibility() != View.VISIBLE) animateView(playerImpl.getBrightnessTextView(), true, 200); |                 playerImpl.getBrightnessImageView().setImageDrawable( | ||||||
|                 if (playerImpl.getVolumeTextView().getVisibility() == View.VISIBLE) playerImpl.getVolumeTextView().setVisibility(View.GONE); |                         AppCompatResources.getDrawable(getApplicationContext(), resId) | ||||||
|  |                 ); | ||||||
|  |  | ||||||
|  |                 if (playerImpl.getBrightnessRelativeLayout().getVisibility() != View.VISIBLE) { | ||||||
|  |                     animateView(playerImpl.getBrightnessRelativeLayout(), SCALE_AND_ALPHA, true, 200); | ||||||
|  |                 } | ||||||
|  |                 if (playerImpl.getVolumeRelativeLayout().getVisibility() == View.VISIBLE) { | ||||||
|  |                     playerImpl.getVolumeRelativeLayout().setVisibility(View.GONE); | ||||||
|  |                 } | ||||||
|             } |             } | ||||||
|             return true; |             return true; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         private void onScrollEnd() { |         private void onScrollEnd() { | ||||||
|             if (DEBUG) Log.d(TAG, "onScrollEnd() called"); |             if (DEBUG) Log.d(TAG, "onScrollEnd() called"); | ||||||
|             triggered = false; |  | ||||||
|             eventsNum = 0; |             if (playerImpl.getVolumeRelativeLayout().getVisibility() == View.VISIBLE) { | ||||||
|             /* if (playerImpl.getVolumeTextView().getVisibility() == View.VISIBLE) playerImpl.getVolumeTextView().setVisibility(View.GONE); |                 animateView(playerImpl.getVolumeRelativeLayout(), SCALE_AND_ALPHA, false, 200, 200); | ||||||
|             if (playerImpl.getBrightnessTextView().getVisibility() == View.VISIBLE) playerImpl.getBrightnessTextView().setVisibility(View.GONE);*/ |  | ||||||
|             if (playerImpl.getVolumeTextView().getVisibility() == View.VISIBLE) { |  | ||||||
|                 animateView(playerImpl.getVolumeTextView(), false, 200, 200); |  | ||||||
|             } |             } | ||||||
|             if (playerImpl.getBrightnessTextView().getVisibility() == View.VISIBLE) { |             if (playerImpl.getBrightnessRelativeLayout().getVisibility() == View.VISIBLE) { | ||||||
|                 animateView(playerImpl.getBrightnessTextView(), false, 200, 200); |                 animateView(playerImpl.getBrightnessRelativeLayout(), SCALE_AND_ALPHA, false, 200, 200); | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             if (playerImpl.isControlsVisible() && playerImpl.getCurrentState() == STATE_PLAYING) { |             if (playerImpl.isControlsVisible() && playerImpl.getCurrentState() == STATE_PLAYING) { | ||||||
|   | |||||||
| @@ -19,6 +19,8 @@ | |||||||
|  |  | ||||||
| package org.schabi.newpipe.player; | package org.schabi.newpipe.player; | ||||||
|  |  | ||||||
|  | import android.animation.Animator; | ||||||
|  | import android.animation.AnimatorListenerAdapter; | ||||||
| import android.annotation.SuppressLint; | import android.annotation.SuppressLint; | ||||||
| import android.app.NotificationManager; | import android.app.NotificationManager; | ||||||
| import android.app.PendingIntent; | import android.app.PendingIntent; | ||||||
| @@ -34,6 +36,7 @@ import android.os.Build; | |||||||
| import android.os.IBinder; | import android.os.IBinder; | ||||||
| import android.preference.PreferenceManager; | import android.preference.PreferenceManager; | ||||||
| import android.support.annotation.NonNull; | import android.support.annotation.NonNull; | ||||||
|  | import android.support.design.widget.FloatingActionButton; | ||||||
| import android.support.v4.app.NotificationCompat; | import android.support.v4.app.NotificationCompat; | ||||||
| import android.util.DisplayMetrics; | import android.util.DisplayMetrics; | ||||||
| import android.util.Log; | import android.util.Log; | ||||||
| @@ -41,7 +44,9 @@ import android.view.GestureDetector; | |||||||
| import android.view.Gravity; | import android.view.Gravity; | ||||||
| import android.view.MotionEvent; | import android.view.MotionEvent; | ||||||
| import android.view.View; | import android.view.View; | ||||||
|  | import android.view.ViewGroup; | ||||||
| import android.view.WindowManager; | import android.view.WindowManager; | ||||||
|  | import android.view.animation.AnticipateInterpolator; | ||||||
| import android.widget.ImageButton; | import android.widget.ImageButton; | ||||||
| import android.widget.ImageView; | import android.widget.ImageView; | ||||||
| import android.widget.PopupMenu; | import android.widget.PopupMenu; | ||||||
| @@ -104,10 +109,13 @@ public final class PopupVideoPlayer extends Service { | |||||||
|             WindowManager.LayoutParams.FLAG_KEEP_SCREEN_ON; |             WindowManager.LayoutParams.FLAG_KEEP_SCREEN_ON; | ||||||
|  |  | ||||||
|     private WindowManager windowManager; |     private WindowManager windowManager; | ||||||
|     private WindowManager.LayoutParams windowLayoutParams; |     private WindowManager.LayoutParams popupLayoutParams; | ||||||
|     private GestureDetector gestureDetector; |     private GestureDetector popupGestureDetector; | ||||||
|  |  | ||||||
|  |     private View closeOverlayView; | ||||||
|  |     private FloatingActionButton closeOverlayButton; | ||||||
|  |     private WindowManager.LayoutParams closeOverlayLayoutParams; | ||||||
|  |  | ||||||
|     private int shutdownFlingVelocity; |  | ||||||
|     private int tossFlingVelocity; |     private int tossFlingVelocity; | ||||||
|  |  | ||||||
|     private float screenWidth, screenHeight; |     private float screenWidth, screenHeight; | ||||||
| @@ -122,6 +130,7 @@ public final class PopupVideoPlayer extends Service { | |||||||
|  |  | ||||||
|     private VideoPlayerImpl playerImpl; |     private VideoPlayerImpl playerImpl; | ||||||
|     private LockManager lockManager; |     private LockManager lockManager; | ||||||
|  |     private boolean isPopupClosing = false; | ||||||
|  |  | ||||||
|     /*////////////////////////////////////////////////////////////////////////// |     /*////////////////////////////////////////////////////////////////////////// | ||||||
|     // Service-Activity Binder |     // Service-Activity Binder | ||||||
| @@ -150,7 +159,10 @@ public final class PopupVideoPlayer extends Service { | |||||||
|     public int onStartCommand(final Intent intent, int flags, int startId) { |     public int onStartCommand(final Intent intent, int flags, int startId) { | ||||||
|         if (DEBUG) |         if (DEBUG) | ||||||
|             Log.d(TAG, "onStartCommand() called with: intent = [" + intent + "], flags = [" + flags + "], startId = [" + startId + "]"); |             Log.d(TAG, "onStartCommand() called with: intent = [" + intent + "], flags = [" + flags + "], startId = [" + startId + "]"); | ||||||
|         if (playerImpl.getPlayer() == null) initPopup(); |         if (playerImpl.getPlayer() == null) { | ||||||
|  |             initPopup(); | ||||||
|  |             initPopupCloseOverlay(); | ||||||
|  |         } | ||||||
|         if (!playerImpl.isPlaying()) playerImpl.getPlayer().setPlayWhenReady(true); |         if (!playerImpl.isPlaying()) playerImpl.getPlayer().setPlayWhenReady(true); | ||||||
|  |  | ||||||
|         playerImpl.handleIntent(intent); |         playerImpl.handleIntent(intent); | ||||||
| @@ -160,15 +172,16 @@ public final class PopupVideoPlayer extends Service { | |||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public void onConfigurationChanged(Configuration newConfig) { |     public void onConfigurationChanged(Configuration newConfig) { | ||||||
|  |         if (DEBUG) Log.d(TAG, "onConfigurationChanged() called with: newConfig = [" + newConfig + "]"); | ||||||
|         updateScreenSize(); |         updateScreenSize(); | ||||||
|         updatePopupSize(windowLayoutParams.width, -1); |         updatePopupSize(popupLayoutParams.width, -1); | ||||||
|         checkPositionBounds(); |         checkPopupPositionBounds(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public void onDestroy() { |     public void onDestroy() { | ||||||
|         if (DEBUG) Log.d(TAG, "onDestroy() called"); |         if (DEBUG) Log.d(TAG, "onDestroy() called"); | ||||||
|         onClose(); |         closePopup(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
| @@ -186,7 +199,6 @@ public final class PopupVideoPlayer extends Service { | |||||||
|         View rootView = View.inflate(this, R.layout.player_popup, null); |         View rootView = View.inflate(this, R.layout.player_popup, null); | ||||||
|         playerImpl.setup(rootView); |         playerImpl.setup(rootView); | ||||||
|  |  | ||||||
|         shutdownFlingVelocity = PlayerHelper.getShutdownFlingVelocity(this); |  | ||||||
|         tossFlingVelocity = PlayerHelper.getTossFlingVelocity(this); |         tossFlingVelocity = PlayerHelper.getTossFlingVelocity(this); | ||||||
|  |  | ||||||
|         updateScreenSize(); |         updateScreenSize(); | ||||||
| @@ -200,27 +212,52 @@ public final class PopupVideoPlayer extends Service { | |||||||
|                 WindowManager.LayoutParams.TYPE_PHONE : |                 WindowManager.LayoutParams.TYPE_PHONE : | ||||||
|                 WindowManager.LayoutParams.TYPE_APPLICATION_OVERLAY; |                 WindowManager.LayoutParams.TYPE_APPLICATION_OVERLAY; | ||||||
|  |  | ||||||
|         windowLayoutParams = new WindowManager.LayoutParams( |         popupLayoutParams = new WindowManager.LayoutParams( | ||||||
|                 (int) popupWidth, (int) getMinimumVideoHeight(popupWidth), |                 (int) popupWidth, (int) getMinimumVideoHeight(popupWidth), | ||||||
|                 layoutParamType, |                 layoutParamType, | ||||||
|                 IDLE_WINDOW_FLAGS, |                 IDLE_WINDOW_FLAGS, | ||||||
|                 PixelFormat.TRANSLUCENT); |                 PixelFormat.TRANSLUCENT); | ||||||
|         windowLayoutParams.gravity = Gravity.LEFT | Gravity.TOP; |         popupLayoutParams.gravity = Gravity.LEFT | Gravity.TOP; | ||||||
|         windowLayoutParams.softInputMode = WindowManager.LayoutParams.SOFT_INPUT_ADJUST_RESIZE; |         popupLayoutParams.softInputMode = WindowManager.LayoutParams.SOFT_INPUT_ADJUST_RESIZE; | ||||||
|  |  | ||||||
|         int centerX = (int) (screenWidth / 2f - popupWidth / 2f); |         int centerX = (int) (screenWidth / 2f - popupWidth / 2f); | ||||||
|         int centerY = (int) (screenHeight / 2f - popupHeight / 2f); |         int centerY = (int) (screenHeight / 2f - popupHeight / 2f); | ||||||
|         windowLayoutParams.x = popupRememberSizeAndPos ? sharedPreferences.getInt(POPUP_SAVED_X, centerX) : centerX; |         popupLayoutParams.x = popupRememberSizeAndPos ? sharedPreferences.getInt(POPUP_SAVED_X, centerX) : centerX; | ||||||
|         windowLayoutParams.y = popupRememberSizeAndPos ? sharedPreferences.getInt(POPUP_SAVED_Y, centerY) : centerY; |         popupLayoutParams.y = popupRememberSizeAndPos ? sharedPreferences.getInt(POPUP_SAVED_Y, centerY) : centerY; | ||||||
|  |  | ||||||
|         checkPositionBounds(); |         checkPopupPositionBounds(); | ||||||
|  |  | ||||||
|         MySimpleOnGestureListener listener = new MySimpleOnGestureListener(); |         PopupWindowGestureListener listener = new PopupWindowGestureListener(); | ||||||
|         gestureDetector = new GestureDetector(this, listener); |         popupGestureDetector = new GestureDetector(this, listener); | ||||||
|         rootView.setOnTouchListener(listener); |         rootView.setOnTouchListener(listener); | ||||||
|         playerImpl.getLoadingPanel().setMinimumWidth(windowLayoutParams.width); |  | ||||||
|         playerImpl.getLoadingPanel().setMinimumHeight(windowLayoutParams.height); |         playerImpl.getLoadingPanel().setMinimumWidth(popupLayoutParams.width); | ||||||
|         windowManager.addView(rootView, windowLayoutParams); |         playerImpl.getLoadingPanel().setMinimumHeight(popupLayoutParams.height); | ||||||
|  |         windowManager.addView(rootView, popupLayoutParams); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @SuppressLint("RtlHardcoded") | ||||||
|  |     private void initPopupCloseOverlay() { | ||||||
|  |         if (DEBUG) Log.d(TAG, "initPopupCloseOverlay() called"); | ||||||
|  |         closeOverlayView = View.inflate(this, R.layout.player_popup_close_overlay, null); | ||||||
|  |         closeOverlayButton = closeOverlayView.findViewById(R.id.closeButton); | ||||||
|  |  | ||||||
|  |         final int layoutParamType = Build.VERSION.SDK_INT < android.os.Build.VERSION_CODES.O ? | ||||||
|  |                 WindowManager.LayoutParams.TYPE_PHONE : | ||||||
|  |                 WindowManager.LayoutParams.TYPE_APPLICATION_OVERLAY; | ||||||
|  |         final int flags = WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE | WindowManager.LayoutParams.FLAG_NOT_TOUCHABLE | ||||||
|  |                 | WindowManager.LayoutParams.FLAG_ALT_FOCUSABLE_IM; | ||||||
|  |  | ||||||
|  |         closeOverlayLayoutParams = new WindowManager.LayoutParams( | ||||||
|  |                 ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.MATCH_PARENT, | ||||||
|  |                 layoutParamType, | ||||||
|  |                 flags, | ||||||
|  |                 PixelFormat.TRANSLUCENT); | ||||||
|  |         closeOverlayLayoutParams.gravity = Gravity.LEFT | Gravity.TOP; | ||||||
|  |         closeOverlayLayoutParams.softInputMode = WindowManager.LayoutParams.SOFT_INPUT_ADJUST_RESIZE; | ||||||
|  |  | ||||||
|  |         closeOverlayButton.setVisibility(View.GONE); | ||||||
|  |         windowManager.addView(closeOverlayView, closeOverlayLayoutParams); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /*////////////////////////////////////////////////////////////////////////// |     /*////////////////////////////////////////////////////////////////////////// | ||||||
| @@ -280,44 +317,105 @@ public final class PopupVideoPlayer extends Service { | |||||||
|     // Misc |     // Misc | ||||||
|     //////////////////////////////////////////////////////////////////////////*/ |     //////////////////////////////////////////////////////////////////////////*/ | ||||||
|  |  | ||||||
|     public void onClose() { |     public void closePopup() { | ||||||
|         if (DEBUG) Log.d(TAG, "onClose() called"); |         if (DEBUG) Log.d(TAG, "closePopup() called, isPopupClosing = " + isPopupClosing); | ||||||
|  |         if (isPopupClosing) return; | ||||||
|  |         isPopupClosing = true; | ||||||
|  |  | ||||||
|         if (playerImpl != null) { |         if (playerImpl != null) { | ||||||
|             if (playerImpl.getRootView() != null) { |             if (playerImpl.getRootView() != null) { | ||||||
|                 windowManager.removeView(playerImpl.getRootView()); |                 windowManager.removeView(playerImpl.getRootView()); | ||||||
|                 playerImpl.setRootView(null); |  | ||||||
|             } |             } | ||||||
|  |             playerImpl.setRootView(null); | ||||||
|             playerImpl.stopActivityBinding(); |             playerImpl.stopActivityBinding(); | ||||||
|             playerImpl.destroy(); |             playerImpl.destroy(); | ||||||
|  |             playerImpl = null; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|  |         mBinder = null; | ||||||
|         if (lockManager != null) lockManager.releaseWifiAndCpu(); |         if (lockManager != null) lockManager.releaseWifiAndCpu(); | ||||||
|         if (notificationManager != null) notificationManager.cancel(NOTIFICATION_ID); |         if (notificationManager != null) notificationManager.cancel(NOTIFICATION_ID); | ||||||
|         mBinder = null; |  | ||||||
|         playerImpl = null; |         animateOverlayAndFinishService(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private void animateOverlayAndFinishService() { | ||||||
|  |         final int targetTranslationY = (int) (closeOverlayButton.getRootView().getHeight() - closeOverlayButton.getY()); | ||||||
|  |  | ||||||
|  |         closeOverlayButton.animate().setListener(null).cancel(); | ||||||
|  |         closeOverlayButton.animate() | ||||||
|  |                 .setInterpolator(new AnticipateInterpolator()) | ||||||
|  |                 .translationY(targetTranslationY) | ||||||
|  |                 .setDuration(400) | ||||||
|  |                 .setListener(new AnimatorListenerAdapter() { | ||||||
|  |                     @Override | ||||||
|  |                     public void onAnimationCancel(Animator animation) { | ||||||
|  |                         end(); | ||||||
|  |                     } | ||||||
|  |  | ||||||
|  |                     @Override | ||||||
|  |                     public void onAnimationEnd(Animator animation) { | ||||||
|  |                         end(); | ||||||
|  |                     } | ||||||
|  |  | ||||||
|  |                     private void end() { | ||||||
|  |                         windowManager.removeView(closeOverlayView); | ||||||
|  |  | ||||||
|                         stopForeground(true); |                         stopForeground(true); | ||||||
|                         stopSelf(); |                         stopSelf(); | ||||||
|                     } |                     } | ||||||
|  |                 }).start(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     /*////////////////////////////////////////////////////////////////////////// |     /*////////////////////////////////////////////////////////////////////////// | ||||||
|     // Utils |     // Utils | ||||||
|     //////////////////////////////////////////////////////////////////////////*/ |     //////////////////////////////////////////////////////////////////////////*/ | ||||||
|  |  | ||||||
|     private void checkPositionBounds() { |     /** | ||||||
|         if (windowLayoutParams.x > screenWidth - windowLayoutParams.width) |      * @see #checkPopupPositionBounds(float, float) | ||||||
|             windowLayoutParams.x = (int) (screenWidth - windowLayoutParams.width); |      */ | ||||||
|         if (windowLayoutParams.x < 0) windowLayoutParams.x = 0; |     @SuppressWarnings("UnusedReturnValue") | ||||||
|         if (windowLayoutParams.y > screenHeight - windowLayoutParams.height) |     private boolean checkPopupPositionBounds() { | ||||||
|             windowLayoutParams.y = (int) (screenHeight - windowLayoutParams.height); |         return checkPopupPositionBounds(screenWidth, screenHeight); | ||||||
|         if (windowLayoutParams.y < 0) windowLayoutParams.y = 0; |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Check if {@link #popupLayoutParams}' position is within a arbitrary boundary that goes from (0,0) to (boundaryWidth,boundaryHeight). | ||||||
|  |      * <p> | ||||||
|  |      * If it's out of these boundaries, {@link #popupLayoutParams}' position is changed and {@code true} is returned | ||||||
|  |      * to represent this change. | ||||||
|  |      * | ||||||
|  |      * @return if the popup was out of bounds and have been moved back to it | ||||||
|  |      */ | ||||||
|  |     private boolean checkPopupPositionBounds(final float boundaryWidth, final float boundaryHeight) { | ||||||
|  |         if (DEBUG) { | ||||||
|  |             Log.d(TAG, "checkPopupPositionBounds() called with: boundaryWidth = [" + boundaryWidth + "], boundaryHeight = [" + boundaryHeight + "]"); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if (popupLayoutParams.x < 0) { | ||||||
|  |             popupLayoutParams.x = 0; | ||||||
|  |             return true; | ||||||
|  |         } else if (popupLayoutParams.x > boundaryWidth - popupLayoutParams.width) { | ||||||
|  |             popupLayoutParams.x = (int) (boundaryWidth - popupLayoutParams.width); | ||||||
|  |             return true; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if (popupLayoutParams.y < 0) { | ||||||
|  |             popupLayoutParams.y = 0; | ||||||
|  |             return true; | ||||||
|  |         } else if (popupLayoutParams.y > boundaryHeight - popupLayoutParams.height) { | ||||||
|  |             popupLayoutParams.y = (int) (boundaryHeight - popupLayoutParams.height); | ||||||
|  |             return true; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         return false; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private void savePositionAndSize() { |     private void savePositionAndSize() { | ||||||
|         SharedPreferences sharedPreferences = PreferenceManager.getDefaultSharedPreferences(PopupVideoPlayer.this); |         SharedPreferences sharedPreferences = PreferenceManager.getDefaultSharedPreferences(PopupVideoPlayer.this); | ||||||
|         sharedPreferences.edit().putInt(POPUP_SAVED_X, windowLayoutParams.x).apply(); |         sharedPreferences.edit().putInt(POPUP_SAVED_X, popupLayoutParams.x).apply(); | ||||||
|         sharedPreferences.edit().putInt(POPUP_SAVED_Y, windowLayoutParams.y).apply(); |         sharedPreferences.edit().putInt(POPUP_SAVED_Y, popupLayoutParams.y).apply(); | ||||||
|         sharedPreferences.edit().putFloat(POPUP_SAVED_WIDTH, windowLayoutParams.width).apply(); |         sharedPreferences.edit().putFloat(POPUP_SAVED_WIDTH, popupLayoutParams.width).apply(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private float getMinimumVideoHeight(float width) { |     private float getMinimumVideoHeight(float width) { | ||||||
| @@ -352,13 +450,13 @@ public final class PopupVideoPlayer extends Service { | |||||||
|         if (height == -1) height = (int) getMinimumVideoHeight(width); |         if (height == -1) height = (int) getMinimumVideoHeight(width); | ||||||
|         else height = (int) (height > maximumHeight ? maximumHeight : height < minimumHeight ? minimumHeight : height); |         else height = (int) (height > maximumHeight ? maximumHeight : height < minimumHeight ? minimumHeight : height); | ||||||
|  |  | ||||||
|         windowLayoutParams.width = width; |         popupLayoutParams.width = width; | ||||||
|         windowLayoutParams.height = height; |         popupLayoutParams.height = height; | ||||||
|         popupWidth = width; |         popupWidth = width; | ||||||
|         popupHeight = height; |         popupHeight = height; | ||||||
|  |  | ||||||
|         if (DEBUG) Log.d(TAG, "updatePopupSize() updated values:  width = [" + width + "], height = [" + height + "]"); |         if (DEBUG) Log.d(TAG, "updatePopupSize() updated values:  width = [" + width + "], height = [" + height + "]"); | ||||||
|         windowManager.updateViewLayout(playerImpl.getRootView(), windowLayoutParams); |         windowManager.updateViewLayout(playerImpl.getRootView(), popupLayoutParams); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     protected void setRepeatModeRemote(final RemoteViews remoteViews, final int repeatMode) { |     protected void setRepeatModeRemote(final RemoteViews remoteViews, final int repeatMode) { | ||||||
| @@ -380,10 +478,10 @@ public final class PopupVideoPlayer extends Service { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     private void updateWindowFlags(final int flags) { |     private void updateWindowFlags(final int flags) { | ||||||
|         if (windowLayoutParams == null || windowManager == null || playerImpl == null) return; |         if (popupLayoutParams == null || windowManager == null || playerImpl == null) return; | ||||||
|  |  | ||||||
|         windowLayoutParams.flags = flags; |         popupLayoutParams.flags = flags; | ||||||
|         windowManager.updateViewLayout(playerImpl.getRootView(), windowLayoutParams); |         windowManager.updateViewLayout(playerImpl.getRootView(), popupLayoutParams); | ||||||
|     } |     } | ||||||
|     /////////////////////////////////////////////////////////////////////////// |     /////////////////////////////////////////////////////////////////////////// | ||||||
|  |  | ||||||
| @@ -393,6 +491,7 @@ public final class PopupVideoPlayer extends Service { | |||||||
|         private ImageView videoPlayPause; |         private ImageView videoPlayPause; | ||||||
|  |  | ||||||
|         private View extraOptionsView; |         private View extraOptionsView; | ||||||
|  |         private View closingOverlayView; | ||||||
|  |  | ||||||
|         @Override |         @Override | ||||||
|         public void handleIntent(Intent intent) { |         public void handleIntent(Intent intent) { | ||||||
| @@ -413,12 +512,18 @@ public final class PopupVideoPlayer extends Service { | |||||||
|             fullScreenButton = rootView.findViewById(R.id.fullScreenButton); |             fullScreenButton = rootView.findViewById(R.id.fullScreenButton); | ||||||
|             fullScreenButton.setOnClickListener(v -> onFullScreenButtonClicked()); |             fullScreenButton.setOnClickListener(v -> onFullScreenButtonClicked()); | ||||||
|             videoPlayPause = rootView.findViewById(R.id.videoPlayPause); |             videoPlayPause = rootView.findViewById(R.id.videoPlayPause); | ||||||
|             videoPlayPause.setOnClickListener(this::onPlayPauseButtonPressed); |  | ||||||
|  |  | ||||||
|             extraOptionsView = rootView.findViewById(R.id.extraOptionsView); |             extraOptionsView = rootView.findViewById(R.id.extraOptionsView); | ||||||
|  |             closingOverlayView = rootView.findViewById(R.id.closingOverlay); | ||||||
|             rootView.addOnLayoutChangeListener(this); |             rootView.addOnLayoutChangeListener(this); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|  |         @Override | ||||||
|  |         public void initListeners() { | ||||||
|  |             super.initListeners(); | ||||||
|  |             videoPlayPause.setOnClickListener(v -> onPlayPause()); | ||||||
|  |         } | ||||||
|  |  | ||||||
|         @Override |         @Override | ||||||
|         protected void setupSubtitleView(@NonNull SubtitleView view, |         protected void setupSubtitleView(@NonNull SubtitleView view, | ||||||
|                                          final float captionScale, |                                          final float captionScale, | ||||||
| @@ -429,10 +534,6 @@ public final class PopupVideoPlayer extends Service { | |||||||
|             view.setStyle(captionStyle); |             view.setStyle(captionStyle); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         private void onPlayPauseButtonPressed(View ib) { |  | ||||||
|             onPlayPause(); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         @Override |         @Override | ||||||
|         public void onLayoutChange(final View view, int left, int top, int right, int bottom, |         public void onLayoutChange(final View view, int left, int top, int right, int bottom, | ||||||
|                                    int oldLeft, int oldTop, int oldRight, int oldBottom) { |                                    int oldLeft, int oldTop, int oldRight, int oldBottom) { | ||||||
| @@ -476,7 +577,7 @@ public final class PopupVideoPlayer extends Service { | |||||||
|                 intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK); |                 intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK); | ||||||
|             } |             } | ||||||
|             context.startActivity(intent); |             context.startActivity(intent); | ||||||
|             onClose(); |             closePopup(); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         @Override |         @Override | ||||||
| @@ -634,7 +735,7 @@ public final class PopupVideoPlayer extends Service { | |||||||
|         @Override |         @Override | ||||||
|         public void onPlaybackShutdown() { |         public void onPlaybackShutdown() { | ||||||
|             super.onPlaybackShutdown(); |             super.onPlaybackShutdown(); | ||||||
|             onClose(); |             closePopup(); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         /*////////////////////////////////////////////////////////////////////////// |         /*////////////////////////////////////////////////////////////////////////// | ||||||
| @@ -660,7 +761,7 @@ public final class PopupVideoPlayer extends Service { | |||||||
|             if (DEBUG) Log.d(TAG, "onBroadcastReceived() called with: intent = [" + intent + "]"); |             if (DEBUG) Log.d(TAG, "onBroadcastReceived() called with: intent = [" + intent + "]"); | ||||||
|             switch (intent.getAction()) { |             switch (intent.getAction()) { | ||||||
|                 case ACTION_CLOSE: |                 case ACTION_CLOSE: | ||||||
|                     onClose(); |                     closePopup(); | ||||||
|                     break; |                     break; | ||||||
|                 case ACTION_PLAY_PAUSE: |                 case ACTION_PLAY_PAUSE: | ||||||
|                     onPlayPause(); |                     onPlayPause(); | ||||||
| @@ -791,12 +892,15 @@ public final class PopupVideoPlayer extends Service { | |||||||
|         public TextView getResizingIndicator() { |         public TextView getResizingIndicator() { | ||||||
|             return resizingIndicator; |             return resizingIndicator; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|  |         public View getClosingOverlayView() { | ||||||
|  |             return closingOverlayView; | ||||||
|  |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private class MySimpleOnGestureListener extends GestureDetector.SimpleOnGestureListener implements View.OnTouchListener { |     private class PopupWindowGestureListener extends GestureDetector.SimpleOnGestureListener implements View.OnTouchListener { | ||||||
|         private int initialPopupX, initialPopupY; |         private int initialPopupX, initialPopupY; | ||||||
|         private boolean isMoving; |         private boolean isMoving; | ||||||
|  |  | ||||||
|         private boolean isResizing; |         private boolean isResizing; | ||||||
|  |  | ||||||
|         @Override |         @Override | ||||||
| @@ -832,10 +936,15 @@ public final class PopupVideoPlayer extends Service { | |||||||
|         @Override |         @Override | ||||||
|         public boolean onDown(MotionEvent e) { |         public boolean onDown(MotionEvent e) { | ||||||
|             if (DEBUG) Log.d(TAG, "onDown() called with: e = [" + e + "]"); |             if (DEBUG) Log.d(TAG, "onDown() called with: e = [" + e + "]"); | ||||||
|             initialPopupX = windowLayoutParams.x; |  | ||||||
|             initialPopupY = windowLayoutParams.y; |             // Fix popup position when the user touch it, it may have the wrong one | ||||||
|             popupWidth = windowLayoutParams.width; |             // because the soft input is visible (the draggable area is currently resized). | ||||||
|             popupHeight = windowLayoutParams.height; |             checkPopupPositionBounds(closeOverlayView.getWidth(), closeOverlayView.getHeight()); | ||||||
|  |  | ||||||
|  |             initialPopupX = popupLayoutParams.x; | ||||||
|  |             initialPopupY = popupLayoutParams.y; | ||||||
|  |             popupWidth = popupLayoutParams.width; | ||||||
|  |             popupHeight = popupLayoutParams.height; | ||||||
|             return super.onDown(e); |             return super.onDown(e); | ||||||
|         } |         } | ||||||
|  |  | ||||||
| @@ -843,20 +952,22 @@ public final class PopupVideoPlayer extends Service { | |||||||
|         public void onLongPress(MotionEvent e) { |         public void onLongPress(MotionEvent e) { | ||||||
|             if (DEBUG) Log.d(TAG, "onLongPress() called with: e = [" + e + "]"); |             if (DEBUG) Log.d(TAG, "onLongPress() called with: e = [" + e + "]"); | ||||||
|             updateScreenSize(); |             updateScreenSize(); | ||||||
|             checkPositionBounds(); |             checkPopupPositionBounds(); | ||||||
|             updatePopupSize((int) screenWidth, -1); |             updatePopupSize((int) screenWidth, -1); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         @Override |         @Override | ||||||
|         public boolean onScroll(MotionEvent e1, MotionEvent e2, float distanceX, float distanceY) { |         public boolean onScroll(MotionEvent initialEvent, MotionEvent movingEvent, float distanceX, float distanceY) { | ||||||
|             if (isResizing || playerImpl == null) return super.onScroll(e1, e2, distanceX, distanceY); |             if (isResizing || playerImpl == null) return super.onScroll(initialEvent, movingEvent, distanceX, distanceY); | ||||||
|  |  | ||||||
|  |             if (!isMoving) { | ||||||
|  |                 animateView(closeOverlayButton, true, 200); | ||||||
|  |             } | ||||||
|  |  | ||||||
|             if (playerImpl.getCurrentState() != BasePlayer.STATE_BUFFERING |  | ||||||
|                     && (!isMoving || playerImpl.getControlsRoot().getAlpha() != 1f)) playerImpl.showControls(0); |  | ||||||
|             isMoving = true; |             isMoving = true; | ||||||
|  |  | ||||||
|             float diffX = (int) (e2.getRawX() - e1.getRawX()), posX = (int) (initialPopupX + diffX); |             float diffX = (int) (movingEvent.getRawX() - initialEvent.getRawX()), posX = (int) (initialPopupX + diffX); | ||||||
|             float diffY = (int) (e2.getRawY() - e1.getRawY()), posY = (int) (initialPopupY + diffY); |             float diffY = (int) (movingEvent.getRawY() - initialEvent.getRawY()), posY = (int) (initialPopupY + diffY); | ||||||
|  |  | ||||||
|             if (posX > (screenWidth - popupWidth)) posX = (int) (screenWidth - popupWidth); |             if (posX > (screenWidth - popupWidth)) posX = (int) (screenWidth - popupWidth); | ||||||
|             else if (posX < 0) posX = 0; |             else if (posX < 0) posX = 0; | ||||||
| @@ -864,26 +975,49 @@ public final class PopupVideoPlayer extends Service { | |||||||
|             if (posY > (screenHeight - popupHeight)) posY = (int) (screenHeight - popupHeight); |             if (posY > (screenHeight - popupHeight)) posY = (int) (screenHeight - popupHeight); | ||||||
|             else if (posY < 0) posY = 0; |             else if (posY < 0) posY = 0; | ||||||
|  |  | ||||||
|             windowLayoutParams.x = (int) posX; |             popupLayoutParams.x = (int) posX; | ||||||
|             windowLayoutParams.y = (int) posY; |             popupLayoutParams.y = (int) posY; | ||||||
|  |  | ||||||
|  |             final View closingOverlayView = playerImpl.getClosingOverlayView(); | ||||||
|  |             if (isInsideClosingRadius(movingEvent)) { | ||||||
|  |                 if (closingOverlayView.getVisibility() == View.GONE) { | ||||||
|  |                     animateView(closingOverlayView, true, 250); | ||||||
|  |                 } | ||||||
|  |             } else { | ||||||
|  |                 if (closingOverlayView.getVisibility() == View.VISIBLE) { | ||||||
|  |                     animateView(closingOverlayView, false, 0); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |  | ||||||
|             //noinspection PointlessBooleanExpression |             //noinspection PointlessBooleanExpression | ||||||
|             if (DEBUG && false) Log.d(TAG, "PopupVideoPlayer.onScroll = " + |             if (DEBUG && false) { | ||||||
|                     ", e1.getRaw = [" + e1.getRawX() + ", " + e1.getRawY() + "]" + |                 Log.d(TAG, "PopupVideoPlayer.onScroll = " + | ||||||
|                     ", e2.getRaw = [" + e2.getRawX() + ", " + e2.getRawY() + "]" + |                         ", e1.getRaw = [" + initialEvent.getRawX() + ", " + initialEvent.getRawY() + "]" + ", e1.getX,Y = [" + initialEvent.getX() + ", " + initialEvent.getY() + "]" + | ||||||
|                     ", distanceXy = [" + distanceX + ", " + distanceY + "]" + |                         ", e2.getRaw = [" + movingEvent.getRawX() + ", " + movingEvent.getRawY() + "]" + ", e2.getX,Y = [" + movingEvent.getX() + ", " + movingEvent.getY() + "]" + | ||||||
|                     ", posXy = [" + posX + ", " + posY + "]" + |                         ", distanceX,Y = [" + distanceX + ", " + distanceY + "]" + | ||||||
|                     ", popupWh = [" + popupWidth + " x " + popupHeight + "]"); |                         ", posX,Y = [" + posX + ", " + posY + "]" + | ||||||
|             windowManager.updateViewLayout(playerImpl.getRootView(), windowLayoutParams); |                         ", popupW,H = [" + popupWidth + " x " + popupHeight + "]"); | ||||||
|  |             } | ||||||
|  |             windowManager.updateViewLayout(playerImpl.getRootView(), popupLayoutParams); | ||||||
|             return true; |             return true; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         private void onScrollEnd() { |         private void onScrollEnd(MotionEvent event) { | ||||||
|             if (DEBUG) Log.d(TAG, "onScrollEnd() called"); |             if (DEBUG) Log.d(TAG, "onScrollEnd() called"); | ||||||
|             if (playerImpl == null) return; |             if (playerImpl == null) return; | ||||||
|             if (playerImpl.isControlsVisible() && playerImpl.getCurrentState() == STATE_PLAYING) { |             if (playerImpl.isControlsVisible() && playerImpl.getCurrentState() == STATE_PLAYING) { | ||||||
|                 playerImpl.hideControls(DEFAULT_CONTROLS_DURATION, DEFAULT_CONTROLS_HIDE_TIME); |                 playerImpl.hideControls(DEFAULT_CONTROLS_DURATION, DEFAULT_CONTROLS_HIDE_TIME); | ||||||
|             } |             } | ||||||
|  |  | ||||||
|  |             if (isInsideClosingRadius(event)) { | ||||||
|  |                 closePopup(); | ||||||
|  |             } else { | ||||||
|  |                 animateView(playerImpl.getClosingOverlayView(), false, 0); | ||||||
|  |  | ||||||
|  |                 if (!isPopupClosing) { | ||||||
|  |                     animateView(closeOverlayButton, false, 200); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         @Override |         @Override | ||||||
| @@ -893,14 +1027,11 @@ public final class PopupVideoPlayer extends Service { | |||||||
|  |  | ||||||
|             final float absVelocityX = Math.abs(velocityX); |             final float absVelocityX = Math.abs(velocityX); | ||||||
|             final float absVelocityY = Math.abs(velocityY); |             final float absVelocityY = Math.abs(velocityY); | ||||||
|             if (absVelocityX > shutdownFlingVelocity) { |             if (Math.max(absVelocityX, absVelocityY) > tossFlingVelocity) { | ||||||
|                 onClose(); |                 if (absVelocityX > tossFlingVelocity) popupLayoutParams.x = (int) velocityX; | ||||||
|                 return true; |                 if (absVelocityY > tossFlingVelocity) popupLayoutParams.y = (int) velocityY; | ||||||
|             } else if (Math.max(absVelocityX, absVelocityY) > tossFlingVelocity) { |                 checkPopupPositionBounds(); | ||||||
|                 if (absVelocityX > tossFlingVelocity) windowLayoutParams.x = (int) velocityX; |                 windowManager.updateViewLayout(playerImpl.getRootView(), popupLayoutParams); | ||||||
|                 if (absVelocityY > tossFlingVelocity) windowLayoutParams.y = (int) velocityY; |  | ||||||
|                 checkPositionBounds(); |  | ||||||
|                 windowManager.updateViewLayout(playerImpl.getRootView(), windowLayoutParams); |  | ||||||
|                 return true; |                 return true; | ||||||
|             } |             } | ||||||
|             return false; |             return false; | ||||||
| @@ -908,7 +1039,7 @@ public final class PopupVideoPlayer extends Service { | |||||||
|  |  | ||||||
|         @Override |         @Override | ||||||
|         public boolean onTouch(View v, MotionEvent event) { |         public boolean onTouch(View v, MotionEvent event) { | ||||||
|             gestureDetector.onTouchEvent(event); |             popupGestureDetector.onTouchEvent(event); | ||||||
|             if (playerImpl == null) return false; |             if (playerImpl == null) return false; | ||||||
|             if (event.getPointerCount() == 2 && !isResizing) { |             if (event.getPointerCount() == 2 && !isResizing) { | ||||||
|                 if (DEBUG) Log.d(TAG, "onTouch() 2 finger pointer detected, enabling resizing."); |                 if (DEBUG) Log.d(TAG, "onTouch() 2 finger pointer detected, enabling resizing."); | ||||||
| @@ -931,7 +1062,7 @@ public final class PopupVideoPlayer extends Service { | |||||||
|                     Log.d(TAG, "onTouch() ACTION_UP > v = [" + v + "],  e1.getRaw = [" + event.getRawX() + ", " + event.getRawY() + "]"); |                     Log.d(TAG, "onTouch() ACTION_UP > v = [" + v + "],  e1.getRaw = [" + event.getRawX() + ", " + event.getRawY() + "]"); | ||||||
|                 if (isMoving) { |                 if (isMoving) { | ||||||
|                     isMoving = false; |                     isMoving = false; | ||||||
|                     onScrollEnd(); |                     onScrollEnd(event); | ||||||
|                 } |                 } | ||||||
|  |  | ||||||
|                 if (isResizing) { |                 if (isResizing) { | ||||||
| @@ -939,8 +1070,11 @@ public final class PopupVideoPlayer extends Service { | |||||||
|                     animateView(playerImpl.getResizingIndicator(), false, 100, 0); |                     animateView(playerImpl.getResizingIndicator(), false, 100, 0); | ||||||
|                     playerImpl.changeState(playerImpl.getCurrentState()); |                     playerImpl.changeState(playerImpl.getCurrentState()); | ||||||
|                 } |                 } | ||||||
|  |  | ||||||
|  |                 if (!isPopupClosing) { | ||||||
|                     savePositionAndSize(); |                     savePositionAndSize(); | ||||||
|                 } |                 } | ||||||
|  |             } | ||||||
|  |  | ||||||
|             v.performClick(); |             v.performClick(); | ||||||
|             return true; |             return true; | ||||||
| @@ -955,13 +1089,13 @@ public final class PopupVideoPlayer extends Service { | |||||||
|             final float diff = Math.abs(firstPointerX - secondPointerX); |             final float diff = Math.abs(firstPointerX - secondPointerX); | ||||||
|             if (firstPointerX > secondPointerX) { |             if (firstPointerX > secondPointerX) { | ||||||
|                 // second pointer is the anchor (the leftmost pointer) |                 // second pointer is the anchor (the leftmost pointer) | ||||||
|                 windowLayoutParams.x = (int) (event.getRawX() - diff); |                 popupLayoutParams.x = (int) (event.getRawX() - diff); | ||||||
|             } else { |             } else { | ||||||
|                 // first pointer is the anchor |                 // first pointer is the anchor | ||||||
|                 windowLayoutParams.x = (int) event.getRawX(); |                 popupLayoutParams.x = (int) event.getRawX(); | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             checkPositionBounds(); |             checkPopupPositionBounds(); | ||||||
|             updateScreenSize(); |             updateScreenSize(); | ||||||
|  |  | ||||||
|             final int width = (int) Math.min(screenWidth, diff); |             final int width = (int) Math.min(screenWidth, diff); | ||||||
| @@ -969,5 +1103,29 @@ public final class PopupVideoPlayer extends Service { | |||||||
|  |  | ||||||
|             return true; |             return true; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|  |         /*////////////////////////////////////////////////////////////////////////// | ||||||
|  |         // Utils | ||||||
|  |         //////////////////////////////////////////////////////////////////////////*/ | ||||||
|  |  | ||||||
|  |         private int distanceFromCloseButton(MotionEvent popupMotionEvent) { | ||||||
|  |             final int closeOverlayButtonX = closeOverlayButton.getLeft() + closeOverlayButton.getWidth() / 2; | ||||||
|  |             final int closeOverlayButtonY = closeOverlayButton.getTop() + closeOverlayButton.getHeight() / 2; | ||||||
|  |  | ||||||
|  |             float fingerX = popupLayoutParams.x + popupMotionEvent.getX(); | ||||||
|  |             float fingerY = popupLayoutParams.y + popupMotionEvent.getY(); | ||||||
|  |  | ||||||
|  |             return (int) Math.sqrt(Math.pow(closeOverlayButtonX - fingerX, 2) + Math.pow(closeOverlayButtonY - fingerY, 2)); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         private float getClosingRadius() { | ||||||
|  |             final int buttonRadius = closeOverlayButton.getWidth() / 2; | ||||||
|  |             // 20% wider than the button itself | ||||||
|  |             return buttonRadius * 1.2f; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         private boolean isInsideClosingRadius(MotionEvent popupMotionEvent) { | ||||||
|  |             return distanceFromCloseButton(popupMotionEvent) <= getClosingRadius(); | ||||||
|  |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
| @@ -16,6 +16,7 @@ import android.util.Log; | |||||||
| import android.view.Menu; | import android.view.Menu; | ||||||
| import android.view.MenuItem; | import android.view.MenuItem; | ||||||
| import android.view.View; | import android.view.View; | ||||||
|  | import android.view.ViewGroup; | ||||||
| import android.widget.ImageButton; | import android.widget.ImageButton; | ||||||
| import android.widget.LinearLayout; | import android.widget.LinearLayout; | ||||||
| import android.widget.PopupMenu; | import android.widget.PopupMenu; | ||||||
| @@ -562,6 +563,12 @@ public abstract class ServicePlayerActivity extends AppCompatActivity | |||||||
|         if (player != null) { |         if (player != null) { | ||||||
|             progressLiveSync.setClickable(!player.isLiveEdge()); |             progressLiveSync.setClickable(!player.isLiveEdge()); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|  |         // this will make shure progressCurrentTime has the same width as progressEndTime | ||||||
|  |         final ViewGroup.LayoutParams endTimeParams = progressEndTime.getLayoutParams(); | ||||||
|  |         final ViewGroup.LayoutParams currentTimeParams = progressCurrentTime.getLayoutParams(); | ||||||
|  |         currentTimeParams.width = progressEndTime.getWidth(); | ||||||
|  |         progressCurrentTime.setLayoutParams(currentTimeParams); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|   | |||||||
| @@ -251,10 +251,6 @@ public class PlayerHelper { | |||||||
|         return true; |         return true; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public static int getShutdownFlingVelocity(@NonNull final Context context) { |  | ||||||
|         return 6000; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public static int getTossFlingVelocity(@NonNull final Context context) { |     public static int getTossFlingVelocity(@NonNull final Context context) { | ||||||
|         return 2500; |         return 2500; | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -6,6 +6,7 @@ import android.util.Log; | |||||||
|  |  | ||||||
| import org.reactivestreams.Subscriber; | import org.reactivestreams.Subscriber; | ||||||
| import org.reactivestreams.Subscription; | import org.reactivestreams.Subscription; | ||||||
|  | import org.schabi.newpipe.BuildConfig; | ||||||
| import org.schabi.newpipe.player.playqueue.events.AppendEvent; | import org.schabi.newpipe.player.playqueue.events.AppendEvent; | ||||||
| import org.schabi.newpipe.player.playqueue.events.ErrorEvent; | import org.schabi.newpipe.player.playqueue.events.ErrorEvent; | ||||||
| import org.schabi.newpipe.player.playqueue.events.InitEvent; | import org.schabi.newpipe.player.playqueue.events.InitEvent; | ||||||
| @@ -41,7 +42,7 @@ import io.reactivex.subjects.BehaviorSubject; | |||||||
| public abstract class PlayQueue implements Serializable { | public abstract class PlayQueue implements Serializable { | ||||||
|     private final String TAG = "PlayQueue@" + Integer.toHexString(hashCode()); |     private final String TAG = "PlayQueue@" + Integer.toHexString(hashCode()); | ||||||
|  |  | ||||||
|     public static final boolean DEBUG = true; |     public static final boolean DEBUG = !BuildConfig.BUILD_TYPE.equals("release"); | ||||||
|  |  | ||||||
|     private ArrayList<PlayQueueItem> backup; |     private ArrayList<PlayQueueItem> backup; | ||||||
|     private ArrayList<PlayQueueItem> streams; |     private ArrayList<PlayQueueItem> streams; | ||||||
|   | |||||||
| @@ -15,7 +15,8 @@ public enum UserAction { | |||||||
|     REQUESTED_CHANNEL("requested channel"), |     REQUESTED_CHANNEL("requested channel"), | ||||||
|     REQUESTED_PLAYLIST("requested playlist"), |     REQUESTED_PLAYLIST("requested playlist"), | ||||||
|     REQUESTED_KIOSK("requested kiosk"), |     REQUESTED_KIOSK("requested kiosk"), | ||||||
|     DELETE_FROM_HISTORY("delete from history"); |     DELETE_FROM_HISTORY("delete from history"), | ||||||
|  |     PLAY_STREAM("Play stream"); | ||||||
|  |  | ||||||
|  |  | ||||||
|     private final String message; |     private final String message; | ||||||
|   | |||||||
| @@ -0,0 +1,41 @@ | |||||||
|  | package org.schabi.newpipe.settings; | ||||||
|  |  | ||||||
|  | import android.app.Activity; | ||||||
|  | import android.app.AlertDialog; | ||||||
|  | import android.content.Context; | ||||||
|  | import android.content.DialogInterface; | ||||||
|  | import android.support.annotation.NonNull; | ||||||
|  | import android.support.annotation.Nullable; | ||||||
|  | import android.view.View; | ||||||
|  | import android.widget.TextView; | ||||||
|  |  | ||||||
|  | import org.schabi.newpipe.R; | ||||||
|  | import org.schabi.newpipe.extractor.stream.StreamInfoItem; | ||||||
|  |  | ||||||
|  | public class AddTabsDialog { | ||||||
|  |     private final AlertDialog dialog; | ||||||
|  |  | ||||||
|  |     public AddTabsDialog(@NonNull final Context context, | ||||||
|  |                           @NonNull final String title, | ||||||
|  |                           @NonNull final String[] commands, | ||||||
|  |                           @NonNull final DialogInterface.OnClickListener actions) { | ||||||
|  |  | ||||||
|  |         final View bannerView = View.inflate(context, R.layout.dialog_title, null); | ||||||
|  |         bannerView.setSelected(true); | ||||||
|  |  | ||||||
|  |         TextView titleView = bannerView.findViewById(R.id.itemTitleView); | ||||||
|  |         titleView.setText(title); | ||||||
|  |  | ||||||
|  |         TextView detailsView = bannerView.findViewById(R.id.itemAdditionalDetails); | ||||||
|  |         detailsView.setVisibility(View.GONE); | ||||||
|  |  | ||||||
|  |         dialog = new AlertDialog.Builder(context) | ||||||
|  |                 .setCustomTitle(bannerView) | ||||||
|  |                 .setItems(commands, actions) | ||||||
|  |                 .create(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public void show() { | ||||||
|  |         dialog.show(); | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,291 @@ | |||||||
|  | package org.schabi.newpipe.settings; | ||||||
|  |  | ||||||
|  | import android.app.Dialog; | ||||||
|  | import android.content.SharedPreferences; | ||||||
|  | import android.content.res.ColorStateList; | ||||||
|  | import android.os.Bundle; | ||||||
|  | import android.support.annotation.NonNull; | ||||||
|  | import android.support.annotation.Nullable; | ||||||
|  | import android.support.design.widget.FloatingActionButton; | ||||||
|  | import android.support.v4.app.Fragment; | ||||||
|  | import android.support.v7.app.AppCompatActivity; | ||||||
|  | import android.support.v7.widget.CardView; | ||||||
|  | import android.support.v7.widget.LinearLayoutManager; | ||||||
|  | import android.support.v7.widget.RecyclerView; | ||||||
|  | import android.support.v7.widget.helper.ItemTouchHelper; | ||||||
|  | import android.util.TypedValue; | ||||||
|  | import android.view.LayoutInflater; | ||||||
|  | import android.view.MotionEvent; | ||||||
|  | import android.view.View; | ||||||
|  | import android.view.ViewGroup; | ||||||
|  | import android.widget.ImageView; | ||||||
|  | import android.widget.TextView; | ||||||
|  |  | ||||||
|  | import org.schabi.newpipe.R; | ||||||
|  | import org.schabi.newpipe.util.ThemeHelper; | ||||||
|  |  | ||||||
|  | import java.util.ArrayList; | ||||||
|  | import java.util.Arrays; | ||||||
|  | import java.util.List; | ||||||
|  |  | ||||||
|  | public class ChoseTabsFragment extends Fragment { | ||||||
|  |  | ||||||
|  |     public ChoseTabsFragment.SelectedTabsAdapter selectedTabsAdapter; | ||||||
|  |  | ||||||
|  |     RecyclerView selectedTabsView; | ||||||
|  |  | ||||||
|  |     List<String> selectedTabs = new ArrayList<>(); | ||||||
|  |     private String saveString; | ||||||
|  |     public String[] availableTabs = new String[7]; | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public View onCreateView(@NonNull LayoutInflater inflater, ViewGroup container, | ||||||
|  |                              Bundle savedInstanceState) { | ||||||
|  |         ((AppCompatActivity)getContext()).getSupportActionBar().setTitle(R.string.main_page_content); | ||||||
|  |         return inflater.inflate(R.layout.fragment_chose_tabs, container, false); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public void onViewCreated(@NonNull View rootView, @Nullable Bundle savedInstanceState) { | ||||||
|  |         super.onViewCreated(rootView, savedInstanceState); | ||||||
|  |  | ||||||
|  |         tabNames(); | ||||||
|  |         initUsedTabs(); | ||||||
|  |         initButton(rootView); | ||||||
|  |  | ||||||
|  |         selectedTabsView = rootView.findViewById(R.id.usedTabs); | ||||||
|  |         selectedTabsView.setLayoutManager(new LinearLayoutManager(getContext())); | ||||||
|  |         selectedTabsAdapter = new ChoseTabsFragment.SelectedTabsAdapter(); | ||||||
|  |  | ||||||
|  |  | ||||||
|  |         ItemTouchHelper itemTouchHelper = new ItemTouchHelper(getItemTouchCallback()); | ||||||
|  |         itemTouchHelper.attachToRecyclerView(selectedTabsView); | ||||||
|  |         selectedTabsAdapter.setOnItemSelectedListener(itemTouchHelper); | ||||||
|  |  | ||||||
|  |         selectedTabsView.setAdapter(selectedTabsAdapter); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private void saveChanges() { | ||||||
|  |         StringBuilder save = new StringBuilder(); | ||||||
|  |         if(selectedTabs.size()==0) { | ||||||
|  |             save = new StringBuilder("0"); | ||||||
|  |         } else { | ||||||
|  |             for(String s: selectedTabs) { | ||||||
|  |                 save.append(s); | ||||||
|  |                 save.append("\n"); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         saveString = save.toString(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public void onPause() { | ||||||
|  |         saveChanges(); | ||||||
|  |         SharedPreferences sharedPreferences  = android.preference.PreferenceManager.getDefaultSharedPreferences(getContext()); | ||||||
|  |         SharedPreferences.Editor editor = sharedPreferences.edit(); | ||||||
|  |         editor.putString("saveUsedTabs", saveString); | ||||||
|  |         editor.commit(); | ||||||
|  |         super.onPause(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private void initUsedTabs() { | ||||||
|  |         String save = android.preference.PreferenceManager.getDefaultSharedPreferences(getContext()).getString("saveUsedTabs", "1\tTrending\t0\n2\n4\n"); | ||||||
|  |         String tabs[] = save.trim().split("\n"); | ||||||
|  |         selectedTabs.addAll(Arrays.asList(tabs)); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private void tabNames() { | ||||||
|  |         availableTabs[0] = getString(R.string.blank_page_summary); | ||||||
|  |         availableTabs[1] = getString(R.string.kiosk_page_summary); | ||||||
|  |         availableTabs[2] = getString(R.string.subscription_page_summary); | ||||||
|  |         availableTabs[3] = getString(R.string.feed_page_summary); | ||||||
|  |         availableTabs[4] = getString(R.string.tab_bookmarks); | ||||||
|  |         availableTabs[5] = getString(R.string.title_activity_history); | ||||||
|  |         availableTabs[6] = getString(R.string.channel_page_summary); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private void initButton(View rootView) { | ||||||
|  |         FloatingActionButton fab = rootView.findViewById(R.id.floatingActionButton); | ||||||
|  |         fab.setImageResource(ThemeHelper.getIconByAttr(R.attr.ic_add, getContext())); | ||||||
|  |         fab.setOnClickListener(v -> { | ||||||
|  |             Dialog.OnClickListener onClickListener = (dialog, which) -> addTab(which); | ||||||
|  |  | ||||||
|  |             new AddTabsDialog(getContext(), | ||||||
|  |                     getString(R.string.tab_chose), | ||||||
|  |                     availableTabs, | ||||||
|  |                     onClickListener) | ||||||
|  |                     .show(); | ||||||
|  |         }); | ||||||
|  |  | ||||||
|  |         TypedValue typedValue = new TypedValue(); | ||||||
|  |         getActivity().getTheme().resolveAttribute(R.attr.colorPrimary, typedValue, true); | ||||||
|  |         int color = typedValue.data; | ||||||
|  |         fab.setBackgroundTintList(ColorStateList.valueOf(color)); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     private void addTab(int position) { | ||||||
|  |         if(position==6) { | ||||||
|  |             SelectChannelFragment selectChannelFragment = new SelectChannelFragment(); | ||||||
|  |             selectChannelFragment.setOnSelectedLisener((String url, String name, int service) -> { | ||||||
|  |                 selectedTabs.add(position+"\t"+url+"\t"+name+"\t"+service); | ||||||
|  |                 selectedTabsAdapter.notifyDataSetChanged(); | ||||||
|  |                 saveChanges(); | ||||||
|  |             }); | ||||||
|  |             selectChannelFragment.show(getFragmentManager(), "select_channel"); | ||||||
|  |         } else if(position==1) { | ||||||
|  |             SelectKioskFragment selectKioskFragment = new SelectKioskFragment(); | ||||||
|  |             selectKioskFragment.setOnSelectedLisener((String kioskId, int service_id) -> { | ||||||
|  |                 selectedTabs.add(position+"\t"+kioskId+"\t"+service_id); | ||||||
|  |                 selectedTabsAdapter.notifyDataSetChanged(); | ||||||
|  |                 saveChanges(); | ||||||
|  |             }); | ||||||
|  |             selectKioskFragment.show(getFragmentManager(), "select_kiosk"); | ||||||
|  |         } else { | ||||||
|  |             selectedTabs.add(String.valueOf(position)); | ||||||
|  |             selectedTabsAdapter.notifyDataSetChanged(); | ||||||
|  |             saveChanges(); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public class SelectedTabsAdapter extends RecyclerView.Adapter<ChoseTabsFragment.SelectedTabsAdapter.TabViewHolder>{ | ||||||
|  |         private ItemTouchHelper itemTouchHelper; | ||||||
|  |  | ||||||
|  |         public void setOnItemSelectedListener(ItemTouchHelper mItemTouchHelper) { | ||||||
|  |             itemTouchHelper = mItemTouchHelper; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         public void swapItems(int fromPosition, int toPosition) { | ||||||
|  |             String temp = selectedTabs.get(fromPosition); | ||||||
|  |             selectedTabs.set(fromPosition, selectedTabs.get(toPosition)); | ||||||
|  |             selectedTabs.set(toPosition, temp); | ||||||
|  |             notifyItemMoved(fromPosition, toPosition); | ||||||
|  |             saveChanges(); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         @Override | ||||||
|  |         public ChoseTabsFragment.SelectedTabsAdapter.TabViewHolder onCreateViewHolder(ViewGroup parent, int viewType) { | ||||||
|  |  | ||||||
|  |             LayoutInflater inflater = LayoutInflater.from(getContext()); | ||||||
|  |             View view = inflater.inflate(R.layout.viewholder_chose_tabs, parent, false); | ||||||
|  |             return new ChoseTabsFragment.SelectedTabsAdapter.TabViewHolder(view); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         @Override | ||||||
|  |         public void onBindViewHolder(@NonNull ChoseTabsFragment.SelectedTabsAdapter.TabViewHolder holder, int position) { | ||||||
|  |             holder.bind(position, holder); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         @Override | ||||||
|  |         public int getItemCount() { | ||||||
|  |             return selectedTabs.size(); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         class TabViewHolder extends RecyclerView.ViewHolder { | ||||||
|  |  | ||||||
|  |             TextView text; | ||||||
|  |             View view; | ||||||
|  |             CardView cardView; | ||||||
|  |             ImageView handle; | ||||||
|  |  | ||||||
|  |             public TabViewHolder(View itemView) { | ||||||
|  |                 super(itemView); | ||||||
|  |  | ||||||
|  |                 text = itemView.findViewById(R.id.tabName); | ||||||
|  |                 cardView = itemView.findViewById(R.id.layoutCard); | ||||||
|  |                 handle = itemView.findViewById(R.id.handle); | ||||||
|  |                 view = itemView; | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             void bind(int position, TabViewHolder holder) { | ||||||
|  |                 handle.setImageResource(ThemeHelper.getIconByAttr(R.attr.drag_handle, getContext())); | ||||||
|  |                 handle.setOnTouchListener(getOnTouchListener(holder)); | ||||||
|  |  | ||||||
|  |                 view.setOnLongClickListener(getOnLongClickListener(holder)); | ||||||
|  |  | ||||||
|  |                 if(selectedTabs.get(position).startsWith("6\t")) { | ||||||
|  |                     String channelInfo[] = selectedTabs.get(position).split("\t"); | ||||||
|  |                     String channelName = ""; | ||||||
|  |                     if (channelInfo.length == 4) channelName = channelInfo[2]; | ||||||
|  |                     String textToSet = availableTabs[6] + ": " + channelName; | ||||||
|  |                     text.setText(textToSet); | ||||||
|  |                 } else if(selectedTabs.get(position).startsWith("1\t")) { | ||||||
|  |                     String kioskInfo[] = selectedTabs.get(position).split("\t"); | ||||||
|  |                     String kioskName = ""; | ||||||
|  |                     if (kioskInfo.length == 3) kioskName = kioskInfo[1]; | ||||||
|  |                     String textToSet = availableTabs[1] + ": " + kioskName; | ||||||
|  |                     text.setText(textToSet); | ||||||
|  |                 } else { | ||||||
|  |                     text.setText(availableTabs[Integer.parseInt(selectedTabs.get(position))]); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             private View.OnTouchListener getOnTouchListener(final RecyclerView.ViewHolder item) { | ||||||
|  |                 return (view, motionEvent) -> { | ||||||
|  |                     view.performClick(); | ||||||
|  |                     if (motionEvent.getActionMasked() == MotionEvent.ACTION_DOWN) { | ||||||
|  |                         if(itemTouchHelper != null) itemTouchHelper.startDrag(item); | ||||||
|  |                     } | ||||||
|  |                     return false; | ||||||
|  |                 }; | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             private View.OnLongClickListener getOnLongClickListener(TabViewHolder holder) { | ||||||
|  |                 return (view) -> { | ||||||
|  |                     if(itemTouchHelper != null) itemTouchHelper.startSwipe(holder); | ||||||
|  |                     return false; | ||||||
|  |                 }; | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private ItemTouchHelper.SimpleCallback getItemTouchCallback() { | ||||||
|  |         return new ItemTouchHelper.SimpleCallback(ItemTouchHelper.UP | ItemTouchHelper.DOWN, | ||||||
|  |                 ItemTouchHelper.START | ItemTouchHelper.END) { | ||||||
|  |             @Override | ||||||
|  |             public int interpolateOutOfBoundsScroll(RecyclerView recyclerView, int viewSize, | ||||||
|  |                                                     int viewSizeOutOfBounds, int totalSize, | ||||||
|  |                                                     long msSinceStartScroll) { | ||||||
|  |                 final int standardSpeed = super.interpolateOutOfBoundsScroll(recyclerView, viewSize, | ||||||
|  |                         viewSizeOutOfBounds, totalSize, msSinceStartScroll); | ||||||
|  |                 final int minimumAbsVelocity = Math.max(12, | ||||||
|  |                         Math.abs(standardSpeed)); | ||||||
|  |                 return minimumAbsVelocity * (int) Math.signum(viewSizeOutOfBounds); | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             @Override | ||||||
|  |             public boolean onMove(RecyclerView recyclerView, RecyclerView.ViewHolder source, | ||||||
|  |                                   RecyclerView.ViewHolder target) { | ||||||
|  |                 if (source.getItemViewType() != target.getItemViewType() || | ||||||
|  |                         selectedTabsAdapter == null) { | ||||||
|  |                     return false; | ||||||
|  |                 } | ||||||
|  |  | ||||||
|  |                 final int sourceIndex = source.getAdapterPosition(); | ||||||
|  |                 final int targetIndex = target.getAdapterPosition(); | ||||||
|  |                 selectedTabsAdapter.swapItems(sourceIndex, targetIndex); | ||||||
|  |                 return true; | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             @Override | ||||||
|  |             public boolean isLongPressDragEnabled() { | ||||||
|  |                 return false; | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             @Override | ||||||
|  |             public boolean isItemViewSwipeEnabled() { | ||||||
|  |                 return false; | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             @Override | ||||||
|  |             public void onSwiped(RecyclerView.ViewHolder viewHolder, int swipeDir) { | ||||||
|  |                 int position = viewHolder.getAdapterPosition(); | ||||||
|  |                 selectedTabs.remove(position); | ||||||
|  |                 selectedTabsAdapter.notifyItemRemoved(position); | ||||||
|  |                 saveChanges(); | ||||||
|  |             } | ||||||
|  |         }; | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -9,6 +9,7 @@ import android.os.Bundle; | |||||||
| import android.preference.PreferenceManager; | import android.preference.PreferenceManager; | ||||||
| import android.support.annotation.NonNull; | import android.support.annotation.NonNull; | ||||||
| import android.support.annotation.Nullable; | import android.support.annotation.Nullable; | ||||||
|  | import android.support.v4.app.Fragment; | ||||||
| import android.support.v7.preference.ListPreference; | import android.support.v7.preference.ListPreference; | ||||||
| import android.support.v7.preference.Preference; | import android.support.v7.preference.Preference; | ||||||
| import android.util.Log; | import android.util.Log; | ||||||
| @@ -20,15 +21,12 @@ import com.nostra13.universalimageloader.core.ImageLoader; | |||||||
| import org.schabi.newpipe.R; | import org.schabi.newpipe.R; | ||||||
| import org.schabi.newpipe.extractor.NewPipe; | import org.schabi.newpipe.extractor.NewPipe; | ||||||
| import org.schabi.newpipe.extractor.StreamingService; | import org.schabi.newpipe.extractor.StreamingService; | ||||||
| import org.schabi.newpipe.extractor.exceptions.ExtractionException; |  | ||||||
| import org.schabi.newpipe.report.ErrorActivity; | import org.schabi.newpipe.report.ErrorActivity; | ||||||
| import org.schabi.newpipe.report.UserAction; | import org.schabi.newpipe.report.UserAction; | ||||||
| import org.schabi.newpipe.util.Constants; |  | ||||||
| import org.schabi.newpipe.util.FilePickerActivityHelper; | import org.schabi.newpipe.util.FilePickerActivityHelper; | ||||||
| import org.schabi.newpipe.util.KioskTranslator; | import org.schabi.newpipe.util.KioskTranslator; | ||||||
| import org.schabi.newpipe.util.ZipHelper; | import org.schabi.newpipe.util.ZipHelper; | ||||||
|  |  | ||||||
| import java.io.BufferedInputStream; |  | ||||||
| import java.io.BufferedOutputStream; | import java.io.BufferedOutputStream; | ||||||
| import java.io.File; | import java.io.File; | ||||||
| import java.io.FileInputStream; | import java.io.FileInputStream; | ||||||
| @@ -42,11 +40,8 @@ import java.util.Date; | |||||||
| import java.util.Locale; | import java.util.Locale; | ||||||
| import java.util.Map; | import java.util.Map; | ||||||
| import java.util.zip.ZipFile; | import java.util.zip.ZipFile; | ||||||
| import java.util.zip.ZipInputStream; |  | ||||||
| import java.util.zip.ZipOutputStream; | import java.util.zip.ZipOutputStream; | ||||||
|  |  | ||||||
| import static android.content.Context.MODE_PRIVATE; |  | ||||||
|  |  | ||||||
| public class ContentSettingsFragment extends BasePreferenceFragment { | public class ContentSettingsFragment extends BasePreferenceFragment { | ||||||
|  |  | ||||||
|     private static final int REQUEST_IMPORT_PATH = 8945; |     private static final int REQUEST_IMPORT_PATH = 8945; | ||||||
| @@ -98,68 +93,6 @@ public class ContentSettingsFragment extends BasePreferenceFragment { | |||||||
|  |  | ||||||
|         addPreferencesFromResource(R.xml.content_settings); |         addPreferencesFromResource(R.xml.content_settings); | ||||||
|  |  | ||||||
|         final ListPreference mainPageContentPref =  (ListPreference) findPreference(getString(R.string.main_page_content_key)); |  | ||||||
|         mainPageContentPref.setOnPreferenceChangeListener((Preference preference, Object newValueO) -> { |  | ||||||
|             final String newValue = newValueO.toString(); |  | ||||||
|  |  | ||||||
|             final String mainPrefOldValue = |  | ||||||
|                     defaultPreferences.getString(getString(R.string.main_page_content_key), "blank_page"); |  | ||||||
|             final String mainPrefOldSummary = getMainPagePrefSummery(mainPrefOldValue, mainPageContentPref); |  | ||||||
|  |  | ||||||
|             if(newValue.equals(getString(R.string.kiosk_page_key))) { |  | ||||||
|                 SelectKioskFragment selectKioskFragment = new SelectKioskFragment(); |  | ||||||
|                 selectKioskFragment.setOnSelectedLisener((String kioskId, int service_id) -> { |  | ||||||
|                     defaultPreferences.edit() |  | ||||||
|                             .putInt(getString(R.string.main_page_selected_service), service_id).apply(); |  | ||||||
|                     defaultPreferences.edit() |  | ||||||
|                             .putString(getString(R.string.main_page_selectd_kiosk_id), kioskId).apply(); |  | ||||||
|                     String serviceName = ""; |  | ||||||
|                     try { |  | ||||||
|                         serviceName = NewPipe.getService(service_id).getServiceInfo().getName(); |  | ||||||
|                     } catch (ExtractionException e) { |  | ||||||
|                         onError(e); |  | ||||||
|                     } |  | ||||||
|                     String kioskName = KioskTranslator.getTranslatedKioskName(kioskId, |  | ||||||
|                             getContext()); |  | ||||||
|  |  | ||||||
|                     String summary = |  | ||||||
|                             String.format(getString(R.string.service_kiosk_string), |  | ||||||
|                                     serviceName, |  | ||||||
|                                     kioskName); |  | ||||||
|  |  | ||||||
|                     mainPageContentPref.setSummary(summary); |  | ||||||
|                 }); |  | ||||||
|                 selectKioskFragment.setOnCancelListener(() -> { |  | ||||||
|                     mainPageContentPref.setSummary(mainPrefOldSummary); |  | ||||||
|                     mainPageContentPref.setValue(mainPrefOldValue); |  | ||||||
|                 }); |  | ||||||
|                 selectKioskFragment.show(getFragmentManager(), "select_kiosk"); |  | ||||||
|             } else if(newValue.equals(getString(R.string.channel_page_key))) { |  | ||||||
|                 SelectChannelFragment selectChannelFragment = new SelectChannelFragment(); |  | ||||||
|                 selectChannelFragment.setOnSelectedLisener((String url, String name, int service) -> { |  | ||||||
|                     defaultPreferences.edit() |  | ||||||
|                             .putInt(getString(R.string.main_page_selected_service), service).apply(); |  | ||||||
|                     defaultPreferences.edit() |  | ||||||
|                             .putString(getString(R.string.main_page_selected_channel_url), url).apply(); |  | ||||||
|                     defaultPreferences.edit() |  | ||||||
|                             .putString(getString(R.string.main_page_selected_channel_name), name).apply(); |  | ||||||
|  |  | ||||||
|                     mainPageContentPref.setSummary(name); |  | ||||||
|                 }); |  | ||||||
|                 selectChannelFragment.setOnCancelListener(() -> { |  | ||||||
|                     mainPageContentPref.setSummary(mainPrefOldSummary); |  | ||||||
|                     mainPageContentPref.setValue(mainPrefOldValue); |  | ||||||
|                 }); |  | ||||||
|                 selectChannelFragment.show(getFragmentManager(), "select_channel"); |  | ||||||
|             } else { |  | ||||||
|                 mainPageContentPref.setSummary(getMainPageSummeryByKey(newValue)); |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             defaultPreferences.edit().putBoolean(Constants.KEY_MAIN_PAGE_CHANGE, true).apply(); |  | ||||||
|  |  | ||||||
|             return true; |  | ||||||
|         }); |  | ||||||
|  |  | ||||||
|         Preference importDataPreference = findPreference(getString(R.string.import_data)); |         Preference importDataPreference = findPreference(getString(R.string.import_data)); | ||||||
|         importDataPreference.setOnPreferenceClickListener((Preference p) -> { |         importDataPreference.setOnPreferenceClickListener((Preference p) -> { | ||||||
|             Intent i = new Intent(getActivity(), FilePickerActivityHelper.class) |             Intent i = new Intent(getActivity(), FilePickerActivityHelper.class) | ||||||
| @@ -349,66 +282,6 @@ public class ContentSettingsFragment extends BasePreferenceFragment { | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |  | ||||||
|     public void onResume() { |  | ||||||
|         super.onResume(); |  | ||||||
|  |  | ||||||
|         final String mainPageContentKey = getString(R.string.main_page_content_key); |  | ||||||
|         final Preference mainPagePref = findPreference(getString(R.string.main_page_content_key)); |  | ||||||
|         final String bpk = getString(R.string.blank_page_key); |  | ||||||
|         if(defaultPreferences.getString(mainPageContentKey, bpk) |  | ||||||
|                 .equals(getString(R.string.channel_page_key))) { |  | ||||||
|             mainPagePref.setSummary(defaultPreferences.getString(getString(R.string.main_page_selected_channel_name), "error")); |  | ||||||
|         } else if(defaultPreferences.getString(mainPageContentKey, bpk) |  | ||||||
|                 .equals(getString(R.string.kiosk_page_key))) { |  | ||||||
|             try { |  | ||||||
|                 StreamingService service = NewPipe.getService( |  | ||||||
|                         defaultPreferences.getInt( |  | ||||||
|                                 getString(R.string.main_page_selected_service), 0)); |  | ||||||
|  |  | ||||||
|                 String kioskName = KioskTranslator.getTranslatedKioskName( |  | ||||||
|                         defaultPreferences.getString( |  | ||||||
|                                 getString(R.string.main_page_selectd_kiosk_id), "Trending"), |  | ||||||
|                         getContext()); |  | ||||||
|  |  | ||||||
|                 String summary = |  | ||||||
|                         String.format(getString(R.string.service_kiosk_string), |  | ||||||
|                                 service.getServiceInfo().getName(), |  | ||||||
|                                 kioskName); |  | ||||||
|  |  | ||||||
|                 mainPagePref.setSummary(summary); |  | ||||||
|             } catch (Exception e) { |  | ||||||
|                 onError(e); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /*////////////////////////////////////////////////////////////////////////// |  | ||||||
|     // Utils |  | ||||||
|     //////////////////////////////////////////////////////////////////////////*/ |  | ||||||
|     private String getMainPagePrefSummery(final String mainPrefOldValue, final ListPreference mainPageContentPref) { |  | ||||||
|         if(mainPrefOldValue.equals(getString(R.string.channel_page_key))) { |  | ||||||
|             return defaultPreferences.getString(getString(R.string.main_page_selected_channel_name), "error"); |  | ||||||
|         } else { |  | ||||||
|             return mainPageContentPref.getSummary().toString(); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     private int getMainPageSummeryByKey(final String key) { |  | ||||||
|         if(key.equals(getString(R.string.blank_page_key))) { |  | ||||||
|             return R.string.blank_page_summary; |  | ||||||
|         } else if(key.equals(getString(R.string.kiosk_page_key))) { |  | ||||||
|             return R.string.kiosk_page_summary; |  | ||||||
|         } else if(key.equals(getString(R.string.feed_page_key))) { |  | ||||||
|             return R.string.feed_page_summary; |  | ||||||
|         } else if(key.equals(getString(R.string.subscription_page_key))) { |  | ||||||
|             return R.string.subscription_page_summary; |  | ||||||
|         } else if(key.equals(getString(R.string.channel_page_key))) { |  | ||||||
|             return R.string.channel_page_summary; |  | ||||||
|         } |  | ||||||
|         return R.string.blank_page_summary; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /*////////////////////////////////////////////////////////////////////////// |     /*////////////////////////////////////////////////////////////////////////// | ||||||
|     // Error |     // Error | ||||||
|     //////////////////////////////////////////////////////////////////////////*/ |     //////////////////////////////////////////////////////////////////////////*/ | ||||||
|   | |||||||
| @@ -71,7 +71,7 @@ public class NewPipeSettings { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     public static File getVideoDownloadFolder(Context context) { |     public static File getVideoDownloadFolder(Context context) { | ||||||
|         return getFolder(context, R.string.download_path_key, Environment.DIRECTORY_MOVIES); |         return getDir(context, R.string.download_path_key, Environment.DIRECTORY_MOVIES); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public static String getVideoDownloadPath(Context context) { |     public static String getVideoDownloadPath(Context context) { | ||||||
| @@ -81,7 +81,7 @@ public class NewPipeSettings { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     public static File getAudioDownloadFolder(Context context) { |     public static File getAudioDownloadFolder(Context context) { | ||||||
|         return getFolder(context, R.string.download_path_audio_key, Environment.DIRECTORY_MUSIC); |         return getDir(context, R.string.download_path_audio_key, Environment.DIRECTORY_MUSIC); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public static String getAudioDownloadPath(Context context) { |     public static String getAudioDownloadPath(Context context) { | ||||||
| @@ -90,21 +90,37 @@ public class NewPipeSettings { | |||||||
|         return prefs.getString(key, Environment.DIRECTORY_MUSIC); |         return prefs.getString(key, Environment.DIRECTORY_MUSIC); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private static File getFolder(Context context, int keyID, String defaultDirectoryName) { |     private static File getDir(Context context, int keyID, String defaultDirectoryName) { | ||||||
|         SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(context); |         SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(context); | ||||||
|         final String key = context.getString(keyID); |         final String key = context.getString(keyID); | ||||||
|         String downloadPath = prefs.getString(key, null); |         String downloadPath = prefs.getString(key, null); | ||||||
|         if ((downloadPath != null) && (!downloadPath.isEmpty())) return new File(downloadPath.trim()); |         if ((downloadPath != null) && (!downloadPath.isEmpty())) return new File(downloadPath.trim()); | ||||||
|  |  | ||||||
|         final File folder = getFolder(defaultDirectoryName); |         final File dir = getDir(defaultDirectoryName); | ||||||
|         SharedPreferences.Editor spEditor = prefs.edit(); |         SharedPreferences.Editor spEditor = prefs.edit(); | ||||||
|         spEditor.putString(key, new File(folder, "NewPipe").getAbsolutePath()); |         spEditor.putString(key, getNewPipeChildFolderPathForDir(dir)); | ||||||
|         spEditor.apply(); |         spEditor.apply(); | ||||||
|         return folder; |         return dir; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @NonNull |     @NonNull | ||||||
|     private static File getFolder(String defaultDirectoryName) { |     private static File getDir(String defaultDirectoryName) { | ||||||
|         return new File(Environment.getExternalStorageDirectory(), defaultDirectoryName); |         return new File(Environment.getExternalStorageDirectory(), defaultDirectoryName); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     public static void resetDownloadFolders(Context context) { | ||||||
|  |         SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(context); | ||||||
|  |         resetDownloadFolder(prefs, context.getString(R.string.download_path_audio_key), Environment.DIRECTORY_MUSIC); | ||||||
|  |         resetDownloadFolder(prefs, context.getString(R.string.download_path_key), Environment.DIRECTORY_MOVIES); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private static void resetDownloadFolder(SharedPreferences prefs, String key, String defaultDirectoryName) { | ||||||
|  |         SharedPreferences.Editor spEditor = prefs.edit(); | ||||||
|  |         spEditor.putString(key, getNewPipeChildFolderPathForDir(getDir(defaultDirectoryName))); | ||||||
|  |         spEditor.apply(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private static String getNewPipeChildFolderPathForDir(File dir) { | ||||||
|  |         return new File(dir, "NewPipe").getAbsolutePath(); | ||||||
|  |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -74,7 +74,7 @@ public final class ExtractorHelper { | |||||||
|         return Single.fromCallable(() -> |         return Single.fromCallable(() -> | ||||||
|             SearchInfo.getInfo(NewPipe.getService(serviceId), |             SearchInfo.getInfo(NewPipe.getService(serviceId), | ||||||
|                     NewPipe.getService(serviceId) |                     NewPipe.getService(serviceId) | ||||||
|                         .getSearchQIHFactory() |                         .getSearchQHFactory() | ||||||
|                         .fromQuery(searchString, contentFilter, sortFilter), |                         .fromQuery(searchString, contentFilter, sortFilter), | ||||||
|                     contentCountry)); |                     contentCountry)); | ||||||
|     } |     } | ||||||
| @@ -89,7 +89,7 @@ public final class ExtractorHelper { | |||||||
|         return Single.fromCallable(() -> |         return Single.fromCallable(() -> | ||||||
|                 SearchInfo.getMoreItems(NewPipe.getService(serviceId), |                 SearchInfo.getMoreItems(NewPipe.getService(serviceId), | ||||||
|                         NewPipe.getService(serviceId) |                         NewPipe.getService(serviceId) | ||||||
|                             .getSearchQIHFactory() |                             .getSearchQHFactory() | ||||||
|                             .fromQuery(searchString, contentFilter, sortFilter), |                             .fromQuery(searchString, contentFilter, sortFilter), | ||||||
|                         contentCountry, |                         contentCountry, | ||||||
|                         pageUrl)); |                         pageUrl)); | ||||||
|   | |||||||
| @@ -35,4 +35,17 @@ public class KioskTranslator { | |||||||
|                 return kioskId; |                 return kioskId; | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     public static int getKioskIcons(String kioskId, Context c) { | ||||||
|  |         switch(kioskId) { | ||||||
|  |             case "Trending": | ||||||
|  |                 return ThemeHelper.resolveResourceIdFromAttr(c, R.attr.ic_hot); | ||||||
|  |             case "Top 50": | ||||||
|  |                 return ThemeHelper.resolveResourceIdFromAttr(c, R.attr.ic_hot); | ||||||
|  |             case "New & hot": | ||||||
|  |                 return ThemeHelper.resolveResourceIdFromAttr(c, R.attr.ic_hot); | ||||||
|  |             default: | ||||||
|  |                 return 0; | ||||||
|  |         } | ||||||
|  |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -31,17 +31,17 @@ import org.schabi.newpipe.extractor.stream.AudioStream; | |||||||
| import org.schabi.newpipe.extractor.stream.Stream; | import org.schabi.newpipe.extractor.stream.Stream; | ||||||
| import org.schabi.newpipe.extractor.stream.StreamInfo; | import org.schabi.newpipe.extractor.stream.StreamInfo; | ||||||
| import org.schabi.newpipe.extractor.stream.VideoStream; | import org.schabi.newpipe.extractor.stream.VideoStream; | ||||||
| import org.schabi.newpipe.extractor.linkhandler.ListLinkHandler; |  | ||||||
| import org.schabi.newpipe.extractor.linkhandler.SearchQueryHandler; |  | ||||||
| import org.schabi.newpipe.fragments.MainFragment; | import org.schabi.newpipe.fragments.MainFragment; | ||||||
| import org.schabi.newpipe.fragments.detail.VideoDetailFragment; | import org.schabi.newpipe.fragments.detail.VideoDetailFragment; | ||||||
| import org.schabi.newpipe.fragments.list.channel.ChannelFragment; | import org.schabi.newpipe.fragments.list.channel.ChannelFragment; | ||||||
|  | import org.schabi.newpipe.local.bookmark.BookmarkFragment; | ||||||
| import org.schabi.newpipe.local.feed.FeedFragment; | import org.schabi.newpipe.local.feed.FeedFragment; | ||||||
| import org.schabi.newpipe.fragments.list.kiosk.KioskFragment; | import org.schabi.newpipe.fragments.list.kiosk.KioskFragment; | ||||||
| import org.schabi.newpipe.fragments.list.playlist.PlaylistFragment; | import org.schabi.newpipe.fragments.list.playlist.PlaylistFragment; | ||||||
| import org.schabi.newpipe.fragments.list.search.SearchFragment; | import org.schabi.newpipe.fragments.list.search.SearchFragment; | ||||||
| import org.schabi.newpipe.local.history.StatisticsPlaylistFragment; | import org.schabi.newpipe.local.history.StatisticsPlaylistFragment; | ||||||
| import org.schabi.newpipe.local.playlist.LocalPlaylistFragment; | import org.schabi.newpipe.local.playlist.LocalPlaylistFragment; | ||||||
|  | import org.schabi.newpipe.local.subscription.SubscriptionFragment; | ||||||
| import org.schabi.newpipe.local.subscription.SubscriptionsImportFragment; | import org.schabi.newpipe.local.subscription.SubscriptionsImportFragment; | ||||||
| import org.schabi.newpipe.player.BackgroundPlayer; | import org.schabi.newpipe.player.BackgroundPlayer; | ||||||
| import org.schabi.newpipe.player.BackgroundPlayerActivity; | import org.schabi.newpipe.player.BackgroundPlayerActivity; | ||||||
| @@ -349,6 +349,20 @@ public class NavigationHelper { | |||||||
|                 .commit(); |                 .commit(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     public static void openBookmarksFragment(FragmentManager fragmentManager) { | ||||||
|  |         defaultTransaction(fragmentManager) | ||||||
|  |                 .replace(R.id.fragment_holder, new BookmarkFragment()) | ||||||
|  |                 .addToBackStack(null) | ||||||
|  |                 .commit(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public static void openSubscriptionFragment(FragmentManager fragmentManager) { | ||||||
|  |         defaultTransaction(fragmentManager) | ||||||
|  |                 .replace(R.id.fragment_holder, new SubscriptionFragment()) | ||||||
|  |                 .addToBackStack(null) | ||||||
|  |                 .commit(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     public static void openKioskFragment(FragmentManager fragmentManager, int serviceId, String kioskId) throws ExtractionException { |     public static void openKioskFragment(FragmentManager fragmentManager, int serviceId, String kioskId) throws ExtractionException { | ||||||
|         defaultTransaction(fragmentManager) |         defaultTransaction(fragmentManager) | ||||||
|                 .replace(R.id.fragment_holder, KioskFragment.getInstance(serviceId, kioskId)) |                 .replace(R.id.fragment_holder, KioskFragment.getInstance(serviceId, kioskId)) | ||||||
|   | |||||||
| @@ -1,10 +1,17 @@ | |||||||
| package us.shandian.giga.get; | package us.shandian.giga.get; | ||||||
|  |  | ||||||
|  | import android.content.Context; | ||||||
|  | import android.content.Intent; | ||||||
|  | import android.os.Handler; | ||||||
|  | import android.support.annotation.NonNull; | ||||||
| import android.support.annotation.Nullable; | import android.support.annotation.Nullable; | ||||||
| import android.util.Log; | import android.util.Log; | ||||||
|  |  | ||||||
|  | import org.schabi.newpipe.download.ExtSDDownloadFailedActivity; | ||||||
|  |  | ||||||
| import java.io.File; | import java.io.File; | ||||||
| import java.io.FilenameFilter; | import java.io.FilenameFilter; | ||||||
|  | import java.io.IOException; | ||||||
| import java.io.RandomAccessFile; | import java.io.RandomAccessFile; | ||||||
| import java.net.HttpURLConnection; | import java.net.HttpURLConnection; | ||||||
| import java.net.URL; | import java.net.URL; | ||||||
| @@ -23,7 +30,9 @@ public class DownloadManagerImpl implements DownloadManager { | |||||||
|     private static final String TAG = DownloadManagerImpl.class.getSimpleName(); |     private static final String TAG = DownloadManagerImpl.class.getSimpleName(); | ||||||
|     private final DownloadDataSource mDownloadDataSource; |     private final DownloadDataSource mDownloadDataSource; | ||||||
|  |  | ||||||
|     private final ArrayList<DownloadMission> mMissions = new ArrayList<DownloadMission>(); |     private final ArrayList<DownloadMission> mMissions = new ArrayList<>(); | ||||||
|  |     @NonNull | ||||||
|  |     private final Context context; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Create a new instance |      * Create a new instance | ||||||
| @@ -33,6 +42,13 @@ public class DownloadManagerImpl implements DownloadManager { | |||||||
|      */ |      */ | ||||||
|     public DownloadManagerImpl(Collection<String> searchLocations, DownloadDataSource downloadDataSource) { |     public DownloadManagerImpl(Collection<String> searchLocations, DownloadDataSource downloadDataSource) { | ||||||
|         mDownloadDataSource = downloadDataSource; |         mDownloadDataSource = downloadDataSource; | ||||||
|  |         this.context = null; | ||||||
|  |         loadMissions(searchLocations); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public DownloadManagerImpl(Collection<String> searchLocations, DownloadDataSource downloadDataSource, Context context) { | ||||||
|  |         mDownloadDataSource = downloadDataSource; | ||||||
|  |         this.context = context; | ||||||
|         loadMissions(searchLocations); |         loadMissions(searchLocations); | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -277,10 +293,12 @@ public class DownloadManagerImpl implements DownloadManager { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     private class Initializer extends Thread { |     private class Initializer extends Thread { | ||||||
|         private DownloadMission mission; |         private final DownloadMission mission; | ||||||
|  |         private final Handler handler; | ||||||
|  |  | ||||||
|         public Initializer(DownloadMission mission) { |         public Initializer(DownloadMission mission) { | ||||||
|             this.mission = mission; |             this.mission = mission; | ||||||
|  |             this.handler = new Handler(); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         @Override |         @Override | ||||||
| @@ -335,6 +353,13 @@ public class DownloadManagerImpl implements DownloadManager { | |||||||
|                 af.close(); |                 af.close(); | ||||||
|  |  | ||||||
|                 mission.start(); |                 mission.start(); | ||||||
|  |             } catch (IOException ie) { | ||||||
|  |                 if(context == null) throw new RuntimeException(ie); | ||||||
|  |  | ||||||
|  |                 if(ie.getMessage().contains("Permission denied")) { | ||||||
|  |                     handler.post(() -> | ||||||
|  |                         context.startActivity(new Intent(context, ExtSDDownloadFailedActivity.class))); | ||||||
|  |                 } else throw new RuntimeException(ie); | ||||||
|             } catch (Exception e) { |             } catch (Exception e) { | ||||||
|                 // TODO Notify |                 // TODO Notify | ||||||
|                 throw new RuntimeException(e); |                 throw new RuntimeException(e); | ||||||
|   | |||||||
| @@ -81,7 +81,7 @@ public class DownloadManagerService extends Service { | |||||||
|             ArrayList<String> paths = new ArrayList<>(2); |             ArrayList<String> paths = new ArrayList<>(2); | ||||||
|             paths.add(NewPipeSettings.getVideoDownloadPath(this)); |             paths.add(NewPipeSettings.getVideoDownloadPath(this)); | ||||||
|             paths.add(NewPipeSettings.getAudioDownloadPath(this)); |             paths.add(NewPipeSettings.getAudioDownloadPath(this)); | ||||||
|             mManager = new DownloadManagerImpl(paths, mDataSource); |             mManager = new DownloadManagerImpl(paths, mDataSource, this); | ||||||
|             if (DEBUG) { |             if (DEBUG) { | ||||||
|                 Log.d(TAG, "mManager == null"); |                 Log.d(TAG, "mManager == null"); | ||||||
|                 Log.d(TAG, "Download directory: " + paths); |                 Log.d(TAG, "Download directory: " + paths); | ||||||
|   | |||||||
							
								
								
									
										
											BIN
										
									
								
								app/src/main/res/drawable-hdpi/ic_add.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 223 B | 
							
								
								
									
										
											BIN
										
									
								
								app/src/main/res/drawable-hdpi/ic_arrow_down_white.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 267 B | 
							
								
								
									
										
											BIN
										
									
								
								app/src/main/res/drawable-hdpi/ic_arrow_up_white.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 261 B | 
							
								
								
									
										
											BIN
										
									
								
								app/src/main/res/drawable-hdpi/ic_remove.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 363 B | 
							
								
								
									
										
											BIN
										
									
								
								app/src/main/res/drawable-mdpi/ic_add.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 152 B | 
							
								
								
									
										
											BIN
										
									
								
								app/src/main/res/drawable-mdpi/ic_arrow_down_white.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 210 B | 
							
								
								
									
										
											BIN
										
									
								
								app/src/main/res/drawable-mdpi/ic_arrow_up_white.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 201 B | 
							
								
								
									
										
											BIN
										
									
								
								app/src/main/res/drawable-mdpi/ic_remove.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 230 B | 
							
								
								
									
										
											BIN
										
									
								
								app/src/main/res/drawable-xhdpi/ic_add.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 197 B | 
							
								
								
									
										
											BIN
										
									
								
								app/src/main/res/drawable-xhdpi/ic_arrow_down_white.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 425 B | 
							
								
								
									
										
											BIN
										
									
								
								app/src/main/res/drawable-xhdpi/ic_arrow_up_white.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 415 B | 
							
								
								
									
										
											BIN
										
									
								
								app/src/main/res/drawable-xhdpi/ic_remove.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 380 B | 
							
								
								
									
										
											BIN
										
									
								
								app/src/main/res/drawable-xxhdpi/ic_add.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 351 B | 
							
								
								
									
										
											BIN
										
									
								
								app/src/main/res/drawable-xxhdpi/ic_arrow_down_white.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 470 B | 
							
								
								
									
										
											BIN
										
									
								
								app/src/main/res/drawable-xxhdpi/ic_arrow_up_white.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 458 B | 
							
								
								
									
										
											BIN
										
									
								
								app/src/main/res/drawable-xxhdpi/ic_remove.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 611 B | 
| @@ -0,0 +1,5 @@ | |||||||
|  | <?xml version="1.0" encoding="utf-8"?> | ||||||
|  | <shape xmlns:android="http://schemas.android.com/apk/res/android" | ||||||
|  |     android:shape="oval"> | ||||||
|  |     <solid android:color="#64000000" /> | ||||||
|  | </shape> | ||||||
							
								
								
									
										9
									
								
								app/src/main/res/drawable/ic_add_black_24dp.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,9 @@ | |||||||
|  | <vector xmlns:android="http://schemas.android.com/apk/res/android" | ||||||
|  |         android:width="24dp" | ||||||
|  |         android:height="24dp" | ||||||
|  |         android:viewportWidth="24.0" | ||||||
|  |         android:viewportHeight="24.0"> | ||||||
|  |     <path | ||||||
|  |         android:fillColor="#FF000000" | ||||||
|  |         android:pathData="M19,13h-6v6h-2v-6H5v-2h6V5h2v6h6v2z"/> | ||||||
|  | </vector> | ||||||
							
								
								
									
										5
									
								
								app/src/main/res/drawable/ic_add_white_24dp.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,5 @@ | |||||||
|  | <vector android:height="24dp" android:tint="#FFFFFF" | ||||||
|  |     android:viewportHeight="24.0" android:viewportWidth="24.0" | ||||||
|  |     android:width="24dp" xmlns:android="http://schemas.android.com/apk/res/android"> | ||||||
|  |     <path android:fillColor="#FF000000" android:pathData="M19,13h-6v6h-2v-6H5v-2h6V5h2v6h6v2z"/> | ||||||
|  | </vector> | ||||||
							
								
								
									
										10
									
								
								app/src/main/res/drawable/ic_brightness_high_white_72dp.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,10 @@ | |||||||
|  | <vector xmlns:android="http://schemas.android.com/apk/res/android" | ||||||
|  |     android:width="72dp" | ||||||
|  |     android:height="72dp" | ||||||
|  |     android:tint="#FFFFFF" | ||||||
|  |     android:viewportWidth="24.0" | ||||||
|  |     android:viewportHeight="24.0"> | ||||||
|  |     <path | ||||||
|  |         android:fillColor="#FF000000" | ||||||
|  |         android:pathData="M20,8.69L20,4h-4.69L12,0.69 8.69,4L4,4v4.69L0.69,12 4,15.31L4,20h4.69L12,23.31 15.31,20L20,20v-4.69L23.31,12 20,8.69zM12,18c-3.31,0 -6,-2.69 -6,-6s2.69,-6 6,-6 6,2.69 6,6 -2.69,6 -6,6zM12,8c-2.21,0 -4,1.79 -4,4s1.79,4 4,4 4,-1.79 4,-4 -1.79,-4 -4,-4z" /> | ||||||
|  | </vector> | ||||||
							
								
								
									
										10
									
								
								app/src/main/res/drawable/ic_brightness_low_white_72dp.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,10 @@ | |||||||
|  | <vector xmlns:android="http://schemas.android.com/apk/res/android" | ||||||
|  |     android:width="72dp" | ||||||
|  |     android:height="72dp" | ||||||
|  |     android:tint="#FFFFFF" | ||||||
|  |     android:viewportWidth="24.0" | ||||||
|  |     android:viewportHeight="24.0"> | ||||||
|  |     <path | ||||||
|  |         android:fillColor="#FF000000" | ||||||
|  |         android:pathData="M20,15.31L23.31,12 20,8.69V4h-4.69L12,0.69 8.69,4H4v4.69L0.69,12 4,15.31V20h4.69L12,23.31 15.31,20H20v-4.69zM12,18c-3.31,0 -6,-2.69 -6,-6s2.69,-6 6,-6 6,2.69 6,6 -2.69,6 -6,6z" /> | ||||||
|  | </vector> | ||||||
| @@ -0,0 +1,10 @@ | |||||||
|  | <vector xmlns:android="http://schemas.android.com/apk/res/android" | ||||||
|  |     android:width="72dp" | ||||||
|  |     android:height="72dp" | ||||||
|  |     android:tint="#FFFFFF" | ||||||
|  |     android:viewportWidth="24.0" | ||||||
|  |     android:viewportHeight="24.0"> | ||||||
|  |     <path | ||||||
|  |         android:fillColor="#FF000000" | ||||||
|  |         android:pathData="M20,15.31L23.31,12 20,8.69V4h-4.69L12,0.69 8.69,4H4v4.69L0.69,12 4,15.31V20h4.69L12,23.31 15.31,20H20v-4.69zM12,18V6c3.31,0 6,2.69 6,6s-2.69,6 -6,6z" /> | ||||||
|  | </vector> | ||||||
							
								
								
									
										10
									
								
								app/src/main/res/drawable/ic_volume_down_white_72dp.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,10 @@ | |||||||
|  | <vector xmlns:android="http://schemas.android.com/apk/res/android" | ||||||
|  |     android:width="72dp" | ||||||
|  |     android:height="72dp" | ||||||
|  |     android:tint="#FFFFFF" | ||||||
|  |     android:viewportWidth="24.0" | ||||||
|  |     android:viewportHeight="24.0"> | ||||||
|  |     <path | ||||||
|  |         android:fillColor="#FF000000" | ||||||
|  |         android:pathData="M18.5,12c0,-1.77 -1.02,-3.29 -2.5,-4.03v8.05c1.48,-0.73 2.5,-2.25 2.5,-4.02zM5,9v6h4l5,5V4L9,9H5z" /> | ||||||
|  | </vector> | ||||||
							
								
								
									
										10
									
								
								app/src/main/res/drawable/ic_volume_mute_white_72dp.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,10 @@ | |||||||
|  | <vector xmlns:android="http://schemas.android.com/apk/res/android" | ||||||
|  |     android:width="72dp" | ||||||
|  |     android:height="72dp" | ||||||
|  |     android:tint="#FFFFFF" | ||||||
|  |     android:viewportWidth="24.0" | ||||||
|  |     android:viewportHeight="24.0"> | ||||||
|  |     <path | ||||||
|  |         android:fillColor="#FF000000" | ||||||
|  |         android:pathData="M7,9v6h4l5,5V4l-5,5H7z" /> | ||||||
|  | </vector> | ||||||
							
								
								
									
										10
									
								
								app/src/main/res/drawable/ic_volume_off_white_72dp.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,10 @@ | |||||||
|  | <vector xmlns:android="http://schemas.android.com/apk/res/android" | ||||||
|  |     android:width="72dp" | ||||||
|  |     android:height="72dp" | ||||||
|  |     android:tint="#FFFFFF" | ||||||
|  |     android:viewportWidth="24.0" | ||||||
|  |     android:viewportHeight="24.0"> | ||||||
|  |     <path | ||||||
|  |         android:fillColor="#FF000000" | ||||||
|  |         android:pathData="M16.5,12c0,-1.77 -1.02,-3.29 -2.5,-4.03v2.21l2.45,2.45c0.03,-0.2 0.05,-0.41 0.05,-0.63zM19,12c0,0.94 -0.2,1.82 -0.54,2.64l1.51,1.51C20.63,14.91 21,13.5 21,12c0,-4.28 -2.99,-7.86 -7,-8.77v2.06c2.89,0.86 5,3.54 5,6.71zM4.27,3L3,4.27 7.73,9L3,9v6h4l5,5v-6.73l4.25,4.25c-0.67,0.52 -1.42,0.93 -2.25,1.18v2.06c1.38,-0.31 2.63,-0.95 3.69,-1.81L19.73,21 21,19.73l-9,-9L4.27,3zM12,4L9.91,6.09 12,8.18L12,4z" /> | ||||||
|  | </vector> | ||||||
							
								
								
									
										10
									
								
								app/src/main/res/drawable/ic_volume_up_white_72dp.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,10 @@ | |||||||
|  | <vector xmlns:android="http://schemas.android.com/apk/res/android" | ||||||
|  |     android:width="72dp" | ||||||
|  |     android:height="72dp" | ||||||
|  |     android:tint="#FFFFFF" | ||||||
|  |     android:viewportWidth="24.0" | ||||||
|  |     android:viewportHeight="24.0"> | ||||||
|  |     <path | ||||||
|  |         android:fillColor="#FF000000" | ||||||
|  |         android:pathData="M3,9v6h4l5,5L12,4L7,9L3,9zM16.5,12c0,-1.77 -1.02,-3.29 -2.5,-4.03v8.05c1.48,-0.73 2.5,-2.25 2.5,-4.02zM14,3.23v2.06c2.89,0.86 5,3.54 5,6.71s-2.11,5.85 -5,6.71v2.06c4.01,-0.91 7,-4.49 7,-8.77s-2.99,-7.86 -7,-8.77z" /> | ||||||
|  | </vector> | ||||||
							
								
								
									
										12
									
								
								app/src/main/res/drawable/progress_circular_white.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,12 @@ | |||||||
|  | <?xml version="1.0" encoding="utf-8"?> | ||||||
|  | <rotate xmlns:android="http://schemas.android.com/apk/res/android" | ||||||
|  |     android:fromDegrees="-90" | ||||||
|  |     android:toDegrees="-90"> | ||||||
|  |     <shape xmlns:android="http://schemas.android.com/apk/res/android" | ||||||
|  |         android:innerRadiusRatio="2.25" | ||||||
|  |         android:shape="ring" | ||||||
|  |         android:thicknessRatio="17.75" | ||||||
|  |         android:useLevel="true"> | ||||||
|  |         <solid android:color="@android:color/white" /> | ||||||
|  |     </shape> | ||||||
|  | </rotate> | ||||||
							
								
								
									
										69
									
								
								app/src/main/res/layout-v21/drawer_header.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,69 @@ | |||||||
|  | <?xml version="1.0" encoding="utf-8"?> | ||||||
|  | <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" | ||||||
|  |     android:layout_width="wrap_content" | ||||||
|  |     android:layout_height="150dp" | ||||||
|  |     android:clickable="true" | ||||||
|  |     android:focusable="true"> | ||||||
|  |  | ||||||
|  |     <Button | ||||||
|  |         android:id="@+id/drawer_header_action_button" | ||||||
|  |         android:layout_width="match_parent" | ||||||
|  |         android:layout_height="match_parent" | ||||||
|  |         android:background="?android:attr/selectableItemBackground" /> | ||||||
|  |  | ||||||
|  |     <ImageView | ||||||
|  |         android:layout_width="match_parent" | ||||||
|  |         android:layout_height="match_parent" | ||||||
|  |         android:background="?attr/colorPrimary" | ||||||
|  |         android:scaleType="centerCrop" | ||||||
|  |         android:src="@drawable/background_header" /> | ||||||
|  |  | ||||||
|  |     <ImageView | ||||||
|  |         android:id="@+id/drawer_header_np_nude_view" | ||||||
|  |         android:layout_width="70dp" | ||||||
|  |         android:layout_height="70dp" | ||||||
|  |         android:layout_marginLeft="30dp" | ||||||
|  |  | ||||||
|  |         android:layout_marginStart="30dp" | ||||||
|  |         android:layout_marginTop="30dp" | ||||||
|  |         android:src="@drawable/np_logo_nude_shadow" /> | ||||||
|  |  | ||||||
|  |     <TextView | ||||||
|  |         android:id="@+id/drawer_header_np_text_view" | ||||||
|  |         android:layout_width="wrap_content" | ||||||
|  |         android:layout_height="0dp" | ||||||
|  |         android:layout_alignBottom="@id/drawer_header_np_nude_view" | ||||||
|  |         android:layout_alignTop="@id/drawer_header_np_nude_view" | ||||||
|  |         android:layout_toEndOf="@id/drawer_header_np_nude_view" | ||||||
|  |         android:layout_toRightOf="@id/drawer_header_np_nude_view" | ||||||
|  |         android:gravity="center" | ||||||
|  |         android:text="@string/app_name" | ||||||
|  |         android:textSize="30dp" | ||||||
|  |         android:textStyle="bold|italic" /> | ||||||
|  |  | ||||||
|  |     <TextView | ||||||
|  |         android:id="@+id/drawer_header_service_view" | ||||||
|  |         android:layout_width="100dp" | ||||||
|  |         android:layout_height="100dp" | ||||||
|  |         android:layout_alignLeft="@id/drawer_header_np_text_view" | ||||||
|  |         android:layout_alignStart="@id/drawer_header_np_text_view" | ||||||
|  |         android:layout_below="@id/drawer_header_np_text_view" | ||||||
|  |         android:text="YouTube" | ||||||
|  |         android:textSize="18dp" | ||||||
|  |         android:textStyle="italic" /> | ||||||
|  |  | ||||||
|  |     <ImageView | ||||||
|  |         android:id="@+id/drawer_arrow" | ||||||
|  |         android:layout_width="30dp" | ||||||
|  |         android:layout_height="35dp" | ||||||
|  |         android:layout_alignParentBottom="true" | ||||||
|  |         android:layout_alignParentEnd="true" | ||||||
|  |         android:layout_alignParentRight="true" | ||||||
|  |         android:layout_gravity="bottom" | ||||||
|  |         android:layout_marginBottom="0dp" | ||||||
|  |         android:paddingBottom="20dp" | ||||||
|  |         android:paddingEnd="20dp" | ||||||
|  |         android:paddingRight="20dp" | ||||||
|  |         android:src="@drawable/ic_arrow_down_white" /> | ||||||
|  |  | ||||||
|  | </RelativeLayout> | ||||||
| @@ -4,7 +4,8 @@ | |||||||
|     xmlns:tools="http://schemas.android.com/tools" |     xmlns:tools="http://schemas.android.com/tools" | ||||||
|     android:id="@+id/drawer_layout" |     android:id="@+id/drawer_layout" | ||||||
|     android:layout_width="match_parent" |     android:layout_width="match_parent" | ||||||
|     android:layout_height="match_parent"> |     android:layout_height="match_parent" | ||||||
|  |     android:fitsSystemWindows="true"> | ||||||
|  |  | ||||||
|  |  | ||||||
|     <FrameLayout |     <FrameLayout | ||||||
|   | |||||||
| @@ -471,9 +471,10 @@ | |||||||
|             android:id="@+id/controlAnimationView" |             android:id="@+id/controlAnimationView" | ||||||
|             android:layout_width="100dp" |             android:layout_width="100dp" | ||||||
|             android:layout_height="100dp" |             android:layout_height="100dp" | ||||||
|             android:src="@drawable/ic_action_av_fast_rewind" |             android:background="@drawable/background_oval_black_transparent" | ||||||
|             android:visibility="gone" |             android:visibility="gone" | ||||||
|             tools:ignore="ContentDescription" |             tools:ignore="ContentDescription" | ||||||
|  |             tools:src="@drawable/ic_action_av_fast_rewind" | ||||||
|             tools:visibility="visible" /> |             tools:visibility="visible" /> | ||||||
|     </LinearLayout> |     </LinearLayout> | ||||||
|  |  | ||||||
| @@ -503,43 +504,57 @@ | |||||||
|         android:layout_toRightOf="@+id/loading_panel" |         android:layout_toRightOf="@+id/loading_panel" | ||||||
|         tools:ignore="RtlHardcoded"> |         tools:ignore="RtlHardcoded"> | ||||||
|  |  | ||||||
|         <TextView |         <RelativeLayout | ||||||
|             android:id="@+id/volumeTextView" |             android:id="@+id/volumeRelativeLayout" | ||||||
|             android:layout_width="wrap_content" |             android:layout_width="wrap_content" | ||||||
|             android:layout_height="wrap_content" |             android:layout_height="wrap_content" | ||||||
|             android:layout_alignParentLeft="true" |  | ||||||
|             android:layout_centerInParent="true" |             android:layout_centerInParent="true" | ||||||
|             android:layout_marginLeft="20dp" |             android:background="@drawable/background_oval_black_transparent" | ||||||
|             android:background="#64000000" |  | ||||||
|             android:paddingBottom="10dp" |  | ||||||
|             android:paddingLeft="30dp" |  | ||||||
|             android:paddingRight="30dp" |  | ||||||
|             android:paddingTop="10dp" |  | ||||||
|             android:textColor="@android:color/white" |  | ||||||
|             android:textSize="35sp" |  | ||||||
|             android:visibility="gone" |             android:visibility="gone" | ||||||
|             tools:ignore="RtlHardcoded" |             tools:visibility="visible"> | ||||||
|             tools:text="Volume 0" |  | ||||||
|             tools:visibility="visible" /> |  | ||||||
|  |  | ||||||
|         <TextView |             <ProgressBar | ||||||
|             android:id="@+id/brightnessTextView" |                 android:id="@+id/volumeProgressBar" | ||||||
|  |                 style="?android:progressBarStyleHorizontal" | ||||||
|  |                 android:layout_width="128dp" | ||||||
|  |                 android:layout_height="128dp" | ||||||
|  |                 android:indeterminate="false" | ||||||
|  |                 android:progressDrawable="@drawable/progress_circular_white" /> | ||||||
|  |  | ||||||
|  |             <ImageView | ||||||
|  |                 android:id="@+id/volumeImageView" | ||||||
|                 android:layout_width="wrap_content" |                 android:layout_width="wrap_content" | ||||||
|                 android:layout_height="wrap_content" |                 android:layout_height="wrap_content" | ||||||
|             android:layout_alignParentRight="true" |  | ||||||
|                 android:layout_centerInParent="true" |                 android:layout_centerInParent="true" | ||||||
|             android:layout_marginRight="20dp" |                 tools:ignore="ContentDescription" | ||||||
|             android:background="#64000000" |                 tools:src="@drawable/ic_volume_up_white_72dp" /> | ||||||
|             android:paddingBottom="10dp" |         </RelativeLayout> | ||||||
|             android:paddingLeft="30dp" |  | ||||||
|             android:paddingRight="30dp" |         <RelativeLayout | ||||||
|             android:paddingTop="10dp" |             android:id="@+id/brightnessRelativeLayout" | ||||||
|             android:textColor="@android:color/white" |             android:layout_width="wrap_content" | ||||||
|             android:textSize="35sp" |             android:layout_height="wrap_content" | ||||||
|  |             android:layout_centerInParent="true" | ||||||
|  |             android:background="@drawable/background_oval_black_transparent" | ||||||
|             android:visibility="gone" |             android:visibility="gone" | ||||||
|             tools:ignore="RtlHardcoded" |             tools:visibility="visible"> | ||||||
|             tools:text="Brightness  0" |  | ||||||
|             tools:visibility="visible" /> |             <ProgressBar | ||||||
|  |                 android:id="@+id/brightnessProgressBar" | ||||||
|  |                 style="?android:progressBarStyleHorizontal" | ||||||
|  |                 android:layout_width="128dp" | ||||||
|  |                 android:layout_height="128dp" | ||||||
|  |                 android:indeterminate="false" | ||||||
|  |                 android:progressDrawable="@drawable/progress_circular_white" /> | ||||||
|  |  | ||||||
|  |             <ImageView | ||||||
|  |                 android:id="@+id/brightnessImageView" | ||||||
|  |                 android:layout_width="wrap_content" | ||||||
|  |                 android:layout_height="wrap_content" | ||||||
|  |                 android:layout_centerInParent="true" | ||||||
|  |                 tools:ignore="ContentDescription" | ||||||
|  |                 tools:src="@drawable/ic_brightness_high_white_72dp" /> | ||||||
|  |         </RelativeLayout> | ||||||
|  |  | ||||||
|         <TextView |         <TextView | ||||||
|             android:id="@+id/currentDisplaySeek" |             android:id="@+id/currentDisplaySeek" | ||||||
|   | |||||||
							
								
								
									
										69
									
								
								app/src/main/res/layout/drawer_header.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,69 @@ | |||||||
|  | <?xml version="1.0" encoding="utf-8"?> | ||||||
|  | <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" | ||||||
|  | android:layout_width="wrap_content" | ||||||
|  | android:layout_height="150dp" | ||||||
|  | android:clickable="true" | ||||||
|  | android:focusable="true"> | ||||||
|  |  | ||||||
|  |     <Button | ||||||
|  |         android:id="@+id/drawer_header_action_button" | ||||||
|  |         android:layout_width="match_parent" | ||||||
|  |         android:layout_height="match_parent" | ||||||
|  |         android:background="?android:attr/selectableItemBackground"/> | ||||||
|  |  | ||||||
|  |     <ImageView | ||||||
|  |         android:layout_width="match_parent" | ||||||
|  |         android:layout_height="match_parent" | ||||||
|  |         android:background="?attr/colorPrimary" | ||||||
|  |         android:src="@drawable/background_header" | ||||||
|  |         android:scaleType="centerCrop"/> | ||||||
|  |  | ||||||
|  |     <ImageView | ||||||
|  |         android:id="@+id/drawer_header_np_nude_view" | ||||||
|  |         android:layout_marginLeft="30dp" | ||||||
|  |         android:layout_marginStart="30dp" | ||||||
|  |         android:layout_marginTop="20dp" | ||||||
|  |  | ||||||
|  |         android:layout_width="70dp" | ||||||
|  |         android:layout_height="70dp" | ||||||
|  |         android:src="@drawable/np_logo_nude_shadow"/> | ||||||
|  |  | ||||||
|  |     <TextView | ||||||
|  |         android:id="@+id/drawer_header_np_text_view" | ||||||
|  |         android:layout_width="wrap_content" | ||||||
|  |         android:layout_height="0dp" | ||||||
|  |         android:text="@string/app_name" | ||||||
|  |         android:layout_toRightOf="@id/drawer_header_np_nude_view" | ||||||
|  |         android:layout_toEndOf="@id/drawer_header_np_nude_view" | ||||||
|  |         android:layout_alignTop="@id/drawer_header_np_nude_view" | ||||||
|  |         android:layout_alignBottom="@id/drawer_header_np_nude_view" | ||||||
|  |         android:gravity="center" | ||||||
|  |         android:textSize="30dp" | ||||||
|  |         android:textStyle="bold|italic"/> | ||||||
|  |  | ||||||
|  |     <TextView | ||||||
|  |         android:id="@+id/drawer_header_service_view" | ||||||
|  |         android:layout_width="100dp" | ||||||
|  |         android:layout_height="100dp" | ||||||
|  |         android:text="YouTube" | ||||||
|  |         android:layout_below="@id/drawer_header_np_text_view" | ||||||
|  |         android:layout_alignLeft="@id/drawer_header_np_text_view" | ||||||
|  |         android:layout_alignStart="@id/drawer_header_np_text_view" | ||||||
|  |         android:textSize="18dp" | ||||||
|  |         android:textStyle="italic"/> | ||||||
|  |  | ||||||
|  |     <ImageView | ||||||
|  |         android:id="@+id/drawer_arrow" | ||||||
|  |         android:layout_width="30dp" | ||||||
|  |         android:layout_height="35dp" | ||||||
|  |         android:layout_alignParentBottom="true" | ||||||
|  |         android:layout_alignParentEnd="true" | ||||||
|  |         android:layout_alignParentRight="true" | ||||||
|  |         android:layout_gravity="bottom" | ||||||
|  |         android:layout_marginBottom="0dp" | ||||||
|  |         android:paddingBottom="20dp" | ||||||
|  |         android:paddingRight="20dp" | ||||||
|  |         android:src="@drawable/ic_arrow_down_white" | ||||||
|  |         android:paddingEnd="20dp" /> | ||||||
|  |  | ||||||
|  | </RelativeLayout> | ||||||
| @@ -1,81 +1,22 @@ | |||||||
| <?xml version="1.0" encoding="utf-8"?> | <?xml version="1.0" encoding="utf-8"?> | ||||||
| <RelativeLayout android:id="@+id/navigation_layout" | <android.support.design.widget.NavigationView android:id="@+id/navigation_layout" | ||||||
|     android:orientation="vertical" |     android:orientation="vertical" | ||||||
|     xmlns:android="http://schemas.android.com/apk/res/android" |     xmlns:android="http://schemas.android.com/apk/res/android" | ||||||
|     android:layout_height="match_parent" |     android:layout_height="match_parent" | ||||||
|     android:layout_width="wrap_content" |     android:layout_width="wrap_content" | ||||||
|     android:layout_gravity="start" |     android:layout_gravity="start" | ||||||
|     xmlns:app="http://schemas.android.com/apk/res-auto" |     xmlns:app="http://schemas.android.com/apk/res-auto" | ||||||
|     android:background="?attr/android:windowBackground" |  | ||||||
|     android:clickable="true" |     android:clickable="true" | ||||||
|     android:focusable="true"> |     android:focusable="true" | ||||||
|  |     > | ||||||
|     <RelativeLayout |  | ||||||
|         android:id="@+id/drawer_header" |  | ||||||
|         android:layout_width="0dp" |  | ||||||
|         android:layout_height="150dp" |  | ||||||
|         android:layout_alignLeft="@id/navigation" |  | ||||||
|         android:layout_alignRight="@id/navigation" |  | ||||||
|         android:layout_alignStart="@id/navigation" |  | ||||||
|         android:layout_alignEnd="@id/navigation" |  | ||||||
|         android:clickable="true" |  | ||||||
|         android:focusable="true"> |  | ||||||
|  |  | ||||||
|         <Button |  | ||||||
|             android:id="@+id/drawer_header_action_button" |  | ||||||
|             android:layout_width="match_parent" |  | ||||||
|             android:layout_height="match_parent" |  | ||||||
|             android:background="?android:attr/selectableItemBackground"/> |  | ||||||
|  |  | ||||||
|         <ImageView |  | ||||||
|             android:layout_width="match_parent" |  | ||||||
|             android:layout_height="match_parent" |  | ||||||
|             android:background="?attr/colorPrimary" |  | ||||||
|             android:src="@drawable/background_header" |  | ||||||
|             android:scaleType="centerCrop"/> |  | ||||||
|  |  | ||||||
|         <ImageView |  | ||||||
|             android:id="@+id/drawer_header_np_nude_view" |  | ||||||
|             android:layout_marginLeft="30dp" |  | ||||||
|             android:layout_marginStart="30dp" |  | ||||||
|             android:layout_marginTop="20dp" |  | ||||||
|  |  | ||||||
|             android:layout_width="70dp" |  | ||||||
|             android:layout_height="70dp" |  | ||||||
|             android:src="@drawable/np_logo_nude_shadow"/> |  | ||||||
|  |  | ||||||
|         <TextView |  | ||||||
|             android:id="@+id/drawer_header_np_text_view" |  | ||||||
|             android:layout_width="wrap_content" |  | ||||||
|             android:layout_height="0dp" |  | ||||||
|             android:text="@string/app_name" |  | ||||||
|             android:layout_toRightOf="@id/drawer_header_np_nude_view" |  | ||||||
|             android:layout_toEndOf="@id/drawer_header_np_nude_view" |  | ||||||
|             android:layout_alignTop="@id/drawer_header_np_nude_view" |  | ||||||
|             android:layout_alignBottom="@id/drawer_header_np_nude_view" |  | ||||||
|             android:gravity="center" |  | ||||||
|             android:textSize="30dp" |  | ||||||
|             android:textStyle="bold|italic"/> |  | ||||||
|  |  | ||||||
|         <TextView |  | ||||||
|             android:id="@+id/drawer_header_service_view" |  | ||||||
|             android:layout_width="100dp" |  | ||||||
|             android:layout_height="100dp" |  | ||||||
|             android:text="YouTube" |  | ||||||
|             android:layout_below="@id/drawer_header_np_text_view" |  | ||||||
|             android:layout_alignLeft="@id/drawer_header_np_text_view" |  | ||||||
|             android:layout_alignStart="@id/drawer_header_np_text_view" |  | ||||||
|             android:textSize="18dp" |  | ||||||
|             android:textStyle="italic"/> |  | ||||||
|  |  | ||||||
|     </RelativeLayout> |  | ||||||
|  |  | ||||||
|     <android.support.design.widget.NavigationView |     <android.support.design.widget.NavigationView | ||||||
|         android:id="@+id/navigation" |         android:id="@+id/navigation" | ||||||
|         android:layout_below="@id/drawer_header" |  | ||||||
|         android:layout_width="wrap_content" |         android:layout_width="wrap_content" | ||||||
|         android:layout_height="wrap_content" |         android:layout_height="wrap_content" | ||||||
|         app:elevation="0dp"/> |         app:elevation="0dp" | ||||||
|  |         android:background="?attr/android:windowBackground" | ||||||
|  |         app:headerLayout="@layout/drawer_header"/> | ||||||
|         <!-- app:menu="@menu/drawer_items" --> |         <!-- app:menu="@menu/drawer_items" --> | ||||||
|  |  | ||||||
|     <LinearLayout |     <LinearLayout | ||||||
| @@ -89,29 +30,6 @@ | |||||||
|         android:layout_alignEnd="@id/navigation" |         android:layout_alignEnd="@id/navigation" | ||||||
|  |  | ||||||
|         android:layout_alignParentBottom="true"> |         android:layout_alignParentBottom="true"> | ||||||
|         <ImageButton |  | ||||||
|             android:id="@+id/drawer_settings" |  | ||||||
|             android:layout_width="0dp" |  | ||||||
|             android:layout_height="match_parent" |  | ||||||
|             android:layout_weight="1" |  | ||||||
|             android:background="?attr/selectableItemBackgroundBorderless" |  | ||||||
|             android:src="?attr/settings"/> |  | ||||||
|  |  | ||||||
|         <ImageButton |  | ||||||
|             android:id="@+id/drawer_downloads" |  | ||||||
|             android:layout_width="0dp" |  | ||||||
|             android:layout_height="match_parent" |  | ||||||
|             android:layout_weight="1" |  | ||||||
|             android:background="?attr/selectableItemBackgroundBorderless" |  | ||||||
|             android:src="?attr/download" /> |  | ||||||
|  |  | ||||||
|         <ImageButton |  | ||||||
|             android:id="@+id/drawer_history" |  | ||||||
|             android:layout_width="0dp" |  | ||||||
|             android:layout_height="match_parent" |  | ||||||
|             android:layout_weight="1" |  | ||||||
|             android:background="?attr/selectableItemBackgroundBorderless" |  | ||||||
|             android:src="?attr/history"/> |  | ||||||
|     </LinearLayout> |     </LinearLayout> | ||||||
|  |  | ||||||
| </RelativeLayout> | </android.support.design.widget.NavigationView> | ||||||
							
								
								
									
										32
									
								
								app/src/main/res/layout/fragment_chose_tabs.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,32 @@ | |||||||
|  | <?xml version="1.0" encoding="utf-8"?> | ||||||
|  | <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" | ||||||
|  |     xmlns:app="http://schemas.android.com/apk/res-auto" | ||||||
|  |     android:id="@+id/relLay" | ||||||
|  |     android:layout_width="match_parent" | ||||||
|  |     android:layout_height="match_parent" | ||||||
|  |     android:orientation="vertical"> | ||||||
|  |  | ||||||
|  |     <android.support.v7.widget.RecyclerView | ||||||
|  |         android:id="@+id/usedTabs" | ||||||
|  |         android:layout_width="match_parent" | ||||||
|  |         android:layout_height="match_parent" | ||||||
|  |         android:layout_margin="0dp" | ||||||
|  |         android:paddingBottom="0dp" | ||||||
|  |         android:paddingTop="0dp" > | ||||||
|  |  | ||||||
|  |     </android.support.v7.widget.RecyclerView> | ||||||
|  |  | ||||||
|  |     <android.support.design.widget.FloatingActionButton | ||||||
|  |         android:id="@+id/floatingActionButton" | ||||||
|  |         android:layout_width="wrap_content" | ||||||
|  |         android:layout_height="wrap_content" | ||||||
|  |         android:layout_alignParentBottom="true" | ||||||
|  |         android:layout_alignParentEnd="true" | ||||||
|  |         android:layout_marginBottom="16dp" | ||||||
|  |         android:layout_marginEnd="16dp" | ||||||
|  |         android:clickable="true" | ||||||
|  |         android:layout_alignParentRight="true" | ||||||
|  |         android:layout_marginRight="16dp" | ||||||
|  |         android:focusable="true" /> | ||||||
|  |  | ||||||
|  | </RelativeLayout> | ||||||
| @@ -9,7 +9,6 @@ | |||||||
|     tools:layout_height="84dp" |     tools:layout_height="84dp" | ||||||
|     tools:layout_width="@dimen/popup_minimum_width"> |     tools:layout_width="@dimen/popup_minimum_width"> | ||||||
|  |  | ||||||
|  |  | ||||||
|     <com.google.android.exoplayer2.ui.AspectRatioFrameLayout |     <com.google.android.exoplayer2.ui.AspectRatioFrameLayout | ||||||
|         android:id="@+id/aspectRatioLayout" |         android:id="@+id/aspectRatioLayout" | ||||||
|         android:layout_width="match_parent" |         android:layout_width="match_parent" | ||||||
| @@ -290,4 +289,11 @@ | |||||||
|         android:visibility="gone" |         android:visibility="gone" | ||||||
|         tools:ignore="RtlHardcoded" |         tools:ignore="RtlHardcoded" | ||||||
|         tools:visibility="gone"/> |         tools:visibility="gone"/> | ||||||
|  |  | ||||||
|  |     <View | ||||||
|  |         android:id="@+id/closingOverlay" | ||||||
|  |         android:layout_width="match_parent" | ||||||
|  |         android:layout_height="match_parent" | ||||||
|  |         android:background="#AAFF0000" | ||||||
|  |         android:visibility="gone"/> | ||||||
| </FrameLayout> | </FrameLayout> | ||||||
							
								
								
									
										18
									
								
								app/src/main/res/layout/player_popup_close_overlay.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,18 @@ | |||||||
|  | <?xml version="1.0" encoding="utf-8"?> | ||||||
|  | <FrameLayout | ||||||
|  |     xmlns:android="http://schemas.android.com/apk/res/android" | ||||||
|  |     xmlns:app="http://schemas.android.com/apk/res-auto" | ||||||
|  |     android:layout_width="match_parent" | ||||||
|  |     android:layout_height="match_parent"> | ||||||
|  |  | ||||||
|  |     <android.support.design.widget.FloatingActionButton | ||||||
|  |         android:id="@+id/closeButton" | ||||||
|  |         android:layout_width="wrap_content" | ||||||
|  |         android:layout_height="wrap_content" | ||||||
|  |         android:layout_gravity="bottom|center_horizontal" | ||||||
|  |         android:layout_marginBottom="24dp" | ||||||
|  |         android:src="@drawable/ic_close_white_24dp" | ||||||
|  |         app:backgroundTint="@color/light_youtube_primary_color" | ||||||
|  |         app:borderWidth="0dp" | ||||||
|  |         app:fabSize="normal"/> | ||||||
|  | </FrameLayout> | ||||||
							
								
								
									
										40
									
								
								app/src/main/res/layout/viewholder_chose_tabs.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,40 @@ | |||||||
|  | <?xml version="1.0" encoding="utf-8"?> | ||||||
|  | <android.support.v7.widget.CardView xmlns:android="http://schemas.android.com/apk/res/android" | ||||||
|  |     xmlns:app="http://schemas.android.com/apk/res-auto" | ||||||
|  |     android:id="@+id/layoutCard" | ||||||
|  |     android:layout_width="match_parent" | ||||||
|  |     android:layout_height="wrap_content" | ||||||
|  |     android:layout_marginBottom="3dp" | ||||||
|  |     android:layout_marginLeft="5dp" | ||||||
|  |     android:layout_marginRight="5dp" | ||||||
|  |     android:layout_marginTop="3dp" | ||||||
|  |     android:orientation="horizontal" | ||||||
|  |     android:paddingBottom="4dp" | ||||||
|  |     android:paddingTop="4dp" | ||||||
|  |     app:cardCornerRadius="5dp" | ||||||
|  |     app:cardElevation="4dp"> | ||||||
|  |  | ||||||
|  |     <ImageView | ||||||
|  |         android:id="@+id/handle" | ||||||
|  |         android:layout_width="30dp" | ||||||
|  |         android:layout_height="30dp" | ||||||
|  |         android:layout_gravity="center_vertical|end" | ||||||
|  |         android:layout_marginRight="10dp" /> | ||||||
|  |  | ||||||
|  |     <TextView | ||||||
|  |         android:id="@+id/tabName" | ||||||
|  |         android:layout_width="match_parent" | ||||||
|  |         android:layout_height="wrap_content" | ||||||
|  |         android:layout_alignParentEnd="true" | ||||||
|  |         android:layout_alignParentLeft="true" | ||||||
|  |         android:layout_alignParentStart="true" | ||||||
|  |         android:layout_centerVertical="true" | ||||||
|  |         android:paddingBottom="9dp" | ||||||
|  |         android:paddingLeft="15dp" | ||||||
|  |         android:paddingRight="15dp" | ||||||
|  |         android:paddingStart="3dp" | ||||||
|  |         android:paddingTop="9dp" | ||||||
|  |         android:textAppearance="?android:attr/textAppearanceListItem" | ||||||
|  |         android:textSize="16sp" /> | ||||||
|  |  | ||||||
|  | </android.support.v7.widget.CardView> | ||||||
| @@ -2,13 +2,11 @@ | |||||||
| <menu xmlns:android="http://schemas.android.com/apk/res/android"> | <menu xmlns:android="http://schemas.android.com/apk/res/android"> | ||||||
|     <group |     <group | ||||||
|         android:id="@+id/menu_services_group"> |         android:id="@+id/menu_services_group"> | ||||||
|         <item |     </group> | ||||||
|             android:id="@+id/menu_service_youtube" |     <group | ||||||
|             android:icon="@drawable/place_holder_youtube" |         android:id="@+id/menu_tabs_group"> | ||||||
|             android:title="@string/youtube"/> |     </group> | ||||||
|         <item |     <group | ||||||
|             android:id="@+id/menu_service_soundcloud" |         android:id="@+id/menu_options_about_group"> | ||||||
|             android:icon="@drawable/place_holder_circle" |  | ||||||
|             android:title="@string/soundcloud"/> |  | ||||||
|     </group> |     </group> | ||||||
| </menu> | </menu> | ||||||
| @@ -19,9 +19,4 @@ | |||||||
|         android:orderInCategory="990" |         android:orderInCategory="990" | ||||||
|         android:title="@string/settings" |         android:title="@string/settings" | ||||||
|         app:showAsAction="never"/> |         app:showAsAction="never"/> | ||||||
|  |  | ||||||
|     <item |  | ||||||
|         android:id="@+id/action_about" |  | ||||||
|         android:orderInCategory="1000" |  | ||||||
|         android:title="@string/action_about"/> |  | ||||||
| </menu> | </menu> | ||||||
							
								
								
									
										34
									
								
								app/src/main/res/values-v21/styles_services.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,34 @@ | |||||||
|  | <?xml version="1.0" encoding="utf-8"?> | ||||||
|  | <resources> | ||||||
|  |     <!-- YouTube --> | ||||||
|  |     <style name="LightTheme.YouTube" parent="LightTheme.Switchable"> | ||||||
|  |         <item name="colorPrimaryDark">@color/light_youtube_statusbar_color</item> | ||||||
|  |     </style> | ||||||
|  |  | ||||||
|  |     <style name="DarkTheme.YouTube" parent="DarkTheme.Switchable"> | ||||||
|  |         <item name="colorPrimaryDark">@color/dark_youtube_statusbar_color</item> | ||||||
|  |     </style> | ||||||
|  |  | ||||||
|  |     <style name="BlackTheme.YouTube" parent="BlackTheme.Switchable"> | ||||||
|  |         <item name="colorPrimaryDark">@color/dark_youtube_statusbar_color</item> | ||||||
|  |     </style> | ||||||
|  |     <!-- SoundCloud --> | ||||||
|  |     <style name="LightTheme.SoundCloud" parent="LightTheme.Switchable"> | ||||||
|  |         <item name="colorPrimary">@color/light_soundcloud_primary_color</item> | ||||||
|  |         <item name="colorPrimaryDark">@color/light_soundcloud_statusbar_color</item> | ||||||
|  |         <item name="colorAccent">@color/light_soundcloud_accent_color</item> | ||||||
|  |     </style> | ||||||
|  |  | ||||||
|  |     <style name="DarkTheme.SoundCloud" parent="DarkTheme.Switchable"> | ||||||
|  |         <item name="colorPrimary">@color/dark_soundcloud_primary_color</item> | ||||||
|  |         <item name="colorPrimaryDark">@color/dark_soundcloud_statusbar_color</item> | ||||||
|  |         <item name="colorAccent">@color/dark_soundcloud_accent_color</item> | ||||||
|  |     </style> | ||||||
|  |  | ||||||
|  |     <style name="BlackTheme.SoundCloud" parent="BlackTheme.Switchable"> | ||||||
|  |         <item name="colorPrimary">@color/dark_soundcloud_primary_color</item> | ||||||
|  |         <item name="colorPrimaryDark">@color/dark_soundcloud_statusbar_color</item> | ||||||
|  |         <item name="colorAccent">@color/dark_soundcloud_accent_color</item> | ||||||
|  |     </style> | ||||||
|  |  | ||||||
|  | </resources> | ||||||
| @@ -35,6 +35,7 @@ | |||||||
|     <attr name="ic_import_export" format="reference"/> |     <attr name="ic_import_export" format="reference"/> | ||||||
|     <attr name="ic_save" format="reference"/> |     <attr name="ic_save" format="reference"/> | ||||||
|     <attr name="ic_backup" format="reference"/> |     <attr name="ic_backup" format="reference"/> | ||||||
|  |     <attr name="ic_add" format="reference"/> | ||||||
|  |  | ||||||
|     <!-- Can't refer to colors directly into drawable's xml--> |     <!-- Can't refer to colors directly into drawable's xml--> | ||||||
|     <attr name="toolbar_shadow_drawable" format="reference"/> |     <attr name="toolbar_shadow_drawable" format="reference"/> | ||||||
|   | |||||||
| @@ -29,13 +29,13 @@ | |||||||
|     <color name="dark_queue_background_color">#af000000</color> |     <color name="dark_queue_background_color">#af000000</color> | ||||||
|  |  | ||||||
|     <!-- Black Theme --> |     <!-- Black Theme --> | ||||||
|     <color name="black_background_color">#000</color> |     <color name="black_background_color">#000000</color> | ||||||
|     <color name="black_settings_accent_color">@color/dark_settings_accent_color</color> |     <color name="black_settings_accent_color">@color/dark_settings_accent_color</color> | ||||||
|     <color name="black_separator_color">#1effffff</color> |     <color name="black_separator_color">#1effffff</color> | ||||||
|     <color name="black_contrast_background_color">#23454545</color> |     <color name="black_contrast_background_color">#23454545</color> | ||||||
|  |  | ||||||
|     <!-- Miscellaneous --> |     <!-- Miscellaneous --> | ||||||
|     <color name="duration_background_color">#AA000000</color> |     <color name="duration_background_color">#aa000000</color> | ||||||
|     <color name="live_duration_background_color">#c8E53935</color> |     <color name="live_duration_background_color">#c8E53935</color> | ||||||
|     <color name="playlist_stream_count_background_color">#e6000000</color> |     <color name="playlist_stream_count_background_color">#e6000000</color> | ||||||
|     <color name="duration_text_color">#EEFFFFFF</color> |     <color name="duration_text_color">#EEFFFFFF</color> | ||||||
| @@ -70,5 +70,6 @@ | |||||||
|     <color name="gray">#616161</color> |     <color name="gray">#616161</color> | ||||||
|  |  | ||||||
|     <color name="black">#000</color> |     <color name="black">#000</color> | ||||||
|  |     <color name="gray_transparent">#be757575</color> | ||||||
|  |  | ||||||
| </resources> | </resources> | ||||||
|   | |||||||
| @@ -2,20 +2,24 @@ | |||||||
| <resources> | <resources> | ||||||
|     <!-- YouTube --> |     <!-- YouTube --> | ||||||
|     <color name="light_youtube_primary_color">#e53935</color> |     <color name="light_youtube_primary_color">#e53935</color> | ||||||
|     <color name="light_youtube_dark_color">#d32f2f</color> |     <color name="light_youtube_dark_color">#992722</color> | ||||||
|     <color name="light_youtube_accent_color">#000000</color> |     <color name="light_youtube_accent_color">#000000</color> | ||||||
|  |     <color name="light_youtube_statusbar_color">#ff4336</color> | ||||||
|  |  | ||||||
|     <color name="dark_youtube_primary_color">#CD322E</color> |     <color name="dark_youtube_primary_color">#CD322E</color> | ||||||
|     <color name="dark_youtube_dark_color">#BC211D</color> |     <color name="dark_youtube_dark_color">#992722</color> | ||||||
|     <color name="dark_youtube_accent_color">#FFFFFF</color> |     <color name="dark_youtube_accent_color">#FFFFFF</color> | ||||||
|  |     <color name="dark_youtube_statusbar_color">#ff4336</color> | ||||||
|  |  | ||||||
|     <!-- SoundCloud --> |     <!-- SoundCloud --> | ||||||
|     <color name="light_soundcloud_primary_color">#f57c00</color> |     <color name="light_soundcloud_primary_color">#f57c00</color> | ||||||
|     <color name="light_soundcloud_dark_color">#ef6c00</color> |     <color name="light_soundcloud_dark_color">#995700</color> | ||||||
|     <color name="light_soundcloud_accent_color">#000000</color> |     <color name="light_soundcloud_accent_color">#000000</color> | ||||||
|  |     <color name="light_soundcloud_statusbar_color">#ff9100</color> | ||||||
|  |  | ||||||
|     <color name="dark_soundcloud_primary_color">#f57c00</color> |     <color name="dark_soundcloud_primary_color">#f57c00</color> | ||||||
|     <color name="dark_soundcloud_dark_color">#ef6c00</color> |     <color name="dark_soundcloud_dark_color">#995700</color> | ||||||
|     <color name="dark_soundcloud_accent_color">#FFFFFF</color> |     <color name="dark_soundcloud_accent_color">#FFFFFF</color> | ||||||
|  |     <color name="dark_soundcloud_statusbar_color">#ff9100</color> | ||||||
|  |  | ||||||
| </resources> | </resources> | ||||||
| @@ -36,6 +36,8 @@ | |||||||
|     <string name="tab_main">Main</string> |     <string name="tab_main">Main</string> | ||||||
|     <string name="tab_subscriptions">Subscriptions</string> |     <string name="tab_subscriptions">Subscriptions</string> | ||||||
|     <string name="tab_bookmarks">Bookmarks</string> |     <string name="tab_bookmarks">Bookmarks</string> | ||||||
|  |     <string name="tab_new">New Tab</string> | ||||||
|  |     <string name="tab_chose">Chose Tab</string> | ||||||
|  |  | ||||||
|     <string name="fragment_whats_new">What\'s New</string> |     <string name="fragment_whats_new">What\'s New</string> | ||||||
|  |  | ||||||
| @@ -169,6 +171,8 @@ | |||||||
|     <string name="search_history_deleted">Search history deleted.</string> |     <string name="search_history_deleted">Search history deleted.</string> | ||||||
|     <!-- error strings --> |     <!-- error strings --> | ||||||
|     <string name="general_error">Error</string> |     <string name="general_error">Error</string> | ||||||
|  |     <string name="download_to_sdcard_error_title">External storage not available.</string> | ||||||
|  |     <string name="download_to_sdcard_error_message">Download to external SD Card is not possible yet. Should the download place be reset?</string> | ||||||
|     <string name="network_error">Network error</string> |     <string name="network_error">Network error</string> | ||||||
|     <string name="could_not_load_thumbnails">Could not load all thumbnails</string> |     <string name="could_not_load_thumbnails">Could not load all thumbnails</string> | ||||||
|     <string name="youtube_signature_decryption_error">Could not decrypt video URL signature</string> |     <string name="youtube_signature_decryption_error">Could not decrypt video URL signature</string> | ||||||
| @@ -358,6 +362,9 @@ | |||||||
|  |  | ||||||
|     <!-- Content --> |     <!-- Content --> | ||||||
|     <string name="main_page_content">Content of main page</string> |     <string name="main_page_content">Content of main page</string> | ||||||
|  |     <string name="main_page_content_summary">What tabs are shown on the main page</string> | ||||||
|  |     <string name="selection">Selection</string> | ||||||
|  |     <string name="chosenTabs">Your tabs</string> | ||||||
|     <string name="blank_page_summary">Blank Page</string> |     <string name="blank_page_summary">Blank Page</string> | ||||||
|     <string name="kiosk_page_summary">Kiosk Page</string> |     <string name="kiosk_page_summary">Kiosk Page</string> | ||||||
|     <string name="subscription_page_summary">Subscription Page</string> |     <string name="subscription_page_summary">Subscription Page</string> | ||||||
|   | |||||||
| @@ -51,6 +51,7 @@ | |||||||
|         <item name="ic_import_export">@drawable/ic_import_export_black_24dp</item> |         <item name="ic_import_export">@drawable/ic_import_export_black_24dp</item> | ||||||
|         <item name="ic_save">@drawable/ic_save_black_24dp</item> |         <item name="ic_save">@drawable/ic_save_black_24dp</item> | ||||||
|         <item name="ic_backup">@drawable/ic_backup_black_24dp</item> |         <item name="ic_backup">@drawable/ic_backup_black_24dp</item> | ||||||
|  |         <item name="ic_add">@drawable/ic_add_black_24dp</item> | ||||||
|  |  | ||||||
|         <item name="separator_color">@color/light_separator_color</item> |         <item name="separator_color">@color/light_separator_color</item> | ||||||
|         <item name="contrast_background_color">@color/light_contrast_background_color</item> |         <item name="contrast_background_color">@color/light_contrast_background_color</item> | ||||||
| @@ -108,6 +109,7 @@ | |||||||
|         <item name="ic_import_export">@drawable/ic_import_export_white_24dp</item> |         <item name="ic_import_export">@drawable/ic_import_export_white_24dp</item> | ||||||
|         <item name="ic_save">@drawable/ic_save_white_24dp</item> |         <item name="ic_save">@drawable/ic_save_white_24dp</item> | ||||||
|         <item name="ic_backup">@drawable/ic_backup_white_24dp</item> |         <item name="ic_backup">@drawable/ic_backup_white_24dp</item> | ||||||
|  |         <item name="ic_add">@drawable/ic_add_white_24dp</item> | ||||||
|  |  | ||||||
|         <item name="separator_color">@color/dark_separator_color</item> |         <item name="separator_color">@color/dark_separator_color</item> | ||||||
|         <item name="contrast_background_color">@color/dark_contrast_background_color</item> |         <item name="contrast_background_color">@color/dark_contrast_background_color</item> | ||||||
|   | |||||||
| @@ -1,12 +1,14 @@ | |||||||
| <?xml version="1.0" encoding="utf-8"?> | <?xml version="1.0" encoding="utf-8"?> | ||||||
| <resources> | <resources> | ||||||
|     <!-- YouTube --> |     <!-- YouTube --> | ||||||
|     <style name="LightTheme.YouTube" parent="LightTheme.Switchable"/> |     <style name="LightTheme.YouTube" parent="LightTheme.Switchable"> | ||||||
|  |     </style> | ||||||
|  |  | ||||||
|     <style name="DarkTheme.YouTube" parent="DarkTheme.Switchable"/> |     <style name="DarkTheme.YouTube" parent="DarkTheme.Switchable"> | ||||||
|  |     </style> | ||||||
|     <style name="BlackTheme.YouTube" parent="BlackTheme.Switchable"/> |  | ||||||
|  |  | ||||||
|  |     <style name="BlackTheme.YouTube" parent="BlackTheme.Switchable"> | ||||||
|  |     </style> | ||||||
|     <!-- SoundCloud --> |     <!-- SoundCloud --> | ||||||
|     <style name="LightTheme.SoundCloud" parent="LightTheme.Switchable"> |     <style name="LightTheme.SoundCloud" parent="LightTheme.Switchable"> | ||||||
|         <item name="colorPrimary">@color/light_soundcloud_primary_color</item> |         <item name="colorPrimary">@color/light_soundcloud_primary_color</item> | ||||||
|   | |||||||
| @@ -26,4 +26,11 @@ | |||||||
|         android:key="@string/caption_settings_key" |         android:key="@string/caption_settings_key" | ||||||
|         android:title="@string/caption_setting_title" |         android:title="@string/caption_setting_title" | ||||||
|         android:summary="@string/caption_setting_description"/> |         android:summary="@string/caption_setting_description"/> | ||||||
|  |  | ||||||
|  |     <PreferenceScreen | ||||||
|  |         android:fragment="org.schabi.newpipe.settings.ChoseTabsFragment" | ||||||
|  |         android:summary="@string/main_page_content_summary" | ||||||
|  |         android:key="@string/main_page_content_key" | ||||||
|  |         android:title="@string/main_page_content"/> | ||||||
|  |  | ||||||
| </PreferenceScreen> | </PreferenceScreen> | ||||||
|   | |||||||
| @@ -43,14 +43,6 @@ | |||||||
|         android:title="@string/download_thumbnail_title" |         android:title="@string/download_thumbnail_title" | ||||||
|         android:summary="@string/download_thumbnail_summary"/> |         android:summary="@string/download_thumbnail_summary"/> | ||||||
|  |  | ||||||
|     <ListPreference |  | ||||||
|         android:defaultValue="@string/kiosk_page_key" |  | ||||||
|         android:entries="@array/main_page_content_names" |  | ||||||
|         android:entryValues="@array/main_page_content_pages" |  | ||||||
|         android:key="@string/main_page_content_key" |  | ||||||
|         android:title="@string/main_page_content" |  | ||||||
|         android:summary="%s"/> |  | ||||||
|  |  | ||||||
|     <Preference |     <Preference | ||||||
|         android:summary="@string/import_data_summary" |         android:summary="@string/import_data_summary" | ||||||
|         android:key="@string/import_data" |         android:key="@string/import_data" | ||||||
|   | |||||||
| @@ -6,7 +6,7 @@ buildscript { | |||||||
|         google() |         google() | ||||||
|     } |     } | ||||||
|     dependencies { |     dependencies { | ||||||
|         classpath 'com.android.tools.build:gradle:3.1.3' |         classpath 'com.android.tools.build:gradle:3.1.4' | ||||||
|  |  | ||||||
|         // NOTE: Do not place your application dependencies here; they belong |         // NOTE: Do not place your application dependencies here; they belong | ||||||
|         // in the individual module build.gradle files |         // in the individual module build.gradle files | ||||||
|   | |||||||
 Christian Schabesberger
					Christian Schabesberger